/*
 * Copyright (c) 2002, The Tendra Project <http://www.ten15.org/>
 * 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 unmodified, 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.
 *
 * 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.
 *
 *
 *    		 Crown Copyright (c) 1997
 *
 *    This TenDRA(r) Computer Program is subject to Copyright
 *    owned by the United Kingdom Secretary of State for Defence
 *    acting through the Defence Evaluation and Research Agency
 *    (DERA).  It is made available to Recipients with a
 *    royalty-free licence for its use, reproduction, transfer
 *    to other parties and amendment for any purpose not excluding
 *    product development provided that any such use et cetera
 *    shall be deemed to be acceptance of the following conditions:-
 *
 *        (1) Its Recipients shall ensure that this Notice is
 *        reproduced upon any copies or amended versions of it;
 *
 *        (2) Any amended version of it shall be clearly marked to
 *        show both the nature of and the organisation responsible
 *        for the relevant amendment or amendments;
 *
 *        (3) Its onward transfer from a recipient to another
 *        party shall be deemed to be that party's acceptance of
 *        these conditions;
 *
 *        (4) DERA gives no warranty or assurance as to its
 *        quality or suitability for any purpose and DERA accepts
 *        no liability whatsoever in relation to any use to which
 *        it may be put.
 *
 * $TenDRA: tendra/src/tools/disp/capsule.c,v 1.11 2005/11/05 12:03:14 stefanf Exp $
 */


#include "config.h"
#include "fmm.h"
#include "msgcat.h"
#include "tdf_types.h"
#include "tdf_stream.h"

#include "types.h"
#include "basic.h"
#include "binding.h"
#include "capsule.h"
#include "file.h"
#include "sort.h"
#include "tdf.h"
#include "tree.h"
#include "unit.h"


/*
 *    EXPANSION MODE
 *
 *    The flag dumb_mode determines whether the pretty-printer will
 *    attempt to rationalize its input, or whether it will just blindly
 *    output what it reads.  If show_stuff is true information on the
 *    number of tags and tokens etc will be output.
 */

BoolT dumb_mode = FALSE;
BoolT show_stuff = FALSE;


/*
 *    SKIP PASS FOR TOKEN DEFINITIONS FLAG
 *
 *    Tokens may be defined recursively without the necessary previous
 *    declarations to make this simple.  Thus the token definitions are
 *    given a preliminary pass to extract the declaration information.
 *    If show_skip is true the information read on the skip pass is
 *    displayed.  skipping is true iff we are in the skip pass.
 */

BoolT show_skip = 0;
int skipping = 0;


/*
 *    ARRAYS OF EQUATION AND VARIABLE NAMES
 *
 *    These arrays store the names of the types of equations (e.g. tokdefs)
 *    and the types of variables these equations are in (e.g. tags and tokens).
 *    The number of types of variables is given by no_variables.
 */

string *eqn_types = null;
string *var_types = null;
char *var_letters = "";
long *var_count = null;
long no_variables = 0;


/*
 *    CURRENT BINDINGS
 *
 *    The current binding table is given by crt_binding.
 */

binding *crt_binding;


/*
 *    DECODE AN EXTERNAL LINKAGE
 *
 *    An external name is decoded and associated with a variable of type v.
 *    r gives the order information.
 */

static void
de_linkextern(long v, long r)
{
	object *p;
	external ext;
	long n = tdf_int ();
	ext = de_extern_name ();
	p = new_object (v);
	set_binding (crt_binding, v, n, p);
	p->named = 1;
	p->name = ext;
	p->order = r;
	if (v == var_token && ext.simple) {
		/* Look for special tokens */
		char *nm = ext.val.str;
		if (nm [0] == '~' && diagnostics) {
			if (streq (nm, "~dg_exp")) {
				token_sort (p, sort_exp, "xG", n);
			} else if (streq (nm, "~exp_to_source")) {
				token_sort (p, sort_exp, "xMM", n);
			} else if (streq (nm, "~diag_id_scope")) {
				token_sort (p, sort_exp, "x$xd", n);
			} else if (streq (nm, "~diag_type_scope")) {
				token_sort (p, sort_exp, "x$d", n);
			} else if (streq (nm, "~diag_tag_scope")) {
				token_sort (p, sort_exp, "x$d", n);
			}
		}
	}
	if (dumb_mode) {
		word *w1, *w2;
		w1 = new_word (HORIZ_NONE);
		out_string (var_types [v]);
		w2 = new_word (HORIZ_BRACKETS);
		out_int (n);
		end_word (w2);
		out_string ("represents ");
		if (ext.simple) {
			out (ext.val.str);
		} else {
			out_unique (ext.val.uniq);
		}
		end_word (w1);
	}
	return;
}


