mirror of
https://github.com/mirror/tinycc.git
synced 2025-01-17 05:20:08 +08:00
x86-64: Fix psABI stdarg prologue
If there were more than 6 integer arguments before the ellipsis, or there were used more than 8 slots used until the ellipsis (e.g. by a large intermediate struct) we generated wrong code. See testcase.
This commit is contained in:
parent
53c5fc2246
commit
7cd1ae7710
@ -558,6 +558,36 @@ static int stdarg_test(void) {
|
|||||||
return run_callback(src, stdarg_test_callback);
|
return run_callback(src, stdarg_test_callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
typedef struct {long long a, b;} stdarg_many_test_struct_type;
|
||||||
|
typedef void (*stdarg_many_test_function_type) (int, int, int, int, int,
|
||||||
|
stdarg_many_test_struct_type,
|
||||||
|
int, int, ...);
|
||||||
|
|
||||||
|
static int stdarg_many_test_callback(void *ptr)
|
||||||
|
{
|
||||||
|
stdarg_many_test_function_type f = (stdarg_many_test_function_type)ptr;
|
||||||
|
int x;
|
||||||
|
stdarg_many_test_struct_type l = {10, 11};
|
||||||
|
f(1, 2, 3, 4, 5, l, 6, 7, &x, 44);
|
||||||
|
return x == 44 ? 0 : -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int stdarg_many_test(void)
|
||||||
|
{
|
||||||
|
const char *src =
|
||||||
|
"#include <stdarg.h>\n"
|
||||||
|
"typedef struct {long long a, b;} stdarg_many_test_struct_type;\n"
|
||||||
|
"void f (int a, int b, int c, int d, int e, stdarg_many_test_struct_type l, int f, int g, ...){\n"
|
||||||
|
" va_list ap;\n"
|
||||||
|
" int *p;\n"
|
||||||
|
" va_start (ap, g);\n"
|
||||||
|
" p = va_arg(ap, int*);\n"
|
||||||
|
" *p = va_arg(ap, int);\n"
|
||||||
|
" va_end (ap);\n"
|
||||||
|
"}\n";
|
||||||
|
return run_callback(src, stdarg_many_test_callback);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Test Win32 stdarg handling, since the calling convention will pass a pointer
|
* 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.
|
* to the struct and the stdarg pointer must point to that pointer initially.
|
||||||
@ -654,6 +684,7 @@ int main(int argc, char **argv) {
|
|||||||
RUN_TEST(many_struct_test_2);
|
RUN_TEST(many_struct_test_2);
|
||||||
RUN_TEST(many_struct_test_3);
|
RUN_TEST(many_struct_test_3);
|
||||||
RUN_TEST(stdarg_test);
|
RUN_TEST(stdarg_test);
|
||||||
|
RUN_TEST(stdarg_many_test);
|
||||||
RUN_TEST(stdarg_struct_test);
|
RUN_TEST(stdarg_struct_test);
|
||||||
RUN_TEST(arg_align_test);
|
RUN_TEST(arg_align_test);
|
||||||
return retval;
|
return retval;
|
||||||
|
14
x86_64-gen.c
14
x86_64-gen.c
@ -1423,21 +1423,15 @@ void gfunc_prolog(CType *func_type)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case x86_64_mode_integer:
|
case x86_64_mode_integer:
|
||||||
if (seen_reg_num + reg_count <= 8) {
|
if (seen_reg_num + reg_count > REGN)
|
||||||
seen_reg_num += reg_count;
|
|
||||||
} else {
|
|
||||||
seen_reg_num = 8;
|
|
||||||
goto stack_arg;
|
goto stack_arg;
|
||||||
}
|
seen_reg_num += reg_count;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case x86_64_mode_sse:
|
case x86_64_mode_sse:
|
||||||
if (seen_sse_num + reg_count <= 8) {
|
if (seen_sse_num + reg_count > 8)
|
||||||
seen_sse_num += reg_count;
|
|
||||||
} else {
|
|
||||||
seen_sse_num = 8;
|
|
||||||
goto stack_arg;
|
goto stack_arg;
|
||||||
}
|
seen_sse_num += reg_count;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user