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.