2020-05-19 14:11:07 +03:00
|
|
|
#include <io.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <windows.h>
|
2021-05-25 18:45:46 +03:00
|
|
|
#include <process.h>
|
2020-05-19 14:11:07 +03:00
|
|
|
|
|
|
|
// THis file exists to avoid clashes between windows.h and idris_rts.h
|
|
|
|
//
|
|
|
|
|
|
|
|
int win_fpoll(FILE *f)
|
|
|
|
{
|
|
|
|
HANDLE wh =(HANDLE) _get_osfhandle(_fileno(f));
|
|
|
|
if (wh == INVALID_HANDLE_VALUE) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
DWORD ret = WaitForSingleObject(wh, 1000);
|
|
|
|
// Imitate the return values of select()
|
|
|
|
if (ret == WAIT_OBJECT_0)
|
|
|
|
return 1;
|
|
|
|
if (ret == WAIT_TIMEOUT)
|
|
|
|
return 0;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int widen_utf8(const char *filename_utf8, LPWSTR *filename_w)
|
|
|
|
{
|
|
|
|
int num_chars = MultiByteToWideChar(CP_UTF8, 0, filename_utf8, -1, 0, 0);
|
|
|
|
int size = sizeof(WCHAR);
|
|
|
|
*filename_w = (LPWSTR)malloc(size * num_chars);
|
|
|
|
MultiByteToWideChar(CP_UTF8, 0, filename_utf8, -1, *filename_w, num_chars);
|
|
|
|
return num_chars;
|
|
|
|
}
|
|
|
|
|
|
|
|
FILE *win32_u8fopen(const char *path, const char *mode)
|
|
|
|
{
|
|
|
|
LPWSTR wpath, wmode;
|
|
|
|
widen_utf8(path, &wpath);
|
|
|
|
widen_utf8(mode, &wmode);
|
|
|
|
FILE *f = _wfopen(wpath, wmode);
|
|
|
|
free(wpath);
|
|
|
|
free(wmode);
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
|
|
|
FILE *win32_u8popen(const char *path, const char *mode)
|
|
|
|
{
|
|
|
|
LPWSTR wpath, wmode;
|
|
|
|
widen_utf8(path, &wpath);
|
|
|
|
widen_utf8(mode, &wmode);
|
|
|
|
FILE *f = _wpopen(wpath, wmode);
|
|
|
|
free(wpath);
|
|
|
|
free(wmode);
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
|
|
|
void win32_gettime(int64_t* sec, int64_t* nsec)
|
|
|
|
{
|
|
|
|
FILETIME ft;
|
2020-06-02 05:30:44 +03:00
|
|
|
#ifdef _OLD_WIN
|
|
|
|
GetSystemTimeAsFileTime(&ft);
|
|
|
|
#else
|
|
|
|
// For Windows NT 6.2 or higher
|
2020-05-19 14:11:07 +03:00
|
|
|
GetSystemTimePreciseAsFileTime(&ft);
|
2020-06-02 05:30:44 +03:00
|
|
|
#endif
|
2020-05-19 14:11:07 +03:00
|
|
|
ULARGE_INTEGER t;
|
|
|
|
t.HighPart = ft.dwHighDateTime;
|
|
|
|
t.LowPart = ft.dwLowDateTime;
|
|
|
|
|
|
|
|
*nsec = (t.QuadPart % 10000000)*100;
|
|
|
|
*sec = t.QuadPart / 10000000;
|
2020-06-16 13:09:22 +03:00
|
|
|
*sec -= 11644473600; // LDAP epoch to Unix epoch
|
2020-05-19 14:11:07 +03:00
|
|
|
}
|
2020-05-24 20:54:41 +03:00
|
|
|
|
|
|
|
void win32_sleep(int ms) {
|
|
|
|
Sleep(ms);
|
|
|
|
}
|
2020-06-16 13:09:22 +03:00
|
|
|
|
2020-06-16 14:41:40 +03:00
|
|
|
int win32_modenv(const char* name, const char* value, int overwrite) {
|
2020-06-16 14:58:49 +03:00
|
|
|
if (!overwrite && getenv(name)) return 0;
|
2020-06-16 15:09:12 +03:00
|
|
|
return _putenv_s(name, value);
|
2021-01-22 16:25:54 +03:00
|
|
|
}
|
2021-03-26 21:27:25 +03:00
|
|
|
|
|
|
|
int win32_getErrno() {
|
|
|
|
return GetLastError();
|
|
|
|
}
|
|
|
|
|
2021-05-25 18:45:46 +03:00
|
|
|
int win32_getPID() {
|
|
|
|
return _getpid();
|
|
|
|
}
|
|
|
|
|
2021-03-26 21:27:25 +03:00
|
|
|
typedef BOOL (WINAPI *LPFN_GLPI)(
|
|
|
|
PSYSTEM_LOGICAL_PROCESSOR_INFORMATION,
|
|
|
|
PDWORD);
|
|
|
|
|
|
|
|
long win32_getNProcessors() {
|
|
|
|
// largely taken from
|
|
|
|
// https://docs.microsoft.com/en-us/windows/win32/api/sysinfoapi/nf-sysinfoapi-getlogicalprocessorinformation
|
|
|
|
|
|
|
|
BOOL done = FALSE;
|
|
|
|
long nPhysicalProcessors = 0;
|
|
|
|
long nSMTProcessors = 0;
|
|
|
|
|
|
|
|
// length of array for storing the information structs
|
|
|
|
DWORD returnLength = 0;
|
|
|
|
// structs for storing the information
|
|
|
|
PSYSTEM_LOGICAL_PROCESSOR_INFORMATION buffer = NULL;
|
|
|
|
PSYSTEM_LOGICAL_PROCESSOR_INFORMATION ptr = NULL;
|
|
|
|
|
|
|
|
// shortcut to a function (?)
|
|
|
|
LPFN_GLPI glpi;
|
|
|
|
glpi = (LPFN_GLPI) GetProcAddress( GetModuleHandle(TEXT("kernel32"))
|
|
|
|
, "GetLogicalProcessorInformation"
|
|
|
|
);
|
|
|
|
// to keep track of whether we're at the end of the array
|
|
|
|
DWORD byteOffset = 0;
|
|
|
|
|
|
|
|
// repeatedly try to malloc and retrieve the information until we have a
|
|
|
|
// large enough array of information structs, or we fail to malloc
|
|
|
|
while (!done) {
|
|
|
|
DWORD rc = glpi(buffer, &returnLength);
|
|
|
|
if (rc == FALSE) {
|
|
|
|
if (GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
|
|
|
|
// this will happen initially since buffer = NULL and we need to
|
|
|
|
// somehow retrieve the required size
|
|
|
|
|
|
|
|
if (buffer) {
|
|
|
|
free(buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
buffer = (PSYSTEM_LOGICAL_PROCESSOR_INFORMATION) malloc(returnLength);
|
|
|
|
|
|
|
|
if (NULL == buffer) {
|
|
|
|
// memory allocation error
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// something else went wrong
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
done = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ptr = buffer;
|
|
|
|
|
|
|
|
while ((byteOffset + sizeof(SYSTEM_LOGICAL_PROCESSOR_INFORMATION))
|
|
|
|
<= returnLength) {
|
|
|
|
// if we have a processor core, count the processors
|
|
|
|
if (ptr->Relationship == RelationProcessorCore) {
|
|
|
|
nPhysicalProcessors++;
|
|
|
|
|
|
|
|
// if we have an SMT-enabled CPU, we need to count the virtual
|
|
|
|
// cores as well
|
|
|
|
DWORD lshift = sizeof(ULONG_PTR) * 8 - 1;
|
|
|
|
ULONG_PTR bitTest = (ULONG_PTR) 1 << lshift;
|
|
|
|
DWORD i;
|
|
|
|
for (i = 0; i <= lshift; ++i) {
|
|
|
|
// count the bit if it is set
|
|
|
|
nSMTProcessors += ((ptr->ProcessorMask & bitTest) ? 1 : 0);
|
|
|
|
// move the test to the next bit
|
|
|
|
bitTest /= 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// move to next info element
|
|
|
|
byteOffset += sizeof(SYSTEM_LOGICAL_PROCESSOR_INFORMATION);
|
|
|
|
ptr++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// return the larger number of physical vs SMT cores
|
|
|
|
// (if this bothers you, overhaul this implementation with a solution that
|
|
|
|
// distinguishes between types of cores on _both_ *NIX and Windows!)
|
|
|
|
return nPhysicalProcessors == nSMTProcessors ? nPhysicalProcessors
|
|
|
|
: nSMTProcessors
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|