/*
 *    COUNT OF TOTAL NUMBER OF EQUATIONS OF A PARTICULAR TYPE
 *
 *    This should be increased appropriately by each unit decoding function.
 */

long total = 0;
long blank_lines = 0;


/*
 *    DECODE A SET OF EQUATIONS
 *
 *    A set of equations is decoded.  f gives the procedure which is to be
 *    used to decode the equation body.
 */

static void
de_equation(equation_func f)
{
	long i, n;
	long no_var;
	int needs_it = 0;
	static long unitno = 1;

	/* Record old bindings */
	binding *old_binding = crt_binding;

	/* Read the number of each type of variable */
	no_var = tdf_int ();
	if (no_var) {
		if (no_var != no_variables) {
			MSG_number_of_local_variables_wrong ();
		}
		crt_binding = new_binding_table ();
		for (i = 0; i < no_var; i++) {
			long sz = tdf_int ();
			set_binding_size (crt_binding, i, sz);
			if (show_stuff) {
				out_string (var_types [i]);
				out_string (" x ");
				out_int (sz);
			}
		}
		if (show_stuff) blank_line ();
	}

	/* Read linkage for each type of variable */
	n = tdf_int ();
	if (n != no_var) MSG_number_of_linkage_units_wrong ();
	if (no_var) {
		if (dumb_mode) {
			word *w = new_word (HORIZ_NONE);
			out_string ("Bindings for Unit ");
			out_int (unitno);
			out ("(inner->outer)");
			end_word (w);
			blank_line ();
		}
		for (i = 0; i < no_var; i++) {
			long j, no_links = tdf_int ();
			for (j = 0; j < no_links; j++) {
				object *p;
				long inner = tdf_int ();
				long outer = tdf_int ();
				if (dumb_mode) {
					/* Output the linkage information */
					word *w1, *w2;
					w1 = new_word (HORIZ_NONE);
					out_string (var_types [i]);
					w2 = new_word (HORIZ_BRACKETS);
					out_int (inner);
					end_word (w2);
					out_string ("is bound to ");
					out_string (var_types [i]);
					w2 = new_word (HORIZ_BRACKETS);
					out_int (outer);
					end_word (w2);
					end_word (w1);
					needs_it = 1;
				}
				p = find_binding (old_binding, i, outer);
				set_binding (crt_binding, i, inner, p);
			}
		}
		if (dumb_mode) {
			if (needs_it) blank_line ();
			blank_lines = 1;
		}

		/* Complete the bindings */
		complete_binding (crt_binding);
	}

	/* Read the unit body */
	n = BYTESIZE * tdf_int ();
	tdf_de_align (tdfr);
	if (f == null) {
		tdf_skip_bits (tdfr, n);
		if (dumb_mode) {
			out ("(skipped)");
			blank_line ();
			blank_lines = 1;
		}
		total++;
	} else {
		tdf_pos end = tdf_stream_tell (tdfr) + n;
		(*f) ();
		tdf_de_align (tdfr);
		if (tdf_stream_tell (tdfr) != end) MSG_unit_length_wrong ();
	}

	/* Restore old bindings */
	if (no_var) {
		free_binding_table (crt_binding);
		crt_binding = old_binding;
		if (dumb_mode) {
			for (i = blank_lines; i < 2; i++) blank_line ();
			out_string ("End of Bindings for Unit ");
			out_int (unitno++);
			blank_line ();
			blank_line ();
			blank_lines = 2;
		}
	}
	return;
}


/*
 *    DECODE A CAPSULE
 *
 *    A capsule consists of a number of equation types, a number of variable
 *    sorts, a number of external names for variables and a number of
 *    equations of certain types.
 */

