#define _GNU_SOURCE /* for Linux O_DIRECT */ #include <u.h> #define NOPLAN9DEFINES #include <sys/file.h> #include <libc.h> #ifndef O_DIRECT #define O_DIRECT 0 #endif int p9open(char *name, int mode) { int cexec, rclose; int fd, umode, lock, rdwr; struct flock fl; rdwr = mode&3; umode = rdwr; cexec = mode&OCEXEC; rclose = mode&ORCLOSE; lock = mode&OLOCK; mode &= ~(3|OCEXEC|ORCLOSE|OLOCK); if(mode&OTRUNC){ umode |= O_TRUNC; mode ^= OTRUNC; } if(mode&ODIRECT){ umode |= O_DIRECT; mode ^= ODIRECT; } if(mode&ONONBLOCK){ umode |= O_NONBLOCK; mode ^= ONONBLOCK; } if(mode&OAPPEND){ umode |= O_APPEND; mode ^= OAPPEND; } if(mode){ werrstr("mode 0x%x not supported", mode); return -1; } fd = open(name, umode); 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); return -1; } } if(cexec) fcntl(fd, F_SETFL, FD_CLOEXEC); if(rclose) remove(name); } return fd; }