home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
C/C++ Interactive Guide
/
c-cplusplus-interactive-guide.iso
/
c_ref
/
csource4
/
204_01
/
outcode.c
< prev
next >
Wrap
Text File
|
1979-12-31
|
12KB
|
407 lines
#include <stdio.h>
#include "c.h"
#include "expr.h"
#include "gen.h"
#include "cglbdec.h"
/*
* 68000 C compiler
*
* Copyright 1984, 1985, 1986 Matthew Brandt.
* all commercial rights reserved.
*
* This compiler is intended as an instructive tool for personal use. Any
* use for profit without the written consent of the author is prohibited.
*
* This compiler may be distributed freely for non-commercial use as long
* as this notice stays intact. Please forward any enhancements or questions
* to:
*
* Matthew Brandt
* Box 920337
* Norcross, Ga 30092
*/
/* variable initialization */
enum e_gt { nogen, bytegen, wordgen, longgen };
enum e_sg { noseg, codeseg, dataseg };
int gentype = nogen;
int curseg = noseg;
int outcol = 0;
struct oplst {
char *s;
int ov;
} opl[] =
{ {"mov",op_move}, {"mov",op_moveq}, {"add",op_add},
{"add",op_addi}, {"add",op_addq}, {"sub",op_sub},
{"sub",op_subi}, {"sub",op_subq}, {"and",op_and},
{"or",op_or}, {"eor",op_eor}, {"muls",op_muls},
{"divs",op_divs}, {"swap",op_swap}, {"beq",op_beq},
{"bhi",op_bhi}, {"bhs",op_bhs}, {"blo",op_blo},
{"bls",op_bls}, {"mulu",op_mulu}, {"divu",op_divu},
{"bne",op_bne}, {"blt",op_blt}, {"ble",op_ble},
{"bgt",op_bgt}, {"bge",op_bge}, {"neg",op_neg},
{"not",op_not}, {"cmp",op_cmp}, {"ext",op_ext},
{"jmp",op_jmp}, {"jsr",op_jsr}, {"rts",op_rts},
{"lea",op_lea}, {"asr",op_asr}, {"asl",op_asl},
{"clr",op_clr}, {"link",op_link}, {"unlk",op_unlk},
{"bra",op_bra}, {"movm",op_movem}, {"pea",op_pea},
{"cmp",op_cmpi}, {"tst",op_tst}, {"dc",op_dc},
{0,0} };
putop(op)
int op;
{ int i;
i = 0;
while( opl[i].s )
{
if( opl[i].ov == op )
{
fprintf(output,"\t%s",opl[i].s);
return;
}
++i;
}
printf("DIAG - illegal opcode.\n");
}
putconst(offset)
/*
* put a constant to the output file.
*/
struct enode *offset;
{ switch( offset->nodetype )
{
case en_autocon:
case en_icon:
fprintf(output,"%d",offset->v.i);
break;
case en_labcon:
fprintf(output,"L%%%d",offset->v.i);
break;
case en_nacon:
fprintf(output,"%s",offset->v.p[0]);
break;
case en_add:
putconst(offset->v.p[0]);
fprintf(output,"+");
putconst(offset->v.p[1]);
break;
case en_sub:
putconst(offset->v.p[0]);
fprintf(output,"-");
putconst(offset->v.p[1]);
break;
case en_uminus:
fprintf(output,"-");
putconst(offset->v.p[0]);
break;
default:
printf("DIAG - illegal constant node.\n");
break;
}
}
putlen(l)
/*
* append the length field to an instruction.
*/
int l;
{ switch( l )
{
case 0:
break; /* no length field */
case 1:
fprintf(output,".b");
break;
case 2:
fprintf(output,".w");
break;
case 4:
fprintf(output,".l");
break;
default:
printf("DIAG - illegal length field.\n");
break;
}
}
putamode(ap)
/*
* output a general addressing mode.
*/
struct amode *ap;
{ switch( ap->mode )
{
case am_immed:
fprintf(output,"&");
case am_direct:
putconst(ap->offset);
break;
case am_areg:
fprintf(output,"%%a%d",ap->preg);
break;
case am_dreg:
fprintf(output,"%%d%d",ap->preg);
break;
case am_ind:
fprintf(output,"(%%a%d)",ap->preg);
break;
case am_ainc:
fprintf(output,"(%%a%d)+",ap->preg);
break;
case am_adec:
fprintf(output,"-(%%a%d)",ap->preg);
break;
case am_indx:
putconst(ap->offset);
fprintf(output,"(%%a%d)",ap->preg);
break;
case am_xpc:
putconst(ap->offset);
fprintf(output,"(%%d%d,%%pc)",ap->preg);
break;
case am_indx2:
putconst(ap->offset);
fprintf(output,"(%%a%d,%%d%d.l)",ap->preg,ap->sreg);
break;
case am_indx3:
putconst(ap->offset);
fprintf(output,"(%%a%d,%%a%d.l)",ap->preg,ap->sreg);
break;
case am_mask:
put_mask(ap->offset);
break;
default:
printf("DIAG - illegal address mode.\n");
break;
}
}
put_code(op,len,aps,apd)
/*
* output a generic instruction.
*/
struct amode *aps, *apd;
int op, len;
{ if( op == op_dc )
{
switch( len )
{
case 1: fprintf(output,"\tbyte"); break;
case 2: fprintf(output,"\tshort"); break;
case 4: fprintf(output,"\tlong"); break;
}
}
else
{
putop(op);
putlen(len);
}
if( aps != 0 )
{
fprintf(output,"\t");
if( op == op_cmp || op == op_cmpi )
putamode( apd );
else
putamode(aps);
if( apd != 0 )
{
fprintf(output,",");
if( op == op_cmp || op == op_cmpi )
putamode( aps );
else
putamode(apd);
}
}
fprintf(output,"\n");
}
put_mask(mask)
/*
* generate a register mask for restore and save.
*/
int mask;
{ int i;
fprintf(output,"&0x%04x",mask);
}
putreg(r)
/*
* generate a register name from a tempref number.
*/
int r;
{ if( r < 8 )
fprintf(output,"D%d",r);
else
fprintf(output,"A%d",r - 8);
}
gen_strlab(s)
/*
* generate a named label.
*/
char *s;
{ fprintf(output,"%s:\n",s);
}
put_label(lab)
/*
* output a compiler generated label.
*/
int lab;
{ fprintf(output,"L%%%d:\n",lab);
}
genbyte(val)
int val;
{ if( gentype == bytegen && outcol < 60) {
fprintf(output,",%d",val & 0x00ff);
outcol += 4;
}
else {
nl();
fprintf(output,"\tbyte\t%d",val & 0x00ff);
gentype = bytegen;
outcol = 19;
}
}
genword(val)
int val;
{ if( gentype == wordgen && outcol < 58) {
fprintf(output,",%d",val &