aboutsummaryrefslogtreecommitdiff
path: root/include
diff options
context:
space:
mode:
Diffstat (limited to 'include')
-rw-r--r--include/ejit/ejit.h232
1 files changed, 169 insertions, 63 deletions
diff --git a/include/ejit/ejit.h b/include/ejit/ejit.h
index b0d8651..0c7f4c5 100644
--- a/include/ejit/ejit.h
+++ b/include/ejit/ejit.h
@@ -25,17 +25,47 @@ enum ejit_type
EJIT_DOUBLE,
};
+static inline enum ejit_type ejit_unsigned_type(size_t w)
+{
+ switch (w) {
+ case 1: return EJIT_UINT8;
+ case 2: return EJIT_UINT16;
+ case 4: return EJIT_UINT32;
+ case 8: return EJIT_UINT64;
+ default: assert(0 && "unknown width of unsigned type");
+ }
+
+ /* shut up some warnings */
+ return EJIT_UINT32;
+}
+
+static inline enum ejit_type ejit_signed_type(size_t w)
+{
+ switch (w) {
+ case 1: return EJIT_INT8;
+ case 2: return EJIT_INT16;
+ case 4: return EJIT_INT32;
+ case 8: return EJIT_INT64;
+ default: assert(0 && "unknown width of signed type");
+ }
+
+ /* shut up some warnings */
+ return EJIT_INT32;
+}
+
/* can be kind of dangerous since we default to a pointer, hmm */
#define EJIT_TYPE(x) \
_Generic((typeof(x))(0), \
- int8_t: EJIT_INT8, \
- int16_t: EJIT_INT16, \
- int32_t: EJIT_INT32, \
- int64_t: EJIT_INT64, \
- uint8_t: EJIT_UINT8, \
- uint16_t: EJIT_UINT16, \
- uint32_t: EJIT_UINT32, \
- uint64_t: EJIT_UINT64, \
+ unsigned char : ejit_unsigned_type(sizeof(x)), \
+ unsigned short : ejit_unsigned_type(sizeof(x)), \
+ unsigned int : ejit_unsigned_type(sizeof(x)), \
+ unsigned long : ejit_unsigned_type(sizeof(x)), \
+ unsigned long long: ejit_unsigned_type(sizeof(x)), \
+ signed char : ejit_signed_type(sizeof(x)), \
+ signed short : ejit_signed_type(sizeof(x)), \
+ signed int : ejit_signed_type(sizeof(x)), \
+ signed long : ejit_signed_type(sizeof(x)), \
+ signed long long : ejit_signed_type(sizeof(x)), \
float: EJIT_FLOAT, \
double: EJIT_DOUBLE, \
default: EJIT_POINTER)
@@ -102,21 +132,19 @@ static inline bool ejit_float_type(enum ejit_type t)
static inline struct ejit_arg ejit_build_arg(enum ejit_type type, uint64_t x)
{
- assert(ejit_int_type(type));
-
struct ejit_arg a;
a.type = type;
switch (type) {
- case EJIT_INT8: a.i8 = x; break;
- case EJIT_INT16: a.i16 = x; break;
- case EJIT_INT32: a.i32 = x; break;
- case EJIT_INT64: a.i64 = x; break;
- case EJIT_UINT8: a.u8 = x; break;
- case EJIT_UINT16: a.u16 = x; break;
- case EJIT_UINT32: a.u32 = x; break;
- case EJIT_UINT64: a.u64 = x; break;
- case EJIT_POINTER: a.p = (void *)x; break;
+ case EJIT_INT8: a.u64 = (int8_t)x; break;
+ case EJIT_INT16: a.u64 = (int16_t)x; break;
+ case EJIT_INT32: a.u64 = (int32_t)x; break;
+ case EJIT_INT64: a.u64 = (int64_t)x; break;
+ case EJIT_UINT8: a.u64 = (uint8_t)x; break;
+ case EJIT_UINT16: a.u64 = (uint16_t)x; break;
+ case EJIT_UINT32: a.u64 = (uint32_t)x; break;
+ case EJIT_UINT64: a.u64 = (uint64_t)x; break;
+ case EJIT_POINTER: a.p = (void *)(uintptr_t)x; break;
default: abort();
}
@@ -125,8 +153,6 @@ static inline struct ejit_arg ejit_build_arg(enum ejit_type type, uint64_t x)
static inline struct ejit_arg ejit_build_arg_f(enum ejit_type type, double x)
{
- assert(ejit_float_type(type));
-
struct ejit_arg a;
a.type = type;
@@ -139,8 +165,6 @@ static inline struct ejit_arg ejit_build_arg_f(enum ejit_type type, double x)
return a;
}
-/* register allocator could be just pushing everything above V0 or whatever onto
- * the stack, heh */
struct ejit_gpr {
size_t r;
};
@@ -182,35 +206,34 @@ void ejit_compile_func(struct ejit_func *f);
void ejit_select_compile_func(struct ejit_func *f, size_t gpr, size_t fpr,
bool use_64, bool try_jit);
-int64_t ejit_run_func(struct ejit_func *f, size_t argc,
+long ejit_run_func_i(struct ejit_func *f, size_t argc,
struct ejit_arg args[argc]);
-static inline int64_t ejit_run_func_1(struct ejit_func *f, struct ejit_arg a0)
-{
- return ejit_run_func(f, 1, &a0);
-}
+int64_t ejit_run_func_l(struct ejit_func *f, size_t argc,
+ struct ejit_arg args[argc]);
-static inline int64_t ejit_run_func_2(struct ejit_func *f, struct ejit_arg a0,
- struct ejit_arg a1)
-{
- struct ejit_arg args[2] = {a0, a1};
- return ejit_run_func(f, 2, args);
-}
+float ejit_run_func_f(struct ejit_func *f, size_t argc,
+ struct ejit_arg args[argc]);
-double ejit_run_func_f(struct ejit_func *f, size_t argc,
+double ejit_run_func_d(struct ejit_func *f, size_t argc,
struct ejit_arg args[argc]);
-static inline double ejit_run_func_f_1(struct ejit_func *f, struct ejit_arg a0)
-{
- return ejit_run_func_f(f, 1, &a0);
-}
+struct ejit_arg ejit_run_func(struct ejit_func *f, size_t argc, struct ejit_arg args[argc]);
+
+
+/* currently unused, the idea is that an escape could get the interpreter state
+ * and pass it on to speed up entering/exiting the VM, could be useful for more
+ * dynamic languages */
+struct interp_state;
+
+long ejit_run_func_ctx_i(struct ejit_func *f, size_t argc, struct ejit_arg args[argc], struct interp_state *ctx);
+
+int64_t ejit_run_func_ctx_l(struct ejit_func *f, size_t argc, struct ejit_arg args[argc], struct interp_state *ctx);
+
+float ejit_run_func_ctx_f(struct ejit_func *f, size_t argc, struct ejit_arg args[argc], struct interp_state *ctx);
+
+double ejit_run_func_ctx_d(struct ejit_func *f, size_t argc, struct ejit_arg args[argc], struct interp_state *ctx);
-static inline double ejit_run_func_f_2(struct ejit_func *f, struct ejit_arg a0,
- struct ejit_arg a1)
-{
- struct ejit_arg args[2] = {a0, a1};
- return ejit_run_func_f(f, 2, args);
-}
void ejit_destroy_func(struct ejit_func *s);
@@ -258,6 +281,26 @@ static inline struct ejit_arg ejit_i64(int64_t a)
return (struct ejit_arg){.i64 = a, .type = EJIT_INT64};
}
+static inline struct ejit_arg ejit_u8(uint8_t a)
+{
+ return (struct ejit_arg){.i64 = a, .type = EJIT_UINT8};
+}
+
+static inline struct ejit_arg ejit_u16(uint16_t a)
+{
+ return (struct ejit_arg){.i64 = a, .type = EJIT_UINT16};
+}
+
+static inline struct ejit_arg ejit_u32(uint32_t a)
+{
+ return (struct ejit_arg){.i64 = a, .type = EJIT_UINT32};
+}
+
+static inline struct ejit_arg ejit_u64(uint64_t a)
+{
+ return (struct ejit_arg){.i64 = a, .type = EJIT_UINT64};
+}
+
static inline struct ejit_arg ejit_pointer(void *p)
{
return (struct ejit_arg){.p = p, .type = EJIT_POINTER};
@@ -273,20 +316,70 @@ static inline struct ejit_arg ejit_double(double a)
return (struct ejit_arg){.d = a, .type = EJIT_DOUBLE};
}
+static inline struct ejit_arg ejit_unsigned_arg(uint64_t a, size_t w)
+{
+ switch (w) {
+ case 1: return ejit_u8(a);
+ case 2: return ejit_u16(a);
+ case 4: return ejit_u32(a);
+ case 8: return ejit_u64(a);
+ default: assert(0 && "unknown width for unsigned arg");
+ }
+
+ /* shut up some warnings */
+ return ejit_u32(a);
+}
+
+static inline struct ejit_arg ejit_signed_arg(int64_t a, size_t w)
+{
+ switch (w) {
+ case 1: return ejit_i8(a);
+ case 2: return ejit_i16(a);
+ case 4: return ejit_i32(a);
+ case 8: return ejit_i64(a);
+ default: assert(0 && "unknown width for signed arg");
+ }
+
+ /* shut up some warnings */
+ return ejit_i32(a);
+}
+
+static inline struct ejit_arg ejit_float_arg(float f, size_t w)
+{
+ (void)w;
+ assert(w == 4);
+ return ejit_float(f);
+}
+
+static inline struct ejit_arg ejit_double_arg(double d, size_t w)
+{
+ (void)w;
+ assert(w == 8);
+ return ejit_double(d);
+}
+
+static inline struct ejit_arg ejit_pointer_arg(void *p, size_t w)
+{
+ (void)w;
+ return ejit_pointer(p);
+}
+
#define EJIT_ARG(x, t) \
_Generic((t)(0), \
- int8_t: ejit_i8, \
- int16_t: ejit_i16, \
- int32_t: ejit_i32, \
- int64_t: ejit_i64, \
- uint8_t: ejit_i8, \
- uint16_t: ejit_i16, \
- uint32_t: ejit_i32, \
- uint64_t: ejit_i64, \
- float: ejit_float, \
- double: ejit_double, \
- default: ejit_pointer \
- )(x)
+ signed char : ejit_signed_arg, \
+ signed short : ejit_signed_arg, \
+ signed int : ejit_signed_arg, \
+ signed long : ejit_signed_arg, \
+ signed long long : ejit_signed_arg, \
+ unsigned char : ejit_unsigned_arg, \
+ unsigned short : ejit_unsigned_arg, \
+ unsigned int : ejit_unsigned_arg, \
+ unsigned long : ejit_unsigned_arg, \
+ unsigned long long: ejit_unsigned_arg, \
+ float : ejit_float_arg, \
+ double : ejit_double_arg, \
+ default : ejit_pointer_arg \
+ )((t)(x), sizeof(t))
#define EJIT_AUTO(x) \
EJIT_ARG(x, typeof(x))
@@ -302,30 +395,43 @@ static inline bool ejit_use64(struct ejit_arg a)
#define EJIT_USE64(t) \
ejit_use64(EJIT_ARG(0, t))
-typedef long (*ejit_escape_t)(size_t argc, const struct ejit_arg args[argc]);
-typedef double (*ejit_escape_f_t)(size_t argc,
- const struct ejit_arg args[argc]);
+typedef long (*ejit_escape_i_t)(size_t argc, const struct ejit_arg args[argc]);
+typedef int64_t (*ejit_escape_l_t)(size_t argc, const struct ejit_arg args[argc]);
+typedef float (*ejit_escape_f_t)(size_t argc, const struct ejit_arg args[argc]);
+typedef double (*ejit_escape_d_t)(size_t argc, const struct ejit_arg args[argc]);
struct ejit_label ejit_label(struct ejit_func *s);
-void ejit_calli(struct ejit_func *s, struct ejit_func *f, size_t argc,
+void ejit_calli_i(struct ejit_func *s, struct ejit_func *f, size_t argc,
+ const struct ejit_operand args[argc]);
+
+void ejit_calli_l(struct ejit_func *s, struct ejit_func *f, size_t argc,
const struct ejit_operand args[argc]);
void ejit_calli_f(struct ejit_func *s, struct ejit_func *f, size_t argc,
const struct ejit_operand args[argc]);
-void ejit_escapei(struct ejit_func *s, ejit_escape_t f, size_t argc,
+void ejit_calli_d(struct ejit_func *s, struct ejit_func *f, size_t argc,
+ const struct ejit_operand args[argc]);
+
+void ejit_escapei_i(struct ejit_func *s, ejit_escape_i_t f, size_t argc,
+ const struct ejit_operand args[argc]);
+
+void ejit_escapei_l(struct ejit_func *s, ejit_escape_l_t f, size_t argc,
const struct ejit_operand args[argc]);
void ejit_escapei_f(struct ejit_func *s, ejit_escape_f_t f, size_t argc,
const struct ejit_operand args[argc]);
+void ejit_escapei_d(struct ejit_func *s, ejit_escape_d_t f, size_t argc,
+ const struct ejit_operand args[argc]);
+
void ejit_ret(struct ejit_func *s);
void ejit_retr(struct ejit_func *s, struct ejit_gpr r0);
void ejit_retr_f(struct ejit_func *s, struct ejit_fpr r0);
void ejit_retr_d(struct ejit_func *s, struct ejit_fpr r0);
-void ejit_reti(struct ejit_func *s, long i);
+void ejit_reti(struct ejit_func *s, int64_t i);
void ejit_reti_f(struct ejit_func *s, float f);
void ejit_reti_d(struct ejit_func *s, double f);