]> git.baikalelectronics.ru Git - kernel.git/commit
i2c: stm32f7: fix a race in slave mode with arbitration loss irq
authorFabrice Gasnier <fabrice.gasnier@st.com>
Tue, 1 Oct 2019 08:51:09 +0000 (10:51 +0200)
committerWolfram Sang <wsa@the-dreams.de>
Thu, 24 Oct 2019 18:52:17 +0000 (20:52 +0200)
commit734b45ea1a212d8daf27c7e1905ecbf78b1f8d3f
tree83718c6f89b3dce44c0357ef3025e46a6aa04c19
parent25d58e162783b3388e5c291e473e28df983fec02
i2c: stm32f7: fix a race in slave mode with arbitration loss irq

When in slave mode, an arbitration loss (ARLO) may be detected before the
slave had a chance to detect the stop condition (STOPF in ISR).
This is seen when two master + slave adapters switch their roles. It
provokes the i2c bus to be stuck, busy as SCL line is stretched.
- the I2C_SLAVE_STOP event is never generated due to STOPF flag is set but
  don't generate an irq (race with ARLO irq, STOPIE is masked). STOPF flag
  remains set until next master xfer (e.g. when STOPIE irq get unmasked).
  In this case, completion is generated too early: immediately upon new
  transfer request (then it doesn't send all data).
- Some data get stuck in TXDR register. As a consequence, the controller
  stretches the SCL line: the bus gets busy until a future master transfer
  triggers the bus busy / recovery mechanism (this can take time... and
  may never happen at all)

So choice is to let the STOPF being detected by the slave isr handler,
to properly handle this stop condition. E.g. don't mask IRQs in error
handler, when the slave is running.

Fixes: b819485944c3 ("i2c: i2c-stm32f7: Add slave support")
Signed-off-by: Fabrice Gasnier <fabrice.gasnier@st.com>
Reviewed-by: Pierre-Yves MORDRET <pierre-yves.mordret@st.com>
Signed-off-by: Wolfram Sang <wsa@the-dreams.de>
drivers/i2c/busses/i2c-stm32f7.c