Skip to content

Commit 5600722

Browse files
benoitmoninAndi Shyti
authored andcommitted
i2c: designware: Support of controller with IC_EMPTYFIFO_HOLD_MASTER disabled
If IC_EMPTYFIFO_HOLD_MASTER_EN parameter is 0, "Stop" and "Repeated Start" bits in command register do not exist, thus it is impossible to send several consecutive write messages in a single hardware batch. The existing implementation worked with such configuration incorrectly: all consecutive write messages are joined into a single message without any Start/Stop or Repeated Start conditions. For example, the following command: i2ctransfer -y 0 w1@0x55 0x00 w1@0x55 0x01 does the same as i2ctransfer -y 0 w2@0x55 0x00 0x01 In i2c_dw_msg_is_valid(), we ensure that we do not have such sequence of messages requiring a RESTART, aborting the transfer on controller that cannot emit them explicitly. This behavior is activated by compatible entries because the state of the IC_EMPTYFIFO_HOLD_MASTER_EN parameter cannot be detected at runtime. The new flag emptyfifo_hold_master reflects the state of the parameter, it is set to true for all controllers except those found in Mobileye SoCs. For now, the controllers in Mobileye SoCs are the only ones known to need the workaround. The behavior of the driver is left unmodified for other controllers. There is another possible problem with this controller configuration: When the CPU is putting commands to the FIFO, this process must not be interrupted because if FIFO buffer gets empty, the controller finishes the I2C transaction and generates STOP condition on the bus. If we continue writing the remainder of the message to the FIFO, the controller will start emitting a new transaction with those data. This turns a single message into multiple I2C transactions. To protect against FIFO underrun, two changes are done: First we flag the interrupt with IRQF_NO_THREAD, to prevent it from running in a thread on PREEMPT-RT kernel. This ensures that we are not interrupted when filling the FIFO as it is very time-senstive. For example, being preempted after writing a single byte in the FIFO with a 1MHz bus gives us only 18µs before an underrun. DMA would allow us to keep the interrupt threaded but it is not available on Mobileye SoC for I2C. Second in i2c_dw_process_transfer(), we abort if a STOP is detected while a read or a write is in progress. This can occur when processing a message larger than the FIFO. In that case the message is processed in parts, and rely on the TX EMPTY interrupt to refill the FIFO when it gets below a threshold. If servicing this interrupt is delayed for too long, it can trigger a FIFO underrun, thus an unwanted STOP. Originally-by: Dmitry Guzman <dmitry.guzman@mobileye.com> Signed-off-by: Benoît Monin <benoit.monin@bootlin.com> Acked-by: Mika Westerberg <mika.westerberg@linux.intel.com> Reviewed-by: Andy Shevchenko <andriy.shevchenko@linux.intel.com> Signed-off-by: Andi Shyti <andi.shyti@kernel.org> Link: https://lore.kernel.org/r/20260130-i2c-dw-v6-3-08ca1e9ece07@bootlin.com
1 parent 4a5aa00 commit 5600722

4 files changed

Lines changed: 42 additions & 0 deletions

File tree

drivers/i2c/busses/i2c-designware-common.c

Lines changed: 20 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -492,6 +492,12 @@ int i2c_dw_fw_parse_and_configure(struct dw_i2c_dev *dev)
492492

493493
dev->clk_freq_optimized = device_property_read_bool(device, "snps,clk-freq-optimized");
494494

495+
/* Mobileye controllers do not hold the clock on empty FIFO */
496+
if (device_is_compatible(device, "mobileye,eyeq6lplus-i2c"))
497+
dev->emptyfifo_hold_master = false;
498+
else
499+
dev->emptyfifo_hold_master = true;
500+
495501
i2c_dw_adjust_bus_speed(dev);
496502

497503
if (is_of_node(fwnode))
@@ -918,6 +924,20 @@ int i2c_dw_probe(struct dw_i2c_dev *dev)
918924
else
919925
irq_flags = IRQF_SHARED | IRQF_COND_SUSPEND;
920926

