corda/src/posix.cpp

750 lines
16 KiB
C++
Raw Normal View History

#include "sys/mman.h"
#include "sys/types.h"
#include "sys/stat.h"
#include "sys/time.h"
#include "time.h"
#include "fcntl.h"
#include "dlfcn.h"
#include "errno.h"
2007-09-17 00:13:36 +00:00
#include "unistd.h"
#include "pthread.h"
2007-07-28 21:28:25 +00:00
#include "signal.h"
#include "ucontext.h"
#include "stdint.h"
2007-10-23 01:00:57 +00:00
#include "x86.h"
2007-07-20 14:39:50 +00:00
#include "system.h"
2007-07-28 21:28:25 +00:00
#define ACQUIRE(x) MutexResource MAKE_NAME(mutexResource_) (x)
using namespace vm;
namespace {
2007-12-30 22:24:48 +00:00
System::SignalHandler* segFaultHandler = 0;
struct sigaction oldSegFaultHandler;
2007-07-28 21:28:25 +00:00
class MutexResource {
public:
MutexResource(pthread_mutex_t& m): m(&m) {
pthread_mutex_lock(&m);
}
~MutexResource() {
pthread_mutex_unlock(m);
}
private:
pthread_mutex_t* m;
};
const int InterruptSignal = SIGUSR2;
#ifdef __APPLE__
const int SegFaultSignal = SIGBUS;
#else
const int SegFaultSignal = SIGSEGV;
#endif
2007-07-28 21:28:25 +00:00
2007-12-30 22:24:48 +00:00
#ifdef __x86_64__
# define IP_REGISTER(context) (context->uc_mcontext.gregs[REG_RIP])
# define BASE_REGISTER(context) (context->uc_mcontext.gregs[REG_RBP])
# define STACK_REGISTER(context) (context->uc_mcontext.gregs[REG_RSP])
# define THREAD_REGISTER(context) (context->uc_mcontext.gregs[REG_RBX])
#elif defined __APPLE__
# define IP_REGISTER(context) (context->uc_mcontext->__ss.__eip)
# define BASE_REGISTER(context) (context->uc_mcontext->__ss.__ebp)
# define STACK_REGISTER(context) (context->uc_mcontext->__ss.__esp)
# define THREAD_REGISTER(context) (context->uc_mcontext->__ss.__ebx)
2007-12-30 22:24:48 +00:00
#elif defined __i386__
# define IP_REGISTER(context) (context->uc_mcontext.gregs[REG_EIP])
# define BASE_REGISTER(context) (context->uc_mcontext.gregs[REG_EBP])
# define STACK_REGISTER(context) (context->uc_mcontext.gregs[REG_ESP])
# define THREAD_REGISTER(context) (context->uc_mcontext.gregs[REG_EBX])
2007-12-30 22:24:48 +00:00
#else
# error unsupported architecture
#endif
2007-07-28 21:28:25 +00:00
void
2007-12-30 22:24:48 +00:00
handleSignal(int signal, siginfo_t* info, void* context)
2007-07-28 21:28:25 +00:00
{
if (signal == SegFaultSignal) {
ucontext_t* c = static_cast<ucontext_t*>(context);
2007-12-30 22:24:48 +00:00
2008-01-01 22:36:26 +00:00
void* ip = reinterpret_cast<void*>(IP_REGISTER(c));
void* base = reinterpret_cast<void*>(BASE_REGISTER(c));
void* stack = reinterpret_cast<void*>(STACK_REGISTER(c));
void* thread = reinterpret_cast<void*>(THREAD_REGISTER(c));
bool jump = segFaultHandler->handleSignal
2008-01-01 22:36:26 +00:00
(&ip, &base, &stack, &thread);
2007-12-30 22:24:48 +00:00
if (jump) {
// I'd like to use setcontext here (and get rid of the
// sigprocmask call), but it doesn't work on my Linux x86_64
// system, and I can't tell from the documentation if it's even
// supposed to work.
sigset_t set;
sigemptyset(&set);
sigaddset(&set, SegFaultSignal);
sigprocmask(SIG_UNBLOCK, &set, 0);
2008-01-01 22:36:26 +00:00
vmJump(ip, base, stack, thread);
} else if (oldSegFaultHandler.sa_flags & SA_SIGINFO) {
oldSegFaultHandler.sa_sigaction(signal, info, context);
} else if (oldSegFaultHandler.sa_handler) {
oldSegFaultHandler.sa_handler(signal);
} else {
abort();
2007-12-30 22:24:48 +00:00
}
}
2007-07-28 21:28:25 +00:00
}
void*
2007-07-28 21:28:25 +00:00
run(void* r)
{
2007-07-28 21:28:25 +00:00
static_cast<System::Runnable*>(r)->run();
return 0;
}
const bool Verbose = false;
2007-07-28 21:28:25 +00:00
const unsigned Waiting = 1 << 0;
const unsigned Notified = 1 << 1;
class MySystem: public System {
public:
class CodeAllocator: public Allocator {
public:
CodeAllocator(MySystem* s): s(s) { }
virtual void* tryAllocate(unsigned size) {
assert(s, size % LikelyPageSizeInBytes == 0);
#ifdef __x86_64__
void* p = mmap(0, size, PROT_EXEC | PROT_READ | PROT_WRITE,
MAP_PRIVATE | MAP_ANONYMOUS | MAP_32BIT, -1, 0);
if (p == MAP_FAILED) {
return 0;
} else {
return p;
}
#else
return s->tryAllocate(size);
#endif
}
virtual void* allocate(unsigned size) {
void* p = tryAllocate(size);
expect(s, p);
return p;
}
virtual void free(const void* p, unsigned size) {
#ifdef __x86_64__
int r UNUSED = munmap(const_cast<void*>(p), size);
assert(s, r == 0);
#else
s->free(p, size);
#endif
}
MySystem* s;
};
class Thread: public System::Thread {
public:
2007-07-28 21:28:25 +00:00
Thread(System* s, System::Runnable* r):
s(s),
r(r),
next(0),
flags(0)
{
pthread_mutex_init(&mutex, 0);
pthread_cond_init(&condition, 0);
}
virtual void interrupt() {
ACQUIRE(mutex);
r->setInterrupted(true);
2007-07-28 21:28:25 +00:00
if (flags & Waiting) {
pthread_kill(thread, InterruptSignal);
}
}
virtual void join() {
int rv UNUSED = pthread_join(thread, 0);
2007-11-27 22:23:00 +00:00
expect(s, rv == 0);
}
virtual void dispose() {
s->free(this, sizeof(*this));
}
2007-07-28 21:28:25 +00:00
pthread_t thread;
pthread_mutex_t mutex;
pthread_cond_t condition;
System* s;
System::Runnable* r;
2007-07-28 21:28:25 +00:00
Thread* next;
unsigned flags;
};
class Monitor: public System::Monitor {
public:
2007-07-28 21:28:25 +00:00
Monitor(System* s): s(s), owner_(0), first(0), last(0), depth(0) {
pthread_mutex_init(&mutex, 0);
}
2007-07-28 21:28:25 +00:00
virtual bool tryAcquire(System::Thread* context) {
Thread* t = static_cast<Thread*>(context);
if (owner_ == t) {
++ depth;
return true;
} else {
switch (pthread_mutex_trylock(&mutex)) {
case EBUSY:
return false;
case 0:
2007-07-28 21:28:25 +00:00
owner_ = t;
++ depth;
return true;
default:
sysAbort(s);
}
}
}
2007-07-28 21:28:25 +00:00
virtual void acquire(System::Thread* context) {
Thread* t = static_cast<Thread*>(context);
if (owner_ != t) {
pthread_mutex_lock(&mutex);
2007-07-28 21:28:25 +00:00
owner_ = t;
}
++ depth;
}
2007-07-28 21:28:25 +00:00
virtual void release(System::Thread* context) {
Thread* t = static_cast<Thread*>(context);
if (owner_ == t) {
if (-- depth == 0) {
2007-07-28 21:28:25 +00:00
owner_ = 0;
pthread_mutex_unlock(&mutex);
}
} else {
sysAbort(s);
}
}
2007-07-28 21:28:25 +00:00
void append(Thread* t) {
if (last) {
last->next = t;
2007-11-27 22:23:00 +00:00
last = t;
2007-07-28 21:28:25 +00:00
} else {
first = last = t;
}
}
void remove(Thread* t) {
2007-11-27 23:04:15 +00:00
Thread* previous = 0;
2007-11-27 22:23:00 +00:00
for (Thread* current = first; current;) {
if (t == current) {
if (current == first) {
first = t->next;
} else {
previous->next = t->next;
}
if (current == last) {
last = previous;
2007-07-28 21:28:25 +00:00
}
2007-11-27 22:23:00 +00:00
t->next = 0;
2007-07-28 21:28:25 +00:00
break;
} else {
2007-11-27 22:23:00 +00:00
previous = current;
current = current->next;
2007-07-28 21:28:25 +00:00
}
}
}
virtual bool wait(System::Thread* context, int64_t time) {
Thread* t = static_cast<Thread*>(context);
if (owner_ == t) {
ACQUIRE(t->mutex);
if (t->r->interrupted()) {
t->r->setInterrupted(false);
return true;
}
t->flags |= Waiting;
append(t);
unsigned depth = this->depth;
this->depth = 0;
2007-07-28 21:28:25 +00:00
owner_ = 0;
pthread_mutex_unlock(&mutex);
if (time) {
2007-08-18 22:35:22 +00:00
int64_t then = s->now() + time;
timespec ts = { then / 1000, (then % 1000) * 1000 * 1000 };
int rv UNUSED = pthread_cond_timedwait
2007-07-28 21:28:25 +00:00
(&(t->condition), &(t->mutex), &ts);
2007-11-27 22:23:00 +00:00
expect(s, rv == 0 or rv == ETIMEDOUT or rv == EINTR);
} else {
int rv UNUSED = pthread_cond_wait(&(t->condition), &(t->mutex));
2007-11-27 22:23:00 +00:00
expect(s, rv == 0 or rv == EINTR);
}
2007-07-28 21:28:25 +00:00
pthread_mutex_lock(&mutex);
owner_ = t;
this->depth = depth;
2007-07-28 21:28:25 +00:00
if ((t->flags & Notified) == 0) {
remove(t);
}
t->flags = 0;
t->next = 0;
2007-07-28 21:28:25 +00:00
if (t->r->interrupted()) {
t->r->setInterrupted(false);
return true;
} else {
return false;
}
} else {
sysAbort(s);
}
}
2007-07-28 21:28:25 +00:00
void doNotify(Thread* t) {
ACQUIRE(t->mutex);
t->flags |= Notified;
int rv UNUSED = pthread_cond_signal(&(t->condition));
2007-11-27 22:23:00 +00:00
expect(s, rv == 0);
2007-07-28 21:28:25 +00:00
}
virtual void notify(System::Thread* context) {
Thread* t = static_cast<Thread*>(context);
if (owner_ == t) {
if (first) {
Thread* t = first;
first = first->next;
if (t == last) {
last = 0;
}
doNotify(t);
}
} else {
sysAbort(s);
}
}
2007-07-28 21:28:25 +00:00
virtual void notifyAll(System::Thread* context) {
Thread* t = static_cast<Thread*>(context);
if (owner_ == t) {
for (Thread* t = first; t; t = t->next) {
2007-07-28 21:28:25 +00:00
doNotify(t);
}
first = last = 0;
} else {
sysAbort(s);
}
}
2007-07-28 21:28:25 +00:00
virtual System::Thread* owner() {
return owner_;
}
virtual void dispose() {
2007-11-27 22:23:00 +00:00
expect(s, owner_ == 0);
pthread_mutex_destroy(&mutex);
s->free(this, sizeof(*this));
}
System* s;
pthread_mutex_t mutex;
2007-07-28 21:28:25 +00:00
Thread* owner_;
Thread* first;
Thread* last;
unsigned depth;
};
class Local: public System::Local {
public:
Local(System* s): s(s) {
int r UNUSED = pthread_key_create(&key, 0);
2007-11-27 22:23:00 +00:00
expect(s, r == 0);
}
virtual void* get() {
return pthread_getspecific(key);
}
virtual void set(void* p) {
int r UNUSED = pthread_setspecific(key, p);
2007-11-27 22:23:00 +00:00
expect(s, r == 0);
}
virtual void dispose() {
int r UNUSED = pthread_key_delete(key);
2007-11-27 22:23:00 +00:00
expect(s, r == 0);
s->free(this, sizeof(*this));
}
System* s;
pthread_key_t key;
};
2007-09-17 00:13:36 +00:00
class Region: public System::Region {
public:
Region(System* s, uint8_t* start, size_t length):
s(s),
2007-09-17 00:13:36 +00:00
start_(start),
length_(length)
{ }
virtual const uint8_t* start() {
return start_;
}
virtual size_t length() {
return length_;
}
virtual void dispose() {
if (start_) {
munmap(start_, length_);
}
s->free(this, sizeof(*this));
2007-09-17 00:13:36 +00:00
}
System* s;
2007-09-17 00:13:36 +00:00
uint8_t* start_;
size_t length_;
};
class Library: public System::Library {
public:
Library(System* s, void* p, const char* name, unsigned nameLength,
bool mapName, System::Library* next):
s(s),
p(p),
name_(name),
nameLength(nameLength),
mapName_(mapName),
next_(next)
{ }
virtual void* resolve(const char* function) {
return dlsym(p, function);
}
virtual const char* name() {
return name_;
}
virtual bool mapName() {
return mapName_;
}
virtual System::Library* next() {
return next_;
}
virtual void dispose() {
if (Verbose) {
fprintf(stderr, "close %p\n", p);
}
dlclose(p);
if (next_) {
next_->dispose();
}
if (name_) {
s->free(name_, nameLength + 1);
}
s->free(this, sizeof(*this));
}
System* s;
void* p;
const char* name_;
unsigned nameLength;
bool mapName_;
System::Library* next_;
};
MySystem(unsigned limit): limit(limit), count(0), codeAllocator_(this) {
pthread_mutex_init(&mutex, 0);
2007-07-28 21:28:25 +00:00
struct sigaction sa;
memset(&sa, 0, sizeof(struct sigaction));
sigemptyset(&(sa.sa_mask));
2007-12-30 22:24:48 +00:00
sa.sa_flags = SA_SIGINFO;
sa.sa_sigaction = handleSignal;
2007-07-28 21:28:25 +00:00
int rv UNUSED = sigaction(InterruptSignal, &sa, 0);
2007-11-27 22:23:00 +00:00
expect(this, rv == 0);
}
virtual void* tryAllocate(unsigned size) {
2007-10-23 23:22:56 +00:00
ACQUIRE(mutex);
if (Verbose) {
fprintf(stderr, "try %d; count: %d; limit: %d\n",
size, count, limit);
}
if (count + size > limit) {
return 0;
} else {
#ifndef NDEBUG
uintptr_t* up = static_cast<uintptr_t*>
(malloc(size + sizeof(uintptr_t)));
if (up == 0) {
return 0;
} else {
*up = size;
count += *up;
return up + 1;
}
#else
void* p = malloc(size);
if (p == 0) {
return 0;
} else {
count += size;
return p;
}
#endif
}
}
virtual void free(const void* p, unsigned size) {
2007-10-23 23:22:56 +00:00
ACQUIRE(mutex);
if (Verbose) {
fprintf(stderr, "free %d; count: %d; limit: %d\n",
size, count, limit);
}
if (p) {
#ifndef NDEBUG
const uintptr_t* up = static_cast<const uintptr_t*>(p) - 1;
if (*up != size) abort();
if (count < *up) abort();
count -= *up;
::free(const_cast<uintptr_t*>(up));
#else
if (count < size) abort();
count -= size;
::free(const_cast<void*>(p));
#endif
}
}
virtual Allocator* codeAllocator() {
return &codeAllocator_;
}
virtual bool success(Status s) {
return s == 0;
}
2007-07-28 21:28:25 +00:00
virtual Status attach(Runnable* r) {
Thread* t = new (System::allocate(sizeof(Thread))) Thread(this, r);
t->thread = pthread_self();
2007-07-28 21:28:25 +00:00
r->attach(t);
return 0;
}
virtual Status start(Runnable* r) {
Thread* t = new (System::allocate(sizeof(Thread))) Thread(this, r);
2007-07-28 21:28:25 +00:00
r->attach(t);
int rv UNUSED = pthread_create(&(t->thread), 0, run, r);
2007-11-27 22:23:00 +00:00
expect(this, rv == 0);
return 0;
}
virtual Status make(System::Monitor** m) {
*m = new (System::allocate(sizeof(Monitor))) Monitor(this);
return 0;
}
virtual Status make(System::Local** l) {
*l = new (System::allocate(sizeof(Local))) Local(this);
return 0;
}
2007-12-30 22:24:48 +00:00
virtual Status handleSegFault(SignalHandler* handler) {
if (handler) {
segFaultHandler = handler;
struct sigaction sa;
memset(&sa, 0, sizeof(struct sigaction));
sigemptyset(&(sa.sa_mask));
sa.sa_flags = SA_SIGINFO;
sa.sa_sigaction = handleSignal;
return sigaction(SegFaultSignal, &sa, &oldSegFaultHandler);
2007-12-30 22:24:48 +00:00
} else if (segFaultHandler) {
segFaultHandler = 0;
return sigaction(SegFaultSignal, &oldSegFaultHandler, 0);
2007-12-30 22:24:48 +00:00
} else {
return 1;
}
}
virtual uint64_t call(void* function, uintptr_t* arguments, uint8_t* types,
unsigned count, unsigned size, unsigned returnType)
{
return dynamicCall(function, arguments, types, count, size, returnType);
}
2007-09-17 00:13:36 +00:00
virtual Status map(System::Region** region, const char* name) {
Status status = 1;
int fd = open(name, O_RDONLY);
if (fd != -1) {
struct stat s;
int r = fstat(fd, &s);
if (r != -1) {
void* data = mmap(0, s.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
if (data) {
*region = new (allocate(sizeof(Region)))
Region(this, static_cast<uint8_t*>(data), s.st_size);
status = 0;
}
}
close(fd);
}
return status;
}
virtual FileType identify(const char* name) {
struct stat s;
int r = stat(name, &s);
if (r == 0) {
2007-09-17 00:13:36 +00:00
if (S_ISREG(s.st_mode)) {
return File;
} else if (S_ISDIR(s.st_mode)) {
return Directory;
} else {
return Unknown;
}
} else {
return DoesNotExist;
}
}
virtual Status load(System::Library** lib,
const char* name,
bool mapName,
System::Library* next)
{
void* p;
unsigned nameLength = (name ? strlen(name) : 0);
if (mapName) {
unsigned size = nameLength + 3 + sizeof(SO_SUFFIX);
char buffer[size];
snprintf(buffer, size, "lib%s" SO_SUFFIX, name);
p = dlopen(buffer, RTLD_LAZY);
} else {
p = dlopen(name, RTLD_LAZY);
}
if (p) {
if (Verbose) {
fprintf(stderr, "open %s as %p\n", name, p);
}
char* n;
if (name) {
n = static_cast<char*>(System::allocate(nameLength + 1));
memcpy(n, name, nameLength + 1);
} else {
n = 0;
}
*lib = new (System::allocate(sizeof(Library)))
Library(this, p, n, nameLength, mapName, next);
return 0;
} else {
// fprintf(stderr, "dlerror: %s\n", dlerror());
return 1;
}
}
virtual char pathSeparator() {
return ':';
}
2007-10-23 01:00:57 +00:00
virtual int64_t now() {
2007-08-18 22:35:22 +00:00
timeval tv = { 0, 0 };
gettimeofday(&tv, 0);
return (static_cast<int64_t>(tv.tv_sec) * 1000) +
(static_cast<int64_t>(tv.tv_usec) / 1000);
}
2007-10-23 01:00:57 +00:00
virtual void exit(int code) {
::exit(code);
}
virtual void abort() {
::abort();
}
virtual void dispose() {
assert(this, count == 0);
pthread_mutex_destroy(&mutex);
::free(this);
}
pthread_mutex_t mutex;
unsigned limit;
unsigned count;
CodeAllocator codeAllocator_;
};
} // namespace
namespace vm {
System*
makeSystem(unsigned heapSize)
{
return new (malloc(sizeof(MySystem))) MySystem(heapSize);
}
} // namespace vm