janet/src/core/os.c

297 lines
7.9 KiB
C
Raw Normal View History

2018-03-29 00:50:20 +00:00
/*
* Copyright (c) 2018 Calvin Rose
2018-03-29 00:50:20 +00:00
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to
* deal in the Software without restriction, including without limitation the
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
* sell copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
* IN THE SOFTWARE.
*/
2018-09-06 02:18:42 +00:00
#include <janet/janet.h>
2018-03-29 00:50:20 +00:00
#include <stdlib.h>
#include <time.h>
2018-09-06 02:18:42 +00:00
#ifdef JANET_WINDOWS
#include <Windows.h>
2018-05-19 05:09:56 +00:00
#include <direct.h>
#else
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <stdio.h>
#endif
2018-03-29 00:50:20 +00:00
2018-09-06 02:18:42 +00:00
static int os_which(JanetArgs args) {
#ifdef JANET_WINDOWS
JANET_RETURN_CSYMBOL(args, ":windows");
2018-08-07 04:54:47 +00:00
#elif __APPLE__
2018-09-06 02:18:42 +00:00
JANET_RETURN_CSYMBOL(args, ":macos");
2018-08-07 04:54:47 +00:00
#else
2018-09-06 02:18:42 +00:00
JANET_RETURN_CSYMBOL(args, ":posix");
2018-08-07 04:54:47 +00:00
#endif
}
2018-09-06 02:18:42 +00:00
#ifdef JANET_WINDOWS
static int os_execute(JanetArgs args) {
JANET_MINARITY(args, 1);
JanetBuffer *buffer = janet_buffer(10);
for (int32_t i = 0; i < args.n; i++) {
const uint8_t *argstring;
2018-09-06 02:18:42 +00:00
JANET_ARG_STRING(argstring, args, i);
janet_buffer_push_bytes(buffer, argstring, janet_string_length(argstring));
if (i != args.n - 1) {
2018-09-06 02:18:42 +00:00
janet_buffer_push_u8(buffer, ' ');
}
}
2018-09-06 02:18:42 +00:00
janet_buffer_push_u8(buffer, 0);
/* Convert to wide chars */
wchar_t *sys_str = malloc(buffer->count * sizeof(wchar_t));
if (NULL == sys_str) {
2018-09-06 02:18:42 +00:00
JANET_OUT_OF_MEMORY;
}
int nwritten = MultiByteToWideChar(
CP_UTF8,
MB_PRECOMPOSED,
buffer->data,
buffer->count,
sys_str,
buffer->count);
if (nwritten == 0) {
free(sys_str);
2018-09-06 02:18:42 +00:00
JANET_THROW(args, "could not create process");
}
STARTUPINFO si;
PROCESS_INFORMATION pi;
ZeroMemory(&si, sizeof(si));
si.cb = sizeof(si);
ZeroMemory(&pi, sizeof(pi));
// Start the child process.
if(!CreateProcess(NULL,
2018-08-06 01:13:14 +00:00
(LPSTR) sys_str,
NULL,
NULL,
FALSE,
0,
NULL,
NULL,
&si,
&pi)) {
free(sys_str);
2018-09-06 02:18:42 +00:00
JANET_THROW(args, "could not create process");
}
free(sys_str);
// Wait until child process exits.
WaitForSingleObject(pi.hProcess, INFINITE);
// Close process and thread handles.
WORD status;
2018-08-06 01:13:14 +00:00
GetExitCodeProcess(pi.hProcess, (LPDWORD)&status);
CloseHandle(pi.hProcess);
CloseHandle(pi.hThread);
2018-09-06 02:18:42 +00:00
JANET_RETURN_INTEGER(args, (int32_t)status);
}
#else
2018-09-06 02:18:42 +00:00
static int os_execute(JanetArgs args) {
JANET_MINARITY(args, 1);
const uint8_t **argv = malloc(sizeof(uint8_t *) * (args.n + 1));
if (NULL == argv) {
2018-09-06 02:18:42 +00:00
JANET_OUT_OF_MEMORY;
}
for (int32_t i = 0; i < args.n; i++) {
2018-09-06 02:18:42 +00:00
JANET_ARG_STRING(argv[i], args, i);
}
argv[args.n] = NULL;
/* Fork child process */
pid_t pid;
if (0 == (pid = fork())) {
if (-1 == execve((const char *)argv[0], (char **)argv, NULL)) {
exit(1);
}
}
/* Wait for child process */
int status;
struct timespec waiter;
waiter.tv_sec = 0;
waiter.tv_nsec = 200;
while (0 == waitpid(pid, &status, WNOHANG)) {
waiter.tv_nsec = (waiter.tv_nsec * 3) / 2;
/* Keep increasing sleep time by a factor of 3/2
* until a maximum */
if (waiter.tv_nsec > 4999999)
waiter.tv_nsec = 5000000;
nanosleep(&waiter, NULL);
}
2018-09-06 02:18:42 +00:00
JANET_RETURN_INTEGER(args, status);
}
#endif
2018-09-06 02:18:42 +00:00
static int os_shell(JanetArgs args) {
int nofirstarg = (args.n < 1 || !janet_checktype(args.v[0], JANET_STRING));
const char *cmd = nofirstarg
2018-03-29 00:50:20 +00:00
? NULL
2018-09-06 02:18:42 +00:00
: (const char *) janet_unwrap_string(args.v[0]);
2018-03-29 00:50:20 +00:00
int stat = system(cmd);
2018-09-06 02:18:42 +00:00
JANET_RETURN(args, cmd
? janet_wrap_integer(stat)
: janet_wrap_boolean(stat));
2018-03-29 00:50:20 +00:00
}
2018-09-06 02:18:42 +00:00
static int os_getenv(JanetArgs args) {
2018-06-08 19:58:23 +00:00
const uint8_t *k;
2018-09-06 02:18:42 +00:00
JANET_FIXARITY(args, 1);
JANET_ARG_STRING(k, args, 0);
2018-06-08 19:58:23 +00:00
const char *cstr = (const char *) k;
2018-03-29 00:50:20 +00:00
const char *res = getenv(cstr);
2018-07-11 23:11:34 +00:00
if (!res) {
2018-09-06 02:18:42 +00:00
JANET_RETURN_NIL(args);
2018-07-11 23:11:34 +00:00
}
2018-09-06 02:18:42 +00:00
JANET_RETURN(args, cstr
? janet_cstringv(res)
: janet_wrap_nil());
2018-03-29 00:50:20 +00:00
}
2018-09-06 02:18:42 +00:00
static int os_setenv(JanetArgs args) {
#ifdef JANET_WINDOWS
2018-06-08 19:58:23 +00:00
#define SETENV(K,V) _putenv_s(K, V)
#define UNSETENV(K) _putenv_s(K, "")
2018-03-29 00:50:20 +00:00
#else
2018-06-08 19:58:23 +00:00
#define SETENV(K,V) setenv(K, V, 1)
#define UNSETENV(K) unsetenv(K)
#endif
const uint8_t *k;
const char *ks;
2018-09-06 02:18:42 +00:00
JANET_MAXARITY(args, 2);
JANET_MINARITY(args, 1);
JANET_ARG_STRING(k, args, 0);
2018-06-08 19:58:23 +00:00
ks = (const char *) k;
2018-09-06 02:18:42 +00:00
if (args.n == 1 || janet_checktype(args.v[1], JANET_NIL)) {
2018-06-08 19:58:23 +00:00
UNSETENV(ks);
2018-03-29 00:50:20 +00:00
} else {
2018-06-08 19:58:23 +00:00
const uint8_t *v;
2018-09-06 02:18:42 +00:00
JANET_ARG_STRING(v, args, 1);
2018-06-08 19:58:23 +00:00
const char *vc = (const char *) v;
SETENV(ks, vc);
2018-03-29 00:50:20 +00:00
}
return 0;
}
2018-09-06 02:18:42 +00:00
static int os_exit(JanetArgs args) {
JANET_MAXARITY(args, 1);
if (args.n == 0) {
exit(EXIT_SUCCESS);
2018-09-06 02:18:42 +00:00
} else if (janet_checktype(args.v[0], JANET_INTEGER)) {
exit(janet_unwrap_integer(args.v[0]));
} else {
2018-06-08 19:58:23 +00:00
exit(EXIT_FAILURE);
}
return 0;
}
2018-07-09 00:54:41 +00:00
/* Clock shim for windows */
2018-09-06 02:18:42 +00:00
#ifdef JANET_WINDOWS
2018-07-09 01:14:08 +00:00
static int clock_gettime(int x, struct timespec *spec) {
(void) x;
2018-07-09 01:10:15 +00:00
int64_t wintime = 0LL;
GetSystemTimeAsFileTime((FILETIME*)&wintime);
2018-07-09 01:10:15 +00:00
/* Windows epoch is January 1, 1601 apparently*/
wintime -= 116444736000000000LL;
spec->tv_sec = wintime / 10000000LL;
/* Resolution is 100 nanoseconds. */
spec->tv_nsec = wintime % 10000000LL * 100;
return 0;
}
2018-07-09 01:10:15 +00:00
#define CLOCK_MONOTONIC 0
#endif
2018-09-12 01:33:50 +00:00
static int os_time(JanetArgs args) {
JANET_FIXARITY(args, 0);
double dtime = (double)(time(NULL));
JANET_RETURN_REAL(args, dtime);
}
2018-09-06 02:18:42 +00:00
static int os_clock(JanetArgs args) {
JANET_FIXARITY(args, 0);
struct timespec tv;
2018-07-09 01:10:15 +00:00
if (clock_gettime(CLOCK_MONOTONIC, &tv))
2018-09-06 02:18:42 +00:00
JANET_THROW(args, "could not get time");
double dtime = tv.tv_sec + (tv.tv_nsec / 1E9);
2018-09-06 02:18:42 +00:00
JANET_RETURN_REAL(args, dtime);
}
2018-09-06 02:18:42 +00:00
static int os_sleep(JanetArgs args) {
double delay;
2018-09-06 02:18:42 +00:00
JANET_FIXARITY(args, 1);
JANET_ARG_NUMBER(delay, args, 0);
if (delay < 0) {
2018-09-06 02:18:42 +00:00
JANET_THROW(args, "invalid argument to sleep");
}
2018-09-06 02:18:42 +00:00
#ifdef JANET_WINDOWS
Sleep((DWORD) (delay * 1000));
#else
struct timespec ts;
ts.tv_sec = (time_t) delay;
ts.tv_nsec = (delay <= UINT32_MAX)
? (long)((delay - ((uint32_t)delay)) * 1000000000)
: 0;
nanosleep(&ts, NULL);
#endif
return 0;
2018-03-29 00:50:20 +00:00
}
2018-09-06 02:18:42 +00:00
static int os_cwd(JanetArgs args) {
JANET_FIXARITY(args, 0);
2018-05-19 05:09:56 +00:00
char buf[FILENAME_MAX];
char *ptr;
2018-09-06 02:18:42 +00:00
#ifdef JANET_WINDOWS
2018-05-19 05:09:56 +00:00
ptr = _getcwd(buf, FILENAME_MAX);
#else
ptr = getcwd(buf, FILENAME_MAX);
#endif
if (NULL == ptr) {
2018-09-06 02:18:42 +00:00
JANET_THROW(args, "could not get current directory");
2018-05-19 05:09:56 +00:00
}
2018-09-06 02:18:42 +00:00
JANET_RETURN_CSTRING(args, ptr);
2018-05-19 05:09:56 +00:00
}
2018-09-06 02:18:42 +00:00
static const JanetReg cfuns[] = {
2018-08-07 04:54:47 +00:00
{"os.which", os_which},
{"os.execute", os_execute},
{"os.shell", os_shell},
{"os.exit", os_exit},
{"os.getenv", os_getenv},
{"os.setenv", os_setenv},
2018-09-12 01:33:50 +00:00
{"os.time", os_time},
{"os.clock", os_clock},
{"os.sleep", os_sleep},
2018-05-19 05:09:56 +00:00
{"os.cwd", os_cwd},
2018-03-29 00:50:20 +00:00
{NULL, NULL}
};
/* Module entry point */
2018-09-06 02:18:42 +00:00
int janet_lib_os(JanetArgs args) {
JanetTable *env = janet_env(args);
janet_cfuns(env, NULL, cfuns);
2018-03-29 00:50:20 +00:00
return 0;
}