aboutsummaryrefslogtreecommitdiff
path: root/liblogfs/create.c
diff options
context:
space:
mode:
authorbhgv <bhgv.empire@gmail.com>2018-03-01 16:54:45 +0200
committerbhgv <bhgv.empire@gmail.com>2018-03-01 16:54:45 +0200
commitb786f20bbab5a59046aa78a2c6c2a11536497202 (patch)
tree0851ecdec889eb9b7ba3751cc04d4f0b474e4a9e /liblogfs/create.c
inferno-os tree was separated from the inferno-os-android (separated from the Android driver)
Diffstat (limited to 'liblogfs/create.c')
-rw-r--r--liblogfs/create.c82
1 files changed, 82 insertions, 0 deletions
diff --git a/liblogfs/create.c b/liblogfs/create.c
new file mode 100644
index 0000000..548f519
--- /dev/null
+++ b/liblogfs/create.c
@@ -0,0 +1,82 @@
+#include "logfsos.h"
+#include "logfs.h"
+#include "local.h"
+
+char *
+logfsservercreate(LogfsServer *server, u32int fid, char *name, u32int perm, uchar mode, Qid *qid)
+{
+ Fid *f;
+ char *uid;
+ ulong newpath;
+ char *errmsg;
+ Entry *e, *xe, *pe;
+ Path *pp;
+ LogMessage s;
+
+ if(server->trace > 1)
+ print("logfsservercreate(%ud, %s, 0%uo, %.2ux)\n", fid, name, perm, mode);
+ f = logfsfidmapfindentry(server->fidmap, fid);
+ if(f == nil)
+ return logfsebadfid;
+ if(f->openmode >= 0)
+ return logfsefidopen;
+ pe = f->entry;
+ if((pe->qid.type & QTDIR) == 0)
+ return Enotdir;
+ if((perm & DMDIR) != 0 && ((mode & OTRUNC) != 0 || (mode & 3) != OREAD))
+ return Eperm;
+ if(!logfsuserpermcheck(server, pe, f, DMWRITE))
+ return Eperm;
+ /*
+ * illegal names
+ */
+ if(strcmp(name, ".") == 0 || strcmp(name, "..") == 0)
+ return Eperm;
+ for(xe = pe->u.dir.list; xe; xe = xe->next)
+ if(strcmp(xe->name, name) == 0)
+ return Eexist;
+ newpath = ++server->path;
+ while(logfspathmapfindentry(server->pathmap, newpath))
+ newpath++; /* shouldn't happen */
+ uid = logfsisfindidfromname(server->is, f->uname);
+ errmsg = logfsentrynew(server, 1, newpath,
+ pe, name, uid, f->entry->gid, logfsnow(), uid, perm, 0, 0, &e);
+ if(errmsg)
+ return errmsg;
+ errmsg = logfspathmapnewentry(server->pathmap, newpath, e, &pp);
+ /* pp is guaranteed to be non-null */
+ if(errmsg) {
+ logfsfreemem(e);
+ return errmsg;
+ }
+ s.type = LogfsLogTcreate;
+ s.path = e->parent->qid.path;
+ s.u.create.perm = e->perm;
+ s.u.create.newpath = e->qid.path;
+ s.u.create.mtime = e->mtime;
+ /* TODO - check with forsyth whether cvers is needed in dirs */
+ s.u.create.cvers = (e->qid.type & QTDIR) ? 0 : e->u.file.cvers;
+ s.u.create.name = e->name;
+ s.u.create.uid = e->uid;
+ s.u.create.gid = e->gid;
+ errmsg = logfslog(server, 1, &s);
+ if(errmsg) {
+ logfsfreemem(e);
+ logfspathmapdeleteentry(server->pathmap, newpath);
+ return errmsg;
+ }
+ server->path = newpath;
+ e->inuse++;
+ e->qid.vers++;
+ e->next = pe->u.dir.list;
+ pe->u.dir.list = e;
+ f->openmode = mode;
+ /*
+ * TODO why does forsyth increment inuse for dir? - we're moving the fid onto the new file
+ * so a decrement seems better
+ */
+ logfsentryclunk(pe);
+ f->entry = e;
+ *qid = e->qid;
+ return nil;
+}