diff --git a/makefile b/makefile index b514d1b072..3987673d91 100644 --- a/makefile +++ b/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 diff --git a/src/common.h b/src/common.h index 802e1d36c1..197ad27205 100644 --- a/src/common.h +++ b/src/common.h @@ -28,6 +28,8 @@ #ifdef __APPLE__ # define SO_SUFFIX ".jnilib" +#elseif defined __MINGW32__ +# define SO_SUFFIX ".dll" #else # define SO_SUFFIX ".so" #endif diff --git a/src/posix.cpp b/src/posix.cpp index 493d311833..5dc42bd0bd 100644 --- a/src/posix.cpp +++ b/src/posix.cpp @@ -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 (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(p) - 1; @@ -516,8 +442,6 @@ class MySystem: public System { ::free(const_cast(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(tv.tv_sec) * 1000) + (static_cast(tv.tv_usec) / 1000); } + virtual void exit(int code) { + ::exit(code); + } + virtual void abort() { ::abort(); } diff --git a/src/windows.cpp b/src/windows.cpp index 65b3dba385..febd2dea14 100644 --- a/src/windows.cpp +++ b/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(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(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(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(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(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(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(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 + (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(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(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(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(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 + (((static_cast(time.QuadPart)) * 1000.0) / + (static_cast(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 diff --git a/src/posix.S b/src/x86.S similarity index 98% rename from src/posix.S rename to src/x86.S index b8bfffd221..fc30148917 100644 --- a/src/posix.S +++ b/src/x86.S @@ -105,7 +105,7 @@ exit: #elif defined __i386__ -#ifdef __APPLE__ +#if defined __APPLE__ || defined __MINGW32__ .globl _cdeclCall _cdeclCall: #else diff --git a/src/x86.h b/src/x86.h new file mode 100644 index 0000000000..4b9c877d92 --- /dev/null +++ b/src/x86.h @@ -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