927+
/*
928+
* The first writing to TX FIFO buffer causes transmission start.
929+
* If IC_EMPTYFIFO_HOLD_MASTER_EN is not set, when TX FIFO gets
930+
* empty, I2C controller finishes the transaction. If writing to
931+
* FIFO is interrupted, FIFO can get empty and the transaction will
932+
* be finished prematurely. FIFO buffer is filled in IRQ handler,
933+
* but in PREEMPT_RT kernel IRQ handler by default is executed
934+
* in thread that can be preempted with another higher priority
935+
* thread or an interrupt. So, IRQF_NO_THREAD flag is required in
936+
* order to prevent any preemption when filling the FIFO.
937+
*/
938+
if (!dev->emptyfifo_hold_master)
939+
irq_flags |= IRQF_NO_THREAD;
940+
921941
ret = i2c_dw_acquire_lock(dev);
922942
if (ret)
923943
return ret;

drivers/i2c/busses/i2c-designware-core.h

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -260,6 +260,8 @@ struct reset_control;
260260
* @clk_freq_optimized: if this value is true, it means the hardware reduces
261261
* its internal clock frequency by reducing the internal latency required
262262
* to generate the high period and low period of SCL line.
263+
* @emptyfifo_hold_master: true if the controller acting as master holds
264+
* the clock when the Tx FIFO is empty instead of emitting a stop.
263265
*
264266
* HCNT and LCNT parameters can be used if the platform knows more accurate
265267
* values than the one computed based only on the input clock frequency.
@@ -318,6 +320,7 @@ struct dw_i2c_dev {
318320
struct i2c_bus_recovery_info rinfo;
319321
u32 bus_capacitance_pF;
320322
bool clk_freq_optimized;
323+
bool emptyfifo_hold_master;
321324
};
322325

323326
#define ACCESS_INTR_MASK BIT(0)

drivers/i2c/busses/i2c-designware-master.c

Lines changed: 18 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -652,6 +652,14 @@ static void i2c_dw_process_transfer(struct dw_i2c_dev *dev, unsigned int stat)
652652
if (stat & DW_IC_INTR_TX_EMPTY)
653653
i2c_dw_xfer_msg(dev);
654654

655+
/* Abort if we detect a STOP in the middle of a read or a write */
656+
if ((stat & DW_IC_INTR_STOP_DET) &&
657+
(dev->status & (STATUS_READ_IN_PROGRESS | STATUS_WRITE_IN_PROGRESS))) {
658+
dev_err(dev->dev, "spurious STOP detected\n");
659+
dev->rx_outstanding = 0;
660+
dev->msg_err = -EIO;
661+
}
662+
655663
/*
656664
* No need to modify or disable the interrupt mask here.
657665
* i2c_dw_xfer_msg() will take care of it according to
@@ -834,6 +842,16 @@ i2c_dw_msg_is_valid(struct dw_i2c_dev *dev, const struct i2c_msg *msgs, size_t i
834842
return false;
835843
}
836844

845+
/*
846+
* Make sure we don't need explicit RESTART between two messages
847+
* in the same direction for controllers that cannot emit them.
848+
*/
849+
if (!dev->emptyfifo_hold_master &&
850+
(msgs[idx - 1].flags & I2C_M_RD) == (msgs[idx].flags & I2C_M_RD)) {
851+
dev_err(dev->dev, "cannot emit RESTART\n");
852+
return false;
853+
}
854+
837855
return true;
838856
}
839857

drivers/i2c/busses/i2c-designware-platdrv.c

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -267,6 +267,7 @@ static void dw_i2c_plat_remove(struct platform_device *pdev)
267267
}
268268

269269
static const struct of_device_id dw_i2c_of_match[] = {
270+
{ .compatible = "mobileye,eyeq6lplus-i2c" },
270271
{ .compatible = "mscc,ocelot-i2c" },
271272
{ .compatible = "snps,designware-i2c" },
272273
{}

0 commit comments

Comments
 (0)