mirror of
https://github.com/mirror/tinycc.git
synced 2025-01-29 06:10:09 +08:00
x86_64: Add -mno-sse option
This disables generation of any SSE instructions (in particular in stdarg function prologues). Necessary for kernel compiles.
This commit is contained in:
parent
b5669a952b
commit
ad723a419f
15
libtcc.c
15
libtcc.c
@ -1303,6 +1303,16 @@ static int tcc_set_flag(TCCState *s, const char *flag_name, int value)
|
|||||||
flag_name, value);
|
flag_name, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const FlagDef m_defs[] = {
|
||||||
|
#ifdef TCC_TARGET_X86_64
|
||||||
|
{ offsetof(TCCState, nosse), FD_INVERT, "sse" },
|
||||||
|
#endif
|
||||||
|
{ 0, 0, " no flag" },
|
||||||
|
};
|
||||||
|
static int tcc_set_m_flag(TCCState *s, const char *flag_name, int value)
|
||||||
|
{
|
||||||
|
return set_flag(s, m_defs, countof(m_defs), flag_name, value);
|
||||||
|
}
|
||||||
|
|
||||||
static int strstart(const char *val, const char **str)
|
static int strstart(const char *val, const char **str)
|
||||||
{
|
{
|
||||||
@ -1778,9 +1788,10 @@ reparse:
|
|||||||
s->soname = tcc_strdup(optarg);
|
s->soname = tcc_strdup(optarg);
|
||||||
break;
|
break;
|
||||||
case TCC_OPTION_m:
|
case TCC_OPTION_m:
|
||||||
if (strcmp(optarg, "32") && strcmp(optarg, "64"))
|
if (!strcmp(optarg, "32") || !strcmp(optarg, "64"))
|
||||||
|
s->option_m = tcc_strdup(optarg);
|
||||||
|
else if (tcc_set_m_flag(s, optarg, 1) < 0)
|
||||||
goto unsupported_option;
|
goto unsupported_option;
|
||||||
s->option_m = tcc_strdup(optarg);
|
|
||||||
break;
|
break;
|
||||||
case TCC_OPTION_o:
|
case TCC_OPTION_o:
|
||||||
if (s->outfile) {
|
if (s->outfile) {
|
||||||
|
3
tcc.h
3
tcc.h
@ -643,6 +643,9 @@ struct TCCState {
|
|||||||
#ifdef TCC_TARGET_I386
|
#ifdef TCC_TARGET_I386
|
||||||
int seg_size; /* 32. Can be 16 with i386 assembler (.code16) */
|
int seg_size; /* 32. Can be 16 with i386 assembler (.code16) */
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef TCC_TARGET_X86_64
|
||||||
|
int nosse; /* For -mno-sse support. */
|
||||||
|
#endif
|
||||||
|
|
||||||
/* array of all loaded dlls (including those referenced by loaded dlls) */
|
/* array of all loaded dlls (including those referenced by loaded dlls) */
|
||||||
DLLReference **loaded_dlls;
|
DLLReference **loaded_dlls;
|
||||||
|
15
x86_64-gen.c
15
x86_64-gen.c
@ -851,6 +851,8 @@ void gfunc_call(int nb_args)
|
|||||||
struct_size += size;
|
struct_size += size;
|
||||||
} else {
|
} else {
|
||||||
if (is_sse_float(vtop->type.t)) {
|
if (is_sse_float(vtop->type.t)) {
|
||||||
|
if (tcc_state->nosse)
|
||||||
|
tcc_error("SSE disabled");
|
||||||
gv(RC_XMM0); /* only use one float register */
|
gv(RC_XMM0); /* only use one float register */
|
||||||
if (arg >= REGN) {
|
if (arg >= REGN) {
|
||||||
/* movq %xmm0, j*8(%rsp) */
|
/* movq %xmm0, j*8(%rsp) */
|
||||||
@ -961,6 +963,8 @@ void gfunc_prolog(CType *func_type)
|
|||||||
if (reg_param_index < REGN) {
|
if (reg_param_index < REGN) {
|
||||||
/* save arguments passed by register */
|
/* save arguments passed by register */
|
||||||
if ((bt == VT_FLOAT) || (bt == VT_DOUBLE)) {
|
if ((bt == VT_FLOAT) || (bt == VT_DOUBLE)) {
|
||||||
|
if (tcc_state->nosse)
|
||||||
|
tcc_error("SSE disabled");
|
||||||
o(0xd60f66); /* movq */
|
o(0xd60f66); /* movq */
|
||||||
gen_modrm(reg_param_index, VT_LOCAL, NULL, addr);
|
gen_modrm(reg_param_index, VT_LOCAL, NULL, addr);
|
||||||
} else {
|
} else {
|
||||||
@ -1208,6 +1212,9 @@ void gfunc_call(int nb_args)
|
|||||||
nb_reg_args += reg_count;
|
nb_reg_args += reg_count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (nb_sse_args && tcc_state->nosse)
|
||||||
|
tcc_error("SSE disabled but floating point arguments passed");
|
||||||
|
|
||||||
/* arguments are collected in runs. Each run is a collection of 8-byte aligned arguments
|
/* arguments are collected in runs. Each run is a collection of 8-byte aligned arguments
|
||||||
and ended by a 16-byte aligned argument. This is because, from the point of view of
|
and ended by a 16-byte aligned argument. This is because, from the point of view of
|
||||||
the callee, argument alignment is computed from the bottom up. */
|
the callee, argument alignment is computed from the bottom up. */
|
||||||
@ -1541,8 +1548,10 @@ void gfunc_prolog(CType *func_type)
|
|||||||
/* save all register passing arguments */
|
/* save all register passing arguments */
|
||||||
for (i = 0; i < 8; i++) {
|
for (i = 0; i < 8; i++) {
|
||||||
loc -= 16;
|
loc -= 16;
|
||||||
o(0xd60f66); /* movq */
|
if (!tcc_state->nosse) {
|
||||||
gen_modrm(7 - i, VT_LOCAL, NULL, loc);
|
o(0xd60f66); /* movq */
|
||||||
|
gen_modrm(7 - i, VT_LOCAL, NULL, loc);
|
||||||
|
}
|
||||||
/* movq $0, loc+8(%rbp) */
|
/* movq $0, loc+8(%rbp) */
|
||||||
o(0x85c748);
|
o(0x85c748);
|
||||||
gen_le32(loc + 8);
|
gen_le32(loc + 8);
|
||||||
@ -1572,6 +1581,8 @@ void gfunc_prolog(CType *func_type)
|
|||||||
mode = classify_x86_64_arg(type, NULL, &size, &align, ®_count);
|
mode = classify_x86_64_arg(type, NULL, &size, &align, ®_count);
|
||||||
switch (mode) {
|
switch (mode) {
|
||||||
case x86_64_mode_sse:
|
case x86_64_mode_sse:
|
||||||
|
if (tcc_state->nosse)
|
||||||
|
tcc_error("SSE disabled but floating point arguments used");
|
||||||
if (sse_param_index + reg_count <= 8) {
|
if (sse_param_index + reg_count <= 8) {
|
||||||
/* save arguments passed by register */
|
/* save arguments passed by register */
|
||||||
loc -= reg_count * 8;
|
loc -= reg_count * 8;
|
||||||
|
Loading…
Reference in New Issue
Block a user