Re: hda: error: DMA in progress..

From: Martin Dalecki (dalecki@evision-ventures.com)
Date: Fri Jun 21 2002 - 05:31:49 EST


Użytkownik Jens Axboe napisał:
> On Fri, Jun 21 2002, Martin Dalecki wrote:

>
>>And I was asking about it's possible interactions with TCQ.
>
>
> Haven't even tried TCQ yet, the above is just plain dma (no travelstarts
> can do tcq).

Argh...

>>
>> if (blk_queue_plugged(&drive->queue)) {
>> BUG_ON(!drive->using_tcq);
>> break;

>
> Not exactly, let me see if I remember the race here... The queue can
> become plugged when we queue one request with the drive (the only on the
> queue at that time), and then try to queue another right after (hence
> only a tcq issue). In that time period, we drop the queue lock, so it's
> indeed possible for the block layer to plug the queue before we reach
> the above code again. The drive can be in two states here, 1) IDE_DMA is
> set because the drive didn't release the bus (or it did, and it already
> reconnected), or 2) drive is disconnected from the bus.

OK. We have now just one single place where IDE_DMA gets unset ->
udma_stop. This to too early to reset IDE_BUSY. However it well
may be that ide_dma_intr() simply doesn't care about IDE_BUSY.
Let's have a look...

>
> For non-tcq, hitting IDE_DMA set queue_commands() is a bug. The old
> IDE_BUSY/IDE_DMA worked because IDE_DMA must not be set if IDE_BUSY is
> not set.
>
>
>>This time it's no new damage - just detecting weak code
>>from the past...
>
>
> Smells like new breakage to me :-)

Well lets look at ata_irq_intr, the end of it:

         * Note that handler() may have set things up for another
         * interrupt to occur soon, but it cannot happen until
         * we exit from this routine, because it will be the
         * same irq as is currently being serviced here, and Linux
         * won't allow another of the same (on any CPU) until we return.
         */
        if (startstop == ide_stopped) {
                if (!ch->handler) { /* paranoia */
                        clear_bit(IDE_BUSY, ch->active);
                        do_request(ch);
                } else {
                        printk("%s: %s: huh? expected NULL handler on exit\n", drive->name, __FUNCTION__);
                }
        } else if (startstop == ide_released)
                queue_commands(drive);

I think the above needs more tough now...

-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/



This archive was generated by hypermail 2b29 : Sun Jun 23 2002 - 22:00:24 EST