void
de_capsule(void)
{
	long i, n;
	long no_eqn, no_var;
	if (dumb_mode) show_stuff = 1;

	/* Read the magic number */
	out ("MAGIC NUMBER");
	blank_line ();
	de_magic (version_magic);
	blank_line ();
	blank_line ();

	/* Read the equation types */
	no_eqn = tdf_int ();
	if (no_eqn) {
		if (show_stuff) {
			out ("EQUATION TYPES");
			blank_line ();
		}
		eqn_types = xmalloc_nof (string, no_eqn);
		for (i = 0; i < no_eqn; i++) {
			string s = de_tdfstring_align ();
			eqn_types [i] = s;
			if (show_stuff) out (s);
		}
		if (show_stuff) {
			blank_line ();
			blank_line ();
		}
	}

	/* Read the variable types and initialize the bindings */
	no_var = tdf_int ();
	no_variables = no_var;
	crt_binding = new_binding_table ();
	if (no_var) {
		if (show_stuff) {
			out ("VARIABLE TYPES");
			blank_line ();
		}
		var_types = xmalloc_nof (string, no_var);
		var_letters = xmalloc_nof (char, no_var + 1);
		var_count = xmalloc_nof (long, no_var);
		var_letters [no_var] = 0;

		for (i = 0; i < no_var; i++) {
			string sv = de_tdfstring_align ();
			long sz = tdf_int ();
			var_letters [i] = find_variable (sv, i);
			var_types [i] = sv;
			var_count [i] = 0;
			set_binding_size (crt_binding, i, sz);
			if (show_stuff) {
				out_string (sv);
				out_string (" x ");
				out_int (sz);
			}
		}
		if (show_stuff) {
			blank_line ();
			blank_line ();
		}
	}

	/* Read the external variable names */
	n = tdf_int ();
	if (n != no_var) MSG_number_of_variables_wrong ();
	if (no_var) {
		if (dumb_mode) {
			out ("EXTERNAL NAMES");
			blank_line ();
		}
		for (i = 0; i < no_var; i++) {
			long j, no_links = tdf_int ();
			for (j = 0; j < no_links; j++) de_linkextern (i, j);
		}
		if (dumb_mode) {
			blank_line ();
			blank_line ();
		}
	}

	/* Complete the bindings */
	complete_binding (crt_binding);

	/* Read the equations */
	n = tdf_int ();
	if (n != no_eqn) MSG_number_of_equations_wrong ();
	for (i = 0; i < no_eqn; i++) {
		int used = 0;
		char *title = null;
		long j, no_units = tdf_int ();
		string se = eqn_types [i];
		equation_func f = find_equation (se, &title, &used);
		if (!used) {
			title = null;
			f = null;
		}
		total = 0;
		if (f == de_tokdef_props && no_units) {
			/* Skip pass */
			tdf_pos pl;
			int old_pf = printflag;
			if (!show_skip) printflag = 0;
			skipping = 1;
			pl = tdf_stream_tell (tdfr);
			if (printflag && (dumb_mode || f)) {
				if (title && !show_stuff) {
					out_string (title);
				} else {
					out_string ("EQUATIONS OF TYPE ");
					out_string (se);
				}
				out (" (SKIP PASS)");
				blank_line ();
				blank_lines = 1;
			}
			for (j = 0; j < no_units; j++) de_equation (f);
			if (printflag && (dumb_mode || f)) {
				if (total == 0) {
					out ("(none)");
					blank_lines = 0;
				}
				for (j = blank_lines; j < 2; j++) blank_line ();
				blank_lines = 2;
			}
			total = 0;
			tdf_stream_seek(tdfr, pl);
			skipping = 0;
			printflag = old_pf;
		}

		/* Main pass */
		if (dumb_mode || f) {
			if (title && !show_stuff) {
				out (title);
			} else {
				out_string ("EQUATIONS OF TYPE ");
				out (se);
			}
			blank_line ();
			blank_lines = 1;
		}
		for (j = 0; j < no_units; j++) de_equation (f);
		if (dumb_mode || f) {
			if (total == 0) {
				out ("(none)");
				blank_lines = 0;
			}
			for (j = blank_lines; j < 2; j++) blank_line ();
			blank_lines = 2;
		}
	}
	return;
}


syntax highlighted by Code2HTML, v. 0.9.1