e36d739fa4
--HG-- branch : dtrg-videocore
1560 lines
34 KiB
Text
1560 lines
34 KiB
Text
/*
|
|
* VideoCore IV code generator for the ACK
|
|
* © 2013 David Given
|
|
* This file is redistributable under the terms of the 3-clause BSD license.
|
|
* See the file 'Copying' in the root of the distribution for the full text.
|
|
*/
|
|
|
|
EM_WSIZE = 4
|
|
EM_PSIZE = 4
|
|
EM_BSIZE = 8 /* two words saved in call frame */
|
|
|
|
BYTE = 1 /* Size of values */
|
|
WORD = 2
|
|
QUAD = 4
|
|
|
|
FP_OFFSET = 0 /* Offset of saved FP relative to our FP */
|
|
PC_OFFSET = 4 /* Offset of saved PC relative to our FP */
|
|
|
|
#define COMMENT(n) /* noop */
|
|
|
|
|
|
#define nicesize(x) ((x)==BYTE || (x)==WORD || (x)==QUAD)
|
|
|
|
|
|
|
|
PROPERTIES
|
|
|
|
GPR /* any GPR */
|
|
REG /* any allocatable GPR */
|
|
STACKABLE /* a register than can be used with push/pop */
|
|
|
|
GPR0 GPR1 GPR2 GPR3 GPR4 GPR5 GPR6 GPR7
|
|
GPR8 GPR9 GPR10 GPR11 GPR12 GPR13 GPR14 GPR15
|
|
GPR16 GPR17 GPR18 GPR19 GPR20 GPR21 GPR22 GPR23
|
|
|
|
GPRGP GPRFP GPRSP GPRLR GPRPC
|
|
|
|
REGISTERS
|
|
|
|
R0("r0") : GPR, REG, GPR0, STACKABLE.
|
|
R1("r1") : GPR, REG, GPR1.
|
|
R2("r2") : GPR, REG, GPR2.
|
|
R3("r3") : GPR, REG, GPR3.
|
|
R4("r4") : GPR, REG, GPR4.
|
|
R5("r5") : GPR, REG, GPR5.
|
|
R6("r6") : GPR, GPR6.
|
|
R7("r7") : GPR, REG, GPR7.
|
|
R8("r8") : GPR, REG, GPR8.
|
|
R9("r9") : GPR, REG, GPR9.
|
|
R10("r10") : GPR, REG, GPR10.
|
|
R11("r11") : GPR, REG, GPR11.
|
|
R12("r12") : GPR, REG, GPR12.
|
|
R13("r13") : GPR, REG, GPR13.
|
|
R14("r14") : GPR, REG, GPR14.
|
|
GP("r15") : GPR, GPRGP.
|
|
|
|
R16("r16") : GPR, GPR16.
|
|
|
|
R23("r23") : GPR.
|
|
FP("fp") : GPR, GPRFP.
|
|
SP("sp") : GPR, GPRSP.
|
|
LR("lr") : GPR, GPRLR.
|
|
PC("pc") : GPR, GPRPC.
|
|
/* r26 to r31 are special and the code generator doesn't touch them. */
|
|
|
|
#define SCRATCH R6
|
|
|
|
TOKENS
|
|
|
|
/* Used only in instruction descriptions (to generate the correct syntax). */
|
|
|
|
GPROFFSET = { GPR reg; INT off; } 4 off "(" reg ")".
|
|
GPRGPR = { GPR reg1; GPR reg2; } 4 "(" reg1 "," reg2 ")".
|
|
GPRINC = { GPR reg; } 4 "(" reg ")++".
|
|
ADDCMPB_LL = { GPR rd; INT val; INT vs; ADDR dest; } 4 rd ",#" val ",#" vs "," dest.
|
|
|
|
/* Primitives */
|
|
|
|
LABEL = { ADDR adr; } 4 adr.
|
|
CONST = { INT val; } 4 "#" val.
|
|
|
|
/* Sign extended values. */
|
|
|
|
/* The size refers to the *source*. */
|
|
SIGNEX8 = { GPR reg; } 4 reg.
|
|
SIGNEX16 = { GPR reg; } 4 reg.
|
|
|
|
/* The results of comparisons. */
|
|
|
|
TRISTATE_RC_S = { GPR reg; INT val; } 4.
|
|
TRISTATE_RC_U = { GPR reg; INT val; } 4.
|
|
TRISTATE_RR_S = { GPR reg1; GPR reg2; } 4.
|
|
TRISTATE_RR_U = { GPR reg1; GPR reg2; } 4.
|
|
|
|
|
|
|
|
SETS
|
|
|
|
TOKEN = LABEL + CONST.
|
|
OP = TOKEN + SIGNEX8 + SIGNEX16.
|
|
ANY = GPR + OP.
|
|
|
|
|
|
|
|
INSTRUCTIONS
|
|
|
|
add GPR:wo, GPR:ro, GPR+CONST:ro.
|
|
add GPR:rw, GPR+CONST:ro.
|
|
addcmpbge "addcmpb.ge" ADDCMPB_LL:rw.
|
|
adds2 GPR:rw, GPR+CONST:ro.
|
|
adds4 GPR:rw, GPR+CONST:ro.
|
|
adds8 GPR:rw, GPR+CONST:ro.
|
|
adds16 GPR:rw, GPR+CONST:ro.
|
|
adds256 GPR:rw, GPR:rw, GPR:ro.
|
|
and GPR:rw, GPR+CONST:ro.
|
|
asr GPR:rw, GPR+CONST:ro.
|
|
beq "b.eq" LABEL:ro.
|
|
bne "b.ne" LABEL:ro.
|
|
bgt "b.gt" LABEL:ro.
|
|
blt "b.lt" LABEL:ro.
|
|
bhi "b.hi" LABEL:ro.
|
|
bset GPR:rw, GPR+CONST:ro.
|
|
b GPR+LABEL:ro.
|
|
bl GPR+LABEL:ro.
|
|
cmp GPR:ro, GPR+CONST:ro kills :cc.
|
|
divs GPR:wo, GPR:ro, GPR+CONST:ro.
|
|
divu GPR:wo, GPR:ro, GPR+CONST:ro.
|
|
eor GPR:rw, GPR+CONST:ro.
|
|
exts GPR:wo, GPR:ro, GPR+CONST:ro.
|
|
exts GPR:rw, GPR+CONST:ro.
|
|
fadd GPR:wo, GPR:ro, GPR:ro.
|
|
fcmp GPR:wo, GPR:ro, GPR:ro.
|
|
fdiv GPR:wo, GPR:ro, GPR:ro.
|
|
flts GPR:wo, GPR:ro.
|
|
fltu GPR:wo, GPR:ro.
|
|
fmul GPR:wo, GPR:ro, GPR:ro.
|
|
fsub GPR:wo, GPR:ro, GPR:ro.
|
|
ftrunc GPR:wo, GPR:ro.
|
|
ld GPR:wo, GPRINC:rw.
|
|
ld GPR:wo, GPROFFSET+GPRGPR+LABEL:ro.
|
|
ldb GPR:wo, GPROFFSET+GPRGPR+LABEL:ro.
|
|
ldh GPR:wo, GPROFFSET+GPRGPR+LABEL:ro.
|
|
ldhs GPR:wo, GPROFFSET+GPRGPR+LABEL:ro.
|
|
lea GPR:wo, LABEL:ro.
|
|
lsl GPR:rw, GPR+CONST:ro.
|
|
lsl GPR:wo, GPR:ro, GPR+CONST:ro.
|
|
lsr GPR:rw, GPR+CONST:ro.
|
|
mov GPR:wo, GPR+CONST:ro.
|
|
mul GPR:rw, GPR+CONST:ro.
|
|
mvn GPR:wo, GPR+CONST:ro.
|
|
neg GPR:rw, GPR+CONST:ro.
|
|
or GPR:rw, GPR+CONST:ro.
|
|
pop GPR0+GPR6+GPR16+GPRFP+GPRPC:wo.
|
|
pop GPR0+GPR6+GPR16+GPRFP:wo, GPRPC:wo.
|
|
push GPR0+GPR6+GPR16+GPRFP+GPRLR:ro.
|
|
push GPR0+GPR6+GPR16+GPRFP:ro, GPRLR:ro.
|
|
rsb GPR:rw, GPR+CONST:ro.
|
|
sub GPR:wo, GPR:ro, CONST+GPR:ro.
|
|
sub GPR:rw, GPR+CONST:ro.
|
|
st GPR:ro, GPRINC:rw.
|
|
st GPR:ro, GPROFFSET+GPRGPR+LABEL:ro.
|
|
stb GPR:ro, GPROFFSET+GPRGPR+LABEL:ro.
|
|
sth GPR:ro, GPROFFSET+GPRGPR+LABEL:ro.
|
|
sths GPR:ro, GPROFFSET+GPRGPR+LABEL:ro.
|
|
|
|
invalid "invalid".
|
|
comment "!" LABEL:ro.
|
|
|
|
|
|
|
|
MOVES
|
|
|
|
from GPR to GPR
|
|
gen
|
|
COMMENT("mov GPR->GPR")
|
|
mov %2, %1
|
|
|
|
/* Constants */
|
|
|
|
from CONST to GPR
|
|
gen
|
|
mov %2, %1
|
|
|
|
from LABEL to GPR
|
|
gen
|
|
lea %2, {LABEL, %1.adr}
|
|
sub %2, GP
|
|
|
|
/* Sign extension */
|
|
|
|
from SIGNEX8 to GPR
|
|
gen
|
|
exts %2, %1.reg, {CONST, 8}
|
|
|
|
from SIGNEX16 to GPR
|
|
gen
|
|
exts %2, %1.reg, {CONST, 16}
|
|
|
|
/* Miscellaneous */
|
|
|
|
from CONST+LABEL+GPR to GPR
|
|
gen
|
|
move %1, %2
|
|
|
|
|
|
TESTS
|
|
|
|
to test GPR
|
|
gen
|
|
cmp %1, {CONST, 0}
|
|
|
|
|
|
|
|
STACKINGRULES
|
|
|
|
from GPR0+GPR6+GPR16 to STACK
|
|
gen
|
|
comment {LABEL, "push stackable"}
|
|
push %1
|
|
|
|
from OP+GPR to STACK
|
|
uses GPR0
|
|
gen
|
|
move %1, %a
|
|
push %a
|
|
|
|
from OP to STACK
|
|
uses STACKABLE
|
|
gen
|
|
move %1, %a
|
|
push %a
|
|
|
|
from OP+GPR to STACK
|
|
gen
|
|
comment {LABEL, "push via scratch"}
|
|
move %1, SCRATCH
|
|
push SCRATCH
|
|
|
|
|
|
|
|
|
|
COERCIONS
|
|
|
|
from OP
|
|
uses REG
|
|
gen
|
|
move %1, %a
|
|
yields %a
|
|
|
|
from STACK
|
|
uses REG
|
|
gen
|
|
pop SCRATCH
|
|
move SCRATCH, %a
|
|
yields %a
|
|
|
|
|
|
|
|
PATTERNS
|
|
|
|
/* Intrinsics */
|
|
|
|
pat nop /* Does nothing */
|
|
|
|
pat loc /* Load constant */
|
|
yields {CONST, $1}
|
|
|
|
pat dup $1<=QUAD /* Duplicate word on top of stack */
|
|
with ANY
|
|
yields %1 %1
|
|
|
|
pat dup $1==(2*QUAD) /* Duplicate word pair on top of stack */
|
|
with ANY ANY
|
|
yields %1 %2 %1 %2
|
|
|
|
pat exg $1<=QUAD /* Exchange top two words on stack */
|
|
with ANY ANY
|
|
yields %1 %2
|
|
|
|
pat exg $1==(2*QUAD) /* Exchange top two word pairs on stack */
|
|
with ANY ANY ANY ANY
|
|
yields %2 %1 %4 %3
|
|
|
|
pat stl lol $1==$2 /* Store then load local */
|
|
leaving
|
|
dup QUAD
|
|
stl $1
|
|
|
|
pat lal sti lal loi $1==$3 && $2==$4 /* Store then load local, of a different size */
|
|
leaving
|
|
dup $2
|
|
lal $1
|
|
sti $2
|
|
|
|
pat ste loe $1==$2 /* Store then load external */
|
|
leaving
|
|
dup QUAD
|
|
ste $1
|
|
|
|
|
|
|
|
/* Type conversions */
|
|
|
|
pat loc loc cii loc loc cii $1==$4 && $2==$5 /* madness, generated by the C compiler */
|
|
leaving
|
|
loc $1
|
|
loc $2
|
|
cii
|
|
|
|
pat loc loc cii loc loc cii $2==QUAD && $5==QUAD && $4<$2 /* madness, generated by the C compiler */
|
|
leaving
|
|
loc $4
|
|
loc $5
|
|
cii
|
|
|
|
pat loc loc ciu /* signed X -> unsigned X */
|
|
leaving
|
|
loc $1
|
|
loc $2
|
|
cuu
|
|
|
|
pat loc loc cuu $1==$2 /* unsigned X -> unsigned X */
|
|
/* nop */
|
|
|
|
pat loc loc cii $1==$2 /* signed X -> signed X */
|
|
/* nop */
|
|
|
|
pat loc loc cui $1==$2 /* unsigned X -> signed X */
|
|
/* nop */
|
|
|
|
pat loc loc cui $1==BYTE && $2==QUAD /* unsigned char -> signed int */
|
|
/* nop */
|
|
|
|
pat loc loc cui $1==WORD && $2==QUAD /* unsigned short -> signed int */
|
|
/* nop */
|
|
|
|
pat loc loc cii $1==BYTE && $2>BYTE /* signed char -> anything */
|
|
with GPR
|
|
yields {SIGNEX8, %1}
|
|
with SIGNEX8
|
|
yields {SIGNEX8, %1.reg}
|
|
with SIGNEX16
|
|
yields {SIGNEX8, %1.reg}
|
|
|
|
pat loc loc cii $1==WORD && $2>WORD /* signed short -> anything */
|
|
with GPR
|
|
yields {SIGNEX16, %1}
|
|
with SIGNEX8
|
|
yields {SIGNEX16, %1.reg}
|
|
with SIGNEX16
|
|
yields {SIGNEX16, %1.reg}
|
|
|
|
|
|
|
|
/* Local variables */
|
|
|
|
pat lal /* Load address of local */
|
|
uses REG
|
|
gen
|
|
sub %a, FP, GP
|
|
add %a, {CONST, $1}
|
|
yields %a
|
|
|
|
pat lol /* Load quad from local */
|
|
uses REG
|
|
gen
|
|
ld %a, {GPROFFSET, FP, $1}
|
|
yields %a
|
|
|
|
pat ldl /* Load double-word from local */
|
|
leaving
|
|
lol $1 + QUAD*1
|
|
lol $1 + QUAD*0
|
|
|
|
pat stl /* Store to local */
|
|
with GPR
|
|
gen
|
|
st %1, {GPROFFSET, FP, $1}
|
|
|
|
pat sdl /* Store double-word to local */
|
|
leaving
|
|
stl $1 + QUAD*0
|
|
stl $1 + QUAD*1
|
|
|
|
pat lil /* Load from indirected local */
|
|
leaving
|
|
lol $1
|
|
loi QUAD
|
|
|
|
pat sil /* Save to indirected local */
|
|
leaving
|
|
lol $1
|
|
sti QUAD
|
|
|
|
pat stl lol $1==$2 /* Save then load (generated by C compiler) */
|
|
leaving
|
|
dup QUAD
|
|
stl $1
|
|
|
|
pat zrl /* Zero local */
|
|
leaving
|
|
loc 0
|
|
stl $1
|
|
|
|
pat inl /* Increment local in register */
|
|
leaving
|
|
lol $1
|
|
loc 1
|
|
adi QUAD
|
|
stl $1
|
|
|
|
pat del /* Decrement local in register */
|
|
leaving
|
|
lol $1
|
|
loc 1
|
|
sbi QUAD
|
|
stl $1
|
|
|
|
|
|
|
|
/* Global variables */
|
|
|
|
pat lpi /* Load address of external function */
|
|
leaving
|
|
lae $1
|
|
|
|
pat lae /* Load address of external */
|
|
yields {LABEL, $1}
|
|
|
|
pat loe /* Load word external */
|
|
leaving
|
|
lae $1
|
|
loi QUAD
|
|
|
|
pat ste /* Store word external */
|
|
leaving
|
|
lae $1
|
|
sti QUAD
|
|
|
|
pat zre /* Zero external */
|
|
leaving
|
|
loc 0
|
|
ste $1
|
|
|
|
pat ine /* Increment external */
|
|
leaving
|
|
loe $1
|
|
inc
|
|
ste $1
|
|
|
|
pat dee /* Decrement external */
|
|
leaving
|
|
loe $1
|
|
dec
|
|
ste $1
|
|
|
|
pat lde /* Load double external */
|
|
leaving
|
|
lae $1
|
|
loi QUAD*2
|
|
|
|
pat sde /* Store double external */
|
|
leaving
|
|
lae $1
|
|
sti QUAD*2
|
|
|
|
|
|
/* Structures */
|
|
|
|
pat lof /* Load word offsetted */
|
|
leaving
|
|
adp $1
|
|
loi QUAD
|
|
|
|
pat ldf /* Load double offsetted */
|
|
with GPR
|
|
uses reusing %1, REG=%1, REG
|
|
gen
|
|
add %a, GP
|
|
ld %b, {GPROFFSET, %a, $1+4}
|
|
ld %a, {GPROFFSET, %a, $1+0}
|
|
yields %a %b
|
|
|
|
pat stf /* Store word offsetted */
|
|
leaving
|
|
adp $1
|
|
sti QUAD
|
|
|
|
pat sdf /* Store double offsetted */
|
|
with GPR GPR GPR
|
|
uses reusing %3, REG=%3
|
|
gen
|
|
add %a, GP
|
|
st %1, {GPROFFSET, %a, $1+0}
|
|
st %2, {GPROFFSET, %a, $1+4}
|
|
|
|
|
|
|
|
|
|
/* Loads and stores */
|
|
|
|
pat loi $1==BYTE /* Load byte indirect */
|
|
with LABEL
|
|
uses REG
|
|
gen
|
|
ldb %a, %1
|
|
yields %a
|
|
with GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
ldb %a, {GPRGPR, %1, GP}
|
|
yields %a
|
|
|
|
pat loi loc loc cii $1==WORD && $2==WORD && $3==QUAD /* Load short indirect and sign extend */
|
|
with LABEL
|
|
uses REG
|
|
gen
|
|
ldhs %a, %1
|
|
yields %a
|
|
with GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
add %a, %1, GP
|
|
ldhs %a, {GPROFFSET, %a, 0}
|
|
yields %a
|
|
|
|
pat loi $1==WORD /* Load short indirect */
|
|
with LABEL
|
|
uses REG
|
|
gen
|
|
ldh %a, %1
|
|
yields %a
|
|
with GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
add %a, %1, GP
|
|
ldh %a, {GPROFFSET, %a, 0}
|
|
yields %a
|
|
|
|
pat loi $1==QUAD /* Load quad indirect */
|
|
with LABEL
|
|
uses REG
|
|
gen
|
|
ld %a, %1
|
|
yields %a
|
|
with GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
add %a, %1, GP
|
|
ld %a, {GPROFFSET, %a, 0}
|
|
yields %a
|
|
|
|
pat loi $1==2*QUAD /* Load double-quad indirect */
|
|
with LABEL
|
|
uses REG, REG
|
|
gen
|
|
lea %b, %1
|
|
ld %a, {GPROFFSET, %b, 0}
|
|
ld %b, {GPROFFSET, %b, 4}
|
|
yields %b %a
|
|
with GPR
|
|
uses reusing %1, REG, REG
|
|
gen
|
|
add %b, %1, GP
|
|
ld %a, {GPROFFSET, %b, 0}
|
|
ld %b, {GPROFFSET, %b, 4}
|
|
yields %b %a
|
|
|
|
pat loi $1==3*QUAD /* Load triple-quad indirect */
|
|
with LABEL
|
|
uses REG, REG, REG
|
|
gen
|
|
lea %b, %1
|
|
ld %a, {GPROFFSET, %b, 0}
|
|
ld %b, {GPROFFSET, %b, 4}
|
|
ld %b, {GPROFFSET, %b, 8}
|
|
yields %c %b %a
|
|
with GPR
|
|
uses reusing %1, REG, REG, REG
|
|
gen
|
|
add %b, %1, GP
|
|
ld %a, {GPROFFSET, %b, 0}
|
|
ld %b, {GPROFFSET, %b, 4}
|
|
ld %c, {GPROFFSET, %b, 8}
|
|
yields %c %b %a
|
|
|
|
pat loi /* Load arbitrary size */
|
|
leaving
|
|
loc $1
|
|
los QUAD
|
|
|
|
pat los /* Load arbitrary size */
|
|
leaving
|
|
cal ".los"
|
|
|
|
pat sti $1==BYTE /* Store byte indirect */
|
|
with LABEL GPR
|
|
gen
|
|
stb %2, %1
|
|
with LABEL SIGNEX8+SIGNEX16
|
|
gen
|
|
stb %2.reg, %1
|
|
with GPR GPR
|
|
gen
|
|
stb %2, {GPRGPR, %1, GP}
|
|
with GPR SIGNEX8+SIGNEX16
|
|
gen
|
|
stb %2.reg, {GPRGPR, %1, GP}
|
|
|
|
pat sti $1==WORD /* Store half-word indirect */
|
|
with LABEL GPR
|
|
gen
|
|
sth %2, %1
|
|
with LABEL SIGNEX16
|
|
gen
|
|
sth %2.reg, %1
|
|
with GPR GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
add %a, %1, GP
|
|
sth %2, {GPROFFSET, %a, 0}
|
|
with GPR SIGNEX16
|
|
uses reusing %1, REG
|
|
gen
|
|
add %a, %1, GP
|
|
sth %2.reg, {GPROFFSET, %a, 0}
|
|
|
|
pat sti $1==QUAD /* Store quad indirect */
|
|
with LABEL GPR
|
|
gen
|
|
st %2, %1
|
|
with GPR GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
add %a, %1, GP
|
|
st %2, {GPROFFSET, %a, 0}
|
|
|
|
pat sti $1==2*QUAD /* Load double-quad indirect */
|
|
with LABEL GPR GPR
|
|
uses REG
|
|
gen
|
|
lea %a, %1
|
|
st %2, {GPROFFSET, %a, 0}
|
|
st %3, {GPROFFSET, %a, 4}
|
|
with GPR GPR GPR
|
|
uses reusing %1, REG=%1
|
|
gen
|
|
add %a, GP
|
|
st %2, {GPROFFSET, %a, 0}
|
|
st %3, {GPROFFSET, %a, 4}
|
|
|
|
pat sti $1==3*QUAD /* Load triple-quad indirect */
|
|
with LABEL GPR GPR GPR
|
|
uses REG
|
|
gen
|
|
lea %a, %1
|
|
st %2, {GPROFFSET, %a, 0}
|
|
st %3, {GPROFFSET, %a, 4}
|
|
st %4, {GPROFFSET, %a, 8}
|
|
with GPR GPR GPR GPR
|
|
uses reusing %1, REG=%1
|
|
gen
|
|
add %a, GP
|
|
st %2, {GPROFFSET, %a, 0}
|
|
st %3, {GPROFFSET, %a, 4}
|
|
st %4, {GPROFFSET, %a, 8}
|
|
|
|
pat sti /* Store arbitrary size */
|
|
leaving
|
|
loc $1
|
|
sts QUAD
|
|
|
|
pat sts /* Store arbitrary size */
|
|
leaving
|
|
cal ".sts"
|
|
|
|
|
|
|
|
/* Arithmetic wrappers */
|
|
|
|
pat ads /* Add var to pointer */
|
|
leaving adi $1
|
|
|
|
pat sbs /* Subtract var from pointer */
|
|
leaving sbi $1
|
|
|
|
pat adp /* Add constant to pointer */
|
|
leaving
|
|
loc $1
|
|
adi QUAD
|
|
|
|
pat adu /* Add unsigned */
|
|
leaving
|
|
adi $1
|
|
|
|
pat sbu /* Subtract unsigned */
|
|
leaving
|
|
sbi $1
|
|
|
|
pat inc /* Add 1 */
|
|
leaving
|
|
loc 1
|
|
adi QUAD
|
|
|
|
pat dec /* Subtract 1 */
|
|
leaving
|
|
loc 1
|
|
sbi QUAD
|
|
|
|
pat loc mlu /* Unsigned multiply by constant */
|
|
leaving
|
|
loc $1
|
|
mli QUAD
|
|
|
|
pat mlu /* Unsigned multiply by var */
|
|
leaving
|
|
mli QUAD
|
|
|
|
pat loc slu /* Shift left unsigned by constant amount */
|
|
leaving
|
|
loc $1
|
|
sli $2
|
|
|
|
pat slu /* Shift left unsigned by variable amount */
|
|
leaving
|
|
sli $1
|
|
|
|
|
|
|
|
/* Word arithmetic */
|
|
|
|
pat loc adi $1==0 /* Add nothing */
|
|
/* nop */
|
|
|
|
pat adi $1==QUAD /* Add word (second + top) */
|
|
with GPR+CONST GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
add %a, %1
|
|
yields %a
|
|
with GPR GPR+CONST
|
|
uses reusing %1, REG=%1
|
|
gen
|
|
add %a, %2
|
|
yields %a
|
|
|
|
pat loc sbi $1==0 /* Subtract nothing */
|
|
/* nop */
|
|
|
|
pat sbi $1==QUAD /* Subtract word (second - top) */
|
|
with GPR+CONST GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
sub %a, %1
|
|
yields %a
|
|
|
|
pat mli $1==QUAD /* Multiply word (second * top) */
|
|
with GPR+CONST GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
mul %a, %1
|
|
yields %a
|
|
with GPR GPR+CONST
|
|
uses reusing %1, REG=%1
|
|
gen
|
|
mul %a, %2
|
|
yields %a
|
|
|
|
pat mlu
|
|
leaving
|
|
mli $1
|
|
|
|
pat dvi $1==QUAD /* Divide word (second / top) */
|
|
with GPR GPR
|
|
uses reusing %2, REG
|
|
gen
|
|
divs %a, %2, %1
|
|
yields %a
|
|
|
|
pat dvu $1==QUAD /* Divide unsigned word (second / top) */
|
|
with GPR GPR
|
|
uses reusing %2, REG
|
|
gen
|
|
divu %a, %2, %1
|
|
yields %a
|
|
|
|
pat rmu $1==QUAD /* Remainder unsigned word (second % top) */
|
|
with GPR GPR
|
|
uses REG
|
|
gen
|
|
divu %a, %2, %1
|
|
mul %a, %1
|
|
rsb %a, %2
|
|
yields %a
|
|
|
|
pat rmi $1==QUAD /* Remainder signed word (second % top) */
|
|
with GPR GPR
|
|
uses REG
|
|
gen
|
|
divs %a, %2, %1
|
|
mul %a, %1
|
|
rsb %a, %2
|
|
yields %a
|
|
|
|
pat ngi $1==QUAD /* Negate word */
|
|
with GPR
|
|
uses reusing %1, REG=%1
|
|
gen
|
|
neg %a, %a
|
|
yields %a
|
|
|
|
pat and $1==QUAD /* AND word */
|
|
with GPR+CONST GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
and %a, %1
|
|
yields %a
|
|
with GPR GPR+CONST
|
|
uses reusing %1, REG=%1
|
|
gen
|
|
and %a, %2
|
|
yields %a
|
|
|
|
pat ior $1==QUAD /* OR word */
|
|
with GPR+CONST GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
or %a, %1
|
|
yields %a
|
|
with GPR GPR+CONST
|
|
uses reusing %1, REG=%1
|
|
gen
|
|
or %a, %2
|
|
yields %a
|
|
|
|
pat xor $1==QUAD /* XOR word */
|
|
with GPR+CONST GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
eor %a, %1
|
|
yields %a
|
|
with GPR GPR+CONST
|
|
uses reusing %1, REG=%1
|
|
gen
|
|
eor %a, %2
|
|
yields %a
|
|
|
|
pat com $1==QUAD /* Complement */
|
|
with GPR
|
|
uses reusing %1, REG=%1
|
|
gen
|
|
mvn %a, %1
|
|
yields %a
|
|
|
|
pat dvi $1==QUAD /* Divide word (second / top) */
|
|
with GPR GPR
|
|
uses reusing %2, REG
|
|
gen
|
|
divs %a, %2, %1
|
|
yields %a
|
|
|
|
pat dvu $1==QUAD /* Divide unsigned word (second / top) */
|
|
with GPR GPR
|
|
uses reusing %2, REG
|
|
gen
|
|
divu %a, %2, %1
|
|
yields %a
|
|
|
|
pat rmu $1==QUAD /* Remainder unsigned word (second % top) */
|
|
with GPR GPR
|
|
uses REG
|
|
gen
|
|
divu %a, %2, %1
|
|
mul %a, %1
|
|
sub %a, %2
|
|
yields %a
|
|
|
|
pat rmi $1==QUAD /* Remainder signed word (second % top) */
|
|
with GPR GPR
|
|
uses REG
|
|
gen
|
|
divs %a, %2, %1
|
|
mul %a, %1
|
|
sub %a, %2
|
|
yields %a
|
|
|
|
#if 0
|
|
pat mli $1==4 /* Multiply word (second * top) */
|
|
with REG REG
|
|
uses reusing %2, REG
|
|
gen
|
|
mullw %a, %2, %1
|
|
yields %a
|
|
|
|
|
|
pat xor $1==4 /* XOR word */
|
|
with GPR GPR
|
|
yields {XOR_RR, %1, %2}
|
|
with GPR CONST
|
|
yields {XOR_RC, %1, %2.val}
|
|
with CONST GPR
|
|
yields {XOR_RC, %2, %1.val}
|
|
|
|
pat xor !defined($1) /* XOR set */
|
|
with STACK
|
|
gen
|
|
bl {LABEL, ".xor"}
|
|
|
|
pat com $1==QUAD /* NOT word */
|
|
with AND_RR
|
|
uses REG
|
|
gen
|
|
nand %a, %1.reg1, %1.reg2
|
|
yields %a
|
|
with OR_RR
|
|
uses REG
|
|
gen
|
|
nor %a, %1.reg1, %1.reg2
|
|
yields %a
|
|
with XOR_RR
|
|
uses REG
|
|
gen
|
|
eqv %a, %1.reg1, %1.reg2
|
|
yields %a
|
|
with GPR
|
|
yields {NOT_R, %1}
|
|
|
|
pat com !defined($1) /* NOT set */
|
|
with STACK
|
|
gen
|
|
bl {LABEL, ".com"}
|
|
#endif
|
|
|
|
pat sli $1==4 /* Shift left (second << top) */
|
|
with CONST+GPR GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
lsl %a, %1
|
|
yields %a
|
|
|
|
pat sri $1==4 /* Shift right signed (second >> top) */
|
|
with CONST+GPR GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
asr %2, %1
|
|
yields %a
|
|
|
|
pat sru $1==4 /* Shift right unsigned (second >> top) */
|
|
with CONST+GPR GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
lsr %2, %1
|
|
yields %a
|
|
|
|
|
|
|
|
/* Special arithmetic */
|
|
|
|
pat loc sli adi $1==1 && $2==QUAD && $3==QUAD /* Shift and add (second + top<<1) */
|
|
with GPR+CONST GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
adds2 %a, %1
|
|
yields %a
|
|
|
|
pat loc sli adi $1==2 && $2==QUAD && $3==QUAD /* Shift and add (second + top<<2) */
|
|
with GPR+CONST GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
adds4 %a, %1
|
|
yields %a
|
|
|
|
pat loc sli adi $1==3 && $2==QUAD && $3==QUAD /* Shift and add (second + top<<3) */
|
|
with GPR+CONST GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
adds8 %a, %1
|
|
yields %a
|
|
|
|
pat loc sli adi $1==4 && $2==QUAD && $3==QUAD /* Shift and add (second + top<<4) */
|
|
with GPR+CONST GPR
|
|
uses reusing %2, REG=%2
|
|
gen
|
|
adds16 %a, %1
|
|
yields %a
|
|
|
|
pat loc sli adi $1==8 && $2==QUAD && $3==QUAD /* Shift and add (second + top<<8) */
|
|
with GPR GPR
|
|
uses reusing %2, REG
|
|
gen
|
|
adds256 %a, %2, %1
|
|
yields %a
|
|
|
|
pat loc sli ads
|
|
leaving
|
|
loc $1
|
|
sli $2
|
|
adi $3
|
|
|
|
|
|
|
|
/* Arrays */
|
|
|
|
pat aar $1==QUAD /* Index array */
|
|
with STACK
|
|
uses GPR0
|
|
gen
|
|
bl {LABEL, ".aar4stack"}
|
|
yields R0
|
|
with GPR0 GPR1 GPR2
|
|
uses GPR0
|
|
gen
|
|
bl {LABEL, ".aar4"}
|
|
yields R0
|
|
|
|
pat lae lar $2==QUAD && nicesize(rom($1, 3)) /* Load array */
|
|
leaving
|
|
lae $1
|
|
aar QUAD
|
|
loi rom($1, 3)
|
|
|
|
pat lar $1==QUAD /* Load array */
|
|
with STACK
|
|
uses GPR0
|
|
gen
|
|
bl {LABEL, ".lar4stack"}
|
|
yields R0
|
|
with GPR0 GPR1 GPR2
|
|
uses GPR0
|
|
gen
|
|
bl {LABEL, ".lar4"}
|
|
yields R0
|
|
|
|
pat lae sar $2==QUAD && nicesize(rom($1, 3)) /* Store array */
|
|
leaving
|
|
lae $1
|
|
aar QUAD
|
|
sti rom($1, 3)
|
|
|
|
pat sar $1==QUAD /* Store array */
|
|
with STACK
|
|
uses GPR0
|
|
gen
|
|
bl {LABEL, ".sar4stack"}
|
|
yields R0
|
|
with GPR0 GPR1 GPR2
|
|
uses GPR0
|
|
gen
|
|
bl {LABEL, ".sar4"}
|
|
|
|
|
|
|
|
/* Sets */
|
|
|
|
pat set $1==QUAD /* Create quad with one bit set */
|
|
with GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
bset %a, %1
|
|
yields %a
|
|
|
|
pat set defined($1) /* Any other set */
|
|
leaving
|
|
loc $1
|
|
cal ".set"
|
|
|
|
pat set !defined($1) /* Create structure with set bit (variable) */
|
|
leaving
|
|
cal ".set"
|
|
|
|
pat inn defined($1) /* Test for set bit */
|
|
leaving
|
|
set $1
|
|
and $1
|
|
|
|
pat inn !defined($1) /* Test for set bit (variable) */
|
|
leaving
|
|
cal ".inn"
|
|
|
|
pat ior !nicesize($1) /* OR set */
|
|
leaving
|
|
cal ".ior"
|
|
|
|
pat ior !defined($1) /* OR set */
|
|
leaving
|
|
cal ".ior"
|
|
|
|
pat and !nicesize($1) /* AND set */
|
|
leaving
|
|
loc $1
|
|
cal ".and"
|
|
|
|
pat and !defined($1) /* AND set */
|
|
leaving
|
|
cal ".and"
|
|
|
|
|
|
|
|
/* Boolean resolutions */
|
|
|
|
proc cm_t example teq
|
|
with GPR GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
cmp %1, %2
|
|
mov %a, {CONST, 0}
|
|
add[1] %a, {CONST, 1}
|
|
yields %a
|
|
|
|
pat cmu teq call cm_t("add.eq") /* top = (second == top) */
|
|
pat cmu tne call cm_t("add.ne") /* top = (second != top) */
|
|
pat cmu tlt call cm_t("add.lo") /* top = unsigned (second < top) */
|
|
pat cmu tle call cm_t("add.ls") /* top = unsigned (second <= top) */
|
|
pat cmu tgt call cm_t("add.hi") /* top = unsigned (second < top) */
|
|
pat cmu tge call cm_t("add.hs") /* top = unsigned (second >= top) */
|
|
pat cmi teq call cm_t("add.eq") /* top = (second == top) */
|
|
pat cmi tne call cm_t("add.ne") /* top = (second != top) */
|
|
pat cmi tlt call cm_t("add.lt") /* top = signed (second < top) */
|
|
pat cmi tle call cm_t("add.le") /* top = signed (second <= top) */
|
|
pat cmi tgt call cm_t("add.gt") /* top = signed (second < top) */
|
|
pat cmi tge call cm_t("add.ge") /* top = signed (second >= top) */
|
|
|
|
proc cmf_t example teq
|
|
with GPR GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
fcmp %a, %1, %2
|
|
mov %a, {CONST, 0}
|
|
add[1] %a, {CONST, 1}
|
|
yields %a
|
|
|
|
pat cmf teq call cmf_t("add.eq") /* top = float (second == top) */
|
|
pat cmf tne call cmf_t("add.ne") /* top = float (second != top) */
|
|
pat cmf tlt call cmf_t("add.lo") /* top = float (second < top) */
|
|
pat cmf tle call cmf_t("add.ls") /* top = float (second <= top) */
|
|
pat cmf tgt call cmf_t("add.hi") /* top = float (second > top) */
|
|
pat cmf tge call cmf_t("add.hs") /* top = float (second >= top) */
|
|
|
|
proc fallback_t example teq
|
|
with GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
cmp %1, {CONST, 0}
|
|
mov %a, {CONST, 0}
|
|
add[1] %a, {CONST, 1}
|
|
yields %a
|
|
|
|
pat teq call fallback_t("add.eq") /* top = float (top == 0) */
|
|
pat tne call fallback_t("add.ne") /* top = float (top != 0) */
|
|
pat tlt call fallback_t("add.lo") /* top = float (top < 0) */
|
|
pat tle call fallback_t("add.ls") /* top = float (top <= 0) */
|
|
pat tgt call fallback_t("add.hi") /* top = float (top > 0) */
|
|
pat tge call fallback_t("add.hs") /* top = float (top >= 0) */
|
|
|
|
|
|
|
|
/* Simple branches */
|
|
|
|
proc anyz example zeq
|
|
with GPR STACK
|
|
kills ALL
|
|
gen
|
|
cmp %1, {CONST, 0}
|
|
beq[1] {LABEL, $1}
|
|
|
|
pat zeq call anyz("b.eq") /* Branch if signed top == 0 */
|
|
pat zne call anyz("b.ne") /* Branch if signed top != 0 */
|
|
pat zgt call anyz("b.gt") /* Branch if signed top > 0 */
|
|
pat zlt call anyz("b.lt") /* Branch if signed top < 0 */
|
|
pat zge call anyz("b.ge") /* Branch if signed top >= 0 */
|
|
pat zle call anyz("b.le") /* Branch if signed top <= 0 */
|
|
|
|
proc anyb example beq
|
|
with GPR+CONST GPR STACK
|
|
kills ALL
|
|
gen
|
|
cmp %2, %1
|
|
beq[1] {LABEL, $1}
|
|
|
|
pat beq call anyb("b.eq") /* Branch if signed second == top */
|
|
pat bne call anyb("b.ne") /* Branch if signed second != top */
|
|
pat bgt call anyb("b.gt") /* Branch if signed second > top */
|
|
pat bge call anyb("b.ge") /* Branch if signed second >= top */
|
|
pat blt call anyb("b.lt") /* Branch if signed second < top */
|
|
pat ble call anyb("b.le") /* Branch if signed second <= top */
|
|
|
|
proc cmu_z example cmu zeq
|
|
with GPR+CONST GPR STACK
|
|
kills ALL
|
|
gen
|
|
cmp %2, %1
|
|
beq[1] {LABEL, $2}
|
|
|
|
pat cmu zeq call cmu_z("b.eq") /* Branch if unsigned second == top */
|
|
pat cmu zne call cmu_z("b.ne") /* Branch if unsigned second != top */
|
|
pat cmu zgt call cmu_z("b.hi") /* Branch if unsigned second > top */
|
|
pat cmu zlt call cmu_z("b.lo") /* Branch if unsigned second < top */
|
|
pat cmu zge call cmu_z("b.hs") /* Branch if unsigned second >= top */
|
|
pat cmu zle call cmu_z("b.ls") /* Branch if unsigned second <= top */
|
|
pat cmi zeq call cmu_z("b.eq") /* Branch if signed second == top */
|
|
pat cmi zne call cmu_z("b.ne") /* Branch if signed second != top */
|
|
pat cmi zgt call cmu_z("b.gt") /* Branch if signed second > top */
|
|
pat cmi zlt call cmu_z("b.lt") /* Branch if signed second < top */
|
|
pat cmi zge call cmu_z("b.ge") /* Branch if signed second >= top */
|
|
pat cmi zle call cmu_z("b.le") /* Branch if signed second <= top */
|
|
|
|
proc cmf_z example cmu zeq
|
|
with GPR GPR STACK
|
|
kills ALL
|
|
gen
|
|
fcmp %2, %2, %1
|
|
beq[1] {LABEL, $2}
|
|
|
|
pat cmf zeq call cmf_z("b.eq") /* Branch if float second == top */
|
|
pat cmf zne call cmf_z("b.ne") /* Branch if float second != top */
|
|
pat cmf zgt call cmf_z("b.gt") /* Branch if float second > top */
|
|
pat cmf zlt call cmf_z("b.lt") /* Branch if float second < top */
|
|
pat cmf zge call cmf_z("b.ge") /* Branch if float second >= top */
|
|
pat cmf zle call cmf_z("b.le") /* Branch if float second <= top */
|
|
|
|
pat cmp /* Compare pointers */
|
|
leaving
|
|
cmu QUAD
|
|
|
|
pat cms $1==QUAD /* Compare blocks (word sized) */
|
|
leaving
|
|
cmi QUAD
|
|
|
|
|
|
|
|
|
|
|
|
/* Other branching and labelling */
|
|
|
|
#if 0
|
|
pat lab topeltsize($1)<=4 && !fallthrough($1)
|
|
gen
|
|
labeldef $1
|
|
yields R0
|
|
|
|
pat lab topeltsize($1)<=4 && fallthrough($1)
|
|
with GPR0
|
|
gen
|
|
labeldef $1
|
|
yields %1
|
|
|
|
pat lab topeltsize($1)>4
|
|
with STACK
|
|
kills ALL
|
|
gen
|
|
labeldef $1
|
|
|
|
pat bra topeltsize($1)<=4 /* Unconditional jump with TOS register */
|
|
with GPR0 STACK
|
|
gen
|
|
b {LABEL, $1}
|
|
|
|
pat bra topeltsize($1)>4 /* Unconditional jump without TOS register */
|
|
with STACK
|
|
gen
|
|
b {LABEL, $1}
|
|
#endif
|
|
|
|
pat lab
|
|
with STACK
|
|
kills ALL
|
|
gen
|
|
labeldef $1
|
|
|
|
pat bra
|
|
with STACK
|
|
kills ALL
|
|
gen
|
|
b {LABEL, $1}
|
|
|
|
|
|
|
|
|
|
/* Miscellaneous */
|
|
|
|
pat cal /* Call procedure */
|
|
with STACK
|
|
kills ALL
|
|
gen
|
|
bl {LABEL, $1}
|
|
|
|
pat cai /* Call procedure indirect */
|
|
with GPR STACK
|
|
kills ALL
|
|
gen
|
|
bl %1
|
|
|
|
pat lfr $1==QUAD /* Load function result, word */
|
|
yields R0
|
|
|
|
pat lfr $1==QUAD*2 /* Load function result, word */
|
|
yields R1 R0
|
|
|
|
pat ret $1==0 /* Return from procedure */
|
|
gen
|
|
mov SP, FP
|
|
pop FP, PC
|
|
|
|
pat ret $1==QUAD /* Return from procedure, word */
|
|
with GPR0
|
|
gen
|
|
mov SP, FP
|
|
pop FP, PC
|
|
|
|
pat ret $1==QUAD*2 /* Return from procedure, word */
|
|
with GPR GPR
|
|
gen
|
|
move %1, R0
|
|
move %2, R1
|
|
mov SP, FP
|
|
pop FP, PC
|
|
|
|
pat blm /* Block move constant length */
|
|
leaving
|
|
loc $1
|
|
bls
|
|
|
|
pat bls /* Block move variable length */
|
|
with STACK
|
|
kills ALL
|
|
gen
|
|
bl {LABEL, "_memmove"}
|
|
|
|
pat csa /* Array-lookup switch */
|
|
with GPR0 GPR1 STACK
|
|
kills ALL
|
|
gen
|
|
b {LABEL, ".csa"}
|
|
|
|
pat csb /* Table-lookup switch */
|
|
with GPR0 GPR1 STACK
|
|
kills ALL
|
|
gen
|
|
bl {LABEL, ".csb"}
|
|
|
|
|
|
|
|
/* EM specials */
|
|
|
|
pat fil /* Set current filename */
|
|
leaving
|
|
lae $1
|
|
ste ".filename"
|
|
|
|
pat lin /* Set current line number */
|
|
leaving
|
|
loc $1
|
|
ste ".linenumber"
|
|
|
|
pat lni /* Increment line number */
|
|
leaving
|
|
ine ".linenumber"
|
|
|
|
pat lim /* Load EM trap ignore mask */
|
|
leaving
|
|
lde ".ignmask"
|
|
|
|
pat sim /* Store EM trap ignore mask */
|
|
leaving
|
|
ste ".ignmask"
|
|
|
|
pat trp /* Raise EM trap */
|
|
leaving
|
|
cal ".trap"
|
|
|
|
pat sig /* Set trap handler */
|
|
leaving
|
|
ste ".trppc"
|
|
|
|
pat rtt /* Return from trap */
|
|
leaving
|
|
ret 0
|
|
|
|
pat lxl $1==0 /* Load FP */
|
|
leaving
|
|
lor 0
|
|
|
|
pat lxl $1==1 /* Load caller's FP */
|
|
leaving
|
|
lxl 0
|
|
dch
|
|
|
|
pat dch /* FP -> caller FP */
|
|
with GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
ld %a, {GPROFFSET, %1, FP_OFFSET}
|
|
sub %a, GP
|
|
yields %a
|
|
|
|
pat lpb /* Convert FP to argument address */
|
|
leaving
|
|
adp EM_BSIZE
|
|
|
|
pat lxa /* Load caller's SP */
|
|
leaving
|
|
lxl $1
|
|
lpb
|
|
|
|
pat gto /* longjmp */
|
|
uses REG, REG
|
|
gen
|
|
move {LABEL, $1}, %a
|
|
ld %b, {GPROFFSET, %a, 8}
|
|
add FP, %b, GP
|
|
ld %b, {GPROFFSET, %a, 4}
|
|
add SP, %b, GP
|
|
ld %b, {GPROFFSET, %a, 0}
|
|
add %b, GP
|
|
b %b
|
|
|
|
#if 0
|
|
|
|
pat gto /* longjmp */
|
|
with STACK
|
|
gen
|
|
ld {LABEL, $1+2}
|
|
wspec {CONST, 1}
|
|
ld {LABEL, $1+4}
|
|
wspec {CONST, 0}
|
|
ld {LABEL, $1+0}
|
|
wspec {CONST, 2}
|
|
|
|
pat str $1==1 /* Store special GPRister */
|
|
with GPR0
|
|
gen
|
|
wspec {CONST, $1}
|
|
|
|
#endif
|
|
|
|
pat lor $1==0 /* Load FP */
|
|
uses REG
|
|
gen
|
|
move FP, %a
|
|
yields %a
|
|
|
|
pat lor $1==1 /* Load SP */
|
|
uses REG
|
|
gen
|
|
move SP, %a
|
|
yields %a
|
|
|
|
pat lor $1==2 /* Load HP */
|
|
leaving
|
|
loe ".reghp"
|
|
|
|
pat str $1==0 /* Store FP */
|
|
with GPR
|
|
gen
|
|
sub FP, %1, GP
|
|
|
|
pat str $1==1 /* Store SP */
|
|
with GPR
|
|
gen
|
|
sub SP, %1, GP
|
|
|
|
pat str $1==2 /* Store HP */
|
|
leaving
|
|
ste ".reghp"
|
|
|
|
pat ass /* Adjust stack by variable amount */
|
|
with CONST+GPR STACK
|
|
gen
|
|
add SP, %1
|
|
|
|
pat asp $1==QUAD /* Adjust stack by constant amount */
|
|
with GPR
|
|
/* silently ignore GPR */
|
|
with STACK
|
|
gen
|
|
pop SCRATCH
|
|
|
|
pat asp $1==(2*QUAD) /* Adjust stack by constant amount */
|
|
with GPR GPR
|
|
/* silently ignore GPR */
|
|
with STACK
|
|
gen
|
|
add SP, {CONST, 2*QUAD}
|
|
|
|
pat asp /* Adjust stack by constant amount */
|
|
leaving
|
|
loc $1
|
|
ass
|
|
|
|
|
|
|
|
/* Floating point */
|
|
|
|
pat ngf /* Negate float */
|
|
leaving
|
|
loc 0
|
|
exg QUAD
|
|
sbf QUAD
|
|
|
|
proc simple_f example adf
|
|
with GPR GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
fadd[1] %a, %2, %1
|
|
yields %a
|
|
|
|
pat adf call simple_f("fadd") /* Float subtract (second + top) */
|
|
pat sbf call simple_f("fsub") /* Float subtract (second - top) */
|
|
pat mlf call simple_f("fmul") /* Float multiply (second * top) */
|
|
pat dvf call simple_f("fdiv") /* Float divide (second / top) */
|
|
|
|
pat loc loc cff $1==$2 && $1==QUAD /* Convert float to float */
|
|
leaving
|
|
nop
|
|
|
|
pat loc loc cfi $1==$2 && $1==QUAD /* Convert float -> integer */
|
|
with GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
ftrunc %a, %1
|
|
yields %a
|
|
|
|
pat loc loc cfu $1==$2 && $1==QUAD /* Convert float -> unsigned */
|
|
with GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
ftrunc %a, %1
|
|
yields %a
|
|
|
|
pat loc loc cif $1==$2 && $1==QUAD /* Convert integer -> float */
|
|
with GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
flts %a, %1
|
|
yields %a
|
|
|
|
pat loc loc cuf $1==$2 && $1==QUAD /* Convert unsigned -> float */
|
|
with GPR
|
|
uses reusing %1, REG
|
|
gen
|
|
fltu %a, %1
|
|
yields %a
|
|
|
|
pat fef /* Split float */
|
|
leaving
|
|
loc 0
|
|
loc 0
|
|
#if 0
|
|
cal ".cuf"
|
|
lfr QUAD*2
|
|
#endif
|
|
|
|
pat fif /* Multiply float and split (?) */
|
|
leaving
|
|
mlf QUAD
|
|
fef
|
|
|
|
pat zrf /* Load a floating zero */
|
|
leaving
|
|
loc 0
|