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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <dst/dst.h>
|
|
|
|
#include <stdlib.h>
|
2018-05-13 00:31:28 +00:00
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
#ifdef DST_WINDOWS
|
|
|
|
#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>
|
|
|
|
#endif
|
2018-03-29 00:50:20 +00:00
|
|
|
|
|
|
|
static int os_execute(DstArgs args) {
|
|
|
|
int nofirstarg = (args.n < 1 || !dst_checktype(args.v[0], DST_STRING));
|
2018-05-20 01:16:00 +00:00
|
|
|
const char *cmd = nofirstarg
|
2018-03-29 00:50:20 +00:00
|
|
|
? NULL
|
|
|
|
: (const char *) dst_unwrap_string(args.v[0]);
|
|
|
|
int stat = system(cmd);
|
2018-05-13 00:31:28 +00:00
|
|
|
DST_RETURN(args, cmd
|
2018-03-29 00:50:20 +00:00
|
|
|
? dst_wrap_integer(stat)
|
|
|
|
: dst_wrap_boolean(stat));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int os_getenv(DstArgs args) {
|
2018-06-08 19:58:23 +00:00
|
|
|
const uint8_t *k;
|
|
|
|
DST_FIXARITY(args, 1);
|
|
|
|
DST_ARG_STRING(k, args, 0);
|
|
|
|
const char *cstr = (const char *) k;
|
2018-03-29 00:50:20 +00:00
|
|
|
const char *res = getenv(cstr);
|
2018-05-13 00:31:28 +00:00
|
|
|
DST_RETURN(args, cstr
|
2018-03-29 00:50:20 +00:00
|
|
|
? dst_cstringv(res)
|
|
|
|
: dst_wrap_nil());
|
|
|
|
}
|
|
|
|
|
|
|
|
static int os_setenv(DstArgs args) {
|
|
|
|
#ifdef DST_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;
|
|
|
|
DST_MAXARITY(args, 2);
|
|
|
|
DST_MINARITY(args, 1);
|
|
|
|
DST_ARG_STRING(k, args, 0);
|
|
|
|
ks = (const char *) k;
|
|
|
|
if (args.n == 1 || dst_checktype(args.v[1], DST_NIL)) {
|
|
|
|
UNSETENV(ks);
|
2018-03-29 00:50:20 +00:00
|
|
|
} else {
|
2018-06-08 19:58:23 +00:00
|
|
|
const uint8_t *v;
|
|
|
|
DST_ARG_STRING(v, args, 1);
|
|
|
|
const char *vc = (const char *) v;
|
|
|
|
SETENV(ks, vc);
|
2018-03-29 00:50:20 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int os_exit(DstArgs args) {
|
2018-05-13 00:31:28 +00:00
|
|
|
DST_MAXARITY(args, 1);
|
|
|
|
if (args.n == 0) {
|
|
|
|
exit(EXIT_SUCCESS);
|
2018-06-08 19:58:23 +00:00
|
|
|
} else if (dst_checktype(args.v[0], DST_INTEGER)) {
|
|
|
|
exit(dst_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;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int os_clock(DstArgs args) {
|
|
|
|
DST_FIXARITY(args, 0);
|
|
|
|
clock_t time = clock();
|
|
|
|
double dtime = time / (double) (CLOCKS_PER_SEC);
|
|
|
|
DST_RETURN_REAL(args, dtime);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int os_sleep(DstArgs args) {
|
|
|
|
int32_t delay;
|
|
|
|
DST_FIXARITY(args, 1);
|
|
|
|
DST_ARG_INTEGER(delay, args, 0);
|
|
|
|
if (delay < 0) {
|
|
|
|
DST_THROW(args, "invalid argument to sleep");
|
|
|
|
}
|
|
|
|
#ifdef DST_WINDOWS
|
|
|
|
Sleep(delay);
|
|
|
|
#else
|
|
|
|
sleep((unsigned int) delay);
|
|
|
|
#endif
|
|
|
|
return 0;
|
2018-03-29 00:50:20 +00:00
|
|
|
}
|
|
|
|
|
2018-05-19 05:09:56 +00:00
|
|
|
static int os_cwd(DstArgs args) {
|
|
|
|
DST_FIXARITY(args, 0);
|
|
|
|
char buf[FILENAME_MAX];
|
|
|
|
char *ptr;
|
|
|
|
#ifdef DST_WINDOWS
|
|
|
|
ptr = _getcwd(buf, FILENAME_MAX);
|
|
|
|
#else
|
|
|
|
ptr = getcwd(buf, FILENAME_MAX);
|
|
|
|
#endif
|
|
|
|
if (NULL == ptr) {
|
|
|
|
DST_THROW(args, "could not get current directory");
|
|
|
|
}
|
|
|
|
DST_RETURN_CSTRING(args, ptr);
|
|
|
|
}
|
|
|
|
|
2018-03-29 00:50:20 +00:00
|
|
|
static const DstReg cfuns[] = {
|
2018-05-08 23:40:28 +00:00
|
|
|
{"os.execute", os_execute},
|
|
|
|
{"os.exit", os_exit},
|
|
|
|
{"os.getenv", os_getenv},
|
|
|
|
{"os.setenv", os_setenv},
|
2018-05-13 00:31:28 +00:00
|
|
|
{"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 */
|
|
|
|
int dst_lib_os(DstArgs args) {
|
|
|
|
DstTable *env = dst_env_arg(args);
|
|
|
|
dst_env_cfuns(env, cfuns);
|
|
|
|
return 0;
|
|
|
|
}
|