mirror of
https://github.com/mirror/tinycc.git
synced 2025-04-01 12:30:08 +08:00
float parsing + long double
This commit is contained in:
parent
3917389449
commit
25618c0430
58
i386-gen.c
58
i386-gen.c
@ -1,6 +1,22 @@
|
|||||||
/******************************************************/
|
/*
|
||||||
/* X86 code generator */
|
* X86 code generator for TCC
|
||||||
|
*
|
||||||
|
* Copyright (c) 2001 Fabrice Bellard
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
/* number of available registers */
|
/* number of available registers */
|
||||||
#define NB_REGS 4
|
#define NB_REGS 4
|
||||||
@ -38,6 +54,13 @@ int reg_classes[NB_REGS] = {
|
|||||||
are directly pushed on stack. */
|
are directly pushed on stack. */
|
||||||
//#define FUNC_STRUCT_PARAM_AS_PTR
|
//#define FUNC_STRUCT_PARAM_AS_PTR
|
||||||
|
|
||||||
|
/* pointer size, in bytes */
|
||||||
|
#define PTR_SIZE 4
|
||||||
|
|
||||||
|
/* long double size and alignment, in bytes */
|
||||||
|
#define LDOUBLE_SIZE 12
|
||||||
|
#define LDOUBLE_ALIGN 4
|
||||||
|
|
||||||
/* function call context */
|
/* function call context */
|
||||||
typedef struct GFuncContext {
|
typedef struct GFuncContext {
|
||||||
int args_size;
|
int args_size;
|
||||||
@ -161,6 +184,9 @@ void load(int r, int ft, int fc)
|
|||||||
} else if ((ft & VT_BTYPE) == VT_DOUBLE) {
|
} else if ((ft & VT_BTYPE) == VT_DOUBLE) {
|
||||||
o(0xdd); /* fldl */
|
o(0xdd); /* fldl */
|
||||||
r = 0;
|
r = 0;
|
||||||
|
} else if ((ft & VT_BTYPE) == VT_LDOUBLE) {
|
||||||
|
o(0xdb); /* fldt */
|
||||||
|
r = 5;
|
||||||
} else if ((ft & VT_TYPE) == VT_BYTE)
|
} else if ((ft & VT_TYPE) == VT_BYTE)
|
||||||
o(0xbe0f); /* movsbl */
|
o(0xbe0f); /* movsbl */
|
||||||
else if ((ft & VT_TYPE) == (VT_BYTE | VT_UNSIGNED))
|
else if ((ft & VT_TYPE) == (VT_BYTE | VT_UNSIGNED))
|
||||||
@ -214,12 +240,16 @@ void store(r, ft, fc)
|
|||||||
fr = ft & VT_VALMASK;
|
fr = ft & VT_VALMASK;
|
||||||
bt = ft & VT_BTYPE;
|
bt = ft & VT_BTYPE;
|
||||||
/* XXX: incorrect if reg to reg */
|
/* XXX: incorrect if reg to reg */
|
||||||
|
/* XXX: should not flush float stack */
|
||||||
if (bt == VT_FLOAT) {
|
if (bt == VT_FLOAT) {
|
||||||
o(0xd9); /* fstps */
|
o(0xd9); /* fstps */
|
||||||
r = 3;
|
r = 3;
|
||||||
} else if (bt == VT_DOUBLE) {
|
} else if (bt == VT_DOUBLE) {
|
||||||
o(0xdd); /* fstpl */
|
o(0xdd); /* fstpl */
|
||||||
r = 3;
|
r = 3;
|
||||||
|
} else if (bt == VT_LDOUBLE) {
|
||||||
|
o(0xdb); /* fstpt */
|
||||||
|
r = 7;
|
||||||
} else {
|
} else {
|
||||||
if (bt == VT_SHORT)
|
if (bt == VT_SHORT)
|
||||||
o(0x66);
|
o(0x66);
|
||||||
@ -269,17 +299,24 @@ void gfunc_param(GFuncContext *c)
|
|||||||
vc = fc;
|
vc = fc;
|
||||||
vstore();
|
vstore();
|
||||||
c->args_size += size;
|
c->args_size += size;
|
||||||
} else if ((vt & VT_BTYPE) == VT_DOUBLE ||
|
} else if ((vt & VT_BTYPE) == VT_LDOUBLE ||
|
||||||
|
(vt & VT_BTYPE) == VT_DOUBLE ||
|
||||||
(vt & VT_BTYPE) == VT_FLOAT) {
|
(vt & VT_BTYPE) == VT_FLOAT) {
|
||||||
gv(); /* only one float register */
|
gv(); /* only one float register */
|
||||||
if ((vt & VT_BTYPE) == VT_FLOAT)
|
if ((vt & VT_BTYPE) == VT_FLOAT)
|
||||||
size = 4;
|
size = 4;
|
||||||
else
|
else if ((vt & VT_BTYPE) == VT_DOUBLE)
|
||||||
size = 8;
|
size = 8;
|
||||||
|
else
|
||||||
|
size = 12;
|
||||||
oad(0xec81, size); /* sub $xxx, %esp */
|
oad(0xec81, size); /* sub $xxx, %esp */
|
||||||
o(0x245cd9 + size - 4); /* fstp[s|l] 0(%esp) */
|
if (size == 12)
|
||||||
|
o(0x7cdb);
|
||||||
|
else
|
||||||
|
o(0x5cd9 + size - 4); /* fstp[s|l] 0(%esp) */
|
||||||
|
g(0x24);
|
||||||
g(0x00);
|
g(0x00);
|
||||||
c->args_size += 8;
|
c->args_size += size;
|
||||||
} else {
|
} else {
|
||||||
/* simple type (currently always same size) */
|
/* simple type (currently always same size) */
|
||||||
/* XXX: implicit cast ? */
|
/* XXX: implicit cast ? */
|
||||||
@ -345,6 +382,7 @@ int gtst(int inv, int t)
|
|||||||
if ((vc != 0) != inv)
|
if ((vc != 0) != inv)
|
||||||
t = gjmp(t);
|
t = gjmp(t);
|
||||||
} else {
|
} else {
|
||||||
|
/* XXX: floats */
|
||||||
v = gv();
|
v = gv();
|
||||||
o(0x85);
|
o(0x85);
|
||||||
o(0xc0 + v * 9);
|
o(0xc0 + v * 9);
|
||||||
@ -499,7 +537,7 @@ void gen_opf(int op)
|
|||||||
o(0x05 + a);
|
o(0x05 + a);
|
||||||
gen_addr32(fc, ft);
|
gen_addr32(fc, ft);
|
||||||
} else if (r == VT_LOCAL) {
|
} else if (r == VT_LOCAL) {
|
||||||
oad(0x85 + a, fc);
|
oad(0x85 + a, fc);
|
||||||
} else {
|
} else {
|
||||||
g(0x00 + a + r);
|
g(0x00 + a + r);
|
||||||
}
|
}
|
||||||
@ -512,14 +550,14 @@ void gen_opf(int op)
|
|||||||
void gen_cvtf(int t)
|
void gen_cvtf(int t)
|
||||||
{
|
{
|
||||||
if ((vt & (VT_BTYPE | VT_UNSIGNED)) == (VT_INT | VT_UNSIGNED)) {
|
if ((vt & (VT_BTYPE | VT_UNSIGNED)) == (VT_INT | VT_UNSIGNED)) {
|
||||||
/* unsigned int to float/double */
|
/* unsigned int to float/double/long double */
|
||||||
o(0x6a); /* push $0 */
|
o(0x6a); /* push $0 */
|
||||||
g(0x00);
|
g(0x00);
|
||||||
o(0x50 + (vt & VT_VALMASK)); /* push r */
|
o(0x50 + (vt & VT_VALMASK)); /* push r */
|
||||||
o(0x242cdf); /* fildll (%esp) */
|
o(0x242cdf); /* fildll (%esp) */
|
||||||
o(0x08c483); /* add $8, %esp */
|
o(0x08c483); /* add $8, %esp */
|
||||||
} else {
|
} else {
|
||||||
/* int to float/double */
|
/* int to float/double/long double */
|
||||||
o(0x50 + (vt & VT_VALMASK)); /* push r */
|
o(0x50 + (vt & VT_VALMASK)); /* push r */
|
||||||
o(0x2404db); /* fildl (%esp) */
|
o(0x2404db); /* fildl (%esp) */
|
||||||
o(0x04c483); /* add $4, %esp */
|
o(0x04c483); /* add $4, %esp */
|
||||||
|
Loading…
Reference in New Issue
Block a user