ftp.nice.ch/pub/next/developer/languages/ada/Adaed.1.11.s.tar.gz#/Adaed-1.11.0a/nodes.c

This is nodes.c in view mode; [Download] [Up]

/*
 * Copyright (C) 1985-1992  New York University
 * 
 * This file is part of the Ada/Ed-C system.  See the Ada/Ed README file for
 * warranty (none) and distribution info and also the GNU General Public
 * License for more details.

 */
/* nodes.c: C version of nodes.stl*/
#include "hdr.h"
#include "vars.h"
#include "setprots.h"
#include "nodesprots.h"
#include "miscprots.h"
#include "smiscprots.h"
#include "sspansprots.h"
#include "chapprots.h"

/*
 * Tree construction procedures
 *--------------------
 * 2. Lexical elements
 */

Node new_instance_node(Tuple value)						/*;new_instance_node*/
{
	/* construct an instance node used to hold tuples used for instantiations */

	Node	node;

	node = node_new(as_instance_tuple);
	N_VAL (node) = (char *) value;
	return node;
}

Node new_number_node(int value)							/*;new_number_node*/
{
	/* constructs an number node, used to hold small integer values used for
	 * attributes and return statement depth.
	 */

	Node	node;

	node = node_new(as_number);
	N_VAL (node) = (char *) value;
	return node;
}

Node new_subtype_decl_node(Symbol type_name)		/*;new_subtype_decl_node*/
{
	Node node;

	node         = node_new(as_subtype_decl);
	N_AST1(node) = new_name_node(type_name);
	N_AST2(node) = OPT_NODE;
	return node;
}

Node new_name_node(Symbol name)								/*;new_name_node*/
{
	/* constructs an as_simple_name node. */

	Node	node;

	node	     = node_new(as_simple_name);
	N_UNQ(node) = name;
	return node;
}

Node new_attribute_node(int attr, Node arg1, Node arg2, Symbol typ)
														/*;new_attribute_node*/
{
	/* Creates an attribute node. attr is the attribute's name.*/

	Node	id_node, attr_node;

	id_node	     = node_new(as_number);
	N_VAL (id_node)   = (char *) attr;
	attr_node	     = node_new(as_attribute);
	N_AST1(attr_node) = id_node;
	N_AST2(attr_node) = arg1;
	N_AST3(attr_node) = arg2;
	N_TYPE(attr_node) = typ;
	return attr_node;
}

/*
 *----------------------------------------
 * 4.5 Operators and expression evaluation
 */

Node new_unop_node(Symbol oper, Node arg1, Symbol typ)		/*;new_unop_node*/
{
	/* Creates a unary operator node. Oper is the operator's name */

	Node	id_node, list_node, node;

	id_node = new_name_node(oper);
	list_node = node_new(as_list);
	N_LIST(list_node) = tup_new1((char *) arg1);
	node = node_new(as_un_op);
	N_AST1(node) = id_node;
	N_AST2(node) = list_node;
	N_TYPE(node) = typ;
	return node;
}

/* this new procedure has the same functionalities as
new_check_bounds. It defines the tests that have to be performed at
execution time. We have been inspired by what has been done in
chapter 12 for the checking of parameters to generic units */

Tuple new_check_disc_node (Symbol type1, Symbol type2) /* new_check_disc_node */
{
	Tuple checks, g_list, a_list, g_discr_map;
	int i;
	Fortup ft;
	Node n, t, d;
	Symbol discr;

	checks = tup_new (0);

	g_list = discriminant_list(base_type(type1));
	a_list = discriminant_list(base_type(type2));
	g_discr_map = (Tuple) SIGNATURE(type1)[2];

	FORTUPI(discr=(Symbol), g_list, i, ft)
		n = node_new(as_check_discr);
		t = new_name_node(type2);
		d = new_name_node((Symbol) a_list[i]);
		N_AST1(n) = discr_map_get(g_discr_map, discr);
		N_AST2(n) = t;
		N_AST3(n) = d;
		checks = tup_with (checks, (char *) n);
	ENDFORTUP(ft);
	return checks;
}

Node new_check_bounds_node(Symbol type1, Symbol type2)
													/*;new_check_bounds_node */
{
	/* used to check that component types of array conversions have the same
	 * bounds. Should also be used in check_actual_constraint (12c)
	 */

	 Node n, t1, t2;

	 n  = node_new(as_check_bounds);
	 t1 = new_name_node(type1);
	 t2 = new_name_node(type2);
	 N_AST1(n) = t1;
	 N_AST2(n) = t2;
	 return n;
}

/*
 *------------------------------------------------------------
 * 5.1 Simple and compound statements - Sequence of statements
 */

Node new_statements_node(Tuple stmt_list)			/*;new_statements_node*/
{
	/* Creates an as_statements node, given a list (tuple) of statements */

	Node	stmt_node, list_node;

	stmt_node	     = node_new(as_statements);
	list_node	     = node_new(as_list);
	N_LIST(list_node) = stmt_list;
	N_AST1(stmt_node) = list_node;
	N_AST2(stmt_node) = OPT_NODE;
	return stmt_node;
}

