#include "idris_support.h" #include "idris_file.h" #include #include #include #include #ifndef _WIN32 #include #endif #include #include #if !defined(_SC_NPROCESSORS_ONLN) && defined(_DARWIN_C_SOURCE) #include #include #endif int _argc; char **_argv; #ifdef _WIN32 extern char **_environ; #include "windows/win_utils.h" #define environ _environ #else extern char **environ; #endif int idris2_isNull(void *ptr) { return (ptr == NULL); } void *idris2_getNull() { return NULL; } char *idris2_getString(void *p) { return (char *)p; } int idris2_getErrno() { #ifdef _WIN32 return win32_getErrno(); #else return errno; #endif } char *idris2_strerror(int errnum) { return strerror(errnum); } char *idris2_getStr() { char *inp = idris2_readLine(stdin); // Remove trailing newline; easier to do this than in PrimIO which // doesn't have the relevant functions available yet for (char *c = inp; *c != '\0'; ++c) { if (*c == '\n' || *c == '\r') { *c = '\0'; } } return inp; } void idris2_putStr(char *f) { idris2_writeLine(stdout, f); } void idris2_sleep(int sec) { #ifdef _WIN32 win32_sleep(sec * 1000); #else struct timespec t; t.tv_sec = sec; t.tv_nsec = 0; // TODO: `nanosleep` can fail // TODO: `nanosleep` can return early due to interrupt nanosleep(&t, NULL); #endif } void idris2_usleep(int usec) { #ifdef _WIN32 win32_sleep(usec / 1000); #else struct timespec t; t.tv_sec = usec / 1000000; t.tv_nsec = (usec % 1000000) * 1000; nanosleep(&t, NULL); #endif } int idris2_time() { return time(NULL); // RTS needs to have 32 bit integers at least } void idris2_setArgs(int argc, char *argv[]) { _argc = argc; _argv = argv; } int idris2_getArgCount() { return _argc; } char *idris2_getArg(int n) { return _argv[n]; } char *idris2_getEnvPair(int i) { return *(environ + i); } int idris2_setenv(const char *name, const char *value, int overwrite) { #ifdef _WIN32 return win32_modenv(name, value, overwrite); #else return setenv(name, value, overwrite); #endif } #ifndef _WIN32 // The initial state of stdin is stored here if enableRawMode is called. struct termios *initial_termios = NULL; #endif int idris2_enableRawMode() { #ifndef _WIN32 struct termios ti; int rval = tcgetattr(STDIN_FILENO, &ti); if (rval != 0) return rval; if (!initial_termios) { initial_termios = malloc(sizeof(struct termios)); *initial_termios = ti; } ti.c_lflag &= ~(ECHO | ICANON); return tcsetattr(STDIN_FILENO, TCSAFLUSH, &ti); #else return -1; #endif } void idris2_resetRawMode() { #ifndef _WIN32 if (initial_termios != NULL) { tcsetattr(STDIN_FILENO, TCSAFLUSH, initial_termios); } #endif } int idris2_unsetenv(const char *name) { #ifdef _WIN32 return win32_modenv(name, "", 1); #else return unsetenv(name); #endif } int idris2_getPID() { #ifdef _WIN32 return win32_getPID(); #else return getpid(); #endif } // get the number of processors configured long idris2_getNProcessors() { #if defined(_WIN32) return win32_getNProcessors(); #elif defined(_POSIX_VERSION) && _POSIX_VERSION >= 200112L && \ defined(_SC_NPROCESSORS_ONLN) // Note: Under MacOS with _POSIX_C_SOURCE, _SC_NPROCESSORS_ONLN never defined. return sysconf(_SC_NPROCESSORS_ONLN); #elif defined(_DARWIN_C_SOURCE) || defined(_BSD_SOURCE) // Generic way for BSD style system. #if defined(_DARWIN_C_SOURCE) int xs[] = {CTL_HW, HW_LOGICALCPU}; #else int xs[] = {CTL_HW, HW_NCPU}; #endif int numcpus = 1; size_t n = sizeof(numcpus); return sysctll(xs, 2, &numcpus, &len, NULL, 0) < 0 ? 1 : numcpus; #else return 1 #endif }