On Mon, Sep 12, 2016 at 11:23:18PM -0700, Junio C Hamano wrote: > Kirill Smelkov <kirr@xxxxxxxxxx> writes: > > > +static int want_found_object(int exclude, struct packed_git *p) > > +{ > > + if (exclude) > > + return 1; > > + if (incremental) > > + return 0; > > + > > + /* > > + * When asked to do --local (do not include an object that appears in a > > + * pack we borrow from elsewhere) or --honor-pack-keep (do not include > > + * an object that appears in a pack marked with .keep), finding a pack > > + * that matches the criteria is sufficient for us to decide to omit it. > > + * However, even if this pack does not satisfy the criteria, we need to > > + * make sure no copy of this object appears in _any_ pack that makes us > > + * to omit the object, so we need to check all the packs. > > + * > > + * We can however first check whether these options can possible matter; > > + * if they do not matter we know we want the object in generated pack. > > + * Otherwise, we signal "-1" at the end to tell the caller that we do > > + * not know either way, and it needs to check more packs. > > + */ > > + if (!ignore_packed_keep && > > + (!local || !have_non_local_packs)) > > + return 1; > > + > > + if (local && !p->pack_local) > > + return 0; > > + if (ignore_packed_keep && p->pack_local && p->pack_keep) > > + return 0; > > + > > + /* we don't know yet; keep looking for more packs */ > > + return -1; > > +} > > Moving this logic out to this helper made the main logic in the > caller easier to grasp. > > > @@ -958,15 +993,30 @@ static int want_object_in_pack(const unsigned char *sha1, > > off_t *found_offset) > > { > > struct packed_git *p; > > + int want; > > > > if (!exclude && local && has_loose_object_nonlocal(sha1)) > > return 0; > > > > + /* > > + * If we already know the pack object lives in, start checks from that > > + * pack - in the usual case when neither --local was given nor .keep files > > + * are present we will determine the answer right now. > > + */ > > + if (*found_pack) { > > + want = want_found_object(exclude, *found_pack); > > + if (want != -1) > > + return want; > > + } > > > > for (p = packed_git; p; p = p->next) { > > + off_t offset; > > + > > + if (p == *found_pack) > > + offset = *found_offset; > > + else > > + offset = find_pack_entry_one(sha1, p); > > + > > if (offset) { > > if (!*found_pack) { > > if (!is_pack_valid(p)) > > @@ -974,31 +1024,9 @@ static int want_object_in_pack(const unsigned char *sha1, > > *found_offset = offset; > > *found_pack = p; > > } > > + want = want_found_object(exclude, p); > > + if (want != -1) > > + return want; > > } > > } > > As Peff noted in his earlier review, however, MRU code needed to be > grafted in to the caller (an update to the MRU list was done in the > code that was moved to the want_found_object() helper). I think I > did it correctly, which ended up looking like this: > > want = want_found_object(exclude, p); > if (!exclude && want > 0) > mru_mark(packed_git_mru, entry); > if (want != -1) > return want; > > I somewhat feel that it is ugly that the helper knows about exclude > (i.e. in the original code, we immediately returned 1 without > futzing with the MRU when we find an entry that is to be excluded, > which now is done in the helper), and the caller also knows about > exclude (i.e. the caller knows that the helper may return positive > in two cases, it knows that MRU marking needs to happen only one of > the two cases, and it also knows that "exclude" is what > differentiates between the two cases) at the same time. > > But probably the reason why I feel it ugly is only because I knew > how the original looked like. I dunno. Junio, the code above is correct semantic merge of pack-mru and my topic, because in pack-mru if found and exclude=1, 1 was returned without marking found pack. But I wonder: even if we exclude an object, we were still looking for it in packs, and when we found it, we found the corresponding pack too. So, that pack _was_ most-recently-used, and it is correct to mark it as MRU. We can do the simplification in the follow-up patch after the merge, so merge does not change semantics and it is all bisectable, etc. Jeff?