ack/lang/cem/cemcom/conversion.c

157 lines
3 KiB
C
Raw Normal View History

/*
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
* See the copyright notice in the ACK home directory, in the file "Copyright".
*/
1986-03-10 13:07:55 +00:00
/* $Header$ */
/* C O N V E R S I O N - C O D E G E N E R A T O R */
#include "lint.h"
#ifndef LINT
#include "nofloat.h"
#include <em.h>
1986-03-10 13:07:55 +00:00
#include "arith.h"
#include "type.h"
#include "sizes.h"
#include "Lpars.h"
#define T_SIGNED 1
#define T_UNSIGNED 2
#ifndef NOFLOAT
1986-03-10 13:07:55 +00:00
#define T_FLOATING 3
#endif NOFLOAT
1986-03-10 13:07:55 +00:00
/* conversion() generates the EM code for a conversion between
the types char, short, int, long, float, double and pointer.
1988-08-23 12:45:11 +00:00
There are three conversion types: signed, unsigned and floating.
1986-03-10 13:07:55 +00:00
The EM code to obtain this conversion looks like:
LOC sizeof(from_type)
LOC sizeof(to_type)
C??
*/
1988-08-23 12:45:11 +00:00
static int convtype();
1986-03-10 13:07:55 +00:00
conversion(from_type, to_type)
register struct type *from_type, *to_type;
1986-03-10 13:07:55 +00:00
{
register arith from_size = from_type->tp_size;
register arith to_size = to_type->tp_size;
1988-08-23 12:45:11 +00:00
int from_cnvtype = convtype(from_type);
int to_cnvtype = convtype(to_type);
1986-03-10 13:07:55 +00:00
1988-07-05 18:06:00 +00:00
if ((int)to_size < (int)word_size) to_size = word_size;
1988-08-23 12:45:11 +00:00
if ((int)from_size == (int)to_size && from_cnvtype == to_cnvtype)
1987-03-25 23:14:43 +00:00
return;
1988-08-23 12:45:11 +00:00
switch (from_cnvtype) {
1986-03-10 13:07:55 +00:00
case T_SIGNED:
1988-08-23 12:45:11 +00:00
switch (to_cnvtype) {
1986-03-10 13:07:55 +00:00
case T_SIGNED:
C_loc(from_size);
C_loc(to_size);
1986-03-10 13:07:55 +00:00
C_cii();
break;
case T_UNSIGNED:
#ifndef NOFLOAT
1986-03-10 13:07:55 +00:00
case T_FLOATING:
1987-03-25 23:14:43 +00:00
#endif NOOFLOAT
1988-07-05 18:06:00 +00:00
if ((int)from_size < (int)word_size) {
C_loc(from_size);
C_loc(word_size);
C_cii();
from_size = word_size;
}
C_loc(from_size);
C_loc(to_size);
1988-08-23 12:45:11 +00:00
if (to_cnvtype == T_UNSIGNED) C_ciu();
1987-03-25 23:14:43 +00:00
else C_cif();
1986-03-10 13:07:55 +00:00
break;
}
break;
case T_UNSIGNED:
1988-07-05 18:06:00 +00:00
if ((int)from_size < (int)word_size) from_size = word_size;
C_loc(from_size);
C_loc(to_size);
1988-08-23 12:45:11 +00:00
switch (to_cnvtype) {
1986-03-10 13:07:55 +00:00
case T_SIGNED:
C_cui();
break;
case T_UNSIGNED:
C_cuu();
break;
#ifndef NOFLOAT
1986-03-10 13:07:55 +00:00
case T_FLOATING:
C_cuf();
break;
#endif NOFLOAT
1986-03-10 13:07:55 +00:00
}
break;
#ifndef NOFLOAT
1986-03-10 13:07:55 +00:00
case T_FLOATING:
C_loc(from_size);
C_loc(to_size);
1988-08-23 12:45:11 +00:00
switch (to_cnvtype) {
1986-03-10 13:07:55 +00:00
case T_SIGNED:
C_cfi();
break;
case T_UNSIGNED:
C_cfu();
break;
case T_FLOATING:
C_cff();
break;
}
break;
#endif NOFLOAT
1986-03-10 13:07:55 +00:00
default:
crash("(conversion) illegal type conversion");
1988-09-16 23:19:50 +00:00
/*NOTREACHED*/
1986-03-10 13:07:55 +00:00
}
1988-07-05 18:06:00 +00:00
if ((int)(to_type->tp_size) < (int)word_size
#ifndef NOFLOAT
1988-08-23 12:45:11 +00:00
&& to_cnvtype != T_FLOATING
#endif NOFLOAT
) {
extern long full_mask[];
1988-08-23 12:45:11 +00:00
if (to_cnvtype == T_SIGNED) {
C_loc(to_type->tp_size);
C_loc(word_size);
C_cii();
}
else {
C_loc((arith) full_mask[(int)(to_type->tp_size)]);
C_and(word_size);
}
}
1986-03-10 13:07:55 +00:00
}
1988-08-23 12:45:11 +00:00
/* convtype() returns in which category a given type falls:
1986-03-10 13:07:55 +00:00
signed, unsigned or floating
*/
1988-08-23 12:45:11 +00:00
static int
convtype(tp)
register struct type *tp;
1986-03-10 13:07:55 +00:00
{
switch (tp->tp_fund) {
case CHAR:
case SHORT:
case INT:
case ERRONEOUS:
1986-03-10 13:07:55 +00:00
case LONG:
case ENUM:
return tp->tp_unsigned ? T_UNSIGNED : T_SIGNED;
#ifndef NOFLOAT
1986-03-10 13:07:55 +00:00
case FLOAT:
case DOUBLE:
return T_FLOATING;
#endif NOFLOAT
1988-08-23 12:45:11 +00:00
case POINTER:
return T_UNSIGNED;
1986-03-10 13:07:55 +00:00
}
return 0;
}
1988-08-19 13:55:22 +00:00
#endif LINT