/* Copyright (C) 2019, 2020 Leo Tenenbaum. This file is part of toc. toc is distributed under version 3 of the GNU General Public License, without any warranty whatsoever. You should have received a copy of the GNU General Public License along with toc. If not, see . */ /* WARNING: In this file, you will find crazy macros and dubious usage of avcall. Beware! */ #if CHAR_BIT != 8 #error "Compile-time foreign functions can only be used on systems where CHAR_BIT is 8." #endif /* avcall has some sign conversion problems on BSD */ /* (the macros it defines cause problems too, which is why this is ignored for so long) */ #ifdef __clang__ #pragma clang diagnostic push #pragma clang diagnostic ignored "-Wsign-conversion" #elif defined __GNUC__ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wsign-conversion" #endif #include #if SCHAR_MAX != 127 #error "Foreign function support requires an 8-bit signed type." #else #define av_type_with_limit_127 schar #endif #if UCHAR_MAX != 255 #error "Foreign function support requires an 8-bit unsigned type." #else #define av_type_with_limit_255 uchar #endif #if SHRT_MAX != 32767 #error "Foreign function support requires a 16-bit signed type." #else #define av_type_with_limit_32767 short #endif #if USHRT_MAX != 65535 #error "Foreign function support requires an 16-bit unsigned type." #else #define av_type_with_limit_65535 ushort #endif #if INT_MAX == 2147483647 #define av_type_with_limit_2147483647 int #elif LONG_MAX == 2147483647 #define av_type_with_limit_2147483647 long #else #error "Foreign function support requires an 32-bit signed type." #endif #if UINT_MAX == 4294967295 #define av_type_with_limit_4294967295 uint #elif ULONG_MAX == 4294967295 #define av_type_with_limit_4294967295 ulong #else #error "Foreign function support requires an 32-bit unsigned type." #endif #if LONG_MAX == 9223372036854775807 #define av_type_with_limit_9223372036854775807 long #elif LLONG_MAX == 9223372036854775807 #define av_type_with_limit_9223372036854775807 longlong #else #error "Foreign function support requires a 64-bit signed type." #endif #if ULONG_MAX == 18446744073709551615ULL #define av_type_with_limit_18446744073709551615 ulong #elif ULLONG_MAX == 18446744073709551615ULL #define av_type_with_limit_18446744073709551615 ulonglong #else #error "Foreign function support requires a 64-bit unsigned type." #endif #define av_type_with_limit(x) join(av_type_with_limit_, x) #define toc_av_start(limit) join(av_start_, av_type_with_limit(limit)) #define toc_av_add(limit) join(av_, av_type_with_limit(limit)) #define toc_av_start_f32 av_start_float #define toc_av_start_f64 av_start_double #define toc_av_f32 av_float #define toc_av_f64 av_double static Status arg_list_start(av_alist *arg_list, FnPtr fn, Value *return_val, Type *return_type, Location where) { switch (return_type->kind) { case TYPE_UNKNOWN: err_print(where, "Cannot call foreign function with unknown return type."); return false; case TYPE_TUPLE: err_print(where, "Cannot call foreign function with tuple return type."); return false; case TYPE_ARR: err_print(where, "Foreign functions cannot return arrays."); return false; case TYPE_FN: warn_print(where, "Foreign function returns function pointer. If it returns a C-style function pointer, it won't be called properly by toc."); av_start_ptr(*arg_list, fn, FnExpr *, &return_val->fn); break; case TYPE_PTR: av_start_ptr(*arg_list, fn, void *, &return_val->ptr); break; case TYPE_BUILTIN: switch (return_type->builtin) { case BUILTIN_I8: toc_av_start(127)(*arg_list, fn, &return_val->i8); break; case BUILTIN_U8: toc_av_start(255)(*arg_list, fn, &return_val->u8); break; case BUILTIN_I16: toc_av_start(32767)(*arg_list, fn, &return_val->i16); break; case BUILTIN_U16: toc_av_start(65535)(*arg_list, fn, &return_val->u16); break; case BUILTIN_I32: toc_av_start(2147483647)(*arg_list, fn, &return_val->i32); break; case BUILTIN_U32: toc_av_start(4294967295)(*arg_list, fn, &return_val->u32); break; case BUILTIN_I64: toc_av_start(9223372036854775807)(*arg_list, fn, &return_val->i64); break; case BUILTIN_U64: toc_av_start(18446744073709551615)(*arg_list, fn, &return_val->u64); break; case BUILTIN_BOOL: /* bool is probably just unsigned char.... hopefully... */ av_start_uchar(*arg_list, fn, &return_val->u8); break; case BUILTIN_CHAR: av_start_char(*arg_list, fn, &return_val->charv); break; case BUILTIN_F32: toc_av_start_f32(*arg_list, fn, &return_val->f32); break; case BUILTIN_F64: toc_av_start_f64(*arg_list, fn, &return_val->f64); break; case BUILTIN_TYPE: av_start_ptr(*arg_list, fn, Type *, &return_val->type); break; case BUILTIN_NMS: av_start_ptr(*arg_list, fn, Namespace *, &return_val->nms); break; case BUILTIN_VOID: av_start_void(*arg_list, fn); break; case BUILTIN_VARARGS: assert(0); break; } break; case TYPE_STRUCT: { size_t struct_size = compiler_sizeof(return_type); StructDef *struc = return_type->struc; return_val->struc = err_calloc(1, struct_size); bool splittable; /* hopefully this is right! */ if (struct_size <= sizeof(long)) { splittable = true; } else if (struct_size > 2*sizeof(long)) { splittable = false; } else if (arr_len(struc->fields) > 4) { splittable = false; } else { /* NOTE: this warning is not because splittable is being computed incorrectly! it doesn't handle it right with *either* splittable = 0 or splittable = 1 */ warn_print(where, "Dynamically calling function which returns a struct. avcall seems to not handle structs of size ~2*sizeof(long) correctly."); splittable = true; size_t word_size = sizeof(__avword); arr_foreach(struc->fields, Field, f) { if (f->offset / word_size != (f->offset + compiler_sizeof(f->type) - 1) / word_size) { splittable = false; break; } } } /* getting warning on Debian stretch about splittable being set but not used */ _av_start_struct(*arg_list, fn, struct_size, splittable, return_val->struc); (void)splittable; } break; case TYPE_SLICE: av_start_struct(*arg_list, fn, Slice, av_word_splittable_2(I64, void *), &return_val->slice); break; case TYPE_EXPR: assert(0); return false; } return true; } static Status arg_list_add(av_alist *arg_list, Value val, Type *type, Location where) { switch (type->kind) { case TYPE_TUPLE: case TYPE_UNKNOWN: case TYPE_ARR: { /* @TODO: maybe just pass pointer for arr? */ char *s = type_to_str(type); err_print(where, "Cannot pass type %s to foreign function.", s); free(s); return false; } case TYPE_PTR: av_ptr(*arg_list, void *, val.ptr); break; case TYPE_FN: warn_print(where, "Passing toc function pointer to foreign function. This will not work if the function expects a C-style function pointer."); av_ptr(*arg_list, FnExpr *, val.fn); break; case TYPE_BUILTIN: switch (type->builtin) { case BUILTIN_I8: toc_av_add(127)(*arg_list, val.i8); break; case BUILTIN_U8: toc_av_add(255)(*arg_list, val.u8); break; case BUILTIN_I16: toc_av_add(32767)(*arg_list, val.i16); break; case BUILTIN_U16: toc_av_add(65535)(*arg_list, val.u16); break; case BUILTIN_I32: toc_av_add(2147483647)(*arg_list, val.i32); break; case BUILTIN_U32: toc_av_add(4294967295)(*arg_list, val.u32); break; case BUILTIN_I64: toc_av_add(9223372036854775807)(*arg_list, val.i64); break; case BUILTIN_U64: toc_av_add(18446744073709551615)(*arg_list, val.u64); break; case BUILTIN_CHAR: av_char(*arg_list, val.charv); break; case BUILTIN_BOOL: av_uchar(*arg_list, val.boolv); break; case BUILTIN_F32: toc_av_f32(*arg_list, val.f32); break; case BUILTIN_F64: toc_av_f64(*arg_list, val.f64); break; case BUILTIN_TYPE: av_ptr(*arg_list, Type *, val.type); break; case BUILTIN_NMS: av_ptr(*arg_list, Namespace *, val.nms); break; case BUILTIN_VARARGS: arr_foreach(val.varargs, VarArg, arg) { if (!arg_list_add(arg_list, arg->val, arg->type, where)) return false; } break; case BUILTIN_VOID: err_print(where, "Cannot pass type void to foreign function."); return false; } break; case TYPE_SLICE: av_struct(*arg_list, Slice, val.slice); break; case TYPE_STRUCT: _av_struct(*arg_list, compiler_sizeof(type), compiler_alignof(type), val.struc); break; case TYPE_EXPR: assert(0); return false; } return true; } #ifdef __clang__ #pragma clang diagnostic pop #elif defined __GNUC__ #pragma GCC diagnostic pop #endif static Status foreign_call(ForeignFnManager *ffmgr, FnExpr *fn, Type *ret_type, Type *arg_types, size_t arg_types_stride, Value *args, size_t nargs, Location call_where, Value *ret) { FnPtr fn_ptr = foreign_get_fn_ptr(ffmgr, fn, call_where); av_alist arg_list; if (!arg_list_start(&arg_list, fn_ptr, ret, ret_type, call_where)) return false; char *type = (char *)arg_types; for (size_t i = 0; i < nargs; ++i) { if (!arg_list_add(&arg_list, args[i], (Type *)type, call_where)) return false; type += arg_types_stride; } av_call(arg_list); return true; }