mirror of https://github.com/rust-lang/rust.git
454 lines
8.1 KiB
C
454 lines
8.1 KiB
C
// Helper functions used only in tests
|
|
|
|
#include <stdint.h>
|
|
#include <stdlib.h>
|
|
#include <assert.h>
|
|
#include <stdarg.h>
|
|
|
|
// 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 */
|
|
}
|