Commit 35780e86 authored by Alexander Sverdlin's avatar Alexander Sverdlin Committed by Wolfram Sang

i2c: davinci: Optimize clock generation on Keystone SoC

According to "KeyStone Architecture Inter-IC Control Bus User Guide", fixed
additive part of frequency divisors (referred as "d" in the code and datasheet)
always equals to 6, independent of module clock prescaler.

                         module clock frequency
master clock frequency = ----------------------
                         (ICCL + 6) + (ICCH + 6)

It was not the case with original Davinci IP. Introduce new compatible property
"ti,keystone-i2c", which triggers special handling in the driver.

Without this change Keystone-based systems (having 204.8MHz input clock) choose
prescaler 29 (PSC=28). Using d=5 in this case leads to bus bitrate ~353kHz
instead of requested 400kHz. After correction, assuming d=6 bus rate is ~392kHz.
This gives ~11% transfer rate increase.
Signed-off-by: default avatarAlexander Sverdlin <alexander.sverdlin@nokia.com>
Reviewed-by: default avatarGrygorii Strashko <grygorii.strashko@ti.com>
Tested-by: default avatarHemanth Guruva Reddy <hemanth.guruva_reddy@nokia.com>
Tested-by: default avatarLukasz Gemborowski <lukasz.gemborowski@nokia.com>
Signed-off-by: default avatarWolfram Sang <wsa@the-dreams.de>
parent 064181b0
* Texas Instruments Davinci I2C * Texas Instruments Davinci/Keystone I2C
This file provides information, what the device node for the This file provides information, what the device node for the
davinci i2c interface contain. davinci/keystone i2c interface contains.
Required properties: Required properties:
- compatible: "ti,davinci-i2c"; - compatible: "ti,davinci-i2c" or "ti,keystone-i2c";
- reg : Offset and length of the register set for the device - reg : Offset and length of the register set for the device
Recommended properties : Recommended properties :
......
...@@ -181,6 +181,7 @@ static void i2c_davinci_calc_clk_dividers(struct davinci_i2c_dev *dev) ...@@ -181,6 +181,7 @@ static void i2c_davinci_calc_clk_dividers(struct davinci_i2c_dev *dev)
u32 clkh; u32 clkh;
u32 clkl; u32 clkl;
u32 input_clock = clk_get_rate(dev->clk); u32 input_clock = clk_get_rate(dev->clk);
struct device_node *of_node = dev->dev->of_node;
/* NOTE: I2C Clock divider programming info /* NOTE: I2C Clock divider programming info
* As per I2C specs the following formulas provide prescaler * As per I2C specs the following formulas provide prescaler
...@@ -196,6 +197,9 @@ static void i2c_davinci_calc_clk_dividers(struct davinci_i2c_dev *dev) ...@@ -196,6 +197,9 @@ static void i2c_davinci_calc_clk_dividers(struct davinci_i2c_dev *dev)
* where if PSC == 0, d = 7, * where if PSC == 0, d = 7,
* if PSC == 1, d = 6 * if PSC == 1, d = 6
* if PSC > 1 , d = 5 * if PSC > 1 , d = 5
*
* Note:
* d is always 6 on Keystone I2C controller
*/ */
/* get minimum of 7 MHz clock, but max of 12 MHz */ /* get minimum of 7 MHz clock, but max of 12 MHz */
...@@ -204,6 +208,9 @@ static void i2c_davinci_calc_clk_dividers(struct davinci_i2c_dev *dev) ...@@ -204,6 +208,9 @@ static void i2c_davinci_calc_clk_dividers(struct davinci_i2c_dev *dev)
psc++; /* better to run under spec than over */ psc++; /* better to run under spec than over */
d = (psc >= 2) ? 5 : 7 - psc; d = (psc >= 2) ? 5 : 7 - psc;
if (of_node && of_device_is_compatible(of_node, "ti,keystone-i2c"))
d = 6;
clk = ((input_clock / (psc + 1)) / (pdata->bus_freq * 1000)); clk = ((input_clock / (psc + 1)) / (pdata->bus_freq * 1000));
/* Avoid driving the bus too fast because of rounding errors above */ /* Avoid driving the bus too fast because of rounding errors above */
if (input_clock / (psc + 1) / clk > pdata->bus_freq * 1000) if (input_clock / (psc + 1) / clk > pdata->bus_freq * 1000)
...@@ -726,6 +733,7 @@ static struct i2c_algorithm i2c_davinci_algo = { ...@@ -726,6 +733,7 @@ static struct i2c_algorithm i2c_davinci_algo = {
static const struct of_device_id davinci_i2c_of_match[] = { static const struct of_device_id davinci_i2c_of_match[] = {
{.compatible = "ti,davinci-i2c", }, {.compatible = "ti,davinci-i2c", },
{.compatible = "ti,keystone-i2c", },
{}, {},
}; };
MODULE_DEVICE_TABLE(of, davinci_i2c_of_match); MODULE_DEVICE_TABLE(of, davinci_i2c_of_match);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment