Hi Peff,

On Fri, 9 Nov 2018, Jeff King wrote:

> On Fri, Nov 09, 2018 at 01:34:17AM -0800, Johannes Schindelin via 
> GitGitGadget wrote:
> 
> > diff --git a/builtin/rebase.c b/builtin/rebase.c
> > index 0ee06aa363..6f6d7de156 100644
> > --- a/builtin/rebase.c
> > +++ b/builtin/rebase.c
> > @@ -569,16 +569,13 @@ static int reset_head(struct object_id *oid, const 
> > char *action,
> >     }
> >  
> >     if (!fill_tree_descriptor(&desc, oid)) {
> > -           error(_("failed to find tree of %s"), oid_to_hex(oid));
> > -           rollback_lock_file(&lock);
> > -           free((void *)desc.buffer);
> > -           return -1;
> > +           ret = error(_("failed to find tree of %s"), oid_to_hex(oid));
> > +           goto leave_reset_head;
> >     }
> 
> If fill_tree_descriptor() fails, what is left in desc.buffer? Looking at
> the implementation, I think it's always NULL or a valid buffer. But I
> think all code paths actually die() unless we pass a NULL oid (and in
> that case desc.buffer would be NULL, too).
> 
> So I think the original here that calls free() doesn't ever do anything
> but it did not hurt. After your patch, the leave_reset_head code would
> continue to call free(), and that's OK.

Right, that was my thinking, too.

> There are a few earlier conditionals in reset_head() that do only
> rollback_lock_file() that could similarly be converted to use the goto.
> But they would need desc.buffer to be initialized to NULL. I could go
> either way on converting them or not.

Whoops. I should have checked more carefully?

> > @@ -586,10 +583,9 @@ static int reset_head(struct object_id *oid, const 
> > char *action,
> >  
> >     if (write_locked_index(the_repository->index, &lock, COMMIT_LOCK) < 0)
> >             ret = error(_("could not write index"));
> > -   free((void *)desc.buffer);
> >  
> >     if (ret)
> > -           return ret;
> > +           goto leave_reset_head;
> >  
> >     reflog_action = getenv(GIT_REFLOG_ACTION_ENVIRONMENT);
> >     strbuf_addf(&msg, "%s: ", reflog_action ? reflog_action : "rebase");
> > @@ -622,7 +618,10 @@ static int reset_head(struct object_id *oid, const 
> > char *action,
> >                                      UPDATE_REFS_MSG_ON_ERR);
> >     }
> >  
> > +leave_reset_head:
> >     strbuf_release(&msg);
> > +   rollback_lock_file(&lock);
> > +   free((void *)desc.buffer);
> >     return ret;
> 
> We get here on success, too. So we may call rollback_lock_file() on an
> already-committed lock. This is explicitly documented as a no-op by the
> lock code, so that's OK.

Indeed. I did not check the documentation, but the code, and came to the
same conclusion.

> So overall looks good to me.

Thanks!
Dscho

Reply via email to