mirror of
https://github.com/MaskRay/ccls.git
synced 2024-11-22 07:35:08 +00:00
229 lines
6.4 KiB
C++
229 lines
6.4 KiB
C++
#if defined(_WIN32)
|
|
#include "platform.h"
|
|
|
|
#include "utils.h"
|
|
|
|
#include <loguru.hpp>
|
|
|
|
#include <Windows.h>
|
|
#include <direct.h>
|
|
#include <fcntl.h>
|
|
#include <io.h>
|
|
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
|
|
#include <algorithm>
|
|
#include <cassert>
|
|
#include <iostream>
|
|
#include <string>
|
|
|
|
namespace {
|
|
|
|
DWORD CheckForError(std::vector<DWORD> allow) {
|
|
DWORD error = GetLastError();
|
|
if (error == ERROR_SUCCESS ||
|
|
std::find(allow.begin(), allow.end(), error) != allow.end())
|
|
return error;
|
|
|
|
// See http://stackoverflow.com/a/17387176
|
|
LPSTR message_buffer = nullptr;
|
|
size_t size = FormatMessageA(
|
|
FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM |
|
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
|
NULL, error, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
|
|
(LPSTR)&message_buffer, 0, NULL);
|
|
std::string message(message_buffer, size);
|
|
LocalFree(message_buffer);
|
|
|
|
std::cerr << "Windows error code=" << error << ", message=" << message
|
|
<< std::endl;
|
|
|
|
assert(false); // debugger break
|
|
exit(1);
|
|
}
|
|
|
|
struct PlatformMutexWin : public PlatformMutex {
|
|
HANDLE raw_mutex = INVALID_HANDLE_VALUE;
|
|
|
|
PlatformMutexWin(const std::string& name) {
|
|
std::cerr << "[win] Creating mutex with name " << name << std::endl;
|
|
raw_mutex = CreateMutex(nullptr, false /*initial_owner*/, name.c_str());
|
|
CheckForError({ERROR_ALREADY_EXISTS});
|
|
}
|
|
|
|
~PlatformMutexWin() override {
|
|
CloseHandle(raw_mutex);
|
|
CheckForError({} /*allow*/);
|
|
raw_mutex = INVALID_HANDLE_VALUE;
|
|
}
|
|
};
|
|
|
|
struct PlatformScopedMutexLockWin : public PlatformScopedMutexLock {
|
|
HANDLE raw_mutex;
|
|
|
|
PlatformScopedMutexLockWin(HANDLE raw_mutex) : raw_mutex(raw_mutex) {
|
|
DWORD result = WaitForSingleObject(raw_mutex, INFINITE);
|
|
assert(result == WAIT_OBJECT_0);
|
|
CheckForError({ERROR_NO_MORE_FILES, ERROR_ALREADY_EXISTS} /*allow*/);
|
|
}
|
|
|
|
~PlatformScopedMutexLockWin() override {
|
|
ReleaseMutex(raw_mutex);
|
|
CheckForError({ERROR_NO_MORE_FILES, ERROR_ALREADY_EXISTS} /*allow*/);
|
|
}
|
|
};
|
|
|
|
struct PlatformSharedMemoryWin : public PlatformSharedMemory {
|
|
HANDLE shmem_;
|
|
|
|
PlatformSharedMemoryWin(const std::string& name, size_t capacity) {
|
|
std::cerr << "[win] Creating shared memory with name " << name
|
|
<< " and capacity " << capacity << std::endl;
|
|
this->name = name;
|
|
|
|
shmem_ = CreateFileMapping(INVALID_HANDLE_VALUE, NULL, PAGE_READWRITE, 0,
|
|
(DWORD)capacity, name.c_str());
|
|
CheckForError({ERROR_ALREADY_EXISTS} /*allow*/);
|
|
|
|
data = MapViewOfFile(shmem_, FILE_MAP_ALL_ACCESS, 0, 0, capacity);
|
|
CheckForError({ERROR_ALREADY_EXISTS} /*allow*/);
|
|
|
|
this->capacity = capacity;
|
|
}
|
|
|
|
~PlatformSharedMemoryWin() override {
|
|
UnmapViewOfFile(data);
|
|
CheckForError({} /*allow*/);
|
|
|
|
data = nullptr;
|
|
capacity = 0;
|
|
}
|
|
};
|
|
|
|
} // namespace
|
|
|
|
std::unique_ptr<PlatformMutex> CreatePlatformMutex(const std::string& name) {
|
|
return MakeUnique<PlatformMutexWin>(name);
|
|
}
|
|
|
|
std::unique_ptr<PlatformScopedMutexLock> CreatePlatformScopedMutexLock(
|
|
PlatformMutex* mutex) {
|
|
return MakeUnique<PlatformScopedMutexLockWin>(
|
|
static_cast<PlatformMutexWin*>(mutex)->raw_mutex);
|
|
}
|
|
|
|
std::unique_ptr<PlatformSharedMemory> CreatePlatformSharedMemory(
|
|
const std::string& name,
|
|
size_t size) {
|
|
return MakeUnique<PlatformSharedMemoryWin>(name, size);
|
|
}
|
|
|
|
void PlatformInit() {
|
|
// We need to write to stdout in binary mode because in Windows, writing
|
|
// \n will implicitly write \r\n. Language server API will ignore a
|
|
// \r\r\n split request.
|
|
_setmode(_fileno(stdout), O_BINARY);
|
|
_setmode(_fileno(stdin), O_BINARY);
|
|
}
|
|
|
|
// See http://stackoverflow.com/a/19535628
|
|
std::string GetWorkingDirectory() {
|
|
char result[MAX_PATH];
|
|
std::string binary_path(result, GetModuleFileName(NULL, result, MAX_PATH));
|
|
return binary_path.substr(0, binary_path.find_last_of("\\/") + 1);
|
|
}
|
|
|
|
std::string NormalizePath(const std::string& path) {
|
|
DWORD retval = 0;
|
|
TCHAR buffer[MAX_PATH] = TEXT("");
|
|
TCHAR buf[MAX_PATH] = TEXT("");
|
|
TCHAR** lpp_part = {NULL};
|
|
|
|
retval = GetFullPathName(path.c_str(), MAX_PATH, buffer, lpp_part);
|
|
// fail, return original
|
|
if (retval == 0)
|
|
return path;
|
|
|
|
std::string result = buffer;
|
|
std::replace(result.begin(), result.end(), '\\', '/');
|
|
// std::transform(result.begin(), result.end(), result.begin(), ::tolower);
|
|
return result;
|
|
}
|
|
|
|
bool TryMakeDirectory(const std::string& absolute_path) {
|
|
if (_mkdir(absolute_path.c_str()) == -1) {
|
|
// Success if the directory exists.
|
|
return errno == EEXIST;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// See https://msdn.microsoft.com/en-us/library/xcb2z8hs.aspx
|
|
const DWORD MS_VC_EXCEPTION = 0x406D1388;
|
|
#pragma pack(push, 8)
|
|
typedef struct tagTHREADNAME_INFO {
|
|
DWORD dwType; // Must be 0x1000.
|
|
LPCSTR szName; // Pointer to name (in user addr space).
|
|
DWORD dwThreadID; // Thread ID (-1=caller thread).
|
|
DWORD dwFlags; // Reserved for future use, must be zero.
|
|
} THREADNAME_INFO;
|
|
#pragma pack(pop)
|
|
void SetCurrentThreadName(const std::string& thread_name) {
|
|
loguru::set_thread_name(thread_name.c_str());
|
|
|
|
THREADNAME_INFO info;
|
|
info.dwType = 0x1000;
|
|
info.szName = thread_name.c_str();
|
|
info.dwThreadID = (DWORD)-1;
|
|
info.dwFlags = 0;
|
|
|
|
__try {
|
|
RaiseException(MS_VC_EXCEPTION, 0, sizeof(info) / sizeof(ULONG_PTR),
|
|
(ULONG_PTR*)&info);
|
|
} __except (EXCEPTION_EXECUTE_HANDLER) {
|
|
}
|
|
}
|
|
|
|
optional<int64_t> GetLastModificationTime(const std::string& absolute_path) {
|
|
struct _stat buf;
|
|
if (_stat(absolute_path.c_str(), &buf) != 0) {
|
|
switch (errno) {
|
|
case ENOENT:
|
|
// std::cerr << "GetLastModificationTime: unable to find file " <<
|
|
// absolute_path << std::endl;
|
|
return nullopt;
|
|
case EINVAL:
|
|
// std::cerr << "GetLastModificationTime: invalid param to _stat for
|
|
// file file " << absolute_path << std::endl;
|
|
return nullopt;
|
|
default:
|
|
// std::cerr << "GetLastModificationTime: unhandled for " <<
|
|
// absolute_path << std::endl; exit(1);
|
|
return nullopt;
|
|
}
|
|
}
|
|
|
|
return buf.st_mtime;
|
|
}
|
|
|
|
void MoveFileTo(const std::string& destination, const std::string& source) {
|
|
MoveFile(source.c_str(), destination.c_str());
|
|
}
|
|
|
|
void CopyFileTo(const std::string& destination, const std::string& source) {
|
|
CopyFile(source.c_str(), destination.c_str(), false /*failIfExists*/);
|
|
}
|
|
|
|
bool IsSymLink(const std::string& path) {
|
|
return false;
|
|
}
|
|
|
|
std::vector<std::string> GetPlatformClangArguments() {
|
|
return {"-fms-compatibility", "-fdelayed-template-parsing"};
|
|
}
|
|
|
|
void FreeUnusedMemory() {}
|
|
|
|
#endif
|