#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); | |
werrstr("lock: %r"); | |
return -1; | |
} | |
} | |
if(cexec) | |
fcntl(fd, F_SETFL, FD_CLOEXEC); | |
if(rclose) | |
remove(name); | |
} | |
return fd; | |
} |