2018-03-29 00:50:20 +00:00
|
|
|
/*
|
2018-05-18 03:41: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>
|
2018-05-13 00:31:28 +00:00
|
|
|
#include <time.h>
|
|
|
|
|
2018-09-06 02:18:42 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
2018-05-13 00:31:28 +00:00
|
|
|
#include <Windows.h>
|
2018-05-19 05:09:56 +00:00
|
|
|
#include <direct.h>
|
2018-05-13 00:31:28 +00:00
|
|
|
#else
|
|
|
|
#include <unistd.h>
|
2018-07-07 01:50:59 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/wait.h>
|
|
|
|
#include <stdio.h>
|
2018-05-13 00:31:28 +00:00
|
|
|
#endif
|
2018-03-29 00:50:20 +00:00
|
|
|
|
2018-09-23 01:46:50 +00:00
|
|
|
/* For macos */
|
|
|
|
#ifdef __MACH__
|
|
|
|
#include <mach/clock.h>
|
|
|
|
#include <mach/mach.h>
|
|
|
|
#endif
|
|
|
|
|
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-10-17 03:08:26 +00:00
|
|
|
#elif defined(__EMSCRIPTEN__)
|
|
|
|
JANET_RETURN_CSYMBOL(args, ":web");
|
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);
|
2018-07-07 01:50:59 +00:00
|
|
|
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));
|
2018-07-07 01:50:59 +00:00
|
|
|
if (i != args.n - 1) {
|
2018-09-06 02:18:42 +00:00
|
|
|
janet_buffer_push_u8(buffer, ' ');
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
|
|
|
}
|
2018-09-06 02:18:42 +00:00
|
|
|
janet_buffer_push_u8(buffer, 0);
|
2018-07-07 01:50:59 +00:00
|
|
|
|
|
|
|
/* 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;
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
|
|
|
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");
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
STARTUPINFO si;
|
|
|
|
PROCESS_INFORMATION pi;
|
|
|
|
|
|
|
|
ZeroMemory(&si, sizeof(si));
|
|
|
|
si.cb = sizeof(si);
|
|
|
|
ZeroMemory(&pi, sizeof(pi));
|
|
|
|
|
2018-11-16 21:24:10 +00:00
|
|
|
// Start the child process.
|
2018-07-07 01:50:59 +00:00
|
|
|
if(!CreateProcess(NULL,
|
2018-08-06 01:13:14 +00:00
|
|
|
(LPSTR) sys_str,
|
2018-07-07 01:50:59 +00:00
|
|
|
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");
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
|
|
|
free(sys_str);
|
|
|
|
|
|
|
|
// Wait until child process exits.
|
|
|
|
WaitForSingleObject(pi.hProcess, INFINITE);
|
|
|
|
|
2018-07-08 18:22:40 +00:00
|
|
|
// Close process and thread handles.
|
|
|
|
WORD status;
|
2018-08-06 01:13:14 +00:00
|
|
|
GetExitCodeProcess(pi.hProcess, (LPDWORD)&status);
|
2018-07-07 01:50:59 +00:00
|
|
|
CloseHandle(pi.hProcess);
|
|
|
|
CloseHandle(pi.hThread);
|
2018-09-06 02:18:42 +00:00
|
|
|
JANET_RETURN_INTEGER(args, (int32_t)status);
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
|
|
|
#else
|
2018-09-06 02:18:42 +00:00
|
|
|
static int os_execute(JanetArgs args) {
|
|
|
|
JANET_MINARITY(args, 1);
|
2018-07-07 01:50:59 +00:00
|
|
|
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;
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
|
|
|
for (int32_t i = 0; i < args.n; i++) {
|
2018-09-06 02:18:42 +00:00
|
|
|
JANET_ARG_STRING(argv[i], args, i);
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
|
|
|
argv[args.n] = NULL;
|
|
|
|
|
|
|
|
/* Fork child process */
|
2018-11-16 07:34:50 +00:00
|
|
|
pid_t pid = fork();
|
|
|
|
if (pid < 0) {
|
|
|
|
JANET_THROW(args, "failed to execute");
|
|
|
|
} else if (pid == 0) {
|
2018-07-07 01:50:59 +00:00
|
|
|
if (-1 == execve((const char *)argv[0], (char **)argv, NULL)) {
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int status;
|
2018-11-16 07:34:50 +00:00
|
|
|
waitpid(pid, &status, 0);
|
2018-09-06 02:18:42 +00:00
|
|
|
JANET_RETURN_INTEGER(args, status);
|
2018-07-07 01:50:59 +00:00
|
|
|
}
|
|
|
|
#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));
|
2018-05-20 01:16:00 +00:00
|
|
|
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);
|
2018-05-13 00:31:28 +00:00
|
|
|
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]));
|
2018-05-13 00:31:28 +00:00
|
|
|
} else {
|
2018-06-08 19:58:23 +00:00
|
|
|
exit(EXIT_FAILURE);
|
2018-05-13 00:31:28 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-09-23 01:46:50 +00:00
|
|
|
static int os_time(JanetArgs args) {
|
|
|
|
JANET_FIXARITY(args, 0);
|
|
|
|
double dtime = (double)(time(NULL));
|
|
|
|
JANET_RETURN_REAL(args, dtime);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Clock shims */
|
2018-09-06 02:18:42 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
2018-09-23 01:46:50 +00:00
|
|
|
static int gettime(struct timespec *spec) {
|
2018-07-09 01:10:15 +00:00
|
|
|
int64_t wintime = 0LL;
|
2018-07-08 23:27:11 +00:00
|
|
|
GetSystemTimeAsFileTime((FILETIME*)&wintime);
|
2018-07-09 01:10:15 +00:00
|
|
|
/* Windows epoch is January 1, 1601 apparently*/
|
|
|
|
wintime -= 116444736000000000LL;
|
2018-07-08 23:27:11 +00:00
|
|
|
spec->tv_sec = wintime / 10000000LL;
|
|
|
|
/* Resolution is 100 nanoseconds. */
|
|
|
|
spec->tv_nsec = wintime % 10000000LL * 100;
|
|
|
|
return 0;
|
|
|
|
}
|
2018-09-23 01:46:50 +00:00
|
|
|
#elif defined(__MACH__)
|
|
|
|
static int gettime(struct timespec *spec) {
|
|
|
|
clock_serv_t cclock;
|
|
|
|
mach_timespec_t mts;
|
|
|
|
host_get_clock_service(mach_host_self(), CALENDAR_CLOCK, &cclock);
|
|
|
|
clock_get_time(cclock, &mts);
|
|
|
|
mach_port_deallocate(mach_task_self(), cclock);
|
|
|
|
spec->tv_sec = mts.tv_sec;
|
|
|
|
spec->tv_nsec = mts.tv_nsec;
|
|
|
|
return 0;
|
2018-09-12 01:33:50 +00:00
|
|
|
}
|
2018-11-16 21:24:10 +00:00
|
|
|
#else
|
2018-09-23 01:46:50 +00:00
|
|
|
#define gettime(TV) clock_gettime(CLOCK_MONOTONIC, (TV))
|
|
|
|
#endif
|
2018-09-12 01:33:50 +00:00
|
|
|
|
2018-09-06 02:18:42 +00:00
|
|
|
static int os_clock(JanetArgs args) {
|
|
|
|
JANET_FIXARITY(args, 0);
|
2018-07-08 23:27:11 +00:00
|
|
|
struct timespec tv;
|
2018-09-23 01:50:43 +00:00
|
|
|
if (gettime(&tv))
|
2018-09-06 02:18:42 +00:00
|
|
|
JANET_THROW(args, "could not get time");
|
2018-07-08 23:27:11 +00:00
|
|
|
double dtime = tv.tv_sec + (tv.tv_nsec / 1E9);
|
2018-09-06 02:18:42 +00:00
|
|
|
JANET_RETURN_REAL(args, dtime);
|
2018-05-13 00:31:28 +00:00
|
|
|
}
|
|
|
|
|
2018-09-06 02:18:42 +00:00
|
|
|
static int os_sleep(JanetArgs args) {
|
2018-07-05 16:55:11 +00:00
|
|
|
double delay;
|
2018-09-06 02:18:42 +00:00
|
|
|
JANET_FIXARITY(args, 1);
|
|
|
|
JANET_ARG_NUMBER(delay, args, 0);
|
2018-05-13 00:31:28 +00:00
|
|
|
if (delay < 0) {
|
2018-09-06 02:18:42 +00:00
|
|
|
JANET_THROW(args, "invalid argument to sleep");
|
2018-05-13 00:31:28 +00:00
|
|
|
}
|
2018-09-06 02:18:42 +00:00
|
|
|
#ifdef JANET_WINDOWS
|
2018-07-05 16:55:11 +00:00
|
|
|
Sleep((DWORD) (delay * 1000));
|
2018-05-13 00:31:28 +00:00
|
|
|
#else
|
2018-07-07 01:50:59 +00:00
|
|
|
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);
|
2018-05-13 00:31:28 +00:00
|
|
|
#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-11-16 21:24:10 +00:00
|
|
|
{"os.which", os_which,
|
2018-11-16 07:34:50 +00:00
|
|
|
"(os.which)\n\n"
|
|
|
|
"Check the current operating system. Returns one of:\n\n"
|
|
|
|
"\t:windows - Microsoft Windows\n"
|
|
|
|
"\t:macos - Apple macos\n"
|
|
|
|
"\t:posix - A POSIX compatible system (default)"
|
|
|
|
},
|
2018-11-16 21:24:10 +00:00
|
|
|
{"os.execute", os_execute,
|
2018-11-16 07:34:50 +00:00
|
|
|
"(os.execute program & args)\n\n"
|
|
|
|
"Execute a program on the system and pass it string arguments. Returns "
|
|
|
|
"the exit status of the program."
|
|
|
|
},
|
2018-11-16 21:24:10 +00:00
|
|
|
{"os.shell", os_shell,
|
2018-11-16 07:34:50 +00:00
|
|
|
"(os.shell str)\n\n"
|
|
|
|
"Pass a command string str directly to the system shell."
|
|
|
|
},
|
2018-11-16 21:24:10 +00:00
|
|
|
{"os.exit", os_exit,
|
2018-11-16 07:34:50 +00:00
|
|
|
"(os.exit x)\n\n"
|
|
|
|
"Exit from janet with an exit code equal to x. If x is not an integer, "
|
|
|
|
"the exit with status equal the hash of x."
|
|
|
|
},
|
2018-11-16 21:24:10 +00:00
|
|
|
{"os.getenv", os_getenv,
|
2018-11-16 07:34:50 +00:00
|
|
|
"(os.getenv variable)\n\n"
|
|
|
|
"Get the string value of an environment variable."
|
|
|
|
},
|
2018-11-16 21:24:10 +00:00
|
|
|
{"os.setenv", os_setenv,
|
2018-11-16 07:34:50 +00:00
|
|
|
"(os.setenv variable value)\n\n"
|
|
|
|
"Set an environment variable."
|
|
|
|
},
|
2018-11-16 21:24:10 +00:00
|
|
|
{"os.time", os_time,
|
2018-11-16 07:34:50 +00:00
|
|
|
"(os.time)\n\n"
|
|
|
|
"Get the current time expressed as the number of seconds since "
|
|
|
|
"January 1, 1970, the Unix epoch. Returns a real number."
|
|
|
|
},
|
2018-11-16 21:24:10 +00:00
|
|
|
{"os.clock", os_clock,
|
2018-11-16 07:34:50 +00:00
|
|
|
"(os.clock)\n\n"
|
|
|
|
"Return the number of seconds since some fixed point in time. The clock "
|
|
|
|
"is guaranteed to be non decreased in real time."
|
|
|
|
},
|
2018-11-16 21:24:10 +00:00
|
|
|
{"os.sleep", os_sleep,
|
2018-11-16 07:34:50 +00:00
|
|
|
"(os.sleep nsec)\n\n"
|
|
|
|
"Suspend the program for nsec seconds. 'nsec' can be a real number. Returns "
|
|
|
|
"nil."
|
2018-11-16 21:24:10 +00:00
|
|
|
|
2018-11-16 07:34:50 +00:00
|
|
|
},
|
2018-11-16 21:24:10 +00:00
|
|
|
{"os.cwd", os_cwd,
|
2018-11-16 07:34:50 +00:00
|
|
|
"(os.cwd)\n\n"
|
|
|
|
"Returns the current working directory."
|
|
|
|
},
|
2018-11-15 20:45:41 +00:00
|
|
|
{NULL, NULL, NULL}
|
2018-03-29 00:50:20 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/* 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;
|
|
|
|
}
|