726 lines
16 KiB
C
726 lines
16 KiB
C
/* $Header$ */
|
|
/*
|
|
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
|
|
* See the copyright notice in the ACK home directory, in the file "Copyright".
|
|
*/
|
|
/*
|
|
* Motorola 68020 syntax rules
|
|
*/
|
|
/* Please do not add more terminal symbols. As it is, 127 terminal symbols
|
|
are used, and this is the limit for some "yacc" implementations,
|
|
notably the Ultrix one.
|
|
*/
|
|
|
|
operation
|
|
: { instrp = instr;
|
|
dot_offset = 0;
|
|
curr_instr = curr_token;
|
|
}
|
|
instruction
|
|
{ emit_instr();
|
|
}
|
|
;
|
|
instruction
|
|
: bcdx DREG ',' DREG
|
|
{ emit2($1 | $2 | $4<<9);}
|
|
| bcdx '-' '(' AREG ')' ',' '-' '(' AREG ')'
|
|
{ emit2($1 | $4 | $9<<9 | 010);}
|
|
| ADD sizedef ea_ea
|
|
{ add($1, $2);}
|
|
| AND sizenon ea_ea
|
|
{ and($1, $2);}
|
|
| SHIFT sizedef ea_ea
|
|
{ shift_op($1, $2);}
|
|
| SHIFT sizedef ea /* This syntax is also allowed */
|
|
{ checksize($2, 2);
|
|
T_EMIT2(($1 & 0177700) | mrg_2,0,0,0);
|
|
ea_2(SIZE_W, MEM|ALT);
|
|
}
|
|
| BR expr
|
|
{ branch($1, $2);}
|
|
| DBR DREG ',' expr
|
|
{ T_EMIT2($1 | $2,0,0,0);
|
|
$4.val -= (DOTVAL+dot_offset);
|
|
fit(fitw($4.val));
|
|
T_EMIT2(loww($4.val), $4.typ,
|
|
RELPC|RELO2, relonami);
|
|
}
|
|
| BITOP ea_ea
|
|
{ bitop($1);}
|
|
| BITFIELD ea off_width
|
|
{ bitfield($1, $3);}
|
|
| BF_TO_D ea off_width ',' DREG
|
|
{ bitfield($1, $3 | $5<<12);}
|
|
| BFINS DREG ',' ea off_width
|
|
{ bitfield($1, $5 | $2<<12);}
|
|
| DIVMUL sizedef ea ',' DREG
|
|
{ checksize($2, 2|4);
|
|
if ($2 == SIZE_W) {
|
|
T_EMIT2((0140300^($1<<8))|mrg_2|$5<<9,
|
|
0, 0, 0);
|
|
ea_2(SIZE_W, DTA);
|
|
}
|
|
else { /* 32 bit dividend or product */
|
|
T_EMIT2((046000 | ($1 & ~1)) | mrg_2,
|
|
0, 0, 0);
|
|
T_EMIT2(($1&1)<<11 | $5<<12 | $5,
|
|
0, 0, 0);
|
|
ea_2(SIZE_L, DTA);
|
|
}
|
|
}
|
|
| DIVMUL sizedef ea ',' DREG ':' DREG
|
|
{ /* 64 bit dividend or product */
|
|
checksize($2, 4);
|
|
T_EMIT2((046000 | ($1 & ~1)) | mrg_2, 0, 0, 0);
|
|
T_EMIT2(($1&1)<<11 | $7<<12 | $5 | 02000 ,0,0,0);
|
|
ea_2(SIZE_L, DTA);
|
|
}
|
|
| DIVL sizedef ea ',' DREG ':' DREG
|
|
{ /* 32 bit long division with remainder */
|
|
checksize($2, 4);
|
|
T_EMIT2(($1 & ~1) | mrg_2, 0, 0, 0);
|
|
T_EMIT2(($1 & 1)<<11 | $7<<12 | $5 , 0, 0, 0);
|
|
ea_2(SIZE_L, DTA);
|
|
}
|
|
| LEA ea ',' AREG
|
|
{ T_EMIT2(040700 | mrg_2 | $4<<9,0,0,0);
|
|
ea_2(SIZE_L, CTR);
|
|
}
|
|
| op_ea ea
|
|
{ if (mrg_2==074)
|
|
serror("bad adressing category");
|
|
T_EMIT2(($1&0177700) | mrg_2,0,0,0);
|
|
ea_2($1&0300, $1&017);
|
|
}
|
|
| OP_NOOP
|
|
{ emit2($1);}
|
|
| OP_EXT SIZE DREG
|
|
{ checksize($2, ($1 & 0400) ? 4 : (2|4));
|
|
emit2($1 | $2+0100 | $3);
|
|
}
|
|
| OP_RANGE sizedef ea ',' reg
|
|
{ T_EMIT2(0300 | ($2<<3) | mrg_2,0,0,0);
|
|
T_EMIT2($1 | ($5<<12),0,0,0);
|
|
ea_2($2, CTR);
|
|
}
|
|
| TRAPCC SIZE imm
|
|
{ checksize($2, 2|4);
|
|
T_EMIT2($1 | ($2>>6)+1,0,0,0);
|
|
ea_2($2, 0);
|
|
}
|
|
| TRAPCC { emit2($1 | 4);}
|
|
| PACK '-' '(' AREG ')' ',' '-' '(' AREG ')' ',' imm
|
|
{ T_EMIT2($1 | 8 | $4 | $9<<9, 0, 0, 0);
|
|
ea_2(SIZE_W, 0);
|
|
}
|
|
| PACK DREG ',' DREG ',' imm
|
|
{ T_EMIT2($1 | $2 | $4<<9, 0, 0, 0);
|
|
ea_2(SIZE_W, 0);
|
|
}
|
|
| CMP sizedef ea_ea
|
|
{ cmp($2);}
|
|
| CHK sizedef ea ',' DREG
|
|
{ checksize($2, 2|4);
|
|
T_EMIT2(040000 | mrg_2 | $5<<9 |
|
|
($2==SIZE_W ? 0600 : 0400),
|
|
0, 0, 0);
|
|
ea_2($2, DTA);
|
|
}
|
|
| MOVE sizenon ea_ea
|
|
{ move($2);}
|
|
| MOVEM sizedef regs ',' notimmreg
|
|
{ movem(0, $2, $3);}
|
|
| MOVEM sizedef notimmreg ',' regs
|
|
{ movem(1, $2, $5);}
|
|
| MOVESP sizedef
|
|
{ curr_size = $1; }
|
|
ea_ea
|
|
{ if ($1 == 0) {
|
|
/* movep */
|
|
movep($2);
|
|
} else if (mrg_1 <= 017) {
|
|
T_EMIT2(007000 | $2 | mrg_2,0,0,0);
|
|
T_EMIT2(mrg_1 << 12 | 04000,0,0,0);
|
|
ea_2($2,ALT|MEM);
|
|
} else if (mrg_2 <= 017) {
|
|
T_EMIT2(007000 | $2 | mrg_1,0,0,0);
|
|
T_EMIT2(mrg_2 << 12,0,0,0);
|
|
ea_1($2,ALT|MEM);
|
|
} else
|
|
badoperand();
|
|
}
|
|
| MOVEC creg ',' reg
|
|
{ T_EMIT2(047172,0,0,0);
|
|
T_EMIT2($2 | $4<<12,0,0,0);
|
|
}
|
|
| MOVEC reg ',' creg
|
|
{ T_EMIT2(047173,0,0,0);
|
|
T_EMIT2($4 | $2<<12,0,0,0);
|
|
}
|
|
| EXG reg ',' reg
|
|
{ if (($2 & 010) == 0)
|
|
emit2(
|
|
(0140500|$4|$2<<9)
|
|
+
|
|
(($4&010)<<3)
|
|
);
|
|
else
|
|
emit2(
|
|
(0140610|$2|($4&07)<<9)
|
|
-
|
|
(($4&010)<<3)
|
|
);
|
|
}
|
|
| SWAP DREG
|
|
{ emit2(044100 | $2);}
|
|
| OP_IMM imm
|
|
{ T_EMIT2($1, 0, 0, 0);
|
|
ea_2(SIZE_W, 0);
|
|
}
|
|
| LINK sizenon AREG ',' imm
|
|
{ link_instr($2, $3);}
|
|
| UNLK AREG
|
|
{ emit2(047130 | $2);}
|
|
| TRAP '#' absexp
|
|
{ fit(fit4($3));
|
|
emit2(047100|low4($3));
|
|
}
|
|
| BKPT '#' absexp
|
|
{ fit(($3 & ~07) == 0);
|
|
emit2(044110 | low3($3));
|
|
}
|
|
| CALLM '#' absexp ',' ea
|
|
{ fit(fitb($3));
|
|
T_EMIT2(03300 | mrg_2,0,0,0);
|
|
T_EMIT2((short) $3,0,0,0);
|
|
ea_2(SIZE_L, CTR);
|
|
}
|
|
| RTM reg
|
|
{ emit2(03300 | $2);}
|
|
| CAS sizedef DREG ',' DREG ',' ea
|
|
{ T_EMIT2(04300 | (($2+0100)<<3) | mrg_2,0,0,0);
|
|
T_EMIT2($3 | ($5<<6),0,0,0);
|
|
ea_2($2, MEM|ALT);
|
|
}
|
|
| CAS2 sizedef DREG ':' DREG ',' DREG ':' DREG ','
|
|
'(' reg ')' ':' '(' reg ')'
|
|
{ checksize($2 , 2|4);
|
|
emit2(04374 | (($2+0100)<<3));
|
|
emit2($3 | ($7<<6) | ($12<<12));
|
|
emit2($5 | ($9<<6) | ($16<<12));
|
|
}
|
|
| fp_op
|
|
| mm_op
|
|
;
|
|
bcdx : ABCD
|
|
| ADDX sizedef
|
|
{ $$ = $1 | $2;}
|
|
;
|
|
creg : CREG
|
|
| SPEC { if ($1 != 075)
|
|
badoperand();
|
|
$$ = 04000;
|
|
}
|
|
;
|
|
off_width /* note: these should be curly brackets, but that would
|
|
* leave us without brackets for expressions.
|
|
*/
|
|
: '[' abs31 ':' abs31 ']'
|
|
{ $$ = ($2<<6) | $4;
|
|
}
|
|
;
|
|
abs31 : DREG { $$ = 040 | $1;}
|
|
| absexp { fit(fit5($1));
|
|
$$ = low5($1);
|
|
}
|
|
;
|
|
op_ea : OP_EA
|
|
| SZ_EA sizedef
|
|
{ $$ = $1 | $2;}
|
|
;
|
|
regs : rrange
|
|
| regs '/' rrange
|
|
{ $$ = $1 | $3;}
|
|
;
|
|
rrange : reg
|
|
{ $$ = 1<<$1;}
|
|
| reg '-' reg
|
|
{ if ($1 > $3)
|
|
badoperand();
|
|
for ($$ = 0; $1 <= $3; $1++)
|
|
$$ |= (1<<$1);
|
|
}
|
|
;
|
|
ea : DREG
|
|
{ mrg_2 = $1;}
|
|
| AREG
|
|
{ mrg_2 = 010 | $1;}
|
|
| SPEC
|
|
{ mrg_2 = $1;}
|
|
| notimmreg
|
|
| imm
|
|
;
|
|
notimmreg
|
|
:
|
|
{ mrg_2 = 0; ffew_2 = 0400; /* initialization */}
|
|
notimmreg1
|
|
;
|
|
notimmreg1
|
|
: '(' AREG ')'
|
|
{ mrg_2 = 020 | $2;}
|
|
| '(' AREG ')' '+'
|
|
{ mrg_2 = 030 | $2;}
|
|
| '-' '(' AREG ')'
|
|
{ mrg_2 = 040 | $3;}
|
|
| '(' expr ')' sizenon
|
|
{ bd_2 = $2; ea7071($4);
|
|
RELOMOVE(bd_rel2, relonami);
|
|
}
|
|
| '(' bd_areg_index ')'
|
|
{ if ((mrg_2 & INDEX) == 0)
|
|
ffew_2 |= 0100; /* suppress index */
|
|
if ( !(mrg_2 & PC_MODE) &&
|
|
(ffew_2 & 0300) == 0100 &&
|
|
bd_2.typ==S_ABS && fitw(bd_2.val)
|
|
)
|
|
mrg_2 = (loww(bd_2.val)?050:020) | $2;
|
|
else {
|
|
mrg_2 = (mrg_2&PC_MODE)?073:(060 | $2);
|
|
ffew_2 |= 060; /* long displacement */
|
|
}
|
|
}
|
|
| '(' '[' bd_areg_index ']' index_od ')'
|
|
{ switch(mrg_2 & INDEX) {
|
|
case 0:
|
|
ffew_2 |= 0163; /* suppress index */
|
|
break;
|
|
case DBL_INDEX:
|
|
serror("bad indexing");
|
|
case PRE_INDEX:
|
|
ffew_2 |= 063; break;
|
|
case POST_INDEX:
|
|
ffew_2 |= 067; break;
|
|
}
|
|
mrg_2 = (mrg_2 & PC_MODE) ? 073 : (060 | $3);
|
|
}
|
|
;
|
|
imm : '#' expr
|
|
{ mrg_2 = 074; bd_2 = $2;
|
|
RELOMOVE(bd_rel2, relonami);
|
|
}
|
|
;
|
|
bd_areg_index
|
|
: /* empty */
|
|
{ $$ = 0; ffew_2 |= 0200;
|
|
/* base-reg suppressed */
|
|
bd_2.typ = S_ABS; bd_2.val = (valu_t)0;
|
|
/* zero displacement */
|
|
}
|
|
| expr { $$ = 0; ffew_2 |= 0200;
|
|
bd_2 = $1;
|
|
RELOMOVE(bd_rel2, relonami);
|
|
}
|
|
| areg_index
|
|
{ bd_2.typ = S_ABS; bd_2.val = (valu_t)0;
|
|
}
|
|
| expr ',' areg_index
|
|
{ $$ = $3; bd_2 = $1;
|
|
RELOMOVE(bd_rel2, relonami);
|
|
}
|
|
;
|
|
areg_index
|
|
: areg
|
|
| index { $$ = 0;
|
|
ffew_2 |= 0200; /* base-reg suppressed */
|
|
mrg_2 |= PRE_INDEX;
|
|
}
|
|
| areg ',' index
|
|
{ mrg_2 |= PRE_INDEX;
|
|
}
|
|
;
|
|
areg : AREG
|
|
| PC { mrg_2 |= PC_MODE;
|
|
ffew_2 |= $1; /* base-reg suppressed for zpc */
|
|
$$ = 0;
|
|
}
|
|
;
|
|
index : reg sizedef scale
|
|
{ checksize($2, 2|4);
|
|
ffew_2 |= $1<<12 | ($2&0200)<<4 | $3;
|
|
}
|
|
;
|
|
scale : /* empty */
|
|
{ $$ = 0;}
|
|
| '*' absexp
|
|
{ $$ = checkscale($2);}
|
|
;
|
|
index_od: /* empty */
|
|
{ od_2.typ = S_ABS; od_2.val = (valu_t)0;}
|
|
| ',' index
|
|
{ od_2.typ = S_ABS; od_2.val = (valu_t)0;
|
|
mrg_2 |= POST_INDEX;
|
|
}
|
|
| ',' expr
|
|
{ od_2 = $2;
|
|
RELOMOVE(od_rel2, relonami);
|
|
}
|
|
| ',' index ',' expr
|
|
{ od_2 = $4;
|
|
mrg_2 |= POST_INDEX;
|
|
RELOMOVE(od_rel2, relonami);
|
|
}
|
|
;
|
|
reg : DREG
|
|
| AREG
|
|
{ $$ = $1 | 010;}
|
|
;
|
|
sizedef : /* empty */
|
|
{ $$ = SIZE_DEF; curr_size = SIZE_DEF; }
|
|
| SIZE { curr_size = $1; $$ = $1; }
|
|
;
|
|
sizenon : /* empty */
|
|
{ $$ = SIZE_NON;}
|
|
| SIZE
|
|
;
|
|
ea_ea : ea ','
|
|
{ mrg_1 = mrg_2;
|
|
bd_1 = bd_2;
|
|
od_1 = od_2;
|
|
ffew_1 = ffew_2;
|
|
RELOMOVE(bd_rel1, bd_rel2);
|
|
RELOMOVE(od_rel1, od_rel2);
|
|
}
|
|
ea
|
|
;
|
|
fp_op : CP
|
|
{ co_id = $1; }
|
|
fp_op1
|
|
| { co_id = DEF_FP; }
|
|
fp_op1
|
|
;
|
|
fp_op1 : FMOVE fsize ea ',' FPCR
|
|
{ check_fsize($2, FSIZE_L);
|
|
if ((mrg_2&070) == 010 && $5 != 001)
|
|
badoperand();
|
|
T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2((0100000|($5<<10)),0,0,0);
|
|
ea_2(SIZE_L, 0);
|
|
}
|
|
| FMOVE fsize FPCR ',' ea
|
|
{ check_fsize($2, FSIZE_L);
|
|
if ((mrg_2&070) == 010 && $3 == 001)
|
|
badoperand();
|
|
T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2((0120000|($3<<10)),0,0,0);
|
|
ea_2(SIZE_L, ALT);
|
|
}
|
|
| FMOVE fsize FPREG ',' FPREG
|
|
{ emit2(0170000|co_id);
|
|
emit2(($3<<10)|($5<<7));
|
|
}
|
|
| FMOVE fsize ea ',' FPREG
|
|
{ ch_sz_dreg($2, mrg_2&070);
|
|
T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2((0040000|($2<<10)|($5<<7)),0,0,0);
|
|
ea_2(SIZE_L, DTA);
|
|
}
|
|
| FMOVE fsize FPREG ',' ea
|
|
{ ch_sz_dreg($2, mrg_2&070);
|
|
if ($2 == FSIZE_P)
|
|
serror("packed decimal needs k-factor");
|
|
T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2((0060000|($2<<10)|($3<<7)),0,0,0);
|
|
ea_2(SIZE_L, DTA|ALT);
|
|
}
|
|
| FMOVE fsize FPREG ',' ea '{' '#' absexp '}'
|
|
{ check_fsize($2, FSIZE_P);
|
|
fit(sfit7($8));
|
|
T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2((0066000|($3<<7)|low7($8)),0,0,0);
|
|
ea_2(SIZE_L, MEM|DTA|ALT);
|
|
}
|
|
| FMOVE fsize FPREG ',' ea '{' DREG '}'
|
|
{ check_fsize($2, FSIZE_P);
|
|
T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2((0076000|($3<<7)|($7<<4)),0,0,0);
|
|
ea_2(SIZE_L, MEM|DTA|ALT);
|
|
}
|
|
| FMOVECR fsize '#' absexp ',' FPREG
|
|
{ fit(fit7($4));
|
|
check_fsize($2, FSIZE_X);
|
|
emit2(0170000|co_id);
|
|
emit2(056000|($6<<7)|low7($4));
|
|
}
|
|
| FMOVEM FSIZE fregs ',' notimmreg
|
|
{ check_fsize($2, FSIZE_X);
|
|
if ((mrg_2&070) == 030)
|
|
serror("bad addressing category");
|
|
T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2(0160000 |
|
|
(((mrg_2&070)==040 || ($3&04000)) ?
|
|
$3 :
|
|
(010000|reverse($3,8))),
|
|
0,0,0);
|
|
ea_2(SIZE_L, MEM|ALT);
|
|
}
|
|
| FMOVEM FSIZE notimmreg ',' fregs
|
|
{ check_fsize($2, FSIZE_X);
|
|
if ((mrg_2&070) == 040)
|
|
serror("bad addressing category");
|
|
T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2((0150000|(($5&04000)?$5:reverse($5,8))),0,0,0);
|
|
ea_2(SIZE_L, MEM);
|
|
}
|
|
| FMOVEM SIZE fcregs ',' ea
|
|
{ checksize($2, 4);
|
|
if ((mrg_2&070) == 1 && $3!= 02000)
|
|
serror("bad addressing category");
|
|
if ((mrg_2 & 070) == 0 &&
|
|
$3 != 02000 && $3 != 04000 && $3 != 010000)
|
|
serror("bad addressing category");
|
|
T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2((0120000|$3),0,0,0);
|
|
ea_2(SIZE_L, ALT);
|
|
}
|
|
| FMOVEM SIZE ea ',' fcregs
|
|
{ checksize($2, 4);
|
|
if ((mrg_2&070) == 1 && $5!= 02000)
|
|
serror("bad addressing category");
|
|
if ((mrg_2 & 070) == 0 &&
|
|
$5 != 02000 && $5 != 04000 && $5 != 010000)
|
|
serror("bad addressing category");
|
|
T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2((0100000|$5),0,0,0);
|
|
ea_2(SIZE_L, 0);
|
|
}
|
|
| FDYADIC fsize ea ',' FPREG
|
|
{ T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2((0040000|($2<<10)|($5<<7)|$1),0,0,0);
|
|
ch_sz_dreg($2, mrg_2&070);
|
|
ea_2(SIZE_L, DTA);
|
|
}
|
|
| FDYADIC fsize FPREG ',' FPREG
|
|
{ check_fsize($2, FSIZE_X);
|
|
emit2(0170000|co_id);
|
|
emit2(($3<<10)|($5<<7)|$1);
|
|
}
|
|
| FMONADIC fsize ea ',' FPREG
|
|
{ T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2((0040000|($2<<10)|($5<<7)|$1),0,0,0);
|
|
ch_sz_dreg($2, mrg_2&070);
|
|
ea_2(SIZE_L, DTA);
|
|
}
|
|
| FMONADIC fsize FPREG ',' FPREG
|
|
{ check_fsize($2, FSIZE_X);
|
|
emit2(0170000|co_id);
|
|
emit2(($3<<10)|($5<<7)|$1);
|
|
}
|
|
| FMONADIC fsize FPREG
|
|
{ check_fsize($2, FSIZE_X);
|
|
emit2(0170000|co_id);
|
|
emit2(($3<<10)|($3<<7)|$1);
|
|
}
|
|
| FSINCOS fsize ea ',' FPREG ':' FPREG
|
|
{ T_EMIT2(0170000|co_id|mrg_2,0,0,0);
|
|
T_EMIT2(0040000|($2<<10)|($7<<7)|$1|$5,0,0,0);
|
|
ea_2(SIZE_L, DTA);
|
|
}
|
|
| FSINCOS fsize FPREG ',' FPREG ':' FPREG
|
|
{ check_fsize($2, FSIZE_X);
|
|
emit2(0170000|co_id);
|
|
emit2(($3<<10)|($7<<7)|$1|$5);
|
|
}
|
|
| FBCC expr
|
|
{ fbranch($1, $2);}
|
|
| FDBCC DREG ',' expr
|
|
{ T_EMIT2(0170110|co_id|$2,0,0,0);
|
|
T_EMIT2($1,0,0,0);
|
|
$4.val -= (DOTVAL+dot_offset);
|
|
fit(fitw($4.val));
|
|
T_EMIT2(loww($4.val), $4.typ,
|
|
RELPC|RELO2,relonami);
|
|
}
|
|
| FNOP
|
|
{ emit2(0170200|co_id);
|
|
emit2(0);
|
|
}
|
|
| FSCC ea
|
|
{ T_EMIT2(0170100|co_id|mrg_2,0,0,0);
|
|
T_EMIT2($1,0,0,0);
|
|
ea_2(SIZE_B, DTA|ALT);
|
|
}
|
|
| FTST fsize ea
|
|
{ T_EMIT2((0170000|co_id|mrg_2),0,0,0);
|
|
T_EMIT2((0040072|($2<<10)),0,0,0);
|
|
ch_sz_dreg($2, mrg_2&070);
|
|
ea_2(SIZE_L, DTA);
|
|
}
|
|
| FTST fsize FPREG
|
|
{ check_fsize($2, FSIZE_X);
|
|
emit2(0170000|co_id);
|
|
emit2(($3<<10)|072);
|
|
}
|
|
| FSAVRES ea
|
|
{ if ((mrg_2&070) == ($1&070))
|
|
badoperand();
|
|
T_EMIT2((0170000|co_id|($1&0700)|mrg_2),0,0,0);
|
|
ea_2(0, $1&07);
|
|
}
|
|
| FTRAPCC
|
|
{ emit2(0170174|co_id);
|
|
emit2($1);
|
|
}
|
|
| FTRAPCC SIZE imm
|
|
{ checksize($2, 2|4);
|
|
T_EMIT2((0170170|co_id|($2==SIZE_L?03:02)),
|
|
0,0,0);
|
|
T_EMIT2($1,0,0,0);
|
|
ea_2($2,0);
|
|
}
|
|
;
|
|
fregs : DREG
|
|
{ $$ = 04000 | $1 << 4; }
|
|
| frlist
|
|
;
|
|
frlist : frrange
|
|
| frlist '/' frrange
|
|
{ $$ = $1 | $3;}
|
|
;
|
|
frrange : FPREG
|
|
{ $$ = 1 << $1; }
|
|
| FPREG '-' FPREG
|
|
{ if ($1 > $3)
|
|
badoperand();
|
|
for ($$ = 0; $1 <= $3; $1++)
|
|
$$ |= (1 << $1);
|
|
}
|
|
;
|
|
fcregs : FPCR
|
|
{ $$ = $1 << 10; }
|
|
| fcregs '/' FPCR
|
|
{ $$ = $1 | ($3 << 10); }
|
|
;
|
|
fsize : /* empty */
|
|
{ $$ = FSIZE_X; }
|
|
| SIZE
|
|
{ if ($1 == SIZE_L)
|
|
$$ = FSIZE_L;
|
|
else if ($1 == SIZE_W)
|
|
$$ = FSIZE_W;
|
|
else $$ = FSIZE_B;
|
|
}
|
|
| FSIZE
|
|
;
|
|
mm_op : CP
|
|
{ co_id = $1; }
|
|
mm_op1
|
|
| { co_id = DEF_MM; }
|
|
mm_op1
|
|
;
|
|
mm_op1 : /* Coprocessor instructions; syntax may be changed (please).
|
|
* No coprocessor defined extension words are emitted.
|
|
*/
|
|
CPBCC cp_cond expr
|
|
{ cpbcc($1 | co_id | $2, $3);
|
|
}
|
|
| CPDBCC cp_cond DREG ',' expr
|
|
{ T_EMIT2($1 | co_id | $3,0,0,0);
|
|
$5.val -= (DOTVAL+dot_offset);
|
|
fit(fitw($5.val));
|
|
T_EMIT2(loww($5.val), $5.typ,
|
|
RELPC|RELO2, relonami);
|
|
}
|
|
| CPGEN
|
|
{ T_EMIT2($1 | co_id,0,0,0);
|
|
/* NO COMMAND WORD IS EMITTED;
|
|
* THIS INSTRUCTIONS IS (STILL) ONE BIG RIDDLE.
|
|
* NO EFFECTIVE ADDRESS IS CALCULATED (SYNTAX ?)
|
|
*/
|
|
}
|
|
| CPSAVREST ea
|
|
{ T_EMIT2($1 | co_id | mrg_2,0,0,0);
|
|
if ($1 & 0100) {
|
|
/* restore */
|
|
ea_2(SIZE_W, (mrg_2 & 070)==030 ? 0 : CTR);
|
|
}
|
|
else ea_2(SIZE_W,(mrg_2 & 070)==020 ? 0 : CTR|ALT);
|
|
}
|
|
| CPSCC cp_cond ea
|
|
{ T_EMIT2($1 | co_id | mrg_2,0,0,0);
|
|
T_EMIT2($2,0,0,0);
|
|
ea_2(SIZE_B,DTA|ALT);
|
|
}
|
|
| CPTRAPCC cp_cond SIZE imm
|
|
{ checksize($3,2|4);
|
|
T_EMIT2($1 | co_id | ($3>>6)+1,0,0,0);
|
|
T_EMIT2($2,0,0,0);
|
|
ea_2($3, 0);
|
|
}
|
|
| CPTRAPCC cp_cond
|
|
{ emit2($1 | co_id | 4);
|
|
emit2($2);
|
|
}
|
|
/* M68030 MMU instructions */
|
|
| PFLUSHA
|
|
{ emit2(0170000);
|
|
emit2($1);
|
|
}
|
|
| PFLUSH fc ',' mask
|
|
{ emit2(0170000);
|
|
emit2($1|010000|($4<<5)|$2);
|
|
}
|
|
| PFLUSH fc ',' mask ',' ea
|
|
{ T_EMIT2(0170000|mrg_2, 0, 0, 0);
|
|
T_EMIT2($1|014000|($4<<5)|$2, 0, 0, 0);
|
|
ea_2(SIZE_L, DTA|CTR);
|
|
}
|
|
| PTEST fc ',' ea ',' mask
|
|
{ T_EMIT2(0170000|mrg_2, 0, 0, 0);
|
|
T_EMIT2($1|($6<<10)|$2, 0, 0, 0);
|
|
ea_2(SIZE_L, DTA|CTR);
|
|
}
|
|
| PTEST fc ',' ea ',' mask ',' AREG
|
|
{ T_EMIT2(0170000|mrg_2, 0, 0, 0);
|
|
T_EMIT2($1|($6<<10)|$2|0400|($8<<5), 0, 0, 0);
|
|
ea_2(SIZE_L, DTA|CTR);
|
|
}
|
|
| PLOAD fc ',' ea
|
|
{ T_EMIT2(0170000|mrg_2, 0, 0, 0);
|
|
T_EMIT2($1|$2, 0, 0, 0);
|
|
ea_2(SIZE_L, DTA|CTR);
|
|
}
|
|
| PMOVE MREG ',' ea
|
|
{ T_EMIT2(0170000|mrg_2, 0, 0, 0);
|
|
T_EMIT2($1|$2|01000, 0, 0, 0);
|
|
ea_2(SIZE_L, DTA|CTR);
|
|
}
|
|
| PMOVE ea ',' MREG
|
|
{ T_EMIT2(0170000|mrg_2, 0, 0, 0);
|
|
T_EMIT2($1|$4, 0, 0, 0);
|
|
ea_2(SIZE_L, DTA|CTR);
|
|
}
|
|
;
|
|
|
|
mask : '#' absexp
|
|
{ fit(fit3($2));
|
|
$$ = low3($2);
|
|
}
|
|
;
|
|
|
|
fc : '#' absexp
|
|
{ fit(fit3($2));
|
|
$$ = (020|low3($2));
|
|
}
|
|
| DREG
|
|
{ $$ = (010|$1); }
|
|
| CREG
|
|
{ if ($1 > 1) serror("illegal control register");
|
|
$$ = ($1&01);
|
|
}
|
|
;
|
|
|
|
cp_cond : DOT absexp
|
|
{ fit(fit6($2));
|
|
$$ = low6($2);
|
|
}
|
|
;
|