i2c-for-6.12-rc1-additional_fixes
I2C host fixes for v6.12-rc1 (from Andi) The DesignWare driver now has the correct ENABLE-ABORT sequence, ensuring ABORT can always be sent when needed. In the SynQuacer controller we now check for PCLK as an optional clock, allowing ACPI to directly provide the clock rate. The recent KEBA driver required a dependency fix in Kconfig. The XIIC driver now has a corrected power suspend sequence. -----BEGIN PGP SIGNATURE----- iQIzBAABCgAdFiEEOZGx6rniZ1Gk92RdFA3kzBSgKbYFAmb5Ex0ACgkQFA3kzBSg KbbNIxAAm1GYB7hSOoIZtv7amoMOJYPqEEh037/9RITF9U8zFL9JBWTePR+ksv6W 0FA4aEUYq8RyaJeQEVDHOdONiDJeYwVM8uxF02rwYx4tJYlLeh4J0H+AXnGPxQpO 17woQt1fx5hdtdyx+aJ6Zr4eJCo9XUaCwjvEQH4oZE2PFesP3b/9HSBq82gFBSHA lEnY1F/SfMNeF90y74f93fiZJLQNdfYKtzMS8iZ5eGVjokXcPP7TfrxIOughRlVT h+cNQaGd7g4YrYvMrTTDiNu8Bnc5nJcjyOcaQsXOpCnxljH4gb1V39RPQ6ATvTBQ uEPQCBU2ibZt2AbRcyWNRkw9sAuHdyOm0wv09ZsChpXafcM9BjzfF/xn6DlQuf64 d1COXxpBN5yI/4AU/Juojoy5/C8fMj81VtveXXlaISsyZzUo49XwM263PSXgkO8y 39A50Px1uVdeAjnVOu9Tsg/fJuAJRolIOVpw+/jWRZ1OXidp4MtmzL8bf2gXH7ts 4F7G8BOtoLXRM56AeO15EQ7YLJaw/03jb57EB+lREqy0qLDFJ0d13o+VcRmCvVK2 uK4RSN/y7poOOSaXnf5Wm61XYV8yu4XTyR8hN6EP/hsBcM0o1DLi4nLoI7S9QMjc +ic0In9qQDo90oTeF3tE+ijCQUTAOzR4wids1DjQuUBleUu71C0= =CZPe -----END PGP SIGNATURE----- Merge tag 'i2c-for-6.12-rc1-additional_fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/wsa/linux Pull i2c fixes from Wolfram Sang: - fix DesignWare driver ENABLE-ABORT sequence, ensuring ABORT can always be sent when needed - check for PCLK in the SynQuacer controller as an optional clock, allowing ACPI to directly provide the clock rate - KEBA driver Kconfig dependency fix - fix XIIC driver power suspend sequence * tag 'i2c-for-6.12-rc1-additional_fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/wsa/linux: i2c: xiic: Fix pm_runtime_set_suspended() with runtime pm enabled i2c: keba: I2C_KEBA should depend on KEBA_CP500 i2c: synquacer: Deal with optional PCLK correctly i2c: designware: fix controller is holding SCL low while ENABLE bit is disabled
This commit is contained in:
commit
907537f570
@ -782,6 +782,7 @@ config I2C_JZ4780
|
|||||||
config I2C_KEBA
|
config I2C_KEBA
|
||||||
tristate "KEBA I2C controller support"
|
tristate "KEBA I2C controller support"
|
||||||
depends on HAS_IOMEM
|
depends on HAS_IOMEM
|
||||||
|
depends on KEBA_CP500 || COMPILE_TEST
|
||||||
select AUXILIARY_BUS
|
select AUXILIARY_BUS
|
||||||
help
|
help
|
||||||
This driver supports the I2C controller found in KEBA system FPGA
|
This driver supports the I2C controller found in KEBA system FPGA
|
||||||
|
@ -523,6 +523,7 @@ err_release_lock:
|
|||||||
|
|
||||||
void __i2c_dw_disable(struct dw_i2c_dev *dev)
|
void __i2c_dw_disable(struct dw_i2c_dev *dev)
|
||||||
{
|
{
|
||||||
|
struct i2c_timings *t = &dev->timings;
|
||||||
unsigned int raw_intr_stats;
|
unsigned int raw_intr_stats;
|
||||||
unsigned int enable;
|
unsigned int enable;
|
||||||
int timeout = 100;
|
int timeout = 100;
|
||||||
@ -535,6 +536,19 @@ void __i2c_dw_disable(struct dw_i2c_dev *dev)
|
|||||||
|
|
||||||
abort_needed = raw_intr_stats & DW_IC_INTR_MST_ON_HOLD;
|
abort_needed = raw_intr_stats & DW_IC_INTR_MST_ON_HOLD;
|
||||||
if (abort_needed) {
|
if (abort_needed) {
|
||||||
|
if (!(enable & DW_IC_ENABLE_ENABLE)) {
|
||||||
|
regmap_write(dev->map, DW_IC_ENABLE, DW_IC_ENABLE_ENABLE);
|
||||||
|
/*
|
||||||
|
* Wait 10 times the signaling period of the highest I2C
|
||||||
|
* transfer supported by the driver (for 400KHz this is
|
||||||
|
* 25us) to ensure the I2C ENABLE bit is already set
|
||||||
|
* as described in the DesignWare I2C databook.
|
||||||
|
*/
|
||||||
|
fsleep(DIV_ROUND_CLOSEST_ULL(10 * MICRO, t->bus_freq_hz));
|
||||||
|
/* Set ENABLE bit before setting ABORT */
|
||||||
|
enable |= DW_IC_ENABLE_ENABLE;
|
||||||
|
}
|
||||||
|
|
||||||
regmap_write(dev->map, DW_IC_ENABLE, enable | DW_IC_ENABLE_ABORT);
|
regmap_write(dev->map, DW_IC_ENABLE, enable | DW_IC_ENABLE_ABORT);
|
||||||
ret = regmap_read_poll_timeout(dev->map, DW_IC_ENABLE, enable,
|
ret = regmap_read_poll_timeout(dev->map, DW_IC_ENABLE, enable,
|
||||||
!(enable & DW_IC_ENABLE_ABORT), 10,
|
!(enable & DW_IC_ENABLE_ABORT), 10,
|
||||||
|
@ -108,6 +108,7 @@
|
|||||||
DW_IC_INTR_RX_UNDER | \
|
DW_IC_INTR_RX_UNDER | \
|
||||||
DW_IC_INTR_RD_REQ)
|
DW_IC_INTR_RD_REQ)
|
||||||
|
|
||||||
|
#define DW_IC_ENABLE_ENABLE BIT(0)
|
||||||
#define DW_IC_ENABLE_ABORT BIT(1)
|
#define DW_IC_ENABLE_ABORT BIT(1)
|
||||||
|
|
||||||
#define DW_IC_STATUS_ACTIVITY BIT(0)
|
#define DW_IC_STATUS_ACTIVITY BIT(0)
|
||||||
|
@ -271,6 +271,34 @@ static void i2c_dw_xfer_init(struct dw_i2c_dev *dev)
|
|||||||
__i2c_dw_write_intr_mask(dev, DW_IC_INTR_MASTER_MASK);
|
__i2c_dw_write_intr_mask(dev, DW_IC_INTR_MASTER_MASK);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This function waits for the controller to be idle before disabling I2C
|
||||||
|
* When the controller is not in the IDLE state, the MST_ACTIVITY bit
|
||||||
|
* (IC_STATUS[5]) is set.
|
||||||
|
*
|
||||||
|
* Values:
|
||||||
|
* 0x1 (ACTIVE): Controller not idle
|
||||||
|
* 0x0 (IDLE): Controller is idle
|
||||||
|
*
|
||||||
|
* The function is called after completing the current transfer.
|
||||||
|
*
|
||||||
|
* Returns:
|
||||||
|
* False when the controller is in the IDLE state.
|
||||||
|
* True when the controller is in the ACTIVE state.
|
||||||
|
*/
|
||||||
|
static bool i2c_dw_is_controller_active(struct dw_i2c_dev *dev)
|
||||||
|
{
|
||||||
|
u32 status;
|
||||||
|
|
||||||
|
regmap_read(dev->map, DW_IC_STATUS, &status);
|
||||||
|
if (!(status & DW_IC_STATUS_MASTER_ACTIVITY))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return regmap_read_poll_timeout(dev->map, DW_IC_STATUS, status,
|
||||||
|
!(status & DW_IC_STATUS_MASTER_ACTIVITY),
|
||||||
|
1100, 20000) != 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int i2c_dw_check_stopbit(struct dw_i2c_dev *dev)
|
static int i2c_dw_check_stopbit(struct dw_i2c_dev *dev)
|
||||||
{
|
{
|
||||||
u32 val;
|
u32 val;
|
||||||
@ -806,6 +834,16 @@ i2c_dw_xfer(struct i2c_adapter *adap, struct i2c_msg msgs[], int num)
|
|||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This happens rarely (~1:500) and is hard to reproduce. Debug trace
|
||||||
|
* showed that IC_STATUS had value of 0x23 when STOP_DET occurred,
|
||||||
|
* if disable IC_ENABLE.ENABLE immediately that can result in
|
||||||
|
* IC_RAW_INTR_STAT.MASTER_ON_HOLD holding SCL low. Check if
|
||||||
|
* controller is still ACTIVE before disabling I2C.
|
||||||
|
*/
|
||||||
|
if (i2c_dw_is_controller_active(dev))
|
||||||
|
dev_err(dev->dev, "controller active\n");
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We must disable the adapter before returning and signaling the end
|
* We must disable the adapter before returning and signaling the end
|
||||||
* of the current transfer. Otherwise the hardware might continue
|
* of the current transfer. Otherwise the hardware might continue
|
||||||
|
@ -550,12 +550,13 @@ static int synquacer_i2c_probe(struct platform_device *pdev)
|
|||||||
device_property_read_u32(&pdev->dev, "socionext,pclk-rate",
|
device_property_read_u32(&pdev->dev, "socionext,pclk-rate",
|
||||||
&i2c->pclkrate);
|
&i2c->pclkrate);
|
||||||
|
|
||||||
pclk = devm_clk_get_enabled(&pdev->dev, "pclk");
|
pclk = devm_clk_get_optional_enabled(&pdev->dev, "pclk");
|
||||||
if (IS_ERR(pclk))
|
if (IS_ERR(pclk))
|
||||||
return dev_err_probe(&pdev->dev, PTR_ERR(pclk),
|
return dev_err_probe(&pdev->dev, PTR_ERR(pclk),
|
||||||
"failed to get and enable clock\n");
|
"failed to get and enable clock\n");
|
||||||
|
|
||||||
i2c->pclkrate = clk_get_rate(pclk);
|
if (pclk)
|
||||||
|
i2c->pclkrate = clk_get_rate(pclk);
|
||||||
|
|
||||||
if (i2c->pclkrate < SYNQUACER_I2C_MIN_CLK_RATE ||
|
if (i2c->pclkrate < SYNQUACER_I2C_MIN_CLK_RATE ||
|
||||||
i2c->pclkrate > SYNQUACER_I2C_MAX_CLK_RATE)
|
i2c->pclkrate > SYNQUACER_I2C_MAX_CLK_RATE)
|
||||||
|
@ -1337,8 +1337,8 @@ static int xiic_i2c_probe(struct platform_device *pdev)
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_pm_disable:
|
err_pm_disable:
|
||||||
pm_runtime_set_suspended(&pdev->dev);
|
|
||||||
pm_runtime_disable(&pdev->dev);
|
pm_runtime_disable(&pdev->dev);
|
||||||
|
pm_runtime_set_suspended(&pdev->dev);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user