// Helper functions used only in tests #include #include #include #include // These functions are used in the unit tests for C ABI calls. uint32_t rust_dbg_extern_identity_u32(uint32_t u) { return u; } uint64_t rust_dbg_extern_identity_u64(uint64_t u) { return u; } double rust_dbg_extern_identity_double(double u) { return u; } char rust_dbg_extern_identity_u8(char u) { return u; } typedef uint64_t (*dbg_callback)(uint64_t); uint64_t rust_dbg_call(dbg_callback cb, uint64_t data) { return cb(data); } void rust_dbg_do_nothing() { } struct TwoU8s { uint8_t one; uint8_t two; }; struct TwoU8s rust_dbg_extern_return_TwoU8s() { struct TwoU8s s; s.one = 10; s.two = 20; return s; } struct TwoU8s rust_dbg_extern_identity_TwoU8s(struct TwoU8s u) { return u; } struct TwoU16s { uint16_t one; uint16_t two; }; struct TwoU16s rust_dbg_extern_return_TwoU16s() { struct TwoU16s s; s.one = 10; s.two = 20; return s; } struct TwoU16s rust_dbg_extern_identity_TwoU16s(struct TwoU16s u) { return u; } struct TwoU32s { uint32_t one; uint32_t two; }; struct TwoU32s rust_dbg_extern_return_TwoU32s() { struct TwoU32s s; s.one = 10; s.two = 20; return s; } struct TwoU32s rust_dbg_extern_identity_TwoU32s(struct TwoU32s u) { return u; } struct TwoU64s { uint64_t one; uint64_t two; }; struct TwoU64s rust_dbg_extern_return_TwoU64s() { struct TwoU64s s; s.one = 10; s.two = 20; return s; } struct TwoU64s rust_dbg_extern_identity_TwoU64s(struct TwoU64s u) { return u; } struct TwoDoubles { double one; double two; }; struct TwoDoubles rust_dbg_extern_identity_TwoDoubles(struct TwoDoubles u) { return u; } struct FiveU16s { uint16_t one; uint16_t two; uint16_t three; uint16_t four; uint16_t five; }; struct FiveU16s rust_dbg_extern_return_FiveU16s() { struct FiveU16s s; s.one = 10; s.two = 20; s.three = 30; s.four = 40; s.five = 50; return s; } struct FiveU16s rust_dbg_extern_identity_FiveU16s(struct FiveU16s u) { return u; } struct ManyInts { int8_t arg1; int16_t arg2; int32_t arg3; int16_t arg4; int8_t arg5; struct TwoU8s arg6; }; // MSVC doesn't allow empty structs or unions #ifndef _MSC_VER struct Empty { }; void rust_dbg_extern_empty_struct(struct ManyInts v1, struct Empty e, struct ManyInts v2) { assert(v1.arg1 == v2.arg1 + 1); assert(v1.arg2 == v2.arg2 + 1); assert(v1.arg3 == v2.arg3 + 1); assert(v1.arg4 == v2.arg4 + 1); assert(v1.arg5 == v2.arg5 + 1); assert(v1.arg6.one == v2.arg6.one + 1); assert(v1.arg6.two == v2.arg6.two + 1); } #endif intptr_t rust_get_test_int() { return 1; } char * rust_get_null_ptr() { return 0; } // Debug helpers strictly to verify ABI conformance. struct quad { uint64_t a; uint64_t b; uint64_t c; uint64_t d; }; struct floats { double a; uint8_t b; double c; }; struct char_char_double { uint8_t a; uint8_t b; double c; }; struct char_char_float { uint8_t a; uint8_t b; float c; }; struct quad rust_dbg_abi_1(struct quad q) { struct quad qq = { q.c + 1, q.d - 1, q.a + 1, q.b - 1 }; return qq; } struct floats rust_dbg_abi_2(struct floats f) { struct floats ff = { f.c + 1.0, 0xff, f.a - 1.0 }; return ff; } struct char_char_double rust_dbg_abi_3(struct char_char_double a) { struct char_char_double ccd = { a.a + 1, a.b - 1, a.c + 1.0 }; return ccd; } struct char_char_float rust_dbg_abi_4(struct char_char_float a) { struct char_char_float ccd = { a.a + 1, a.b - 1, a.c + 1.0 }; return ccd; } int rust_dbg_static_mut = 3; void rust_dbg_static_mut_check_four() { assert(rust_dbg_static_mut == 4); } struct S { uint64_t x; uint64_t y; uint64_t z; }; uint64_t get_x(struct S s) { return s.x; } uint64_t get_y(struct S s) { return s.y; } uint64_t get_z(struct S s) { return s.z; } uint64_t get_c_many_params(void *a, void *b, void *c, void *d, struct quad f) { return f.c; } struct quad_floats { float a; float b; float c; float d; }; float get_c_exhaust_sysv64_ints( void *a, void *b, void *c, void *d, void *e, void *f, // `f` used the last integer register, so `g` goes on the stack. // It also used to bring the "count of available integer registers" down to // `-1` which broke the next SSE-only aggregate argument (`h`) - see #62350. void *g, struct quad_floats h ) { return h.c; } // Calculates the average of `(x + y) / n` where x: i64, y: f64. There must be exactly n pairs // passed as variadic arguments. There are two versions of this function: the // variadic one, and the one that takes a `va_list`. double rust_valist_interesting_average(uint64_t n, va_list pairs) { double sum = 0.0; int i; for(i = 0; i < n; i += 1) { sum += (double)va_arg(pairs, int64_t); sum += va_arg(pairs, double); } return sum / n; } double rust_interesting_average(uint64_t n, ...) { double sum; va_list pairs; va_start(pairs, n); sum = rust_valist_interesting_average(n, pairs); va_end(pairs); return sum; } int32_t rust_int8_to_int32(int8_t x) { return (int32_t)x; } typedef union LARGE_INTEGER { struct { uint32_t LowPart; uint32_t HighPart; }; struct { uint32_t LowPart; uint32_t HighPart; } u; uint64_t QuadPart; } LARGE_INTEGER; LARGE_INTEGER increment_all_parts(LARGE_INTEGER li) { li.LowPart += 1; li.HighPart += 1; li.u.LowPart += 1; li.u.HighPart += 1; li.QuadPart += 1; return li; } #if __SIZEOF_INT128__ == 16 unsigned __int128 identity(unsigned __int128 a) { return a; } __int128 square(__int128 a) { return a * a; } __int128 sub(__int128 a, __int128 b) { return a - b; } #endif #define OPTION_TAG_NONE (0) #define OPTION_TAG_SOME (1) struct U8TaggedEnumOptionU64 { uint8_t tag; union { uint64_t some; }; }; struct U8TaggedEnumOptionU64 rust_dbg_new_some_u64(uint64_t some) { struct U8TaggedEnumOptionU64 r = { .tag = OPTION_TAG_SOME, .some = some, }; return r; } struct U8TaggedEnumOptionU64 rust_dbg_new_none_u64(void) { struct U8TaggedEnumOptionU64 r = { .tag = OPTION_TAG_NONE, }; return r; } int32_t rust_dbg_unpack_option_u64(struct U8TaggedEnumOptionU64 o, uint64_t *into) { assert(into); switch (o.tag) { case OPTION_TAG_SOME: *into = o.some; return 1; case OPTION_TAG_NONE: return 0; default: assert(0 && "unexpected tag"); return 0; } } struct U8TaggedEnumOptionU64U64 { uint8_t tag; union { struct { uint64_t a; uint64_t b; } some; }; }; struct U8TaggedEnumOptionU64U64 rust_dbg_new_some_u64u64(uint64_t a, uint64_t b) { struct U8TaggedEnumOptionU64U64 r = { .tag = OPTION_TAG_SOME, .some = { .a = a, .b = b }, }; return r; } struct U8TaggedEnumOptionU64U64 rust_dbg_new_none_u64u64(void) { struct U8TaggedEnumOptionU64U64 r = { .tag = OPTION_TAG_NONE, }; return r; } int32_t rust_dbg_unpack_option_u64u64(struct U8TaggedEnumOptionU64U64 o, uint64_t *a, uint64_t *b) { assert(a); assert(b); switch (o.tag) { case OPTION_TAG_SOME: *a = o.some.a; *b = o.some.b; return 1; case OPTION_TAG_NONE: return 0; default: assert(0 && "unexpected tag"); return 0; } } uint16_t issue_97463_leak_uninit_data(uint32_t a, uint32_t b, uint32_t c) { struct bloc { uint16_t a; uint16_t b; uint16_t c; }; struct bloc *data = malloc(sizeof(struct bloc)); data->a = a & 0xFFFF; data->b = b & 0xFFFF; data->c = c & 0xFFFF; return data->b; /* leak data */ }