mirror of
https://github.com/frida/tinycc
synced 2025-01-26 13:12:02 +03:00
5919da6f05
stdarg_test in abitest.c relies on a sum of some parameters made by both the caller and the callee to reach the same result. However, the variables used to store the temporary result of the additions are not initialized to 0, leading to uncertainty as to the results. This commit add this needed initialization.
456 lines
14 KiB
C
456 lines
14 KiB
C
#include <libtcc.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdarg.h>
|
|
|
|
// MinGW has 80-bit rather than 64-bit long double which isn't compatible with TCC or MSVC
|
|
#if defined(_WIN32) && defined(__GNUC__)
|
|
#define LONG_DOUBLE double
|
|
#define LONG_DOUBLE_LITERAL(x) x
|
|
#else
|
|
#define LONG_DOUBLE long double
|
|
#define LONG_DOUBLE_LITERAL(x) x ## L
|
|
#endif
|
|
|
|
static const char *tccdir = NULL;
|
|
static const char *include_dir = NULL;
|
|
|
|
typedef int (*callback_type) (void*);
|
|
|
|
/*
|
|
* Compile source code and call a callback with a pointer to the symbol "f".
|
|
*/
|
|
static int run_callback(const char *src, callback_type callback) {
|
|
TCCState *s;
|
|
int result;
|
|
void *ptr;
|
|
|
|
s = tcc_new();
|
|
if (!s)
|
|
return -1;
|
|
if (tccdir)
|
|
tcc_set_lib_path(s, tccdir);
|
|
if (include_dir) {
|
|
if (tcc_add_include_path(s, include_dir) == -1)
|
|
return -1;
|
|
}
|
|
if (tcc_set_output_type(s, TCC_OUTPUT_MEMORY) == -1)
|
|
return -1;
|
|
if (tcc_compile_string(s, src) == -1)
|
|
return -1;
|
|
if (tcc_relocate(s, TCC_RELOCATE_AUTO) == -1)
|
|
return -1;
|
|
|
|
ptr = tcc_get_symbol(s, "f");
|
|
if (!ptr)
|
|
return -1;
|
|
result = callback(ptr);
|
|
|
|
tcc_delete(s);
|
|
|
|
return result;
|
|
}
|
|
|
|
#define STR2(x) #x
|
|
#define STR(x) STR2(x)
|
|
|
|
#define RET_PRIMITIVE_TEST(name, type, val) \
|
|
static int ret_ ## name ## _test_callback(void *ptr) { \
|
|
type (*callback) (type) = (type(*)(type))ptr; \
|
|
type x = val; \
|
|
type y = callback(x); \
|
|
return (y == x+x) ? 0 : -1; \
|
|
} \
|
|
\
|
|
static int ret_ ## name ## _test(void) { \
|
|
const char *src = STR(type) " f(" STR(type) " x) {return x+x;}"; \
|
|
return run_callback(src, ret_ ## name ## _test_callback); \
|
|
}
|
|
|
|
RET_PRIMITIVE_TEST(int, int, 70000)
|
|
RET_PRIMITIVE_TEST(longlong, long long, 4333369356528LL)
|
|
RET_PRIMITIVE_TEST(float, float, 63.0)
|
|
RET_PRIMITIVE_TEST(double, double, 14789798.0)
|
|
RET_PRIMITIVE_TEST(longdouble, LONG_DOUBLE, LONG_DOUBLE_LITERAL(378943892.0))
|
|
|
|
/*
|
|
* ret_2float_test:
|
|
*
|
|
* On x86-64, a struct with 2 floats should be packed into a single
|
|
* SSE register (VT_DOUBLE is used for this purpose).
|
|
*/
|
|
typedef struct ret_2float_test_type_s {float x, y;} ret_2float_test_type;
|
|
typedef ret_2float_test_type (*ret_2float_test_function_type) (ret_2float_test_type);
|
|
|
|
static int ret_2float_test_callback(void *ptr) {
|
|
ret_2float_test_function_type f = (ret_2float_test_function_type)ptr;
|
|
ret_2float_test_type a = {10, 35};
|
|
ret_2float_test_type r;
|
|
r = f(a);
|
|
return ((r.x == a.x*5) && (r.y == a.y*3)) ? 0 : -1;
|
|
}
|
|
|
|
static int ret_2float_test(void) {
|
|
const char *src =
|
|
"typedef struct ret_2float_test_type_s {float x, y;} ret_2float_test_type;"
|
|
"ret_2float_test_type f(ret_2float_test_type a) {\n"
|
|
" ret_2float_test_type r = {a.x*5, a.y*3};\n"
|
|
" return r;\n"
|
|
"}\n";
|
|
|
|
return run_callback(src, ret_2float_test_callback);
|
|
}
|
|
|
|
/*
|
|
* ret_2double_test:
|
|
*
|
|
* On x86-64, a struct with 2 doubles should be passed in two SSE
|
|
* registers.
|
|
*/
|
|
typedef struct ret_2double_test_type_s {double x, y;} ret_2double_test_type;
|
|
typedef ret_2double_test_type (*ret_2double_test_function_type) (ret_2double_test_type);
|
|
|
|
static int ret_2double_test_callback(void *ptr) {
|
|
ret_2double_test_function_type f = (ret_2double_test_function_type)ptr;
|
|
ret_2double_test_type a = {10, 35};
|
|
ret_2double_test_type r;
|
|
r = f(a);
|
|
return ((r.x == a.x*5) && (r.y == a.y*3)) ? 0 : -1;
|
|
}
|
|
|
|
static int ret_2double_test(void) {
|
|
const char *src =
|
|
"typedef struct ret_2double_test_type_s {double x, y;} ret_2double_test_type;"
|
|
"ret_2double_test_type f(ret_2double_test_type a) {\n"
|
|
" ret_2double_test_type r = {a.x*5, a.y*3};\n"
|
|
" return r;\n"
|
|
"}\n";
|
|
|
|
return run_callback(src, ret_2double_test_callback);
|
|
}
|
|
|
|
/*
|
|
* reg_pack_test: return a small struct which should be packed into
|
|
* registers (Win32) during return.
|
|
*/
|
|
typedef struct reg_pack_test_type_s {int x, y;} reg_pack_test_type;
|
|
typedef reg_pack_test_type (*reg_pack_test_function_type) (reg_pack_test_type);
|
|
|
|
static int reg_pack_test_callback(void *ptr) {
|
|
reg_pack_test_function_type f = (reg_pack_test_function_type)ptr;
|
|
reg_pack_test_type a = {10, 35};
|
|
reg_pack_test_type r;
|
|
r = f(a);
|
|
return ((r.x == a.x*5) && (r.y == a.y*3)) ? 0 : -1;
|
|
}
|
|
|
|
static int reg_pack_test(void) {
|
|
const char *src =
|
|
"typedef struct reg_pack_test_type_s {int x, y;} reg_pack_test_type;"
|
|
"reg_pack_test_type f(reg_pack_test_type a) {\n"
|
|
" reg_pack_test_type r = {a.x*5, a.y*3};\n"
|
|
" return r;\n"
|
|
"}\n";
|
|
|
|
return run_callback(src, reg_pack_test_callback);
|
|
}
|
|
|
|
/*
|
|
* reg_pack_longlong_test: return a small struct which should be packed into
|
|
* registers (x86-64) during return.
|
|
*/
|
|
typedef struct reg_pack_longlong_test_type_s {long long x, y;} reg_pack_longlong_test_type;
|
|
typedef reg_pack_longlong_test_type (*reg_pack_longlong_test_function_type) (reg_pack_longlong_test_type);
|
|
|
|
static int reg_pack_longlong_test_callback(void *ptr) {
|
|
reg_pack_longlong_test_function_type f = (reg_pack_longlong_test_function_type)ptr;
|
|
reg_pack_longlong_test_type a = {10, 35};
|
|
reg_pack_longlong_test_type r;
|
|
r = f(a);
|
|
return ((r.x == a.x*5) && (r.y == a.y*3)) ? 0 : -1;
|
|
}
|
|
|
|
static int reg_pack_longlong_test(void) {
|
|
const char *src =
|
|
"typedef struct reg_pack_longlong_test_type_s {long long x, y;} reg_pack_longlong_test_type;"
|
|
"reg_pack_longlong_test_type f(reg_pack_longlong_test_type a) {\n"
|
|
" reg_pack_longlong_test_type r = {a.x*5, a.y*3};\n"
|
|
" return r;\n"
|
|
"}\n";
|
|
|
|
return run_callback(src, reg_pack_longlong_test_callback);
|
|
}
|
|
|
|
/*
|
|
* sret_test: Create a struct large enough to be returned via sret
|
|
* (hidden pointer as first function argument)
|
|
*/
|
|
typedef struct sret_test_type_s {long long a, b, c;} sret_test_type;
|
|
typedef sret_test_type (*sret_test_function_type) (sret_test_type);
|
|
|
|
static int sret_test_callback(void *ptr) {
|
|
sret_test_function_type f = (sret_test_function_type)(ptr);
|
|
sret_test_type x = {5436LL, 658277698LL, 43878957LL};
|
|
sret_test_type r = f(x);
|
|
return ((r.a==x.a*35)&&(r.b==x.b*19)&&(r.c==x.c*21)) ? 0 : -1;
|
|
}
|
|
|
|
static int sret_test(void) {
|
|
const char *src =
|
|
"typedef struct sret_test_type_s {long long a, b, c;} sret_test_type;\n"
|
|
"sret_test_type f(sret_test_type x) {\n"
|
|
" sret_test_type r = {x.a*35, x.b*19, x.c*21};\n"
|
|
" return r;\n"
|
|
"}\n";
|
|
|
|
return run_callback(src, sret_test_callback);
|
|
}
|
|
|
|
/*
|
|
* one_member_union_test:
|
|
*
|
|
* In the x86-64 ABI a union should always be passed on the stack. However
|
|
* it appears that a single member union is treated by GCC as its member.
|
|
*/
|
|
typedef union one_member_union_test_type_u {int x;} one_member_union_test_type;
|
|
typedef one_member_union_test_type (*one_member_union_test_function_type) (one_member_union_test_type);
|
|
|
|
static int one_member_union_test_callback(void *ptr) {
|
|
one_member_union_test_function_type f = (one_member_union_test_function_type)ptr;
|
|
one_member_union_test_type a, b;
|
|
a.x = 34;
|
|
b = f(a);
|
|
return (b.x == a.x*2) ? 0 : -1;
|
|
}
|
|
|
|
static int one_member_union_test(void) {
|
|
const char *src =
|
|
"typedef union one_member_union_test_type_u {int x;} one_member_union_test_type;\n"
|
|
"one_member_union_test_type f(one_member_union_test_type a) {\n"
|
|
" one_member_union_test_type b;\n"
|
|
" b.x = a.x * 2;\n"
|
|
" return b;\n"
|
|
"}\n";
|
|
return run_callback(src, one_member_union_test_callback);
|
|
}
|
|
|
|
/*
|
|
* two_member_union_test:
|
|
*
|
|
* In the x86-64 ABI a union should always be passed on the stack.
|
|
*/
|
|
typedef union two_member_union_test_type_u {int x; long y;} two_member_union_test_type;
|
|
typedef two_member_union_test_type (*two_member_union_test_function_type) (two_member_union_test_type);
|
|
|
|
static int two_member_union_test_callback(void *ptr) {
|
|
two_member_union_test_function_type f = (two_member_union_test_function_type)ptr;
|
|
two_member_union_test_type a, b;
|
|
a.x = 34;
|
|
b = f(a);
|
|
return (b.x == a.x*2) ? 0 : -1;
|
|
}
|
|
|
|
static int two_member_union_test(void) {
|
|
const char *src =
|
|
"typedef union two_member_union_test_type_u {int x; long y;} two_member_union_test_type;\n"
|
|
"two_member_union_test_type f(two_member_union_test_type a) {\n"
|
|
" two_member_union_test_type b;\n"
|
|
" b.x = a.x * 2;\n"
|
|
" return b;\n"
|
|
"}\n";
|
|
return run_callback(src, two_member_union_test_callback);
|
|
}
|
|
|
|
/*
|
|
* Win64 calling convetntion test.
|
|
*/
|
|
|
|
typedef struct many_struct_test_type_s {long long a, b, c;} many_struct_test_type;
|
|
typedef many_struct_test_type (*many_struct_test_function_type) (many_struct_test_type,many_struct_test_type,many_struct_test_type,many_struct_test_type,many_struct_test_type,many_struct_test_type);
|
|
|
|
static int many_struct_test_callback(void *ptr) {
|
|
many_struct_test_function_type f = (many_struct_test_function_type)ptr;
|
|
many_struct_test_type v = {1, 2, 3};
|
|
many_struct_test_type r = f(v,v,v,v,v,v);
|
|
return ((r.a == 6) && (r.b == 12) && (r.c == 18))?0:-1;
|
|
}
|
|
|
|
static int many_struct_test(void) {
|
|
const char *src =
|
|
"typedef struct many_struct_test_type_s {long long a, b, c;} many_struct_test_type;\n"
|
|
"many_struct_test_type f(many_struct_test_type x1, many_struct_test_type x2, many_struct_test_type x3, many_struct_test_type x4, many_struct_test_type x5, many_struct_test_type x6) {\n"
|
|
" many_struct_test_type y;\n"
|
|
" y.a = x1.a + x2.a + x3.a + x4.a + x5.a + x6.a;\n"
|
|
" y.b = x1.b + x2.b + x3.b + x4.b + x5.b + x6.b;\n"
|
|
" y.c = x1.c + x2.c + x3.c + x4.c + x5.c + x6.c;\n"
|
|
" return y;\n"
|
|
"}\n";
|
|
return run_callback(src, many_struct_test_callback);
|
|
}
|
|
|
|
/*
|
|
* Win64 calling convention test.
|
|
*/
|
|
|
|
typedef struct many_struct_test_2_type_s {int a, b;} many_struct_test_2_type;
|
|
typedef many_struct_test_2_type (*many_struct_test_2_function_type) (many_struct_test_2_type,many_struct_test_2_type,many_struct_test_2_type,many_struct_test_2_type,many_struct_test_2_type,many_struct_test_2_type);
|
|
|
|
static int many_struct_test_2_callback(void *ptr) {
|
|
many_struct_test_2_function_type f = (many_struct_test_2_function_type)ptr;
|
|
many_struct_test_2_type v = {1,2};
|
|
many_struct_test_2_type r = f(v,v,v,v,v,v);
|
|
return ((r.a == 6) && (r.b == 12))?0:-1;
|
|
}
|
|
|
|
static int many_struct_test_2(void) {
|
|
const char *src =
|
|
"typedef struct many_struct_test_2_type_s {int a, b;} many_struct_test_2_type;\n"
|
|
"many_struct_test_2_type f(many_struct_test_2_type x1, many_struct_test_2_type x2, many_struct_test_2_type x3, many_struct_test_2_type x4, many_struct_test_2_type x5, many_struct_test_2_type x6) {\n"
|
|
" many_struct_test_2_type y;\n"
|
|
" y.a = x1.a + x2.a + x3.a + x4.a + x5.a + x6.a;\n"
|
|
" y.b = x1.b + x2.b + x3.b + x4.b + x5.b + x6.b;\n"
|
|
" return y;\n"
|
|
"}\n";
|
|
return run_callback(src, many_struct_test_2_callback);
|
|
}
|
|
|
|
/*
|
|
* stdarg_test: Test variable argument list ABI
|
|
*/
|
|
|
|
typedef struct {long long a, b, c;} stdarg_test_struct_type;
|
|
typedef void (*stdarg_test_function_type) (int,int,int,...);
|
|
|
|
static int stdarg_test_callback(void *ptr) {
|
|
stdarg_test_function_type f = (stdarg_test_function_type)ptr;
|
|
int x;
|
|
double y;
|
|
stdarg_test_struct_type z = {1, 2, 3}, w;
|
|
f(10, 10, 5,
|
|
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, &x,
|
|
1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, &y,
|
|
z, z, z, z, z, &w);
|
|
return ((x == 55) && (y == 55) && (w.a == 5) && (w.b == 10) && (w.c == 15)) ? 0 : -1;
|
|
}
|
|
|
|
static int stdarg_test(void) {
|
|
const char *src =
|
|
"#include <stdarg.h>\n"
|
|
"typedef struct {long long a, b, c;} stdarg_test_struct_type;\n"
|
|
"void f(int n_int, int n_float, int n_struct, ...) {\n"
|
|
" int i, ti = 0;\n"
|
|
" double td = 0.0;\n"
|
|
" stdarg_test_struct_type ts = {0,0,0}, tmp;\n"
|
|
" va_list ap;\n"
|
|
" va_start(ap, n_struct);\n"
|
|
" for (i = 0, ti = 0; i < n_int; ++i)\n"
|
|
" ti += va_arg(ap, int);\n"
|
|
" *va_arg(ap, int*) = ti;\n"
|
|
" for (i = 0, td = 0; i < n_float; ++i)\n"
|
|
" td += va_arg(ap, double);\n"
|
|
" *va_arg(ap, double*) = td;\n"
|
|
" for (i = 0; i < n_struct; ++i) {\n"
|
|
" tmp = va_arg(ap, stdarg_test_struct_type);\n"
|
|
" ts.a += tmp.a; ts.b += tmp.b; ts.c += tmp.c;"
|
|
" }\n"
|
|
" *va_arg(ap, stdarg_test_struct_type*) = ts;\n"
|
|
" va_end(ap);"
|
|
"}\n";
|
|
return run_callback(src, stdarg_test_callback);
|
|
}
|
|
|
|
/*
|
|
* Test Win32 stdarg handling, since the calling convention will pass a pointer
|
|
* to the struct and the stdarg pointer must point to that pointer initially.
|
|
*/
|
|
|
|
typedef struct {long long a, b, c;} stdarg_struct_test_struct_type;
|
|
typedef int (*stdarg_struct_test_function_type) (stdarg_struct_test_struct_type a, ...);
|
|
|
|
static int stdarg_struct_test_callback(void *ptr) {
|
|
stdarg_struct_test_function_type f = (stdarg_struct_test_function_type)ptr;
|
|
stdarg_struct_test_struct_type v = {10, 35, 99};
|
|
int x = f(v, 234);
|
|
return (x == 378) ? 0 : -1;
|
|
}
|
|
|
|
static int stdarg_struct_test(void) {
|
|
const char *src =
|
|
"#include <stdarg.h>\n"
|
|
"typedef struct {long long a, b, c;} stdarg_struct_test_struct_type;\n"
|
|
"int f(stdarg_struct_test_struct_type a, ...) {\n"
|
|
" va_list ap;\n"
|
|
" va_start(ap, a);\n"
|
|
" int z = va_arg(ap, int);\n"
|
|
" va_end(ap);\n"
|
|
" return z + a.a + a.b + a.c;\n"
|
|
"}\n";
|
|
return run_callback(src, stdarg_struct_test_callback);
|
|
}
|
|
|
|
/* Test that x86-64 arranges the stack correctly for arguments with alignment >8 bytes */
|
|
|
|
typedef LONG_DOUBLE (*arg_align_test_callback_type) (LONG_DOUBLE,int,LONG_DOUBLE,int,LONG_DOUBLE);
|
|
|
|
static int arg_align_test_callback(void *ptr) {
|
|
arg_align_test_callback_type f = (arg_align_test_callback_type)ptr;
|
|
long double x = f(12, 0, 25, 0, 37);
|
|
return (x == 74) ? 0 : -1;
|
|
}
|
|
|
|
static int arg_align_test(void) {
|
|
const char *src =
|
|
"long double f(long double a, int b, long double c, int d, long double e) {\n"
|
|
" return a + c + e;\n"
|
|
"}\n";
|
|
return run_callback(src, arg_align_test_callback);
|
|
}
|
|
|
|
#define RUN_TEST(t) \
|
|
if (!testname || (strcmp(#t, testname) == 0)) { \
|
|
fputs(#t "... ", stdout); \
|
|
fflush(stdout); \
|
|
if (t() == 0) { \
|
|
fputs("success\n", stdout); \
|
|
} else { \
|
|
fputs("failure\n", stdout); \
|
|
retval = EXIT_FAILURE; \
|
|
} \
|
|
}
|
|
|
|
int main(int argc, char **argv) {
|
|
int i;
|
|
const char *testname = NULL;
|
|
int retval = EXIT_SUCCESS;
|
|
|
|
/* if tcclib.h and libtcc1.a are not installed, where can we find them */
|
|
for (i = 1; i < argc; ++i) {
|
|
if (!memcmp(argv[i], "lib_path=",9))
|
|
tccdir = argv[i] + 9;
|
|
else if (!memcmp(argv[i], "run_test=", 9))
|
|
testname = argv[i] + 9;
|
|
else if (!memcmp(argv[i], "include=", 8))
|
|
include_dir = argv[i] + 8;
|
|
}
|
|
|
|
RUN_TEST(ret_int_test);
|
|
RUN_TEST(ret_longlong_test);
|
|
RUN_TEST(ret_float_test);
|
|
RUN_TEST(ret_double_test);
|
|
RUN_TEST(ret_longdouble_test);
|
|
RUN_TEST(ret_2float_test);
|
|
RUN_TEST(ret_2double_test);
|
|
RUN_TEST(reg_pack_test);
|
|
RUN_TEST(reg_pack_longlong_test);
|
|
RUN_TEST(sret_test);
|
|
RUN_TEST(one_member_union_test);
|
|
RUN_TEST(two_member_union_test);
|
|
RUN_TEST(many_struct_test);
|
|
RUN_TEST(many_struct_test_2);
|
|
RUN_TEST(stdarg_test);
|
|
RUN_TEST(stdarg_struct_test);
|
|
RUN_TEST(arg_align_test);
|
|
return retval;
|
|
}
|