/*
 *-------------------------
 * 5.2 Assignment statement
 */

Node new_assign_node(Node lhs, Node rhs)				/*;new_assign_node*/
{
	/* Creates an assign node */

	Node	node;

	node	       = node_new(as_assignment);
	N_AST1(node) = lhs;
	N_AST2(node) = rhs;
	return node;
}

/*
 *-----------------
 * 5.3 If statement
 */

Node new_call_node(Symbol proc_name, Tuple arg_list, Symbol type_name)
															/*;new_call_node*/
{
	/* Creates a call node.*/

	Node	list_node, call_node;

	list_node = node_new(as_list);
	N_LIST(list_node) = arg_list;

	call_node = node_new(as_call);
	N_TYPE(call_node) = type_name;
	N_AST1(call_node) = new_name_node(proc_name);
	N_AST2(call_node) = list_node;
	return call_node;
}

void make_insert_node(Node node, Tuple pre_list, Node post_node)
														/*;make_insert_node*/
{
	/* Transforms node into an insert node */

	int nk;

	nk = N_KIND(node);
	N_KIND(node) = as_insert;
	N_AST1(node) = post_node;
	if (N_AST2_DEFINED(nk)) N_AST2(node) = (Node)0;
	if (N_AST3_DEFINED(nk)) N_AST3(node) = (Node)0;
	if (N_AST4_DEFINED(nk))  /* copy ast4 if defined */
	   N_AST4(node) = (Node)0;
	/* or set n_type if post_node has n_type defined */
	else if (N_TYPE_DEFINED(N_KIND(post_node)))
	   N_TYPE(node) = N_TYPE(post_node);
	if (N_VAL_DEFINED(nk)) N_VAL (node) = (char *)0;
	N_LIST(node) = pre_list; /* TBSL: is copy needed  ds 7 nov */
}

Node copy_node(Node node)	/*;copy_node*/
{
	Node node2;

	node2 = node_new(N_KIND(node));
	copy_attributes(node, node2);
	if (is_terminal_node(N_KIND(node)))
	  copy_span(node, node2);/* TBSL: check whether this is always desirable */
	return node2;
}

void copy_attributes(Node old, Node newn)				/*;copy_attributes*/
{
	/*  copy attributes of old to new, preserving sequence and unit of new */

	char	*np, *op;
	Span	save_span;
	short	terminal;
	int i, n, nseq, nunit;

	nseq = N_SEQ(newn);
	nunit = N_UNIT(newn);
	terminal = (is_terminal_node(N_KIND(old)) && is_terminal_node(N_KIND(newn)));
	if (terminal) save_span = get_left_span(newn);
	n = sizeof(Node_s);
	op = (char *)old; np = (char*) newn;
	for (i=1; i <= n; i++) *np++ = *op++;
	N_UNIT(newn) = nunit;
	N_SEQ(newn) = nseq;
	if (terminal) set_span(newn, save_span);
}

void copy_span(Node old, Node newn)							/*;copy_span */
{
	/* retrieve left spans of old node and set spans fields of new node */
	
	Span span;

	span = get_left_span(old);
	N_SPAN0(newn) = span->line;
	N_SPAN1(newn) = span->col;
	efreet((char *) span, "spans");
}

void set_span(Node node, Span span)								/*;set_span */
{
	/* set span of node to the values in span */
	N_SPAN0(node) = span->line;
	N_SPAN1(node) = span->col;
}

Node copy_tree(Node node)										/*;copy_tree*/
{
	/* Create a full copy of the tree rooted at node, and return the new root*/

	Fortup	ft1;
	Tuple	tup;
	Node n, root;
	int i;

	if (node == (Node)0 || node == OPT_NODE) 
	return (node);
	root = node_new(N_KIND(node));
	N_KIND(root) = N_KIND(node); i = N_KIND(node);
	N_OVERLOADED(root) = N_OVERLOADED(node);
	N_VAL(root) = N_VAL(node);
	N_UNQ(root) = N_UNQ(node);
	N_NAMES(root) = N_NAMES(node);
	N_TYPE(root) = N_TYPE(node);
	N_PTYPES(root) = N_PTYPES(node);
	if (N_AST1_DEFINED(i)) N_AST1(root) = copy_tree(N_AST1(node));
	if (N_AST2_DEFINED(i)) N_AST2(root) = copy_tree(N_AST2(node));
	if (N_AST3_DEFINED(i)) N_AST3(root) = copy_tree(N_AST3(node));
	if (N_AST4_DEFINED(i)) N_AST4(root) = copy_tree(N_AST4(node));
	if (is_terminal_node((unsigned) i)) copy_span(node, root);

	if (N_LIST_DEFINED(i) && N_LIST(node) != (Tuple)0) {
	tup = tup_new(tup_size(N_LIST(node)));
	FORTUPI(n=(Node), N_LIST(node), i, ft1);
	    tup[i] = (char *) copy_tree(n);
	ENDFORTUP(ft1);
	N_LIST(root) = tup;
	}
	return root;
}

These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.