home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Fresh Fish 8
/
FreshFishVol8-CD2.bin
/
bbs
/
gnu
/
f2c-1993.04.28-src.lha
/
f2c-1993.04.28
/
src
/
output.c
< prev
next >
Wrap
C/C++ Source or Header
|
1993-04-28
|
38KB
|
1,492 lines
/****************************************************************
Copyright 1990, 1991, 1992, 1993 by AT&T Bell Laboratories and Bellcore.
Permission to use, copy, modify, and distribute this software
and its documentation for any purpose and without fee is hereby
granted, provided that the above copyright notice appear in all
copies and that both that the copyright notice and this
permission notice and warranty disclaimer appear in supporting
documentation, and that the names of AT&T Bell Laboratories or
Bellcore or any of their entities not be used in advertising or
publicity pertaining to distribution of the software without
specific, written prior permission.
AT&T and Bellcore disclaim all warranties with regard to this
software, including all implied warranties of merchantability
and fitness. In no event shall AT&T or Bellcore be liable for
any special, indirect or consequential damages or any damages
whatsoever resulting from loss of use, data or profits, whether
in an action of contract, negligence or other tortious action,
arising out of or in connection with the use or performance of
this software.
****************************************************************/
#include "defs.h"
#include "names.h"
#include "output.h"
#ifndef TRUE
#define TRUE 1
#endif
#ifndef FALSE
#define FALSE 0
#endif
char _assoc_table[] = { 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0 };
/* Opcode table -- This array is indexed by the OP_____ macros defined in
defines.h; these macros are expected to be adjacent integers, so that
this table is as small as possible. */
table_entry opcode_table[] = {
{ 0, 0, NULL },
/* OPPLUS 1 */ { BINARY_OP, 12, "%l + %r" },
/* OPMINUS 2 */ { BINARY_OP, 12, "%l - %r" },
/* OPSTAR 3 */ { BINARY_OP, 13, "%l * %r" },
/* OPSLASH 4 */ { BINARY_OP, 13, "%l / %r" },
/* OPPOWER 5 */ { BINARY_OP, 0, "power (%l, %r)" },
/* OPNEG 6 */ { UNARY_OP, 14, "-%l" },
/* OPOR 7 */ { BINARY_OP, 4, "%l || %r" },
/* OPAND 8 */ { BINARY_OP, 5, "%l && %r" },
/* OPEQV 9 */ { BINARY_OP, 9, "%l == %r" },
/* OPNEQV 10 */ { BINARY_OP, 9, "%l != %r" },
/* OPNOT 11 */ { UNARY_OP, 14, "! %l" },
/* OPCONCAT 12 */ { BINARY_OP, 0, "concat (%l, %r)" },
/* OPLT 13 */ { BINARY_OP, 10, "%l < %r" },
/* OPEQ 14 */ { BINARY_OP, 9, "%l == %r" },
/* OPGT 15 */ { BINARY_OP, 10, "%l > %r" },
/* OPLE 16 */ { BINARY_OP, 10, "%l <= %r" },
/* OPNE 17 */ { BINARY_OP, 9, "%l != %r" },
/* OPGE 18 */ { BINARY_OP, 10, "%l >= %r" },
/* OPCALL 19 */ { BINARY_OP, 15, SPECIAL_FMT },
/* OPCCALL 20 */ { BINARY_OP, 15, SPECIAL_FMT },
/* Left hand side of an assignment cannot have outermost parens */
/* OPASSIGN 21 */ { BINARY_OP, 2, "%l = %r" },
/* OPPLUSEQ 22 */ { BINARY_OP, 2, "%l += %r" },
/* OPSTAREQ 23 */ { BINARY_OP, 2, "%l *= %r" },
/* OPCONV 24 */ { BINARY_OP, 14, "%l" },
/* OPLSHIFT 25 */ { BINARY_OP, 11, "%l << %r" },
/* OPMOD 26 */ { BINARY_OP, 13, "%l %% %r" },
/* OPCOMMA 27 */ { BINARY_OP, 1, "%l, %r" },
/* Don't want to nest the colon operator in parens */
/* OPQUEST 28 */ { BINARY_OP, 3, "%l ? %r" },
/* OPCOLON 29 */ { BINARY_OP, 3, "%l : %r" },
/* OPABS 30 */ { UNARY_OP, 0, "abs(%l)" },
/* OPMIN 31 */ { BINARY_OP, 0, SPECIAL_FMT },
/* OPMAX 32 */ { BINARY_OP, 0, SPECIAL_FMT },
/* OPADDR 33 */ { UNARY_OP, 14, "&%l" },
/* OPCOMMA_ARG 34 */ { BINARY_OP, 15, SPECIAL_FMT },
/* OPBITOR 35 */ { BINARY_OP, 6, "%l | %r" },
/* OPBITAND 36 */ { BINARY_OP, 8, "%l & %r" },
/* OPBITXOR 37 */ { BINARY_OP, 7, "%l ^ %r" },
/* OPBITNOT 38 */ { UNARY_OP, 14, "~ %l" },
/* OPRSHIFT 39 */ { BINARY_OP, 11, "%l >> %r" },
/* This isn't quite right -- it doesn't handle arrays, for instance */
/* OPWHATSIN 40 */ { UNARY_OP, 14, "*%l" },
/* OPMINUSEQ 41 */ { BINARY_OP, 2, "%l -= %r" },
/* OPSLASHEQ 42 */ { BINARY_OP, 2, "%l /= %r" },
/* OPMODEQ 43 */ { BINARY_OP, 2, "%l %%= %r" },
/* OPLSHIFTEQ 44 */ { BINARY_OP, 2, "%l <<= %r" },
/* OPRSHIFTEQ 45 */ { BINARY_OP, 2, "%l >>= %r" },
/* OPBITANDEQ 46 */ { BINARY_OP, 2, "%l &= %r" },
/* OPBITXOREQ 47 */ { BINARY_OP, 2, "%l ^= %r" },
/* OPBITOREQ 48 */ { BINARY_OP, 2, "%l |= %r" },
/* OPPREINC 49 */ { UNARY_OP, 14, "++%l" },
/* OPPREDEC 50 */ { UNARY_OP, 14, "--%l" },
/* OPDOT 51 */ { BINARY_OP, 15, "%l.%r" },
/* OPARROW 52 */ { BINARY_OP, 15, "%l -> %r"},
/* OPNEG1 53 */ { UNARY_OP, 14, "-%l" },
/* OPDMIN 54 */ { BINARY_OP, 0, "dmin(%l,%r)" },
/* OPDMAX 55 */ { BINARY_OP, 0, "dmax(%l,%r)" },
/* OPASSIGNI 56 */ { BINARY_OP, 2, "%l = &%r" },
/* OPIDENTITY 57 */ { UNARY_OP, 15, "%l" },
/* OPCHARCAST 58 */ { UNARY_OP, 14, "(char *)&%l" },
/* OPDABS 59 */ { UNARY_OP, 0, "dabs(%l)" },
/* OPMIN2 60 */ { BINARY_OP, 0, "min(%l,%r)" },
/* OPMAX2 61 */ { BINARY_OP, 0, "max(%l,%r)" },
/* kludge to imitate (under forcedouble) f77's bizarre treatement of OPNEG... */
/* OPNEG KLUDGE */ { UNARY_OP, 14, "-(doublereal)%l" }
}; /* opcode_table */
#define OPNEG_KLUDGE (sizeof(opcode_table)/sizeof(table_entry) - 1)
static char opeqable[sizeof(opcode_table)/sizeof(table_entry)];
static void output_prim ();
static void output_unary (), output_binary (), output_arg_list ();
static void output_list (), output_literal ();
void expr_out (fp, e)
FILE *fp;
expptr e;
{
if (e == (expptr) NULL)
return;
switch (e -> tag) {
case TNAME: out_name (fp, (struct Nameblock *) e);
return;
case TCONST: out_const(fp, &e->constblock);
goto end_out;
case TEXPR:
break;
case TADDR: out_addr (fp, &(e -> addrblock));
goto end_out;
case TPRIM: warn ("expr_out: got TPRIM");
output_prim (fp, &(e -> primblock));
return;
case TLIST: output_list (fp, &(e -> listblock));
end_out: frexpr(e);
return;
case TIMPLDO: err ("expr_out: got TIMPLDO");
return;
case TERROR:
default:
erri ("expr_out: bad tag '%d'", e -> tag);
} /* switch */
/* Now we know that the tag is TEXPR */
/* Optimize on simple expressions, such as "a = a + b" ==> "a += b" */
if (e -> exprblock.opcode == OPASSIGN && e -> exprblock.rightp &&
e -> exprblock.rightp -> tag == TEXPR) {
int opcode;
opcode = e -> exprblock.rightp -> exprblock.opcode;
if (opeqable[opcode]) {
expptr leftp, rightp;
if ((leftp = e -> exprblock.leftp) &&
(rightp = e -> exprblock.rightp -> exprblock.leftp)) {
if (same_ident (leftp, rightp)) {
expptr temp = e -> exprblock.rightp;
e -> exprblock.opcode = op_assign(opcode);
e -> exprblock.rightp = temp -> exprblock.rightp;
temp->exprblock.rightp = 0;
frexpr(temp);
} /* if same_ident (leftp, rightp) */
} /* if leftp && rightp */
} /* if opcode == OPPLUS || */
} /* if e -> exprblock.opcode == OPASSIGN */
/* Optimize on increment or decrement by 1 */
{
int opcode = e -> exprblock.opcode;
expptr leftp = e -> exprblock.leftp;
expptr rightp = e -> exprblock.rightp;
if (leftp && rightp && (leftp -> headblock.vstg == STGARG ||
ISINT (leftp -> headblock.vtype)) &&
(opcode == OPPLUSEQ || opcode == OPMINUSEQ) &&
ISINT (rightp -> headblock.vtype) &&
ISICON (e -> exprblock.rightp) &&
(ISONE (e -> exprblock.rightp) ||
e -> exprblock.rightp -> constblock.Const.ci == -1)) {
/* Allow for the '-1' constant value */
if (!ISONE (e -> exprblock.rightp))
opcode = (opcode == OPPLUSEQ) ? OPMINUSEQ : OPPLUSEQ;
/* replace the existing opcode */
if (opcode == OPPLUSEQ)
e -> exprblock.opcode = OPPREINC;
else
e -> exprblock.opcode = OPPREDEC;
/* Free up storage used by the right hand side */
frexpr (e -> exprblock.rightp);
e->exprblock.rightp = 0;
} /* if opcode == OPPLUS */
} /* block */
if (is_unary_op (e -> exprblock.opcode))
output_unary (fp, &(e -> exprblock));
else if (is_binary_op (e -> exprblock.opcode))
output_binary (fp, &(e -> exprblock));
else
erri ("expr_out: bad opcode '%d'", (int) e -> exprblock.opcode);
free((char *)e);
} /* expr_out */
void out_and_free_statement (outfile, expr)
FILE *outfile;
expptr expr;
{
if (expr)
expr_out (outfile, expr);
nice_printf (outfile, ";\n");
} /* out_and_free_statement */
int same_ident (left, right)
exppt