mirror of
https://github.com/edk2-porting/linux-next.git
synced 2024-12-18 02:04:05 +08:00
i2c: i2c-sh_mobile: optimize ICCH/ICCL values according to I2C bus speed
ICCH/ICCL values is supposed to be calculated/optimized to strictly meet the timing specs required by the I2C standard. The resulting I2C bus speed does not matter at all, if it's less than 100 or 400 kHz. With this change, sh_mobile_i2c_icch() is virtually identical to sh_mobile_i2c_iccl(), but they're providing good descriptions of SH-/R-Mobile I2C hardware spec, and I'd leave them as separated. Also fix a typo in the comment, print icch/iccl values at probe, etc. Signed-off-by: Shinya Kuribayashi <shinya.kuribayashi.px@renesas.com> [wsa: squashed two patches for bisectability] Signed-off-by: Wolfram Sang <w.sang@pengutronix.de>
This commit is contained in:
parent
7b0e62920a
commit
23a612916a
@ -122,9 +122,9 @@ struct sh_mobile_i2c_data {
|
|||||||
unsigned long bus_speed;
|
unsigned long bus_speed;
|
||||||
struct clk *clk;
|
struct clk *clk;
|
||||||
u_int8_t icic;
|
u_int8_t icic;
|
||||||
u_int8_t iccl;
|
|
||||||
u_int8_t icch;
|
|
||||||
u_int8_t flags;
|
u_int8_t flags;
|
||||||
|
u_int16_t iccl;
|
||||||
|
u_int16_t icch;
|
||||||
|
|
||||||
spinlock_t lock;
|
spinlock_t lock;
|
||||||
wait_queue_head_t wait;
|
wait_queue_head_t wait;
|
||||||
@ -135,7 +135,8 @@ struct sh_mobile_i2c_data {
|
|||||||
|
|
||||||
#define IIC_FLAG_HAS_ICIC67 (1 << 0)
|
#define IIC_FLAG_HAS_ICIC67 (1 << 0)
|
||||||
|
|
||||||
#define NORMAL_SPEED 100000 /* FAST_SPEED 400000 */
|
#define STANDARD_MODE 100000
|
||||||
|
#define FAST_MODE 400000
|
||||||
|
|
||||||
/* Register offsets */
|
/* Register offsets */
|
||||||
#define ICDR 0x00
|
#define ICDR 0x00
|
||||||
@ -187,55 +188,81 @@ static void iic_set_clr(struct sh_mobile_i2c_data *pd, int offs,
|
|||||||
iic_wr(pd, offs, (iic_rd(pd, offs) | set) & ~clr);
|
iic_wr(pd, offs, (iic_rd(pd, offs) | set) & ~clr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static u32 sh_mobile_i2c_iccl(unsigned long count_khz, u32 tLOW, u32 tf, int offset)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Conditional expression:
|
||||||
|
* ICCL >= COUNT_CLK * (tLOW + tf)
|
||||||
|
*
|
||||||
|
* SH-Mobile IIC hardware starts counting the LOW period of
|
||||||
|
* the SCL signal (tLOW) as soon as it pulls the SCL line.
|
||||||
|
* In order to meet the tLOW timing spec, we need to take into
|
||||||
|
* account the fall time of SCL signal (tf). Default tf value
|
||||||
|
* should be 0.3 us, for safety.
|
||||||
|
*/
|
||||||
|
return (((count_khz * (tLOW + tf)) + 5000) / 10000) + offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
static u32 sh_mobile_i2c_icch(unsigned long count_khz, u32 tHIGH, u32 tf, int offset)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Conditional expression:
|
||||||
|
* ICCH >= COUNT_CLK * (tHIGH + tf)
|
||||||
|
*
|
||||||
|
* SH-Mobile IIC hardware is aware of SCL transition period 'tr',
|
||||||
|
* and can ignore it. SH-Mobile IIC controller starts counting
|
||||||
|
* the HIGH period of the SCL signal (tHIGH) after the SCL input
|
||||||
|
* voltage increases at VIH.
|
||||||
|
*
|
||||||
|
* Afterward it turned out calculating ICCH using only tHIGH spec
|
||||||
|
* will result in violation of the tHD;STA timing spec. We need
|
||||||
|
* to take into account the fall time of SDA signal (tf) at START
|
||||||
|
* condition, in order to meet both tHIGH and tHD;STA specs.
|
||||||
|
*/
|
||||||
|
return (((count_khz * (tHIGH + tf)) + 5000) / 10000) + offset;
|
||||||
|
}
|
||||||
|
|
||||||
static void sh_mobile_i2c_init(struct sh_mobile_i2c_data *pd)
|
static void sh_mobile_i2c_init(struct sh_mobile_i2c_data *pd)
|
||||||
{
|
{
|
||||||
unsigned long i2c_clk;
|
unsigned long i2c_clk_khz;
|
||||||
u_int32_t num;
|
u32 tHIGH, tLOW, tf;
|
||||||
u_int32_t denom;
|
int offset;
|
||||||
u_int32_t tmp;
|
|
||||||
|
|
||||||
/* Get clock rate after clock is enabled */
|
/* Get clock rate after clock is enabled */
|
||||||
clk_enable(pd->clk);
|
clk_enable(pd->clk);
|
||||||
i2c_clk = clk_get_rate(pd->clk);
|
i2c_clk_khz = clk_get_rate(pd->clk) / 1000;
|
||||||
|
|
||||||
/* Calculate the value for iccl. From the data sheet:
|
if (pd->bus_speed == STANDARD_MODE) {
|
||||||
* iccl = (p clock / transfer rate) * (L / (L + H))
|
tLOW = 47; /* tLOW = 4.7 us */
|
||||||
* where L and H are the SCL low/high ratio (5/4 in this case).
|
tHIGH = 40; /* tHD;STA = tHIGH = 4.0 us */
|
||||||
* We also round off the result.
|
tf = 3; /* tf = 0.3 us */
|
||||||
*/
|
offset = 0; /* No offset */
|
||||||
num = i2c_clk * 5;
|
} else if (pd->bus_speed == FAST_MODE) {
|
||||||
denom = pd->bus_speed * 9;
|
tLOW = 13; /* tLOW = 1.3 us */
|
||||||
tmp = num * 10 / denom;
|
tHIGH = 6; /* tHD;STA = tHIGH = 0.6 us */
|
||||||
if (tmp % 10 >= 5)
|
tf = 3; /* tf = 0.3 us */
|
||||||
pd->iccl = (u_int8_t)((num/denom) + 1);
|
offset = 0; /* No offset */
|
||||||
else
|
} else {
|
||||||
pd->iccl = (u_int8_t)(num/denom);
|
dev_err(pd->dev, "unrecognized bus speed %lu Hz\n",
|
||||||
|
pd->bus_speed);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
pd->iccl = sh_mobile_i2c_iccl(i2c_clk_khz, tLOW, tf, offset);
|
||||||
/* one more bit of ICCL in ICIC */
|
/* one more bit of ICCL in ICIC */
|
||||||
if (pd->flags & IIC_FLAG_HAS_ICIC67) {
|
if ((pd->iccl > 0xff) && (pd->flags & IIC_FLAG_HAS_ICIC67))
|
||||||
if ((num/denom) > 0xff)
|
pd->icic |= ICIC_ICCLB8;
|
||||||
pd->icic |= ICIC_ICCLB8;
|
|
||||||
else
|
|
||||||
pd->icic &= ~ICIC_ICCLB8;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Calculate the value for icch. From the data sheet:
|
|
||||||
icch = (p clock / transfer rate) * (H / (L + H)) */
|
|
||||||
num = i2c_clk * 4;
|
|
||||||
tmp = num * 10 / denom;
|
|
||||||
if (tmp % 10 >= 5)
|
|
||||||
pd->icch = (u_int8_t)((num/denom) + 1);
|
|
||||||
else
|
else
|
||||||
pd->icch = (u_int8_t)(num/denom);
|
pd->icic &= ~ICIC_ICCLB8;
|
||||||
|
|
||||||
|
pd->icch = sh_mobile_i2c_icch(i2c_clk_khz, tHIGH, tf, offset);
|
||||||
/* one more bit of ICCH in ICIC */
|
/* one more bit of ICCH in ICIC */
|
||||||
if (pd->flags & IIC_FLAG_HAS_ICIC67) {
|
if ((pd->icch > 0xff) && (pd->flags & IIC_FLAG_HAS_ICIC67))
|
||||||
if ((num/denom) > 0xff)
|
pd->icic |= ICIC_ICCHB8;
|
||||||
pd->icic |= ICIC_ICCHB8;
|
else
|
||||||
else
|
pd->icic &= ~ICIC_ICCHB8;
|
||||||
pd->icic &= ~ICIC_ICCHB8;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
out:
|
||||||
clk_disable(pd->clk);
|
clk_disable(pd->clk);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -252,8 +279,8 @@ static void activate_ch(struct sh_mobile_i2c_data *pd)
|
|||||||
iic_wr(pd, ICIC, 0);
|
iic_wr(pd, ICIC, 0);
|
||||||
|
|
||||||
/* Set the clock */
|
/* Set the clock */
|
||||||
iic_wr(pd, ICCL, pd->iccl);
|
iic_wr(pd, ICCL, pd->iccl & 0xff);
|
||||||
iic_wr(pd, ICCH, pd->icch);
|
iic_wr(pd, ICCH, pd->icch & 0xff);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void deactivate_ch(struct sh_mobile_i2c_data *pd)
|
static void deactivate_ch(struct sh_mobile_i2c_data *pd)
|
||||||
@ -457,8 +484,8 @@ static int start_ch(struct sh_mobile_i2c_data *pd, struct i2c_msg *usr_msg)
|
|||||||
iic_set_clr(pd, ICCR, ICCR_ICE, 0);
|
iic_set_clr(pd, ICCR, ICCR_ICE, 0);
|
||||||
|
|
||||||
/* Set the clock */
|
/* Set the clock */
|
||||||
iic_wr(pd, ICCL, pd->iccl);
|
iic_wr(pd, ICCL, pd->iccl & 0xff);
|
||||||
iic_wr(pd, ICCH, pd->icch);
|
iic_wr(pd, ICCH, pd->icch & 0xff);
|
||||||
|
|
||||||
pd->msg = usr_msg;
|
pd->msg = usr_msg;
|
||||||
pd->pos = -1;
|
pd->pos = -1;
|
||||||
@ -627,8 +654,8 @@ static int sh_mobile_i2c_probe(struct platform_device *dev)
|
|||||||
goto err_irq;
|
goto err_irq;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Use platformd data bus speed or NORMAL_SPEED */
|
/* Use platform data bus speed or STANDARD_MODE */
|
||||||
pd->bus_speed = NORMAL_SPEED;
|
pd->bus_speed = STANDARD_MODE;
|
||||||
if (pdata && pdata->bus_speed)
|
if (pdata && pdata->bus_speed)
|
||||||
pd->bus_speed = pdata->bus_speed;
|
pd->bus_speed = pdata->bus_speed;
|
||||||
|
|
||||||
@ -675,8 +702,9 @@ static int sh_mobile_i2c_probe(struct platform_device *dev)
|
|||||||
goto err_all;
|
goto err_all;
|
||||||
}
|
}
|
||||||
|
|
||||||
dev_info(&dev->dev, "I2C adapter %d with bus speed %lu Hz\n",
|
dev_info(&dev->dev,
|
||||||
adap->nr, pd->bus_speed);
|
"I2C adapter %d with bus speed %lu Hz (L/H=%x/%x)\n",
|
||||||
|
adap->nr, pd->bus_speed, pd->iccl, pd->icch);
|
||||||
|
|
||||||
of_i2c_register_devices(adap);
|
of_i2c_register_devices(adap);
|
||||||
return 0;
|
return 0;
|
||||||
|
Loading…
Reference in New Issue
Block a user