diff options
Diffstat (limited to 'lib')
| -rw-r--r-- | lib/nostd/sail.h | 341 | ||||
| -rw-r--r-- | lib/sail.c | 36 | ||||
| -rw-r--r-- | lib/sail.h | 10 |
3 files changed, 369 insertions, 18 deletions
diff --git a/lib/nostd/sail.h b/lib/nostd/sail.h new file mode 100644 index 00000000..52318a61 --- /dev/null +++ b/lib/nostd/sail.h @@ -0,0 +1,341 @@ +#ifndef __SAIL_LIB +#define __SAIL_LIB + +#include <stddef.h> +#include <stdint.h> +#include <stdbool.h> + +void *sail_malloc(size_t); +void sail_free(void *); + +void sail_match_failure(char *message); + +/* + * The Sail compiler expects functions to follow a specific naming + * convention for allocation, deallocation, and (deep)-copying. These + * macros implement this naming convention. + */ +#define CREATE(type) create_ ## type +#define RECREATE(type) recreate_ ## type +#define CREATE_OF(type1, type2) create_ ## type1 ## _of_ ## type2 +#define RECREATE_OF(type1, type2) recreate_ ## type1 ## _of_ ## type2 +#define CONVERT_OF(type1, type2) convert_ ## type1 ## _of_ ## type2 +#define COPY(type) copy_ ## type +#define KILL(type) kill_ ## type +#define UNDEFINED(type) undefined_ ## type +#define EQUAL(type) eq_ ## type + +#define SAIL_BUILTIN_TYPE(type)\ + void create_ ## type(type *);\ + void recreate_ ## type(type *);\ + void copy_ ## type(type *, const type);\ + void kill_ ## type(type *); + +/* ***** Sail unit type ***** */ + +typedef int unit; + +#define UNIT 0 + +unit UNDEFINED(unit)(const unit); +bool EQUAL(unit)(const unit, const unit); + +unit skip(const unit); +unit sail_exit(const unit); + +/* ***** Sail booleans ***** */ + +/* + * and_bool and or_bool are special-cased by the compiler to ensure + * short-circuiting evaluation. + */ +bool not(const bool); +bool EQUAL(bool)(const bool, const bool); +bool UNDEFINED(bool)(const unit); + +/* ***** Sail strings ***** */ + +/* + * Sail strings are just C strings. + */ +typedef char *sail_string; + +SAIL_BUILTIN_TYPE(sail_string); + +void undefined_string(sail_string *str, const unit u); + +bool eq_string(const sail_string, const sail_string); +bool EQUAL(sail_string)(const sail_string, const sail_string); + +void concat_str(sail_string *stro, const sail_string str1, const sail_string str2); +bool string_startswith(sail_string s, sail_string prefix); + +unit sail_assert(bool, sail_string); + +/* ***** Sail integers ***** */ + +typedef int64_t mach_int; + +bool EQUAL(mach_int)(const mach_int, const mach_int); + +typedef __int128 sail_int; + +uint64_t sail_int_get_ui(const sail_int); + +void dec_str(sail_string *str, const sail_int n); +void hex_str(sail_string *str, const sail_int n); + +#define SAIL_INT_FUNCTION(fname, rtype, ...) rtype fname(__VA_ARGS__) + +// SAIL_BUILTIN_TYPE(sail_int); + +sail_int CREATE_OF(sail_int, mach_int)(const mach_int); +mach_int CREATE_OF(mach_int, sail_int)(const sail_int); + +mach_int CONVERT_OF(mach_int, sail_int)(const sail_int); +sail_int CONVERT_OF(sail_int, mach_int)(const mach_int); +sail_int CONVERT_OF(sail_int, sail_string)(const sail_string); + +/* + * Comparison operators for integers + */ +bool eq_int(const sail_int, const sail_int); +bool EQUAL(sail_int)(const sail_int, const sail_int); + +bool lt(const sail_int, const sail_int); +bool gt(const sail_int, const sail_int); +bool lteq(const sail_int, const sail_int); +bool gteq(const sail_int, const sail_int); + +/* + * Left and right shift for integers + */ +mach_int shl_mach_int(const mach_int, const mach_int); +mach_int shr_mach_int(const mach_int, const mach_int); +SAIL_INT_FUNCTION(shl_int, sail_int, const sail_int, const sail_int); +SAIL_INT_FUNCTION(shr_int, sail_int, const sail_int, const sail_int); + +/* + * undefined_int and undefined_range can't use the UNDEFINED(TYPE) + * macro, because they're slightly magical. They take extra parameters + * to ensure that no undefined int can violate any type-guaranteed + * constraints. + */ +SAIL_INT_FUNCTION(undefined_int, sail_int, const int); +SAIL_INT_FUNCTION(undefined_range, sail_int, const sail_int, const sail_int); + +/* + * Arithmetic operations in integers. We include functions for both + * truncating towards zero, and rounding towards -infinity (floor) as + * fdiv/fmod and tdiv/tmod respectively. + */ +SAIL_INT_FUNCTION(add_int, sail_int, const sail_int, const sail_int); +SAIL_INT_FUNCTION(sub_int, sail_int, const sail_int, const sail_int); +SAIL_INT_FUNCTION(sub_nat, sail_int, const sail_int, const sail_int); +SAIL_INT_FUNCTION(mult_int, sail_int, const sail_int, const sail_int); +SAIL_INT_FUNCTION(ediv_int, sail_int, const sail_int, const sail_int); +SAIL_INT_FUNCTION(emod_int, sail_int, const sail_int, const sail_int); +SAIL_INT_FUNCTION(tdiv_int, sail_int, const sail_int, const sail_int); +SAIL_INT_FUNCTION(tmod_int, sail_int, const sail_int, const sail_int); +//SAIL_INT_FUNCTION(fdiv_int, sail_int, const sail_int, const sail_int); +//SAIL_INT_FUNCTION(fmod_int, sail_int, const sail_int, const sail_int); +SAIL_INT_FUNCTION(max_int, sail_int, const sail_int, const sail_int); +SAIL_INT_FUNCTION(min_int, sail_int, const sail_int, const sail_int); +SAIL_INT_FUNCTION(neg_int, sail_int, const sail_int); +SAIL_INT_FUNCTION(abs_int, sail_int, const sail_int); +SAIL_INT_FUNCTION(pow_int, sail_int, const sail_int, const sail_int); +SAIL_INT_FUNCTION(pow2, sail_int, const sail_int); + +SAIL_INT_FUNCTION(make_the_value, sail_int, const sail_int); +SAIL_INT_FUNCTION(size_itself_int, sail_int, const sail_int); + +/* ***** Sail bitvectors ***** */ + +typedef uint64_t fbits; + +bool eq_bit(const fbits a, const fbits b); + +bool EQUAL(fbits)(const fbits, const fbits); + +typedef struct { + uint64_t len; + uint64_t bits; +} sbits; + +// For backwards compatability +typedef uint64_t mach_bits; +typedef sbits lbits; + +lbits append_64(const lbits, const uint64_t); + +lbits add_bits_int(const lbits, const sail_int); +lbits sub_bits_int(const lbits, const sail_int); + +lbits UNDEFINED(lbits)(bool); + +lbits vector_subrange_lbits(const lbits, const sail_int, const sail_int); + +lbits vector_update_subrange_lbits(const lbits, const sail_int, const sail_int, const lbits); + +lbits slice(const lbits, const sail_int, const sail_int); + +lbits set_slice(const sail_int, const sail_int, const lbits, const sail_int, const lbits); + +lbits get_slice_int(const sail_int, const sail_int, const sail_int); + +lbits zeros(const sail_int); +lbits zero_extend(const lbits, const sail_int); + +bool eq_bits(const lbits, const lbits); +bool neq_bits(const lbits, const lbits); +lbits not_bits(const lbits); +lbits xor_bits(const lbits, const lbits); +lbits or_bits(const lbits, const lbits); +lbits and_bits(const lbits, const lbits); +lbits add_bits(const lbits, const lbits); +lbits sub_bits(const lbits, const lbits); + +lbits append(const lbits, const lbits); + +lbits replicate_bits(const lbits, const sail_int); + +sbits CREATE_OF(sbits, lbits)(const lbits op, const bool direction); +fbits CREATE_OF(fbits, lbits)(const lbits op, const bool direction); +sbits CREATE_OF(sbits, fbits)(const fbits op, const uint64_t len, const bool direction); + +/* Bitvector conversions */ + +lbits CONVERT_OF(lbits, fbits)(const fbits, const uint64_t, const bool); +lbits CONVERT_OF(lbits, sbits)(const sbits, const bool); + +fbits CONVERT_OF(fbits, lbits)(const lbits, const bool); +fbits CONVERT_OF(fbits, sbits)(const sbits, const bool); + +sbits CONVERT_OF(sbits, fbits)(const fbits, const uint64_t, const bool); +sbits CONVERT_OF(sbits, lbits)(const lbits, const bool); + +fbits UNDEFINED(fbits)(const unit); + +sbits undefined_sbits(void); + +/* + * Wrapper around >> operator to avoid UB when shift amount is greater + * than or equal to 64. + */ +fbits safe_rshift(const fbits, const fbits); + +fbits fast_zero_extend(const sbits op, const uint64_t n); +fbits fast_sign_extend(const fbits op, const uint64_t n, const uint64_t m); +fbits fast_sign_extend2(const sbits op, const uint64_t m); + +sail_int length_lbits(const lbits op); + +bool eq_bits(const lbits op1, const lbits op2); +bool EQUAL(lbits)(const lbits op1, const lbits op2); +bool neq_bits(const lbits op1, const lbits op2); + +fbits bitvector_access(const lbits op, const sail_int n_mpz); + +sail_int sail_unsigned(const lbits op); +sail_int sail_signed(const lbits op); + +mach_int fast_signed(const fbits, const uint64_t); +mach_int fast_unsigned(const fbits); + +sbits append_sf(const sbits, const fbits, const uint64_t); +sbits append_fs(const fbits, const uint64_t, const sbits); +sbits append_ss(const sbits, const sbits); + +fbits fast_replicate_bits(const fbits shift, const fbits v, const mach_int times); + +sail_int set_slice_int(const sail_int, const sail_int, const sail_int, const lbits); + +fbits fast_update_subrange(const fbits op, + const mach_int n, + const mach_int m, + const fbits slice); + +sbits sslice(const fbits op, const mach_int start, const mach_int len); + +bool eq_sbits(const sbits op1, const sbits op2); +bool neq_sbits(const sbits op1, const sbits op2); +sbits not_sbits(const sbits op); +sbits xor_sbits(const sbits op1, const sbits op2); +sbits or_sbits(const sbits op1, const sbits op2); +sbits and_sbits(const sbits op1, const sbits op2); +sbits add_sbits(const sbits op1, const sbits op2); +sbits sub_sbits(const sbits op1, const sbits op2); + +/* ***** Sail reals ***** */ + +typedef double real; + +SAIL_BUILTIN_TYPE(real); + +void CREATE_OF(real, sail_string)(real *rop, const sail_string op); +void CONVERT_OF(real, sail_string)(real *rop, const sail_string op); + +void UNDEFINED(real)(real *rop, unit u); + +void neg_real(real *rop, const real op); + +void mult_real(real *rop, const real op1, const real op2); +void sub_real(real *rop, const real op1, const real op2); +void add_real(real *rop, const real op1, const real op2); +void div_real(real *rop, const real op1, const real op2); + +void sqrt_real(real *rop, const real op); +void abs_real(real *rop, const real op); + +SAIL_INT_FUNCTION(round_up, sail_int, const real); +SAIL_INT_FUNCTION(round_down, sail_int, const real); + +void to_real(real *rop, const sail_int op); + +bool EQUAL(real)(const real op1, const real op2); + +bool lt_real(const real op1, const real op2); +bool gt_real(const real op1, const real op2); +bool lteq_real(const real op1, const real op2); +bool gteq_real(const real op1, const real op2); + +void real_power(real *rop, const real base, const sail_int exp); + +unit print_real(const sail_string, const real); +unit prerr_real(const sail_string, const real); + +void random_real(real *rop, unit); + +/* ***** String utilities ***** */ + +SAIL_INT_FUNCTION(string_length, sail_int, sail_string); +void string_drop(sail_string *dst, sail_string s, sail_int len); +void string_take(sail_string *dst, sail_string s, sail_int len); + +/* ***** Printing ***** */ + +void string_of_int(sail_string *str, const sail_int i); +void string_of_lbits(sail_string *str, const lbits op); +void string_of_fbits(sail_string *str, const fbits op); +void decimal_string_of_lbits(sail_string *str, const lbits op); +void decimal_string_of_fbits(sail_string *str, const fbits op); + +unit print_bits(const sail_string str, const lbits op); +unit prerr_bits(const sail_string str, const lbits op); + +unit print(const sail_string str); +unit print_endline(const sail_string str); + +unit prerr(const sail_string str); +unit prerr_endline(const sail_string str); + +unit print_int(const sail_string str, const sail_int op); +unit prerr_int(const sail_string str, const sail_int op); + +unit sail_putchar(const sail_int op); + +/* ***** Misc ***** */ + +sail_int get_time_ns(const unit); + +#endif @@ -78,15 +78,15 @@ bool UNDEFINED(bool)(const unit u) { void CREATE(sail_string)(sail_string *str) { - char *istr = (char *) malloc(1 * sizeof(char)); + char *istr = (char *) sail_malloc(1 * sizeof(char)); istr[0] = '\0'; *str = istr; } void RECREATE(sail_string)(sail_string *str) { - free(*str); - char *istr = (char *) malloc(1 * sizeof(char)); + sail_free(*str); + char *istr = (char *) sail_malloc(1 * sizeof(char)); istr[0] = '\0'; *str = istr; } @@ -100,18 +100,18 @@ void COPY(sail_string)(sail_string *str1, const sail_string str2) void KILL(sail_string)(sail_string *str) { - free(*str); + sail_free(*str); } void dec_str(sail_string *str, const mpz_t n) { - free(*str); + sail_free(*str); gmp_asprintf(str, "%Zd", n); } void hex_str(sail_string *str, const mpz_t n) { - free(*str); + sail_free(*str); gmp_asprintf(str, "0x%Zx", n); } @@ -463,7 +463,7 @@ bool EQUAL(fbits)(const fbits op1, const fbits op2) void CREATE(lbits)(lbits *rop) { - rop->bits = malloc(sizeof(mpz_t)); + rop->bits = sail_malloc(sizeof(mpz_t)); rop->len = 0; mpz_init(*rop->bits); } @@ -483,12 +483,12 @@ void COPY(lbits)(lbits *rop, const lbits op) void KILL(lbits)(lbits *rop) { mpz_clear(*rop->bits); - free(rop->bits); + sail_free(rop->bits); } void CREATE_OF(lbits, fbits)(lbits *rop, const uint64_t op, const uint64_t len, const bool direction) { - rop->bits = malloc(sizeof(mpz_t)); + rop->bits = sail_malloc(sizeof(mpz_t)); rop->len = len; mpz_init_set_ui(*rop->bits, op); } @@ -522,7 +522,7 @@ void RECREATE_OF(lbits, fbits)(lbits *rop, const uint64_t op, const uint64_t len void CREATE_OF(lbits, sbits)(lbits *rop, const sbits op, const bool direction) { - rop->bits = malloc(sizeof(mpz_t)); + rop->bits = sail_malloc(sizeof(mpz_t)); rop->len = op.len; mpz_init_set_ui(*rop->bits, op.bits); } @@ -1477,14 +1477,14 @@ void random_real(real *rop, const unit u) void string_of_int(sail_string *str, const sail_int i) { - free(*str); + sail_free(*str); gmp_asprintf(str, "%Zd", i); } /* asprintf is a GNU extension, but it should exist on BSD */ void string_of_fbits(sail_string *str, const fbits op) { - free(*str); + sail_free(*str); int bytes = asprintf(str, "0x%" PRIx64, op); if (bytes == -1) { fprintf(stderr, "Could not print bits 0x%" PRIx64 "\n", op); @@ -1493,11 +1493,11 @@ void string_of_fbits(sail_string *str, const fbits op) void string_of_lbits(sail_string *str, const lbits op) { - free(*str); + sail_free(*str); if ((op.len % 4) == 0) { gmp_asprintf(str, "0x%*0ZX", op.len / 4, *op.bits); } else { - *str = (char *) malloc((op.len + 3) * sizeof(char)); + *str = (char *) sail_malloc((op.len + 3) * sizeof(char)); (*str)[0] = '0'; (*str)[1] = 'b'; for (int i = 1; i <= op.len; ++i) { @@ -1509,7 +1509,7 @@ void string_of_lbits(sail_string *str, const lbits op) void decimal_string_of_fbits(sail_string *str, const fbits op) { - free(*str); + sail_free(*str); int bytes = asprintf(str, "%" PRId64, op); if (bytes == -1) { fprintf(stderr, "Could not print bits %" PRId64 "\n", op); @@ -1518,7 +1518,7 @@ void decimal_string_of_fbits(sail_string *str, const fbits op) void decimal_string_of_lbits(sail_string *str, const lbits op) { - free(*str); + sail_free(*str); gmp_asprintf(str, "%Z", *op.bits); } @@ -1534,7 +1534,7 @@ void fprint_bits(const sail_string pre, mpz_t buf; mpz_init_set(buf, *op.bits); - char *hex = malloc((op.len / 4) * sizeof(char)); + char *hex = sail_malloc((op.len / 4) * sizeof(char)); for (int i = 0; i < op.len / 4; ++i) { char c = (char) ((0xF & mpz_get_ui(buf)) + 0x30); @@ -1546,7 +1546,7 @@ void fprint_bits(const sail_string pre, fputc(hex[i - 1], stream); } - free(hex); + sail_free(hex); mpz_clear(buf); } else { fputs("0b", stream); @@ -8,6 +8,16 @@ #include<time.h> +static inline void *sail_malloc(size_t size) +{ + return malloc(size); +} + +static inline void sail_free(void *ptr) +{ + free(ptr); +} + /* * Called by the RTS to initialise and clear any library state. */ |
