From dba98f649c45df467a48b7c0cfb95ad64fde8d2b Mon Sep 17 00:00:00 2001 From: Ori Bernstein Date: Sat, 16 Apr 2022 23:53:19 +0000 Subject: [PATCH] git/merge: preserve exec bit correctly A while ago, qwx noticed that we clobbered the exec bit when merging files. This is not what we want, so we changed the operator precedence to avoid merging dirty files implicitly. But we do want to merge, because it's convenient for maintaining permissions. So, instead, we should do a 3 way merge of the exec bit. This patch does that, as well as reverting the rollback of that change. While we're here, we adjust the timestamps correctly in git/branch. This requires changes to git/fs, because without an open handler, lib9p allows opening any file with any mode, which confuses 'test -x'. --- sys/src/cmd/git/branch | 16 ++++++++------ sys/src/cmd/git/fs.c | 48 ++++++++++++++++++++++++++++++++++-------- 2 files changed, 48 insertions(+), 16 deletions(-) diff --git a/sys/src/cmd/git/branch b/sys/src/cmd/git/branch index dfbce636d..01c41176b 100755 --- a/sys/src/cmd/git/branch +++ b/sys/src/cmd/git/branch @@ -48,9 +48,12 @@ if(~ $#newbr 0){ modified=`$nl{git/query -c HEAD $base | grep '^[^-]' | subst '^..'} deleted=`$nl{git/query -c HEAD $base | grep '^-' | subst '^..'} -if(! ~ $#modified 0 || {! ~ $#deleted 0 && ~ $#merge 0}){ - git/walk -fRMA $modified $deleted || - die 'uncommitted changes would be clobbered' +# if we're not merging, don't clobber existing changes. +if(~ $#merge 0){ + if(! ~ $#modified 0 || ! ~ $#deleted 0){ + git/walk -fRMA $modified $deleted || + die 'uncommitted changes would be clobbered' + } } if(~ $delete 1){ rm -f .git/$new @@ -97,10 +100,9 @@ for(m in $cleanpaths){ rm -rf .git/index9/tracked/$m } if(~ $b file){ - if(cp -x -- $basedir/tree/$m $m) - walk -eq $m > .git/index9/tracked/$m - if not - echo -n > .git/index9/tracked/$m + cp -x -- $basedir/tree/$m $m + walk -eq $m > .git/index9/tracked/$m + touch $m } } diff --git a/sys/src/cmd/git/fs.c b/sys/src/cmd/git/fs.c index b9710f324..7e6b4bf87 100644 --- a/sys/src/cmd/git/fs.c +++ b/sys/src/cmd/git/fs.c @@ -70,13 +70,14 @@ char *qroot[] = { "ctl", }; -#define Eperm "permission denied"; -#define Eexist "does not exist"; -#define E2long "path too long"; -#define Enodir "not a directory"; -#define Erepo "unable to read repo"; -#define Egreg "wat"; -#define Ebadobj "invalid object"; +#define Eperm "permission denied" +#define Eexist "does not exist" +#define E2long "path too long" +#define Enodir "not a directory" +#define Erepo "unable to read repo" +#define Eobject "invalid object" +#define Egreg "wat" +#define Ebadobj "invalid object" char gitdir[512]; char *username; @@ -624,9 +625,9 @@ gitwalk1(Fid *fid, char *name, Qid *q) e = objwalk1(q, o->obj, o, c, name, Qobject, aux); }else{ if(hparse(&h, name) == -1) - return "invalid object name"; + return Eobject; if((c->obj = readobject(h)) == nil) - return "could not read object"; + return Eobject; if(c->obj->type == GBlob || c->obj->type == GTag){ c->mode = 0644; q->type = 0; @@ -804,6 +805,34 @@ gitread(Req *r) respond(r, e); } +static void +gitopen(Req *r) +{ + Gitaux *aux; + Crumb *c; + + aux = r->fid->aux; + c = crumb(aux, 0); + switch(r->ifcall.mode&3){ + default: + respond(r, "botched mode"); + break; + case OWRITE: + respond(r, Eperm); + break; + case OREAD: + case ORDWR: + respond(r, nil); + break; + case OEXEC: + if((c->mode & 0111) == 0) + respond(r, Eperm); + else + respond(r, nil); + break; + } +} + static void gitstat(Req *r) { @@ -830,6 +859,7 @@ Srv gitsrv = { .attach=gitattach, .walk1=gitwalk1, .clone=gitclone, + .open=gitopen, .read=gitread, .stat=gitstat, .destroyfid=gitdestroyfid,