On 30/11/2019 21:57, Jens Axboe wrote: > On 11/30/19 10:56 AM, Arvind Sankar wrote: >> On Sat, Nov 30, 2019 at 12:22:27PM +0300, Pavel Begunkov wrote: >>> On 30/11/2019 02:24, Arvind Sankar wrote: >>>> On Sat, Nov 30, 2019 at 01:47:16AM +0300, Pavel Begunkov wrote: >>>>> On 30/11/2019 01:17, Arvind Sankar wrote: >>>>>> >>>>>> The loop can be simplified a bit further, as done has to be 0 once we go >>>>>> beyond the current bio_vec. See below for the simplified version. >>>>>> >>>>> >>>>> Thanks for the suggestion! I thought about it, and decided to not >>>>> for several reasons. I prefer to not fine-tune and give compilers >>>>> more opportunity to do their job. And it's already fast enough with >>>>> modern architectures (MOVcc, complex addressing, etc). >>>>> >>>>> Also need to consider code clarity and the fact, that this is inline, >>>>> so should be brief and register-friendly. >>>>> >>>> >>>> It should be more register-friendly, as it uses fewer variables, and I >>>> think it's easier to see what the loop is doing, i.e. that we advance >>>> one bio_vec per iteration: in the existing code, it takes a bit of >>>> thinking to see that we won't spend more than one iteration within the >>>> same bio_vec. >>> >>> Yeah, may be. It's more the matter of preference then. I don't think >>> it's simpler, and performance is entirely depends on a compiler and >>> input. But, that's rather subjective and IMHO not worth of time. >>> >>> Anyway, thanks for thinking this through! >>> >> >> You don't find listing 1 simpler than listing 2? It does save one >> register, as it doesn't have to keep track of done independently from >> bytes. This is always going to be the case unless the compiler can >> eliminate done by transforming Listing 2 into Listing 1. Unfortunately, >> even if it gets much smarter, it's unlikely to be able to do that, >> because they're equivalent only if there is no overflow, so it would >> need to know that bytes + iter->bi_bvec_done cannot overflow, and that >> iter->bi_bvec_done must be smaller than cur->bv_len initially. >> >> Listing 1: >> >> bytes += iter->bi_bvec_done; >> while (bytes) { >> const struct bio_vec *cur = bv + idx; >> >> if (bytes < cur->bv_len) >> break; >> bytes -= cur->bv_len; >> idx++; >> } >> >> iter->bi_idx = idx; >> iter->bi_bvec_done = bytes; >> >> Listing 2: >> >> while (bytes) { >> const struct bio_vec *cur = bv + idx; >> unsigned int len = min(bytes, cur->bv_len - done); >> >> bytes -= len; >> done += len; >> if (done == cur->bv_len) { >> idx++; >> done = 0; >> } >> } >> >> iter->bi_idx = idx; >> iter->bi_bvec_done = done; > > Have yet to take a closer look (and benchmark) and the patches and > the generated code, but fwiw I do agree that case #1 is easier to > read. > Ok, ok, I'm not keen on bike-shedding. I'll resend a simplified version -- Pavel Begunkov
Attachment:
signature.asc
Description: OpenPGP digital signature