NetBSD-5.0.2/usr.sbin/gspa/gspa/gsp_act.c

Compare this file to the similar file:
Show the results in this format:

/*	$NetBSD: gsp_act.c,v 1.5 2006/08/26 18:15:37 christos Exp $	*/
/*
 * GSP assembler - semantic actions
 *
 * Copyright (c) 1993 Paul Mackerras.
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 * 3. All advertising materials mentioning features or use of this software
 *    must display the following acknowledgement:
 *      This product includes software developed by Paul Mackerras.
 * 4. The name of the author may not be used to endorse or promote products
 *    derived from this software without specific prior written permission
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

#include <sys/cdefs.h>
#ifndef lint
__RCSID("$NetBSD: gsp_act.c,v 1.5 2006/08/26 18:15:37 christos Exp $");
#endif

#include <stdlib.h>
#include <string.h>
#include <util.h>
#include "gsp_ass.h"

void
free_operands(operand l)
{
	operand op, oq;

	for( op = l; op != NULL; op = oq ){
		oq = op->next;
		if( op->type == EXPR ||
		    (op->type == EA && op->mode >= M_INDEX) )
			free_expr(op->op_u.value);
		free(op);
	}
}

operand
add_operand(operand first, operand last)
{
	operand p;

	for( p = first; p->next != NULL; p = p->next )
		;
	p->next = last;
	return first;
}

operand
reg_op(int reg)
{
	operand o;

/*	printf("reg_op reg=%d sign=%d\n", reg, sign);	*/
	new(o);
	o->type = REG;
	o->reg_no = reg;
	o->next = NULL;
	return o;
}

operand
expr_op(expr val)
{
	operand o;

/*	printf("immed len=%d\n", len);	*/
	new(o);
	o->type = EXPR;
	o->op_u.value = val;
	o->next = NULL;
	return o;
}

operand
string_op(char *str)
{
	operand o;

/*	printf("string_op str=%s\n", str);	*/
	new(o);
	o->type = STR_OPN;
	o->op_u.string = str;
	o->next = NULL;
	return o;
}

operand
abs_adr(expr adr)
{
	operand o;

/*	printf("abs_adr len=%d\n", len);	*/
	new(o);
	o->type = EA;
	o->mode = M_ABSOLUTE;
	o->op_u.value = adr;
	o->next = NULL;
	return o;
}

operand
reg_ind(int reg, int mode)
{
	operand o;

/*	printf("reg_adr r1=%d r2=%d mode=%d\n", r1, r2, mode);	*/
	new(o);
	o->type = EA;
	o->mode = mode;
	o->reg_no = reg;
	o->next = NULL;
	return o;
}

operand
reg_indxy(int reg, char *xy)
{
	ucasify(xy);
	if( strcmp(xy, ".XY") != 0 )
		perr("Register format must be .XY");
	return reg_ind(reg, M_INDXY);
}

operand
reg_index(int reg, expr disp)
{
	operand o;

	o = reg_ind(reg, M_INDEX);
	o->op_u.value = disp;
	return o;
}

expr
id_expr(char *id)
{
	expr x;

/*	printf("id_expr id=%s\n", id);	*/
	new(x);
	x->e_op = SYM;
	x->e_sym = lookup(id, TRUE);
	return x;
}

expr
num_expr(int val)
{
	expr x;

/*	printf("num_expr val=%d\n", val);	*/
	new(x);
	x->e_op = CONST;
	x->e_val = val;
	return x;
}

expr
here_expr()
{
	expr x;

/*	printf("here_expr()\n");	*/
	new(x);
	x->e_op = '.';
	return x;
}

expr
bexpr(int op, expr l, expr r)
{
	expr x;

/*	printf("bexpr op=%d\n", op);	*/
	new(x);
	x->e_op = op;
	x->e_left = l;
	x->e_right = r;
	return x;
}

void
free_expr(expr x)
{
	if( x->e_op != SYM && x->e_op != CONST && x->e_op != '.' ){
		free_expr(x->e_left);
		if( x->e_right != NULL )
			free_expr(x->e_right);
	}
	free(x);
}