mirror of
https://github.com/corda/corda.git
synced 2025-01-19 11:16:54 +00:00
sketch of windows port
This commit is contained in:
parent
f8b5ec1ee3
commit
58dec050c8
3
makefile
3
makefile
@ -54,6 +54,7 @@ cflags = $(warnings) -fPIC -fno-rtti -fno-exceptions -fvisibility=hidden \
|
||||
lflags = -lpthread -ldl -lm -lz
|
||||
|
||||
system = posix
|
||||
asm = x86
|
||||
|
||||
ifeq ($(platform),darwin)
|
||||
rdynamic =
|
||||
@ -137,7 +138,7 @@ interpreter-sources = \
|
||||
$(src)/jnienv.cpp \
|
||||
$(src)/main.cpp
|
||||
|
||||
interpreter-asm-sources = $(src)/$(system).S
|
||||
interpreter-asm-sources = $(src)/$(asm).S
|
||||
|
||||
ifeq ($(process),compile)
|
||||
interpreter-asm-sources += $(src)/compile.S
|
||||
|
@ -28,6 +28,8 @@
|
||||
|
||||
#ifdef __APPLE__
|
||||
# define SO_SUFFIX ".jnilib"
|
||||
#elseif defined __MINGW32__
|
||||
# define SO_SUFFIX ".dll"
|
||||
#else
|
||||
# define SO_SUFFIX ".so"
|
||||
#endif
|
||||
|
@ -11,81 +11,11 @@
|
||||
#include "signal.h"
|
||||
#include "stdint.h"
|
||||
|
||||
#include "x86.h"
|
||||
#include "system.h"
|
||||
|
||||
#define ACQUIRE(x) MutexResource MAKE_NAME(mutexResource_) (x)
|
||||
|
||||
#ifdef __i386__
|
||||
|
||||
extern "C" uint64_t
|
||||
cdeclCall(void* function, void* stack, unsigned stackSize,
|
||||
unsigned returnType);
|
||||
|
||||
namespace {
|
||||
|
||||
inline uint64_t
|
||||
dynamicCall(void* function, uintptr_t* arguments, uint8_t*,
|
||||
unsigned, unsigned argumentsSize, unsigned returnType)
|
||||
{
|
||||
return cdeclCall(function, arguments, argumentsSize, returnType);
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
#elif defined __x86_64__
|
||||
|
||||
extern "C" uint64_t
|
||||
amd64Call(void* function, void* stack, unsigned stackSize,
|
||||
void* gprTable, void* sseTable, unsigned returnType);
|
||||
|
||||
namespace {
|
||||
|
||||
uint64_t
|
||||
dynamicCall(void* function, uint64_t* arguments, uint8_t* argumentTypes,
|
||||
unsigned argumentCount, unsigned, unsigned returnType)
|
||||
{
|
||||
const unsigned GprCount = 6;
|
||||
uint64_t gprTable[GprCount];
|
||||
unsigned gprIndex = 0;
|
||||
|
||||
const unsigned SseCount = 8;
|
||||
uint64_t sseTable[SseCount];
|
||||
unsigned sseIndex = 0;
|
||||
|
||||
uint64_t stack[argumentCount];
|
||||
unsigned stackIndex = 0;
|
||||
|
||||
for (unsigned i = 0; i < argumentCount; ++i) {
|
||||
switch (argumentTypes[i]) {
|
||||
case FLOAT_TYPE:
|
||||
case DOUBLE_TYPE: {
|
||||
if (sseIndex < SseCount) {
|
||||
sseTable[sseIndex++] = arguments[i];
|
||||
} else {
|
||||
stack[stackIndex++] = arguments[i];
|
||||
}
|
||||
} break;
|
||||
|
||||
default: {
|
||||
if (gprIndex < GprCount) {
|
||||
gprTable[gprIndex++] = arguments[i];
|
||||
} else {
|
||||
stack[stackIndex++] = arguments[i];
|
||||
}
|
||||
} break;
|
||||
}
|
||||
}
|
||||
|
||||
return amd64Call(function, stack, stackIndex * 8, (gprIndex ? gprTable : 0),
|
||||
(sseIndex ? sseTable : 0), returnType);
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
#else
|
||||
# error unsupported platform
|
||||
#endif
|
||||
|
||||
using namespace vm;
|
||||
|
||||
namespace {
|
||||
@ -473,7 +403,7 @@ class MySystem: public System {
|
||||
}
|
||||
|
||||
virtual void* tryAllocate(unsigned size) {
|
||||
pthread_mutex_lock(&mutex);
|
||||
ACQUIRE(&mutex);
|
||||
|
||||
if (Verbose) {
|
||||
fprintf(stderr, "try %d; count: %d; limit: %d\n",
|
||||
@ -481,26 +411,22 @@ class MySystem: public System {
|
||||
}
|
||||
|
||||
if (count + size > limit) {
|
||||
pthread_mutex_unlock(&mutex);
|
||||
return 0;
|
||||
} else {
|
||||
uintptr_t* up = static_cast<uintptr_t*>
|
||||
(malloc(size + sizeof(uintptr_t)));
|
||||
if (up == 0) {
|
||||
pthread_mutex_unlock(&mutex);
|
||||
sysAbort(this);
|
||||
} else {
|
||||
*up = size;
|
||||
count += *up;
|
||||
|
||||
pthread_mutex_unlock(&mutex);
|
||||
return up + 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
virtual void free(const void* p) {
|
||||
pthread_mutex_lock(&mutex);
|
||||
ACQUIRE(&mutex);
|
||||
|
||||
if (p) {
|
||||
const uintptr_t* up = static_cast<const uintptr_t*>(p) - 1;
|
||||
@ -516,8 +442,6 @@ class MySystem: public System {
|
||||
|
||||
::free(const_cast<uintptr_t*>(up));
|
||||
}
|
||||
|
||||
pthread_mutex_unlock(&mutex);
|
||||
}
|
||||
|
||||
virtual Status attach(Runnable* r) {
|
||||
@ -626,17 +550,17 @@ class MySystem: public System {
|
||||
}
|
||||
}
|
||||
|
||||
virtual void exit(int code) {
|
||||
::exit(code);
|
||||
}
|
||||
|
||||
int64_t now() {
|
||||
virtual int64_t now() {
|
||||
timeval tv = { 0, 0 };
|
||||
gettimeofday(&tv, 0);
|
||||
return (static_cast<int64_t>(tv.tv_sec) * 1000) +
|
||||
(static_cast<int64_t>(tv.tv_usec) / 1000);
|
||||
}
|
||||
|
||||
virtual void exit(int code) {
|
||||
::exit(code);
|
||||
}
|
||||
|
||||
virtual void abort() {
|
||||
::abort();
|
||||
}
|
||||
|
625
src/windows.cpp
625
src/windows.cpp
@ -1 +1,624 @@
|
||||
// todo
|
||||
#include "sys/stat.h"
|
||||
#include "windows.h"
|
||||
|
||||
#undef max
|
||||
#undef min
|
||||
|
||||
#include "x86.h"
|
||||
#include "system.h"
|
||||
|
||||
#define ACQUIRE(s, x) MutexResource MAKE_NAME(mutexResource_) (s, x)
|
||||
|
||||
using namespace vm;
|
||||
|
||||
namespace {
|
||||
|
||||
class MutexResource {
|
||||
public:
|
||||
MutexResource(System* s, HANDLE m): s(s), m(m) {
|
||||
int r UNUSED = WaitForSingleObject(m, INFINITE);
|
||||
assert(s, r == WAIT_OBJECT_0);
|
||||
}
|
||||
|
||||
~MutexResource() {
|
||||
int r UNUSED = ReleaseMutex(m);
|
||||
assert(s, r == 0);
|
||||
}
|
||||
|
||||
private:
|
||||
System* s;
|
||||
HANDLE m;
|
||||
};
|
||||
|
||||
DWORD WINAPI
|
||||
run(void* r)
|
||||
{
|
||||
static_cast<System::Runnable*>(r)->run();
|
||||
return 0;
|
||||
}
|
||||
|
||||
const bool Verbose = false;
|
||||
|
||||
const unsigned Waiting = 1 << 0;
|
||||
const unsigned Notified = 1 << 1;
|
||||
|
||||
class MySystem: public System {
|
||||
public:
|
||||
class Thread: public System::Thread {
|
||||
public:
|
||||
Thread(System* s, System::Runnable* r):
|
||||
s(s),
|
||||
r(r),
|
||||
next(0),
|
||||
flags(0)
|
||||
{
|
||||
mutex = CreateMutex(0, false, 0);
|
||||
assert(s, mutex);
|
||||
|
||||
event = CreateEvent(0, true, false, 0);
|
||||
assert(s, event);
|
||||
}
|
||||
|
||||
virtual void interrupt() {
|
||||
ACQUIRE(s, mutex);
|
||||
|
||||
r->setInterrupted(true);
|
||||
|
||||
if (flags & Waiting) {
|
||||
int r UNUSED = SetEvent(event);
|
||||
assert(s, r == 0);
|
||||
}
|
||||
}
|
||||
|
||||
virtual void join() {
|
||||
int r UNUSED = WaitForSingleObject(thread, INFINITE);
|
||||
assert(s, r == WAIT_OBJECT_0);
|
||||
}
|
||||
|
||||
virtual void dispose() {
|
||||
CloseHandle(event);
|
||||
CloseHandle(mutex);
|
||||
CloseHandle(thread);
|
||||
s->free(this);
|
||||
}
|
||||
|
||||
HANDLE thread;
|
||||
HANDLE mutex;
|
||||
HANDLE event;
|
||||
System* s;
|
||||
System::Runnable* r;
|
||||
Thread* next;
|
||||
unsigned flags;
|
||||
};
|
||||
|
||||
class Monitor: public System::Monitor {
|
||||
public:
|
||||
Monitor(System* s): s(s), owner_(0), first(0), last(0), depth(0) {
|
||||
mutex = CreateMutex(0, false, 0);
|
||||
assert(s, mutex);
|
||||
}
|
||||
|
||||
virtual bool tryAcquire(System::Thread* context) {
|
||||
Thread* t = static_cast<Thread*>(context);
|
||||
|
||||
if (owner_ == t) {
|
||||
++ depth;
|
||||
return true;
|
||||
} else {
|
||||
switch (WaitForSingleObject(mutex, 0)) {
|
||||
case WAIT_TIMEOUT:
|
||||
return false;
|
||||
|
||||
case WAIT_OBJECT_0:
|
||||
owner_ = t;
|
||||
++ depth;
|
||||
return true;
|
||||
|
||||
default:
|
||||
sysAbort(s);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
virtual void acquire(System::Thread* context) {
|
||||
Thread* t = static_cast<Thread*>(context);
|
||||
|
||||
if (owner_ != t) {
|
||||
int r UNUSED = WaitForSingleObject(mutex, INFINITE);
|
||||
assert(s, r == WAIT_OBJECT_0);
|
||||
owner_ = t;
|
||||
}
|
||||
++ depth;
|
||||
}
|
||||
|
||||
virtual void release(System::Thread* context) {
|
||||
Thread* t = static_cast<Thread*>(context);
|
||||
|
||||
if (owner_ == t) {
|
||||
if (-- depth == 0) {
|
||||
owner_ = 0;
|
||||
int r UNUSED = ReleaseMutex(mutex);
|
||||
assert(s, r == 0);
|
||||
}
|
||||
} else {
|
||||
sysAbort(s);
|
||||
}
|
||||
}
|
||||
|
||||
void append(Thread* t) {
|
||||
if (last) {
|
||||
last->next = t;
|
||||
} else {
|
||||
first = last = t;
|
||||
}
|
||||
}
|
||||
|
||||
void remove(Thread* t) {
|
||||
for (Thread** p = &first; *p;) {
|
||||
if (t == *p) {
|
||||
*p = t->next;
|
||||
if (last == t) {
|
||||
last = 0;
|
||||
}
|
||||
break;
|
||||
} else {
|
||||
p = &((*p)->next);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
virtual bool wait(System::Thread* context, int64_t time) {
|
||||
Thread* t = static_cast<Thread*>(context);
|
||||
|
||||
if (owner_ == t) {
|
||||
ACQUIRE(s, t->mutex);
|
||||
|
||||
if (t->r->interrupted()) {
|
||||
t->r->setInterrupted(false);
|
||||
return true;
|
||||
}
|
||||
|
||||
t->flags |= Waiting;
|
||||
|
||||
append(t);
|
||||
|
||||
unsigned depth = this->depth;
|
||||
this->depth = 0;
|
||||
owner_ = 0;
|
||||
|
||||
int r UNUSED = ReleaseMutex(mutex);
|
||||
assert(s, r == 0);
|
||||
|
||||
r = ResetEvent(t->event);
|
||||
assert(s, r);
|
||||
|
||||
r = ReleaseMutex(t->mutex);
|
||||
assert(s, r == 0);
|
||||
|
||||
r = WaitForSingleObject(t->event, (time ? time : INFINITE));
|
||||
assert(s, r == WAIT_OBJECT_0);
|
||||
|
||||
r = WaitForSingleObject(t->mutex, INFINITE);
|
||||
assert(s, r == WAIT_OBJECT_0);
|
||||
|
||||
r = WaitForSingleObject(mutex, INFINITE);
|
||||
assert(s, r == WAIT_OBJECT_0);
|
||||
|
||||
owner_ = t;
|
||||
this->depth = depth;
|
||||
|
||||
if ((t->flags & Notified) == 0) {
|
||||
remove(t);
|
||||
}
|
||||
|
||||
t->flags = 0;
|
||||
t->next = 0;
|
||||
|
||||
if (t->r->interrupted()) {
|
||||
t->r->setInterrupted(false);
|
||||
return true;
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
sysAbort(s);
|
||||
}
|
||||
}
|
||||
|
||||
void doNotify(Thread* t) {
|
||||
ACQUIRE(s, t->mutex);
|
||||
|
||||
t->flags |= Notified;
|
||||
|
||||
int r UNUSED = SetEvent(t->event);
|
||||
assert(s, r == 0);
|
||||
}
|
||||
|
||||
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);
|
||||
}
|
||||
}
|
||||
|
||||
virtual void notifyAll(System::Thread* context) {
|
||||
Thread* t = static_cast<Thread*>(context);
|
||||
|
||||
if (owner_ == t) {
|
||||
for (Thread* t = first; t; t = t->next) {
|
||||
doNotify(t);
|
||||
}
|
||||
first = last = 0;
|
||||
} else {
|
||||
sysAbort(s);
|
||||
}
|
||||
}
|
||||
|
||||
virtual System::Thread* owner() {
|
||||
return owner_;
|
||||
}
|
||||
|
||||
virtual void dispose() {
|
||||
assert(s, owner_ == 0);
|
||||
CloseHandle(mutex);
|
||||
s->free(this);
|
||||
}
|
||||
|
||||
System* s;
|
||||
HANDLE mutex;
|
||||
Thread* owner_;
|
||||
Thread* first;
|
||||
Thread* last;
|
||||
unsigned depth;
|
||||
};
|
||||
|
||||
class Local: public System::Local {
|
||||
public:
|
||||
Local(System* s): s(s) {
|
||||
key = TlsAlloc();
|
||||
assert(s, key == TLS_OUT_OF_INDEXES);
|
||||
}
|
||||
|
||||
virtual void* get() {
|
||||
return TlsGetValue(key);
|
||||
}
|
||||
|
||||
virtual void set(void* p) {
|
||||
bool r UNUSED = TlsSetValue(key, p);
|
||||
assert(s, r);
|
||||
}
|
||||
|
||||
virtual void dispose() {
|
||||
bool r UNUSED = TlsFree(key);
|
||||
assert(s, r);
|
||||
|
||||
s->free(this);
|
||||
}
|
||||
|
||||
System* s;
|
||||
unsigned key;
|
||||
};
|
||||
|
||||
class Region: public System::Region {
|
||||
public:
|
||||
Region(System* system, uint8_t* start, size_t length, HANDLE mapping,
|
||||
HANDLE file):
|
||||
system(system),
|
||||
start_(start),
|
||||
length_(length),
|
||||
mapping(mapping),
|
||||
file(file)
|
||||
{ }
|
||||
|
||||
virtual const uint8_t* start() {
|
||||
return start_;
|
||||
}
|
||||
|
||||
virtual size_t length() {
|
||||
return length_;
|
||||
}
|
||||
|
||||
virtual void dispose() {
|
||||
if (start_) {
|
||||
if (start_) UnmapViewOfFile(start_);
|
||||
if (mapping) CloseHandle(mapping);
|
||||
if (file) CloseHandle(file);
|
||||
}
|
||||
system->free(this);
|
||||
}
|
||||
|
||||
System* system;
|
||||
uint8_t* start_;
|
||||
size_t length_;
|
||||
HANDLE mapping;
|
||||
HANDLE file;
|
||||
};
|
||||
|
||||
class Library: public System::Library {
|
||||
public:
|
||||
Library(System* s, HMODULE handle, const char* name, bool mapName,
|
||||
System::Library* next):
|
||||
s(s),
|
||||
handle(handle),
|
||||
name_(name),
|
||||
mapName_(mapName),
|
||||
next_(next)
|
||||
{ }
|
||||
|
||||
virtual void* resolve(const char* function) {
|
||||
void* address;
|
||||
FARPROC p = GetProcAddress(handle, function);
|
||||
memcpy(&address, &p, BytesPerWord);
|
||||
return address;
|
||||
}
|
||||
|
||||
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", handle);
|
||||
}
|
||||
|
||||
FreeLibrary(handle);
|
||||
|
||||
if (next_) {
|
||||
next_->dispose();
|
||||
}
|
||||
|
||||
if (name_) {
|
||||
s->free(name_);
|
||||
}
|
||||
|
||||
s->free(this);
|
||||
}
|
||||
|
||||
System* s;
|
||||
HMODULE handle;
|
||||
const char* name_;
|
||||
bool mapName_;
|
||||
System::Library* next_;
|
||||
};
|
||||
|
||||
MySystem(unsigned limit): limit(limit), count(0) {
|
||||
mutex = CreateMutex(0, false, 0);
|
||||
assert(this, mutex);
|
||||
}
|
||||
|
||||
virtual bool success(Status s) {
|
||||
return s == 0;
|
||||
}
|
||||
|
||||
virtual void* tryAllocate(unsigned size) {
|
||||
ACQUIRE(this, mutex);
|
||||
|
||||
if (Verbose) {
|
||||
fprintf(stderr, "try %d; count: %d; limit: %d\n",
|
||||
size, count, limit);
|
||||
}
|
||||
|
||||
if (count + size > limit) {
|
||||
return 0;
|
||||
} else {
|
||||
uintptr_t* up = static_cast<uintptr_t*>
|
||||
(malloc(size + sizeof(uintptr_t)));
|
||||
if (up == 0) {
|
||||
sysAbort(this);
|
||||
} else {
|
||||
*up = size;
|
||||
count += *up;
|
||||
|
||||
return up + 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
virtual void free(const void* p) {
|
||||
ACQUIRE(this, mutex);
|
||||
|
||||
if (p) {
|
||||
const uintptr_t* up = static_cast<const uintptr_t*>(p) - 1;
|
||||
if (count < *up) {
|
||||
abort();
|
||||
}
|
||||
count -= *up;
|
||||
|
||||
if (Verbose) {
|
||||
fprintf(stderr, "free %"ULD"; count: %d; limit: %d\n",
|
||||
*up, count, limit);
|
||||
}
|
||||
|
||||
::free(const_cast<uintptr_t*>(up));
|
||||
}
|
||||
}
|
||||
|
||||
virtual Status attach(Runnable* r) {
|
||||
Thread* t = new (System::allocate(sizeof(Thread))) Thread(this, r);
|
||||
bool success = DuplicateHandle
|
||||
(GetCurrentProcess(), GetCurrentThread(), GetCurrentProcess(),
|
||||
&(t->thread), 0, false, DUPLICATE_SAME_ACCESS);
|
||||
assert(this, success);
|
||||
r->attach(t);
|
||||
return 0;
|
||||
}
|
||||
|
||||
virtual Status start(Runnable* r) {
|
||||
Thread* t = new (System::allocate(sizeof(Thread))) Thread(this, r);
|
||||
r->attach(t);
|
||||
DWORD id;
|
||||
t->thread = CreateThread(0, 0, run, r, 0, &id);
|
||||
assert(this, t->thread);
|
||||
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;
|
||||
}
|
||||
|
||||
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);
|
||||
}
|
||||
|
||||
virtual Status map(System::Region** region, const char* name) {
|
||||
Status status = 1;
|
||||
|
||||
HANDLE file = CreateFile(name, FILE_READ_DATA, 0, 0, OPEN_EXISTING, 0, 0);
|
||||
if (file != INVALID_HANDLE_VALUE) {
|
||||
unsigned size = GetFileSize(file, 0);
|
||||
if (size != INVALID_FILE_SIZE) {
|
||||
HANDLE mapping = CreateFileMapping(file, 0, PAGE_READONLY, 0, size, 0);
|
||||
if (mapping) {
|
||||
void* data = MapViewOfFile(mapping, FILE_MAP_READ, 0, 0, 0);
|
||||
if (data) {
|
||||
*region = new (allocate(sizeof(Region)))
|
||||
Region(this, static_cast<uint8_t*>(data), size, file, mapping);
|
||||
status = 0;
|
||||
}
|
||||
|
||||
if (status) {
|
||||
CloseHandle(mapping);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (status) {
|
||||
CloseHandle(file);
|
||||
}
|
||||
}
|
||||
|
||||
return status;
|
||||
}
|
||||
|
||||
virtual FileType identify(const char* name) {
|
||||
struct _stat s;
|
||||
int r = _stat(name, &s);
|
||||
if (r == 0) {
|
||||
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)
|
||||
{
|
||||
HMODULE handle;
|
||||
unsigned nameLength = (name ? strlen(name) : 0);
|
||||
if (mapName) {
|
||||
unsigned size = nameLength + sizeof(SO_SUFFIX);
|
||||
char buffer[size];
|
||||
snprintf(buffer, size, "%s" SO_SUFFIX, name);
|
||||
handle = LoadLibrary(buffer);
|
||||
} else {
|
||||
handle = LoadLibrary(name);
|
||||
}
|
||||
|
||||
if (handle) {
|
||||
if (Verbose) {
|
||||
fprintf(stderr, "open %s as %p\n", name, handle);
|
||||
}
|
||||
|
||||
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, handle, n, mapName, next);
|
||||
return 0;
|
||||
} else {
|
||||
// fprintf(stderr, "dlerror: %s\n", dlerror());
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
virtual int64_t now() {
|
||||
static LARGE_INTEGER frequency;
|
||||
static LARGE_INTEGER time;
|
||||
static bool init = true;
|
||||
|
||||
if (init) {
|
||||
QueryPerformanceFrequency(&frequency);
|
||||
|
||||
if (frequency.QuadPart == 0) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
init = false;
|
||||
}
|
||||
|
||||
QueryPerformanceCounter(&time);
|
||||
return static_cast<int64_t>
|
||||
(((static_cast<double>(time.QuadPart)) * 1000.0) /
|
||||
(static_cast<double>(frequency.QuadPart)));
|
||||
}
|
||||
|
||||
virtual void exit(int code) {
|
||||
::exit(code);
|
||||
}
|
||||
|
||||
virtual void abort() {
|
||||
::abort();
|
||||
}
|
||||
|
||||
virtual void dispose() {
|
||||
CloseHandle(mutex);
|
||||
::free(this);
|
||||
}
|
||||
|
||||
HANDLE mutex;
|
||||
unsigned limit;
|
||||
unsigned count;
|
||||
};
|
||||
|
||||
} // namespace
|
||||
|
||||
namespace vm {
|
||||
|
||||
System*
|
||||
makeSystem(unsigned heapSize)
|
||||
{
|
||||
return new (malloc(sizeof(MySystem))) MySystem(heapSize);
|
||||
}
|
||||
|
||||
} // namespace vm
|
||||
|
@ -105,7 +105,7 @@ exit:
|
||||
|
||||
#elif defined __i386__
|
||||
|
||||
#ifdef __APPLE__
|
||||
#if defined __APPLE__ || defined __MINGW32__
|
||||
.globl _cdeclCall
|
||||
_cdeclCall:
|
||||
#else
|
79
src/x86.h
Normal file
79
src/x86.h
Normal file
@ -0,0 +1,79 @@
|
||||
#ifndef X86_H
|
||||
#define X86_H
|
||||
|
||||
#include "types.h"
|
||||
#include "stdint.h"
|
||||
|
||||
#ifdef __i386__
|
||||
|
||||
extern "C" uint64_t
|
||||
cdeclCall(void* function, void* stack, unsigned stackSize,
|
||||
unsigned returnType);
|
||||
|
||||
namespace vm {
|
||||
|
||||
inline uint64_t
|
||||
dynamicCall(void* function, uintptr_t* arguments, uint8_t*,
|
||||
unsigned, unsigned argumentsSize, unsigned returnType)
|
||||
{
|
||||
return cdeclCall(function, arguments, argumentsSize, returnType);
|
||||
}
|
||||
|
||||
} // namespace vm
|
||||
|
||||
#elif defined __x86_64__
|
||||
|
||||
extern "C" uint64_t
|
||||
amd64Call(void* function, void* stack, unsigned stackSize,
|
||||
void* gprTable, void* sseTable, unsigned returnType);
|
||||
|
||||
namespace vm {
|
||||
|
||||
inline uint64_t
|
||||
dynamicCall(void* function, uint64_t* arguments, uint8_t* argumentTypes,
|
||||
unsigned argumentCount, unsigned, unsigned returnType)
|
||||
{
|
||||
const unsigned GprCount = 6;
|
||||
uint64_t gprTable[GprCount];
|
||||
unsigned gprIndex = 0;
|
||||
|
||||
const unsigned SseCount = 8;
|
||||
uint64_t sseTable[SseCount];
|
||||
unsigned sseIndex = 0;
|
||||
|
||||
uint64_t stack[argumentCount];
|
||||
unsigned stackIndex = 0;
|
||||
|
||||
for (unsigned i = 0; i < argumentCount; ++i) {
|
||||
switch (argumentTypes[i]) {
|
||||
case FLOAT_TYPE:
|
||||
case DOUBLE_TYPE: {
|
||||
if (sseIndex < SseCount) {
|
||||
sseTable[sseIndex++] = arguments[i];
|
||||
} else {
|
||||
stack[stackIndex++] = arguments[i];
|
||||
}
|
||||
} break;
|
||||
|
||||
default: {
|
||||
if (gprIndex < GprCount) {
|
||||
gprTable[gprIndex++] = arguments[i];
|
||||
} else {
|
||||
stack[stackIndex++] = arguments[i];
|
||||
}
|
||||
} break;
|
||||
}
|
||||
}
|
||||
|
||||
return amd64Call(function, stack, stackIndex * 8, (gprIndex ? gprTable : 0),
|
||||
(sseIndex ? sseTable : 0), returnType);
|
||||
}
|
||||
|
||||
} // namespace vm
|
||||
|
||||
#else
|
||||
# error unsupported platform
|
||||
#endif
|
||||
|
||||
|
||||
#endif//X86_H
|
Loading…
Reference in New Issue
Block a user