Re: [PATCH 7/8 v3] i2c: i801: enable irq for i801 smbustransactions
From: Jean Delvare
Date: Wed Jul 04 2012 - 16:16:34 EST
Hi again Daniel,
On Wed, 27 Jun 2012 21:54:14 +0800, Daniel Kurtz wrote:
> Add a new 'feature' to i2c-i801 to enable using i801 interrupts.
> When the feature is enabled, then an isr is installed for the device's
> pci irq.
>
> An I2C/SMBus transaction is always terminated by one of the following
> interrupt sources: FAILED, BUS_ERR, DEV_ERR, or on success: INTR.
>
> When the isr fires for one of these cases, it sets the ->status variable
> and wakes up the waitq. The waitq then saves off the status code, and
> clears ->status (in preparation for some future transaction).
> The SMBus controller generates an INTR irq at the end of each
> transaction where INTREN was set in the HST_CNT register.
>
> No locking is needed around accesses to priv->status since all writes to
> it are serialized: it is only ever set once in the isr at the end of a
> transaction, and cleared while no irqs can occur. In addition, the I2C
> adapter lock guarantees that entire I2C transactions for a single
> adapter are always serialized.
>
> For this patch, the INTREN bit is set only for SMBus block, byte and word
> transactions, but not for I2C reads or writes. The use of the DS
> (BYTE_DONE) interrupt with byte-by-byte I2C transactions is implemented in
> a subsequent patch.
Hmm, I have hit a bug with this patch. I was testing it on my ICH3-M
laptop, and while SMBus byte and word reads worked fine, SMBus block
reads did fail once in a while.
The ICH3-M lacks the block buffer, so SMBus block reads use the
i801_block_transaction_byte_by_byte function, which at this point does
not use interrupts. However, the IRQ is heavily shared on this laptop:
additionally to SMBus, it is used for USB, sound, network and even the
graphics chip. So function i801_isr() is called all the time.
If i801_isr() is being called while
i801_block_transaction_byte_by_byte() is running, there is a chance
that the status register will have some flags set, in particular INTR.
If so, the code in i801_isr() will clear the flags and wake up the
wait queue while nobody was actually waiting for. And
i801_block_transaction_byte_by_byte() will wait for an event which was
already processed, leading to a timeout.
You should be able to reproduce this bug by loading i2c-i801 with
option disable_features=0x0002, assuming your SMBus IRQ is shared.
This leads me to several thoughts (feel free to correct me if I'm
wrong, I am getting very tired):
1* This must be the reason why a bit was added to the PCI status
register starting with the ICH5, telling you if an interrupt has been
delivered to you. Checking for it as you originally did was a good idea
after all.
2* Is there any reason why you decided to clear the status bits in
i801_isr(), rather than after wait_event()? I am no expert in
interrupt handling, I admit, but letting the "caller" clear the status
bits would ensure we don't clear status bits when nobody is actually
waiting to catch them.
3* Applying this patch (7/8) without the one adding interrupt support
to i801_block_transaction_byte_by_byte (8/8) is not OK: mixing
interrupts and polling isn't safe. So unless we implement 1* or 2*,
both patches would have to be merged before being pushed upstream.
4* Even then, we have to keep in mind that i801_isr() may be called
before the transaction is finished (if IRQ is shared.) My testing
has shown that error flags may be raised before the BUSY flag is
cleared, so we should use in i801_isr() the same strict conditions
we are now using in the polling code. In other words, if we get an
interrupt but the BUSY flag is still set, then it's not our
interrupt and we should ignore it. This applies even if 2* or 3*
above are implemented, but no longer if 1* is implemented.
5* Your claim that no locking is needed might have to be revisited when
interrupts are shared.
Implementing 1* has the drawback of limiting interrupt support to ICH5
and later chips, but I suspect it is the easiest and safest way, so I
have no objection if you want to do that.
--
Jean Delvare
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/