Re: [PATCH] block: Limit number of items taken from the I/O scheduler in one go

From: Salman Qazi
Date: Fri Feb 07 2020 - 13:45:33 EST


On Fri, Feb 7, 2020 at 7:26 AM Bart Van Assche <bvanassche@xxxxxxx> wrote:
>
> On 2020-02-06 13:12, Salman Qazi wrote:
> > + *
> > + * Returns true if hctx->dispatch was found non-empty and
> > + * run_work has to be run again.
>
> Please elaborate this comment and explain why this is necessary (to
> avoid that flush processing is postponed forever).
>
> > + * Returns true if hctx->dispatch was found non-empty and
> > + * run_work has to be run again.
>
> Same comment here.

Will do.

>
> > +again:
> > + run_again = false;
> > +
> > /*
> > * If we have previous entries on our dispatch list, grab them first for
> > * more fair dispatch.
> > @@ -208,19 +234,28 @@ void blk_mq_sched_dispatch_requests(struct blk_mq_hw_ctx *hctx)
> > blk_mq_sched_mark_restart_hctx(hctx);
> > if (blk_mq_dispatch_rq_list(q, &rq_list, false)) {
> > if (has_sched_dispatch)
> > - blk_mq_do_dispatch_sched(hctx);
> > + run_again = blk_mq_do_dispatch_sched(hctx);
> > else
> > - blk_mq_do_dispatch_ctx(hctx);
> > + run_again = blk_mq_do_dispatch_ctx(hctx);
> > }
> > } else if (has_sched_dispatch) {
> > - blk_mq_do_dispatch_sched(hctx);
> > + run_again = blk_mq_do_dispatch_sched(hctx);
> > } else if (hctx->dispatch_busy) {
> > /* dequeue request one by one from sw queue if queue is busy */
> > - blk_mq_do_dispatch_ctx(hctx);
> > + run_again = blk_mq_do_dispatch_ctx(hctx);
> > } else {
> > blk_mq_flush_busy_ctxs(hctx, &rq_list);
> > blk_mq_dispatch_rq_list(q, &rq_list, false);
> > }
> > +
> > + if (run_again) {
> > + if (!restarted) {
> > + restarted = true;
> > + goto again;
> > + }
> > +
> > + blk_mq_run_hw_queue(hctx, true);
> > + }
>
> So this patch changes blk_mq_sched_dispatch_requests() such that it
> iterates at most two times? How about implementing that loop with an
> explicit for-loop? I think that will make
> blk_mq_sched_dispatch_requests() easier to read. As you may know forward
> goto's are accepted in kernel code but backward goto's are frowned upon.
>

About the goto, I don't know if backwards gotos in general are frowned
upon. There are plenty of examples
in the kernel source. This particular label, 'again' for instance:

$ grep -r again: mm/|wc -l
22
$ grep -r again: block/|wc -l
4

But, just because others have done it doesn't mean I should. So, I
will attempt to explain why I think this is a good idea.
If I were to write this as a for-loop, it will look like this:

for (i = 0; i == 0 || (run_again && i < 2); i++) {
/* another level of 8 character wide indentation */
run_again = false;
/* a bunch of code that possibly sets run_again to true
}

if (run_again)
blk_mq_run_hw_queue(hctx, true);

[Another alternative is to set run_again to true, and simplify the for-loop
condition to run_again && i < 2. But, again, lots of verbiage and a boolean
in the for-loop condition.]

The for-loop is far from idiomatic. It's not clear what it does when
you first look at it.
It distracts from the common path of the code, which is something that
almost always
runs exactly once. There is now an additional level of indentation.
The readers of the
code aren't any better off, because they still have to figure out what
run_again is and if
they care about it. And the only way to do that is to read the entire
body of the loop, and
comments at the top of the functions.

The goto in this case preserves the intent of the code better. It is
dealing with an exceptional
and unusual case. Indeed this kind of use is not unusual in the
kernel, for instance to deal
with possible but unlikely races.

Just my $0.02.

> Thanks,
>
> Bart.