#define _GNU_SOURCE /* for Linux O_DIRECT */ | |
#include <u.h> | |
#define NOPLAN9DEFINES | |
#include <sys/file.h> | |
#include <unistd.h> | |
#include <fcntl.h> | |
#include <libc.h> | |
#include <sys/stat.h> | |
#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; | |
} |