float parsing + long double
This commit is contained in:
		
							parent
							
								
									3917389449
								
							
						
					
					
						commit
						25618c0430
					
				
					 2 changed files with 535 additions and 159 deletions
				
			
		
							
								
								
									
										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…
	
	Add table
		
		Reference in a new issue