mirror of
https://github.com/janet-lang/janet
synced 2025-01-25 14:46:52 +00:00
Add lookups for marshalling and unmarshalling.
Allow generating lookup tables from the current environment.
This commit is contained in:
parent
c3ba613959
commit
98f2c6feab
@ -332,7 +332,6 @@ static void janet_quick_asm(
|
|||||||
}
|
}
|
||||||
memcpy(def->bytecode, bytecode, bytecode_size);
|
memcpy(def->bytecode, bytecode, bytecode_size);
|
||||||
janet_def(env, name, janet_wrap_function(janet_thunk(def)));
|
janet_def(env, name, janet_wrap_function(janet_thunk(def)));
|
||||||
janet_register(name, janet_wrap_function(janet_thunk(def)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Macros for easier inline janet assembly */
|
/* Macros for easier inline janet assembly */
|
||||||
|
@ -376,26 +376,21 @@ static const JanetReg cfuns[] = {
|
|||||||
{NULL, NULL}
|
{NULL, NULL}
|
||||||
};
|
};
|
||||||
|
|
||||||
static void addf(JanetTable *env, const char *name, Janet val) {
|
|
||||||
janet_def(env, name, val);
|
|
||||||
janet_register(name, val);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Module entry point */
|
/* Module entry point */
|
||||||
int janet_lib_io(JanetArgs args) {
|
int janet_lib_io(JanetArgs args) {
|
||||||
JanetTable *env = janet_env(args);
|
JanetTable *env = janet_env(args);
|
||||||
janet_cfuns(env, NULL, cfuns);
|
janet_cfuns(env, NULL, cfuns);
|
||||||
|
|
||||||
/* stdout */
|
/* stdout */
|
||||||
addf(env, "stdout",
|
janet_def(env, "stdout",
|
||||||
makef(stdout, IO_APPEND | IO_NOT_CLOSEABLE | IO_SERIALIZABLE));
|
makef(stdout, IO_APPEND | IO_NOT_CLOSEABLE | IO_SERIALIZABLE));
|
||||||
|
|
||||||
/* stderr */
|
/* stderr */
|
||||||
addf(env, "stderr",
|
janet_def(env, "stderr",
|
||||||
makef(stderr, IO_APPEND | IO_NOT_CLOSEABLE | IO_SERIALIZABLE));
|
makef(stderr, IO_APPEND | IO_NOT_CLOSEABLE | IO_SERIALIZABLE));
|
||||||
|
|
||||||
/* stdin */
|
/* stdin */
|
||||||
addf(env, "stdin",
|
janet_def(env, "stdin",
|
||||||
makef(stdin, IO_READ | IO_NOT_CLOSEABLE | IO_SERIALIZABLE));
|
makef(stdin, IO_READ | IO_NOT_CLOSEABLE | IO_SERIALIZABLE));
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
132
src/core/marsh.c
132
src/core/marsh.c
@ -31,6 +31,7 @@ typedef struct {
|
|||||||
jmp_buf err;
|
jmp_buf err;
|
||||||
JanetBuffer *buf;
|
JanetBuffer *buf;
|
||||||
JanetTable seen;
|
JanetTable seen;
|
||||||
|
JanetTable *rreg;
|
||||||
JanetFuncEnv **seen_envs;
|
JanetFuncEnv **seen_envs;
|
||||||
JanetFuncDef **seen_defs;
|
JanetFuncDef **seen_defs;
|
||||||
int32_t nextid;
|
int32_t nextid;
|
||||||
@ -78,6 +79,59 @@ enum {
|
|||||||
LB_FUNCDEF_REF
|
LB_FUNCDEF_REF
|
||||||
} LeadBytes;
|
} LeadBytes;
|
||||||
|
|
||||||
|
/* Helper to look inside an entry in an environment */
|
||||||
|
static Janet entry_getval(Janet env_entry) {
|
||||||
|
if (janet_checktype(env_entry, JANET_TABLE)) {
|
||||||
|
JanetTable *entry = janet_unwrap_table(env_entry);
|
||||||
|
Janet checkval = janet_table_get(entry, janet_csymbolv(":value"));
|
||||||
|
if (janet_checktype(checkval, JANET_NIL)) {
|
||||||
|
checkval = janet_table_get(entry, janet_csymbolv(":ref"));
|
||||||
|
}
|
||||||
|
return checkval;
|
||||||
|
} else if (janet_checktype(env_entry, JANET_STRUCT)) {
|
||||||
|
const JanetKV *entry = janet_unwrap_struct(env_entry);
|
||||||
|
Janet checkval = janet_struct_get(entry, janet_csymbolv(":value"));
|
||||||
|
if (janet_checktype(checkval, JANET_NIL)) {
|
||||||
|
checkval = janet_struct_get(entry, janet_csymbolv(":ref"));
|
||||||
|
}
|
||||||
|
return checkval;
|
||||||
|
} else {
|
||||||
|
return janet_wrap_nil();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Make a reverse lookup table for an environment (for marshaling) */
|
||||||
|
JanetTable *janet_env_rreg(JanetTable *env) {
|
||||||
|
JanetTable *renv = janet_table(env->count);
|
||||||
|
while (env) {
|
||||||
|
for (int32_t i = 0; i < env->capacity; i++) {
|
||||||
|
if (janet_checktype(env->data[i].key, JANET_SYMBOL)) {
|
||||||
|
janet_table_put(renv,
|
||||||
|
entry_getval(env->data[i].value),
|
||||||
|
env->data[i].key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
env = env->proto;
|
||||||
|
}
|
||||||
|
return renv;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Make a forward lookup table from an environment (for unmarshaling) */
|
||||||
|
JanetTable *janet_env_reg(JanetTable *env) {
|
||||||
|
JanetTable *renv = janet_table(env->count);
|
||||||
|
while (env) {
|
||||||
|
for (int32_t i = 0; i < env->capacity; i++) {
|
||||||
|
if (janet_checktype(env->data[i].key, JANET_SYMBOL)) {
|
||||||
|
janet_table_put(renv,
|
||||||
|
env->data[i].key,
|
||||||
|
entry_getval(env->data[i].value));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
env = env->proto;
|
||||||
|
}
|
||||||
|
return renv;
|
||||||
|
}
|
||||||
|
|
||||||
/* Marshal an integer onto the buffer */
|
/* Marshal an integer onto the buffer */
|
||||||
static void pushint(MarshalState *st, int32_t x) {
|
static void pushint(MarshalState *st, int32_t x) {
|
||||||
if (x >= 0 && x < 200) {
|
if (x >= 0 && x < 200) {
|
||||||
@ -258,7 +312,10 @@ static void marshal_one(MarshalState *st, Janet x, int flags) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check reference */
|
#define MARK_SEEN() \
|
||||||
|
janet_table_put(&st->seen, x, janet_wrap_integer(st->nextid++))
|
||||||
|
|
||||||
|
/* Check reference and registry value */
|
||||||
{
|
{
|
||||||
Janet check = janet_table_get(&st->seen, x);
|
Janet check = janet_table_get(&st->seen, x);
|
||||||
if (janet_checktype(check, JANET_INTEGER)) {
|
if (janet_checktype(check, JANET_INTEGER)) {
|
||||||
@ -266,10 +323,18 @@ static void marshal_one(MarshalState *st, Janet x, int flags) {
|
|||||||
pushint(st, janet_unwrap_integer(check));
|
pushint(st, janet_unwrap_integer(check));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
if (st->rreg) {
|
||||||
|
check = janet_table_get(st->rreg, x);
|
||||||
|
if (janet_checktype(check, JANET_SYMBOL)) {
|
||||||
|
MARK_SEEN();
|
||||||
|
const uint8_t *regname = janet_unwrap_symbol(check);
|
||||||
|
pushbyte(st, LB_REGISTRY);
|
||||||
|
pushint(st, janet_string_length(regname));
|
||||||
|
pushbytes(st, regname, janet_string_length(regname));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#define MARK_SEEN() \
|
|
||||||
janet_table_put(&st->seen, x, janet_wrap_integer(st->nextid++))
|
|
||||||
|
|
||||||
/* Reference types */
|
/* Reference types */
|
||||||
switch (type) {
|
switch (type) {
|
||||||
@ -373,17 +438,7 @@ static void marshal_one(MarshalState *st, Janet x, int flags) {
|
|||||||
return;
|
return;
|
||||||
case JANET_ABSTRACT:
|
case JANET_ABSTRACT:
|
||||||
case JANET_CFUNCTION:
|
case JANET_CFUNCTION:
|
||||||
{
|
|
||||||
MARK_SEEN();
|
|
||||||
Janet regval = janet_table_get(janet_vm_registry, x);
|
|
||||||
if (!janet_checktype(regval, JANET_SYMBOL))
|
|
||||||
goto noregval;
|
goto noregval;
|
||||||
const uint8_t *regname = janet_unwrap_symbol(regval);
|
|
||||||
pushbyte(st, LB_REGISTRY);
|
|
||||||
pushint(st, janet_string_length(regname));
|
|
||||||
pushbytes(st, regname, janet_string_length(regname));
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
case JANET_FUNCTION:
|
case JANET_FUNCTION:
|
||||||
{
|
{
|
||||||
pushbyte(st, LB_FUNCTION);
|
pushbyte(st, LB_FUNCTION);
|
||||||
@ -416,13 +471,14 @@ noregval:
|
|||||||
longjmp(st->err, MR_NRV);
|
longjmp(st->err, MR_NRV);
|
||||||
}
|
}
|
||||||
|
|
||||||
int janet_marshal(JanetBuffer *buf, Janet x, int flags) {
|
int janet_marshal(JanetBuffer *buf, Janet x, JanetTable *rreg, int flags) {
|
||||||
int status;
|
int status;
|
||||||
MarshalState st;
|
MarshalState st;
|
||||||
st.buf = buf;
|
st.buf = buf;
|
||||||
st.nextid = 0;
|
st.nextid = 0;
|
||||||
st.seen_defs = NULL;
|
st.seen_defs = NULL;
|
||||||
st.seen_envs = NULL;
|
st.seen_envs = NULL;
|
||||||
|
st.rreg = rreg;
|
||||||
janet_table_init(&st.seen, 0);
|
janet_table_init(&st.seen, 0);
|
||||||
if (!(status = setjmp(st.err)))
|
if (!(status = setjmp(st.err)))
|
||||||
marshal_one(&st, x, flags);
|
marshal_one(&st, x, flags);
|
||||||
@ -435,6 +491,7 @@ int janet_marshal(JanetBuffer *buf, Janet x, int flags) {
|
|||||||
typedef struct {
|
typedef struct {
|
||||||
jmp_buf err;
|
jmp_buf err;
|
||||||
JanetArray lookup;
|
JanetArray lookup;
|
||||||
|
JanetTable *reg;
|
||||||
JanetFuncEnv **lookup_envs;
|
JanetFuncEnv **lookup_envs;
|
||||||
JanetFuncDef **lookup_defs;
|
JanetFuncDef **lookup_defs;
|
||||||
const uint8_t *end;
|
const uint8_t *end;
|
||||||
@ -897,8 +954,12 @@ static const uint8_t *unmarshal_one(
|
|||||||
const uint8_t *str = janet_symbol(data, len);
|
const uint8_t *str = janet_symbol(data, len);
|
||||||
*out = janet_wrap_symbol(str);
|
*out = janet_wrap_symbol(str);
|
||||||
} else if (lead == LB_REGISTRY) {
|
} else if (lead == LB_REGISTRY) {
|
||||||
|
if (st->reg) {
|
||||||
Janet regkey = janet_symbolv(data, len);
|
Janet regkey = janet_symbolv(data, len);
|
||||||
*out = janet_table_get(janet_vm_registry, regkey);
|
*out = janet_table_get(st->reg, regkey);
|
||||||
|
} else {
|
||||||
|
*out = janet_wrap_nil();
|
||||||
|
}
|
||||||
} else { /* (lead == LB_BUFFER) */
|
} else { /* (lead == LB_BUFFER) */
|
||||||
JanetBuffer *buffer = janet_buffer(len);
|
JanetBuffer *buffer = janet_buffer(len);
|
||||||
buffer->count = len;
|
buffer->count = len;
|
||||||
@ -1004,6 +1065,7 @@ int janet_unmarshal(
|
|||||||
size_t len,
|
size_t len,
|
||||||
int flags,
|
int flags,
|
||||||
Janet *out,
|
Janet *out,
|
||||||
|
JanetTable *reg,
|
||||||
const uint8_t **next) {
|
const uint8_t **next) {
|
||||||
int status;
|
int status;
|
||||||
/* Avoid longjmp clobber warning in GCC */
|
/* Avoid longjmp clobber warning in GCC */
|
||||||
@ -1011,6 +1073,7 @@ int janet_unmarshal(
|
|||||||
st.end = bytes + len;
|
st.end = bytes + len;
|
||||||
st.lookup_defs = NULL;
|
st.lookup_defs = NULL;
|
||||||
st.lookup_envs = NULL;
|
st.lookup_envs = NULL;
|
||||||
|
st.reg = reg;
|
||||||
janet_array_init(&st.lookup, 0);
|
janet_array_init(&st.lookup, 0);
|
||||||
if (!(status = setjmp(st.err))) {
|
if (!(status = setjmp(st.err))) {
|
||||||
const uint8_t *nextbytes = unmarshal_one(&st, bytes, out, flags);
|
const uint8_t *nextbytes = unmarshal_one(&st, bytes, out, flags);
|
||||||
@ -1024,18 +1087,35 @@ int janet_unmarshal(
|
|||||||
|
|
||||||
/* C functions */
|
/* C functions */
|
||||||
|
|
||||||
|
static int cfun_env_lookups(JanetArgs args) {
|
||||||
|
JanetTable *env;
|
||||||
|
Janet tup[2];
|
||||||
|
JANET_FIXARITY(args, 1);
|
||||||
|
JANET_ARG_TABLE(env, args, 0);
|
||||||
|
tup[0] = janet_wrap_table(janet_env_rreg(env));
|
||||||
|
tup[1] = janet_wrap_table(janet_env_reg(env));
|
||||||
|
JANET_RETURN_TUPLE(args, janet_tuple_n(tup, 2));
|
||||||
|
}
|
||||||
|
|
||||||
static int cfun_marshal(JanetArgs args) {
|
static int cfun_marshal(JanetArgs args) {
|
||||||
JanetBuffer *buffer;
|
JanetBuffer *buffer;
|
||||||
|
JanetTable *rreg;
|
||||||
int status;
|
int status;
|
||||||
JANET_MINARITY(args, 1);
|
JANET_MINARITY(args, 1);
|
||||||
JANET_MAXARITY(args, 2);
|
JANET_MAXARITY(args, 3);
|
||||||
if (args.n == 2) {
|
if (args.n > 1) {
|
||||||
|
/* Reverse Registry provided */
|
||||||
|
JANET_ARG_TABLE(rreg, args, 1);
|
||||||
|
} else {
|
||||||
|
rreg = NULL;
|
||||||
|
}
|
||||||
|
if (args.n > 2) {
|
||||||
/* Buffer provided */
|
/* Buffer provided */
|
||||||
JANET_ARG_BUFFER(buffer, args, 1);
|
JANET_ARG_BUFFER(buffer, args, 2);
|
||||||
} else {
|
} else {
|
||||||
buffer = janet_buffer(10);
|
buffer = janet_buffer(10);
|
||||||
}
|
}
|
||||||
status = janet_marshal(buffer, args.v[0], 0);
|
status = janet_marshal(buffer, args.v[0], rreg, 0);
|
||||||
if (status) {
|
if (status) {
|
||||||
JANET_THROW(args, mr_strings[status]);
|
JANET_THROW(args, mr_strings[status]);
|
||||||
}
|
}
|
||||||
@ -1044,11 +1124,18 @@ static int cfun_marshal(JanetArgs args) {
|
|||||||
|
|
||||||
static int cfun_unmarshal(JanetArgs args) {
|
static int cfun_unmarshal(JanetArgs args) {
|
||||||
const uint8_t *bytes;
|
const uint8_t *bytes;
|
||||||
|
JanetTable *reg;
|
||||||
int32_t len;
|
int32_t len;
|
||||||
int status;
|
int status;
|
||||||
JANET_FIXARITY(args, 1);
|
JANET_MINARITY(args, 1);
|
||||||
|
JANET_MAXARITY(args, 2);
|
||||||
JANET_ARG_BYTES(bytes, len, args, 0);
|
JANET_ARG_BYTES(bytes, len, args, 0);
|
||||||
status = janet_unmarshal(bytes, (size_t) len, 0, args.ret, NULL);
|
if (args.n > 1) {
|
||||||
|
JANET_ARG_TABLE(reg, args, 1);
|
||||||
|
} else {
|
||||||
|
reg = NULL;
|
||||||
|
}
|
||||||
|
status = janet_unmarshal(bytes, (size_t) len, 0, args.ret, reg, NULL);
|
||||||
if (status) {
|
if (status) {
|
||||||
JANET_THROW(args, umr_strings[status]);
|
JANET_THROW(args, umr_strings[status]);
|
||||||
}
|
}
|
||||||
@ -1058,6 +1145,7 @@ static int cfun_unmarshal(JanetArgs args) {
|
|||||||
static const JanetReg cfuns[] = {
|
static const JanetReg cfuns[] = {
|
||||||
{"marshal", cfun_marshal},
|
{"marshal", cfun_marshal},
|
||||||
{"unmarshal", cfun_unmarshal},
|
{"unmarshal", cfun_unmarshal},
|
||||||
|
{"env-lookups", cfun_env_lookups},
|
||||||
{NULL, NULL}
|
{NULL, NULL}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -194,7 +194,8 @@ const void *janet_strbinsearch(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Register a value in the global registry */
|
/* Register a value in the global registry */
|
||||||
void janet_register(const char *name, Janet value) {
|
void janet_register(const char *name, JanetCFunction cfun) {
|
||||||
|
Janet value = janet_wrap_cfunction(cfun);
|
||||||
Janet regkey = janet_csymbolv(name);
|
Janet regkey = janet_csymbolv(name);
|
||||||
janet_table_put(janet_vm_registry, regkey, value);
|
janet_table_put(janet_vm_registry, regkey, value);
|
||||||
janet_table_put(janet_vm_registry, value, regkey);
|
janet_table_put(janet_vm_registry, value, regkey);
|
||||||
|
@ -1025,13 +1025,20 @@ JANET_API void *janet_abstract(const JanetAbstractType *type, size_t size);
|
|||||||
JANET_API JanetCFunction janet_native(const char *name, const uint8_t **error);
|
JANET_API JanetCFunction janet_native(const char *name, const uint8_t **error);
|
||||||
|
|
||||||
/* Marshaling */
|
/* Marshaling */
|
||||||
JANET_API int janet_marshal(JanetBuffer *buf, Janet x, int flags);
|
JANET_API int janet_marshal(
|
||||||
|
JanetBuffer *buf,
|
||||||
|
Janet x,
|
||||||
|
JanetTable *rreg,
|
||||||
|
int flags);
|
||||||
JANET_API int janet_unmarshal(
|
JANET_API int janet_unmarshal(
|
||||||
const uint8_t *bytes,
|
const uint8_t *bytes,
|
||||||
size_t len,
|
size_t len,
|
||||||
int flags,
|
int flags,
|
||||||
Janet *out,
|
Janet *out,
|
||||||
|
JanetTable *reg,
|
||||||
const uint8_t **next);
|
const uint8_t **next);
|
||||||
|
JANET_API JanetTable *janet_env_rreg(JanetTable *env);
|
||||||
|
JANET_API JanetTable *janet_env_reg(JanetTable *env);
|
||||||
|
|
||||||
/* GC */
|
/* GC */
|
||||||
JANET_API void janet_mark(Janet x);
|
JANET_API void janet_mark(Janet x);
|
||||||
@ -1075,7 +1082,7 @@ JANET_API void janet_var(JanetTable *env, const char *name, Janet val);
|
|||||||
JANET_API void janet_cfuns(JanetTable *env, const char *regprefix, const JanetReg *cfuns);
|
JANET_API void janet_cfuns(JanetTable *env, const char *regprefix, const JanetReg *cfuns);
|
||||||
JANET_API JanetBindingType janet_resolve(JanetTable *env, const uint8_t *sym, Janet *out);
|
JANET_API JanetBindingType janet_resolve(JanetTable *env, const uint8_t *sym, Janet *out);
|
||||||
JANET_API JanetTable *janet_env(JanetArgs args);
|
JANET_API JanetTable *janet_env(JanetArgs args);
|
||||||
JANET_API void janet_register(const char *name, Janet value);
|
JANET_API void janet_register(const char *name, JanetCFunction cfun);
|
||||||
|
|
||||||
/* C Function helpers */
|
/* C Function helpers */
|
||||||
JANET_API int janet_arity_err(JanetArgs args, int32_t n, const char *prefix);
|
JANET_API int janet_arity_err(JanetArgs args, int32_t n, const char *prefix);
|
||||||
|
@ -42,7 +42,7 @@ int main(int argc, char **argv) {
|
|||||||
|
|
||||||
/* Expose line getter */
|
/* Expose line getter */
|
||||||
janet_def(env, "getline", janet_wrap_cfunction(janet_line_getter));
|
janet_def(env, "getline", janet_wrap_cfunction(janet_line_getter));
|
||||||
janet_register("getline", janet_wrap_cfunction(janet_line_getter));
|
janet_register("getline", janet_line_getter);
|
||||||
janet_line_init();
|
janet_line_init();
|
||||||
|
|
||||||
/* Run startup script */
|
/* Run startup script */
|
||||||
|
@ -74,11 +74,11 @@ void repl_init(void) {
|
|||||||
|
|
||||||
/* Janet line getter */
|
/* Janet line getter */
|
||||||
janet_def(env, "repl-yield", janet_wrap_cfunction(repl_yield));
|
janet_def(env, "repl-yield", janet_wrap_cfunction(repl_yield));
|
||||||
janet_register("repl-yield", janet_wrap_cfunction(repl_yield));
|
janet_register("repl-yield", repl_yield);
|
||||||
|
|
||||||
/* Janet line getter */
|
/* Janet line getter */
|
||||||
janet_def(env, "js", janet_wrap_cfunction(cfun_js));
|
janet_def(env, "js", janet_wrap_cfunction(cfun_js));
|
||||||
janet_register("js", janet_wrap_cfunction(cfun_js));
|
janet_register("js", cfun_js);
|
||||||
|
|
||||||
/* Run startup script */
|
/* Run startup script */
|
||||||
Janet ret;
|
Janet ret;
|
||||||
|
@ -141,9 +141,11 @@
|
|||||||
|
|
||||||
# Marshal
|
# Marshal
|
||||||
|
|
||||||
|
(def [m-lookup um-lookup] (env-lookups _env))
|
||||||
|
|
||||||
(defn testmarsh [x msg]
|
(defn testmarsh [x msg]
|
||||||
(def marshx (marshal x))
|
(def marshx (marshal x m-lookup))
|
||||||
(def out (-> marshx unmarshal marshal))
|
(def out (marshal (unmarshal marshx um-lookup) m-lookup))
|
||||||
(assert (= (string marshx) (string out)) msg))
|
(assert (= (string marshx) (string out)) msg))
|
||||||
|
|
||||||
(testmarsh nil "marshal nil")
|
(testmarsh nil "marshal nil")
|
||||||
|
Loading…
Reference in New Issue
Block a user