#define _GNU_SOURCE /* for Linux O_DIRECT */ #include #define NOPLAN9DEFINES #include #include #include #include #include #ifndef O_DIRECT #define O_DIRECT 0 #endif int p9create(char *path, int mode, ulong perm) { int fd, cexec, umode, rclose, lock, rdwr; struct flock fl; rdwr = mode&3; lock = mode&OLOCK; cexec = mode&OCEXEC; rclose = mode&ORCLOSE; mode &= ~(ORCLOSE|OCEXEC|OLOCK); /* XXX should get mode mask right? */ fd = -1; if(perm&DMDIR){ if(mode != OREAD){ werrstr("bad mode in directory create"); goto out; } if(mkdir(path, perm&0777) < 0) goto out; fd = open(path, O_RDONLY); }else{ umode = (mode&3)|O_CREAT|O_TRUNC; mode &= ~(3|OTRUNC); if(mode&ODIRECT){ umode |= O_DIRECT; mode &= ~ODIRECT; } if(mode&OEXCL){ umode |= O_EXCL; mode &= ~OEXCL; } if(mode&OAPPEND){ umode |= O_APPEND; mode &= ~OAPPEND; } if(mode){ werrstr("unsupported mode in create"); goto out; } fd = open(path, umode, perm); } out: if(fd >= 0){ if(lock){ fl.l_type = (rdwr==OREAD) ? F_RDLCK : F_WRLCK; fl.l_whence = SEEK_SET; fl.l_start = 0; fl.l_len = 0; if(fcntl(fd, F_SETLK, &fl) < 0){ close(fd); werrstr("lock: %r"); return -1; } } if(cexec) fcntl(fd, F_SETFL, FD_CLOEXEC); if(rclose) remove(path); } return fd; }