Commit 1745aded authored by Kenton Varda's avatar Kenton Varda

MinGW: Entire compiler builds and runs.

parent 90cda34c
...@@ -36,12 +36,20 @@ ...@@ -36,12 +36,20 @@
#include <kj/parse/char.h> #include <kj/parse/char.h>
#include <sys/stat.h> #include <sys/stat.h>
#include <sys/types.h> #include <sys/types.h>
#include <sys/wait.h>
#include <capnp/serialize.h> #include <capnp/serialize.h>
#include <capnp/serialize-packed.h> #include <capnp/serialize-packed.h>
#include <errno.h> #include <errno.h>
#include <stdlib.h> #include <stdlib.h>
#if _WIN32
#include <process.h>
#include <io.h>
#include <fcntl.h>
#define pipe(fds) _pipe(fds, 8192, _O_BINARY)
#else
#include <sys/wait.h>
#endif
#if HAVE_CONFIG_H #if HAVE_CONFIG_H
#include "config.h" #include "config.h"
#endif #endif
...@@ -408,49 +416,95 @@ public: ...@@ -408,49 +416,95 @@ public:
exeName = kj::heapString(output.name); exeName = kj::heapString(output.name);
} }
kj::Array<char> pwd = kj::heapArray<char>(256);
while (getcwd(pwd.begin(), pwd.size()) == nullptr) {
KJ_REQUIRE(pwd.size() < 8192, "WTF your working directory path is more than 8k?");
pwd = kj::heapArray<char>(pwd.size() * 2);
}
#if _WIN32
int oldStdin;
KJ_SYSCALL(oldStdin = dup(STDIN_FILENO));
intptr_t child;
#else // _WIN32
pid_t child; pid_t child;
KJ_SYSCALL(child = fork()); KJ_SYSCALL(child = fork());
if (child == 0) { if (child == 0) {
// I am the child! // I am the child!
KJ_SYSCALL(close(pipeFds[1])); KJ_SYSCALL(close(pipeFds[1]));
#endif // _WIN32, else
KJ_SYSCALL(dup2(pipeFds[0], STDIN_FILENO)); KJ_SYSCALL(dup2(pipeFds[0], STDIN_FILENO));
KJ_SYSCALL(close(pipeFds[0])); KJ_SYSCALL(close(pipeFds[0]));
kj::Array<char> pwd = kj::heapArray<char>(256);
while (getcwd(pwd.begin(), pwd.size()) == nullptr) {
KJ_REQUIRE(pwd.size() < 8192, "WTF your working directory path is more than 8k?");
pwd = kj::heapArray<char>(pwd.size() * 2);
}
if (output.dir != nullptr) { if (output.dir != nullptr) {
KJ_SYSCALL(chdir(output.dir.cStr()), output.dir); KJ_SYSCALL(chdir(output.dir.cStr()), output.dir);
} }
if (shouldSearchPath) { if (shouldSearchPath) {
#if _WIN32
child = _spawnlp(_P_NOWAIT, exeName.cStr(), exeName.cStr(), nullptr);
#else
execlp(exeName.cStr(), exeName.cStr(), nullptr); execlp(exeName.cStr(), exeName.cStr(), nullptr);
#endif
} else { } else {
if (!exeName.startsWith("/")) { if (!exeName.startsWith("/")) {
// The name is relative. Prefix it with our original working directory path. // The name is relative. Prefix it with our original working directory path.
exeName = kj::str(pwd.begin(), "/", exeName); exeName = kj::str(pwd.begin(), "/", exeName);
} }
#if _WIN32
child = _spawnl(_P_NOWAIT, exeName.cStr(), exeName.cStr(), nullptr);
#else
execl(exeName.cStr(), exeName.cStr(), nullptr); execl(exeName.cStr(), exeName.cStr(), nullptr);
#endif
} }
#if _WIN32
if (child == -1) {
#endif
int error = errno; int error = errno;
if (error == ENOENT) { if (error == ENOENT) {
context.exitError(kj::str(output.name, ": no such plugin (executable should be '", context.exitError(kj::str(output.name, ": no such plugin (executable should be '",
exeName, "')")); exeName, "')"));
} else { } else {
#if _WIN32
KJ_FAIL_SYSCALL("spawn()", error);
#else
KJ_FAIL_SYSCALL("exec()", error); KJ_FAIL_SYSCALL("exec()", error);
#endif
} }
#if _WIN32
}
// Restore stdin.
KJ_SYSCALL(dup2(oldStdin, STDIN_FILENO));
KJ_SYSCALL(close(oldStdin));
// Restore current directory.
KJ_SYSCALL(chdir(pwd.begin()), pwd.begin());
#else // _WIN32
} }
KJ_SYSCALL(close(pipeFds[0])); KJ_SYSCALL(close(pipeFds[0]));
#endif // _WIN32, else
writeMessageToFd(pipeFds[1], message); writeMessageToFd(pipeFds[1], message);
KJ_SYSCALL(close(pipeFds[1])); KJ_SYSCALL(close(pipeFds[1]));
#if _WIN32
int status;
if (_cwait(&status, child, 0) == -1) {
KJ_FAIL_SYSCALL("_cwait()", errno);
}
if (status != 0) {
context.error(kj::str(output.name, ": plugin failed: exit code ", status));
}
#else // _WIN32
int status; int status;
KJ_SYSCALL(waitpid(child, &status, 0)); KJ_SYSCALL(waitpid(child, &status, 0));
if (WIFSIGNALED(status)) { if (WIFSIGNALED(status)) {
...@@ -458,6 +512,7 @@ public: ...@@ -458,6 +512,7 @@ public:
} else if (WIFEXITED(status) && WEXITSTATUS(status) != 0) { } else if (WIFEXITED(status) && WEXITSTATUS(status) != 0) {
context.error(kj::str(output.name, ": plugin failed: exit code ", WEXITSTATUS(status))); context.error(kj::str(output.name, ": plugin failed: exit code ", WEXITSTATUS(status)));
} }
#endif // _WIN32, else
} }
return true; return true;
......
...@@ -50,6 +50,10 @@ ...@@ -50,6 +50,10 @@
#define VERSION "(unknown)" #define VERSION "(unknown)"
#endif #endif
#if _WIN32
#define mkdir(path, mode) mkdir(path)
#endif
namespace capnp { namespace capnp {
namespace { namespace {
......
...@@ -29,12 +29,17 @@ ...@@ -29,12 +29,17 @@
#include <capnp/message.h> #include <capnp/message.h>
#include <map> #include <map>
#include <unistd.h> #include <unistd.h>
#include <sys/mman.h>
#include <sys/types.h> #include <sys/types.h>
#include <sys/stat.h> #include <sys/stat.h>
#include <fcntl.h> #include <fcntl.h>
#include <errno.h> #include <errno.h>
#if _WIN32
#include <windows.h>
#else
#include <sys/mman.h>
#endif
namespace capnp { namespace capnp {
namespace compiler { namespace compiler {
...@@ -44,7 +49,11 @@ class MmapDisposer: public kj::ArrayDisposer { ...@@ -44,7 +49,11 @@ class MmapDisposer: public kj::ArrayDisposer {
protected: protected:
void disposeImpl(void* firstElement, size_t elementSize, size_t elementCount, void disposeImpl(void* firstElement, size_t elementSize, size_t elementCount,
size_t capacity, void (*destroyElement)(void*)) const { size_t capacity, void (*destroyElement)(void*)) const {
#if _WIN32
KJ_ASSERT(UnmapViewOfFile(firstElement));
#else
munmap(firstElement, elementSize * elementCount); munmap(firstElement, elementSize * elementCount);
#endif
} }
}; };
...@@ -66,10 +75,20 @@ kj::Array<const char> mmapForRead(kj::StringPtr filename) { ...@@ -66,10 +75,20 @@ kj::Array<const char> mmapForRead(kj::StringPtr filename) {
} }
// Regular file. Just mmap() it. // Regular file. Just mmap() it.
#if _WIN32
HANDLE handle = reinterpret_cast<HANDLE>(_get_osfhandle(fd));
KJ_ASSERT(handle != INVALID_HANDLE_VALUE);
HANDLE mappingHandle = CreateFileMapping(
handle, NULL, PAGE_READONLY, 0, stats.st_size, NULL);
KJ_ASSERT(mappingHandle != INVALID_HANDLE_VALUE);
KJ_DEFER(KJ_ASSERT(CloseHandle(mappingHandle)));
const void* mapping = MapViewOfFile(mappingHandle, FILE_MAP_READ, 0, 0, stats.st_size);
#else // _WIN32
const void* mapping = mmap(NULL, stats.st_size, PROT_READ, MAP_SHARED, fd, 0); const void* mapping = mmap(NULL, stats.st_size, PROT_READ, MAP_SHARED, fd, 0);
if (mapping == MAP_FAILED) { if (mapping == MAP_FAILED) {
KJ_FAIL_SYSCALL("mmap", errno, filename); KJ_FAIL_SYSCALL("mmap", errno, filename);
} }
#endif // _WIN32, else
return kj::Array<const char>( return kj::Array<const char>(
reinterpret_cast<const char*>(mapping), stats.st_size, mmapDisposer); reinterpret_cast<const char*>(mapping), stats.st_size, mmapDisposer);
......
...@@ -32,12 +32,17 @@ ...@@ -32,12 +32,17 @@
#include <kj/debug.h> #include <kj/debug.h>
#include <kj/io.h> #include <kj/io.h>
#include <unistd.h> #include <unistd.h>
#include <sys/mman.h>
#include <sys/types.h> #include <sys/types.h>
#include <sys/stat.h> #include <sys/stat.h>
#include <fcntl.h> #include <fcntl.h>
#include <errno.h> #include <errno.h>
#if _WIN32
#include <windows.h>
#else
#include <sys/mman.h>
#endif
namespace capnp { namespace capnp {
namespace { namespace {
...@@ -221,7 +226,11 @@ class MmapDisposer: public kj::ArrayDisposer { ...@@ -221,7 +226,11 @@ class MmapDisposer: public kj::ArrayDisposer {
protected: protected:
void disposeImpl(void* firstElement, size_t elementSize, size_t elementCount, void disposeImpl(void* firstElement, size_t elementSize, size_t elementCount,
size_t capacity, void (*destroyElement)(void*)) const { size_t capacity, void (*destroyElement)(void*)) const {
#if _WIN32
KJ_ASSERT(UnmapViewOfFile(firstElement));
#else
munmap(firstElement, elementSize * elementCount); munmap(firstElement, elementSize * elementCount);
#endif
} }
}; };
...@@ -359,10 +368,20 @@ kj::Array<const char> SchemaFile::DiskFileReader::read(kj::StringPtr path) const ...@@ -359,10 +368,20 @@ kj::Array<const char> SchemaFile::DiskFileReader::read(kj::StringPtr path) const
} }
// Regular file. Just mmap() it. // Regular file. Just mmap() it.
#if _WIN32
HANDLE handle = reinterpret_cast<HANDLE>(_get_osfhandle(fd));
KJ_ASSERT(handle != INVALID_HANDLE_VALUE);
HANDLE mappingHandle = CreateFileMapping(
handle, NULL, PAGE_READONLY, 0, stats.st_size, NULL);
KJ_ASSERT(mappingHandle != INVALID_HANDLE_VALUE);
KJ_DEFER(KJ_ASSERT(CloseHandle(mappingHandle)));
const void* mapping = MapViewOfFile(mappingHandle, FILE_MAP_READ, 0, 0, stats.st_size);
#else // _WIN32
const void* mapping = mmap(NULL, stats.st_size, PROT_READ, MAP_SHARED, fd, 0); const void* mapping = mmap(NULL, stats.st_size, PROT_READ, MAP_SHARED, fd, 0);
if (mapping == MAP_FAILED) { if (mapping == MAP_FAILED) {
KJ_FAIL_SYSCALL("mmap", errno, path); KJ_FAIL_SYSCALL("mmap", errno, path);
} }
#endif // !_WIN32
return kj::Array<const char>( return kj::Array<const char>(
reinterpret_cast<const char*>(mapping), stats.st_size, mmapDisposer); reinterpret_cast<const char*>(mapping), stats.st_size, mmapDisposer);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment