dsIIr4/bin/
dsIIr4/extra/creremote/
dsIIr4/extra/wolfpaw/
dsIIr4/lib/cmds/admins/
dsIIr4/lib/cmds/common/
dsIIr4/lib/cmds/creators/include/
dsIIr4/lib/cmds/creators/include/SCCS/
dsIIr4/lib/daemon/services/
dsIIr4/lib/doc/
dsIIr4/lib/domains/Ylsrim/
dsIIr4/lib/domains/Ylsrim/adm/
dsIIr4/lib/domains/Ylsrim/armor/
dsIIr4/lib/domains/Ylsrim/broken/
dsIIr4/lib/domains/Ylsrim/fish/
dsIIr4/lib/domains/Ylsrim/meal/
dsIIr4/lib/domains/Ylsrim/npc/
dsIIr4/lib/domains/Ylsrim/virtual/
dsIIr4/lib/domains/Ylsrim/weapon/
dsIIr4/lib/domains/campus/adm/
dsIIr4/lib/domains/campus/etc/
dsIIr4/lib/domains/campus/meals/
dsIIr4/lib/domains/campus/npc/
dsIIr4/lib/domains/campus/save/
dsIIr4/lib/domains/campus/txt/
dsIIr4/lib/domains/campus/txt/ai/charles/
dsIIr4/lib/domains/campus/txt/ai/charles/bak2/
dsIIr4/lib/domains/campus/txt/ai/charles/bak2/bak1/
dsIIr4/lib/domains/campus/txt/ai/charly/
dsIIr4/lib/domains/campus/txt/ai/charly/bak/
dsIIr4/lib/domains/campus/txt/jenny/
dsIIr4/lib/domains/default/creator/
dsIIr4/lib/domains/default/doors/
dsIIr4/lib/domains/default/etc/
dsIIr4/lib/domains/default/virtual/
dsIIr4/lib/domains/default/weap/
dsIIr4/lib/domains/town/virtual/
dsIIr4/lib/lib/comp/
dsIIr4/lib/lib/lvs/
dsIIr4/lib/lib/user/
dsIIr4/lib/lib/virtual/
dsIIr4/lib/log/
dsIIr4/lib/obj/book_source/
dsIIr4/lib/obj/include/
dsIIr4/lib/realms/template/
dsIIr4/lib/realms/template/adm/
dsIIr4/lib/realms/template/area/armor/
dsIIr4/lib/realms/template/area/npc/
dsIIr4/lib/realms/template/area/obj/
dsIIr4/lib/realms/template/area/room/
dsIIr4/lib/realms/template/area/weap/
dsIIr4/lib/realms/template/bak/
dsIIr4/lib/realms/template/cmds/
dsIIr4/lib/save/
dsIIr4/lib/save/kills/o/
dsIIr4/lib/secure/cfg/classes/
dsIIr4/lib/secure/cmds/creators/include/
dsIIr4/lib/secure/cmds/players/
dsIIr4/lib/secure/cmds/players/include/
dsIIr4/lib/secure/daemon/include/
dsIIr4/lib/secure/lib/
dsIIr4/lib/secure/lib/include/
dsIIr4/lib/secure/lib/net/include/
dsIIr4/lib/secure/lib/std/
dsIIr4/lib/secure/modules/
dsIIr4/lib/secure/npc/
dsIIr4/lib/secure/obj/include/
dsIIr4/lib/secure/room/
dsIIr4/lib/secure/save/
dsIIr4/lib/secure/save/boards/
dsIIr4/lib/secure/save/players/g/
dsIIr4/lib/secure/tmp/
dsIIr4/lib/secure/verbs/creators/
dsIIr4/lib/shadows/
dsIIr4/lib/spells/
dsIIr4/lib/std/board/
dsIIr4/lib/std/lib/
dsIIr4/lib/tmp/
dsIIr4/lib/verbs/admins/include/
dsIIr4/lib/verbs/common/
dsIIr4/lib/verbs/common/include/
dsIIr4/lib/verbs/creators/include/
dsIIr4/lib/verbs/players/include/SCCS/
dsIIr4/lib/verbs/rooms/
dsIIr4/lib/verbs/rooms/include/
dsIIr4/lib/www/
dsIIr4/v22.2b14-dsouls2/
dsIIr4/v22.2b14-dsouls2/ChangeLog.old/
dsIIr4/v22.2b14-dsouls2/Win32/
dsIIr4/v22.2b14-dsouls2/compat/
dsIIr4/v22.2b14-dsouls2/compat/simuls/
dsIIr4/v22.2b14-dsouls2/include/
dsIIr4/v22.2b14-dsouls2/mudlib/
dsIIr4/v22.2b14-dsouls2/testsuite/
dsIIr4/v22.2b14-dsouls2/testsuite/clone/
dsIIr4/v22.2b14-dsouls2/testsuite/command/
dsIIr4/v22.2b14-dsouls2/testsuite/data/
dsIIr4/v22.2b14-dsouls2/testsuite/etc/
dsIIr4/v22.2b14-dsouls2/testsuite/include/
dsIIr4/v22.2b14-dsouls2/testsuite/inherit/
dsIIr4/v22.2b14-dsouls2/testsuite/inherit/master/
dsIIr4/v22.2b14-dsouls2/testsuite/log/
dsIIr4/v22.2b14-dsouls2/testsuite/single/
dsIIr4/v22.2b14-dsouls2/testsuite/single/tests/compiler/
dsIIr4/v22.2b14-dsouls2/testsuite/single/tests/efuns/
dsIIr4/v22.2b14-dsouls2/testsuite/single/tests/operators/
dsIIr4/v22.2b14-dsouls2/testsuite/u/
dsIIr4/v22.2b14-dsouls2/tmp/
dsIIr4/win32/
#include "std.h"
#include "lpc_incl.h"
#include "compiler.h"
#include "generate.h"
#include "swap.h"
#include "scratchpad.h"
#include "qsort.h"
#include "file.h"
#include "binaries.h"
/* This should be moved with initializers move out of here */
#include "icode.h"
#include "lex.h"
#include "simul_efun.h"

static void clean_parser PROT((void));
static void prolog PROT((int, char *));
static program_t *epilog PROT((void));
static void show_overload_warnings PROT((void));

#define CT(x) (1 << (x))
#define CT_SIMPLE(x) (CT(TYPE_ANY) | CT(x))

short compatible[11] = { 
    /* UNKNOWN */  0,
    /* ANY */      0xfff,
    /* NOVALUE to*/CT_SIMPLE(TYPE_NOVALUE) | CT(TYPE_VOID) | CT(TYPE_NUMBER),
    /* VOID to*/   CT_SIMPLE(TYPE_VOID) | CT(TYPE_NUMBER),
    /* NUMBER to*/ CT_SIMPLE(TYPE_NUMBER) | CT(TYPE_REAL),
    /* STRING */   CT_SIMPLE(TYPE_STRING),
    /* OBJECT */   CT_SIMPLE(TYPE_OBJECT),
    /* MAPPING */  CT_SIMPLE(TYPE_MAPPING),
    /* FUNCTION */ CT_SIMPLE(TYPE_FUNCTION),
    /* REAL */     CT_SIMPLE(TYPE_REAL) | CT(TYPE_NUMBER),
    /* BUFFER */   CT_SIMPLE(TYPE_BUFFER),
};

short is_type[11] = { 
    /* UNKNOWN */ 0,
    /* ANY */     0xfff,
    /* NOVALUE */ CT_SIMPLE(TYPE_NOVALUE) | CT(TYPE_VOID),
    /* VOID */    CT_SIMPLE(TYPE_VOID) | CT(TYPE_NOVALUE),
    /* NUMBER */  CT_SIMPLE(TYPE_NUMBER),
    /* STRING */  CT_SIMPLE(TYPE_STRING),
    /* OBJECT */  CT_SIMPLE(TYPE_OBJECT),
    /* MAPPING */ CT_SIMPLE(TYPE_MAPPING),
    /* FUNCTION */CT_SIMPLE(TYPE_FUNCTION),
    /* REAL */    CT_SIMPLE(TYPE_REAL),
    /* BUFFER */  CT_SIMPLE(TYPE_BUFFER),
};

mem_block_t mem_block[NUMAREAS];

parse_node_t *comp_trees[NUMTREES];
int comp_last_inherited;
int current_tree;

function_context_t function_context;

int exact_types, global_modifiers;

int current_type;
int var_defined;

unsigned short *comp_def_index_map, *func_index_map;
unsigned short *prog_flags, *comp_sorted_funcs;

char *prog_code;
char *prog_code_max;

program_t NULL_program;

program_t *prog;

static short string_idx[0x100];
unsigned char string_tags[0x20];
short freed_string;

/* x_ptr is different inside nested functions */
unsigned short *type_of_locals, *type_of_locals_ptr;
local_info_t *locals, *locals_ptr;

int locals_size = 0;
int type_of_locals_size = 0; 
int current_number_of_locals = 0;
int max_num_locals = 0;

/* This function has strput() semantics; see comments in simulate.c */
char *get_two_types P4(char *, where, char *, end, int, type1, int, type2)
{
    where = strput(where, end, "( ");
    where = get_type_name(where, end, type1);
    where = strput(where, end, "vs ");
    where = get_type_name(where, end, type2);
    where = strput(where, end, ")");

    return where;
}

void init_locals()
{
    type_of_locals = CALLOCATE(CFG_MAX_LOCAL_VARIABLES,unsigned short, 
			       TAG_LOCALS, "init_locals:1");
    locals = CALLOCATE(CFG_MAX_LOCAL_VARIABLES, local_info_t, 
		       TAG_LOCALS, "init_locals:2");
    type_of_locals_ptr = type_of_locals;
    locals_ptr = locals;
    locals_size = type_of_locals_size = CFG_MAX_LOCAL_VARIABLES;
    current_number_of_locals = max_num_locals = 0;
}

void free_all_local_names P1(int, flag)
{
    int i;

    for (i=0; i < current_number_of_locals; i++) {
        if (flag && (type_of_locals_ptr[locals_ptr[i].runtime_index] & LOCAL_MOD_UNUSED)) {
	    char buf[256];
	    char *end = EndOf(buf);
	    char *p;
	  
	    p = strput(buf, end, "Unused local variable '");
	    p = strput(p, end, locals_ptr[i].ihe->name);
	    p = strput(p, end, "'");
	    yywarn(buf);
	}
	locals_ptr[i].ihe->sem_value--;
	locals_ptr[i].ihe->dn.local_num = -1;
    }
    current_number_of_locals = 0;
    max_num_locals = 0;
}

void deactivate_current_locals() {
    int i;

    for (i = 0; i < current_number_of_locals; i++)
	locals_ptr[i].ihe->dn.local_num = -1;
}

void reactivate_current_locals() {
    int i;

    for (i = 0; i < current_number_of_locals; i++) {
	locals_ptr[i].ihe->dn.local_num = locals_ptr[i].runtime_index;
	locals_ptr[i].ihe->sem_value++;
    }
}

void clean_up_locals()
{
    int offset;

    offset = locals_ptr + current_number_of_locals - locals;
    while (offset--) {
	locals[offset].ihe->sem_value--;
	locals[offset].ihe->dn.local_num = -1;
    }
    current_number_of_locals = 0;
    max_num_locals = 0;
    locals_ptr = locals;
    type_of_locals_ptr = type_of_locals;
}

void pop_n_locals P1(int, num) {
    int lcur_start;
    int ltype_start, i1;

    DEBUG_CHECK(num < 0, "pop_n_locals called with num < 0");
    if (num == 0)
	return;
    
    lcur_start = current_number_of_locals -= num;
    ltype_start = locals_ptr[lcur_start].runtime_index;

    i1 = num;
    while (i1--) {
	if (type_of_locals_ptr[ltype_start] & LOCAL_MOD_UNUSED) {
	    char buf[256];
	    char *end = EndOf(buf);
	    char *p;
		    
	    p = strput(buf, end, "Unused local variable '");
	    p = strput(p, end, locals_ptr[lcur_start].ihe->name);
	    p = strput(p, end, "'");
	    yywarn(buf);
	}
	locals_ptr[lcur_start].ihe->sem_value--;
	locals_ptr[lcur_start].ihe->dn.local_num = -1;
        ++lcur_start;
	++ltype_start;
    }
}

int add_local_name P2(char *, str, int, type)
{
    if (max_num_locals == CFG_MAX_LOCAL_VARIABLES) {
	yyerror("Too many local variables");
	return 0;
    } else {
	ident_hash_elem_t *ihe;

	ihe = find_or_add_ident(str,FOA_NEEDS_MALLOC);
	type_of_locals_ptr[max_num_locals] = type;
	locals_ptr[current_number_of_locals].ihe = ihe;
	locals_ptr[current_number_of_locals++].runtime_index = max_num_locals;
	if (ihe->dn.local_num == -1)
	    ihe->sem_value++;
	return ihe->dn.local_num = max_num_locals++;
    }
}

void reallocate_locals() {
    int offset;
    offset = type_of_locals_ptr - type_of_locals;
    type_of_locals = RESIZE(type_of_locals, type_of_locals_size += CFG_MAX_LOCAL_VARIABLES,
			    unsigned short, TAG_LOCALS, "reallocate_locals:1");
    type_of_locals_ptr = type_of_locals + offset;
    offset = locals_ptr - locals;
    locals = RESIZE(locals, locals_size, 
		    local_info_t, TAG_LOCALS, "reallocate_locals:2");
    locals_ptr = locals + offset;
}

/* Fix a inherited class type for the current program */
static void fix_class_type P2(int *, t, program_t *, from) {
    ident_hash_elem_t *ihe;
    int num;

    if ((*t) & TYPE_MOD_CLASS) {
	ihe = lookup_ident(from->strings[from->classes[(*t) & CLASS_NUM_MASK].name]);
	if (ihe && ((num = ihe->dn.class_num) != -1)) {
	    (*t) = (*t & ~CLASS_NUM_MASK) | num;
	} else {
	    fatal("Cannot find class %s (%s,%d).\n",from->strings[from->classes[(*t) & CLASS_NUM_MASK].name], current_file, current_line);
	}
    }
}
		  

/*
 * Copy all variable names from the object that is inherited from.
 * It is very important that they are stored in the same order with the
 * same index.
 */
void copy_variables P2(program_t *, from, int, type) {
    int i;
    
    for (i = 0; i < from->num_inherited; i++) {
	int t = DECL_MODIFY(type, from->inherit[i].type_mod);
	
	/* 'private' vars become 'hidden' */
	if (from->inherit[i].type_mod & DECL_PRIVATE)
	  t = (t & ~DECL_PRIVATE) | DECL_HIDDEN;

	t &= ~FUNC_VARARGS; /* for 'varargs inherit' */

	copy_variables(from->inherit[i].prog, t);
    }
    for (i = 0; i < from->num_variables_defined; i++) {
	int t = DECL_MODIFY(from->variable_types[i], type);

	/* 'private' vars become 'hidden' */
	if (from->variable_types[i] & DECL_PRIVATE)
	    t = (t & ~DECL_PRIVATE) | DECL_HIDDEN;

	t &= ~FUNC_VARARGS; /* for 'varargs inherit' */

	/* Handle classes */
	fix_class_type(&t, from);

	define_variable(from->variable_table[i], t);
    }
}

static int add_new_function_entry() {
    int index = mem_block[A_FUNCTION_DEFS].current_size / sizeof(compiler_temp_t);
    allocate_in_mem_block(A_FUNCTION_DEFS, sizeof(compiler_temp_t));
    
    return index;
}

/* copy a function verbatim into this object, and possibly add it to the
   list of functions in this object, as well
 */

/* This is the first time the function occurs in the current program 
   and it happened by copying from program prog 
   Because program prog is assumed to be dealt with correctly, 
   we know if this _was_ an aliased function, it must point to 
   to the correct definition already in prog 
   If this is not the first time it occurs, use overload_function 
   Otherwise, consistency cannot be guaranteed. The resulting
   first time copy will never be aliased, as it's presence will
   be a magic marker for other similar functions that will get
   removed. - Sym 
 */

static void copy_new_function P5(program_t *, prog, int, index,
				 program_t *, defprog, int, defindex,
				 int, typemod) {
    ident_hash_elem_t *ihe;
    
    int where = add_new_function_entry();
    int f = prog->function_flags[index];
      
    if (f & FUNC_ALIAS) {
	f = FUNC_ALIAS | prog->function_flags[f & ~FUNC_ALIAS];
    }
    
    /* 'private' functions become 'hidden' */
    if (f & DECL_PRIVATE)
	f = (f & ~DECL_PRIVATE) | DECL_HIDDEN;

    f = DECL_MODIFY(typemod, f) | FUNC_INHERITED;
    f &= ~DECL_NOSAVE;

    FUNCTION_FLAGS(where) = f;

    FUNCTION_TEMP(where)->prog = defprog;
    FUNCTION_TEMP(where)->u.func = defprog->function_table + defindex;
    FUNCTION_TEMP(where)->offset = NUM_INHERITS - 1;
    FUNCTION_TEMP(where)->function_index_offset = index;
    FUNCTION_NEXT(where) = 0;
    
    /* add the identifier */
    ihe = find_or_add_ident(defprog->function_table[defindex].name,
			    FOA_GLOBAL_SCOPE);
    ihe->sem_value++; /* we knew it was the first, so dn.function_num = -1 */
    ihe->dn.function_num = where;
}

static int find_class_member P3(int, which, char *, name, unsigned char *, type) {
    int i;
    class_def_t *cd;
    class_member_entry_t *cme;

    DEBUG_CHECK(!findstring(name), "name in find_class_member must be a shared string.\n");

    cd = ((class_def_t *)mem_block[A_CLASS_DEF].block) + which;
    cme = ((class_member_entry_t *)mem_block[A_CLASS_MEMBER].block) + cd->index;
    for (i = 0; i < cd->size; i++) {
	if (PROG_STRING(cme[i].name) == name)
	    break;
    }
    if (i == cd->size) {
	if (type) *type = TYPE_ANY;
	return -1;
    } else {
	if (type) *type = cme[i].type;
	return i;
    }
}

int lookup_any_class_member P2(char *, name, unsigned char *, type) {
    int nc = mem_block[A_CLASS_DEF].current_size / sizeof(class_def_t);
    int i, ret = -1, nret;
    char *s = findstring(name);
    
    if (s) {
	for (i = 0; i < nc; i++) {
	    nret = find_class_member(i, s, type);
	    if (nret == -1) continue;
	    if (ret != -1 && nret != ret) {
		char buf[256];
		char *end = EndOf(buf);
		char *p;
		
		p = strput(buf, end, "More than one class in scope has member '");
		p = strput(p, end, name);
		p = strput(buf, end, "'; use a cast to disambiguate");
		yyerror(buf);
	    }
	    ret = nret;
	}
    }

    if (ret == -1) {
	char buf[256];
	char *end = EndOf(buf);
	char *p;
	
	p = strput(buf, end, "No class in scope has no member '");
	p = strput(p, end, name);
	p = strput(p, end, "'");
	yyerror(buf);
    }

    return ret;
}

int lookup_class_member P3(int, which, char *, name, unsigned char *, type) {
    char *s = findstring(name);
    int ret;
    
    if (s)
	ret = find_class_member(which, s, type);
    else
	ret = -1;

    if (ret == -1) {
	class_def_t *cd = ((class_def_t *)mem_block[A_CLASS_DEF].block) + which;
	char buf[256];
	char *end = EndOf(buf);
	char *p;
	
	p = strput(buf, end, "Class '");
	p = strput(p, end, PROG_STRING(cd->name));
	p = strput(p, end, "' has no member '");
	p = strput(p, end, name);
	p = strput(p, end, "'");

	yyerror(buf);
    }
    return ret;
}	

parse_node_t *reorder_class_values P2(int, which, parse_node_t *, node) {
    class_def_t *cd;
    parse_node_t **tmp;
    int i;
    
    cd = ((class_def_t *)mem_block[A_CLASS_DEF].block) + which;
    tmp = CALLOCATE(cd->size, parse_node_t *, TAG_COMPILER,
		    "reorder_class_values");

    for (i = 0; i < cd->size; i++) 
	tmp[i] = 0;

    while (node) {
	i = lookup_class_member(which, (char *)node->l.expr, 0);
	if (i != -1) {
	    if (tmp[i]) {
		char buf[256];
		char *end = EndOf(buf);
		char *p;

		p = strput(buf, end, "Redefinition of member '");
		p = strput(p, end, (char *) node->l.expr);
		p = strput(p, end, "' in instantiation of class  '");
		p = strput(p, end, PROG_STRING(cd->name));
		p = strput(p, end, "'");
		yyerror(buf);
	    } else {
		tmp[i] = node->v.expr;
	    }
	}
	scratch_free((char *) node->l.expr);

	node = node->r.expr;
    }
    i = cd->size;
    node = 0;
    while (i--) {
	parse_node_t *newnode;
	if (tmp[i]) {
	    CREATE_STATEMENTS(newnode, tmp[i], node);
	} else {
	    CREATE_STATEMENTS(newnode, 0, node);
	    CREATE_NUMBER(newnode->l.expr, 0);
	}
	node = newnode;
    }

    FREE(tmp);
    return node;
}

static void check_class P4(char *, name, program_t *, prog, int, idx, int, nidx) {
    class_def_t *sd1, *sd2;
    class_member_entry_t *sme1, *sme2;
    int i, n;
    
    sd1 = &prog->classes[idx];
    sd2 = CLASS(nidx);
    
    n = sd1->size;
    if (sd1->size != sd2->size) {
	char buf[256];
	char *end = EndOf(buf);
	char *p;
	
	p = strput(buf, end, "Definitions of class '");
	p = strput(p, end, name);
	p = strput(p, end, "' differ in size.");
	yyerror(buf);
	return;
    }

    sme1 = &prog->class_members[sd1->index];
    sme2 = (class_member_entry_t *)mem_block[A_CLASS_MEMBER].block + sd2->index;

    for (i = 0; i < n; i++) {
	int newtype;

	newtype = sme2[i].type;


	fix_class_type(&newtype, prog);

	if (sme1[i].type != newtype ||
	    prog->strings[sme1[i].name] != PROG_STRING(sme2[i].name)) {
	    char buf[512];
	    char *end = EndOf(buf);
	    char *p;
	    
	    p = strput(buf, end, "Definitions of class '");
	    p = strput(p, end, name);
	    p = strput(p, end, "' disagree.");
	    yyerror(buf);
	    return;
	}
    }
}

void copy_structures P1(program_t *, prog) {
    class_def_t *sd;
    class_member_entry_t *sme;
    ident_hash_elem_t *ihe;
    char *str;
    int sm_off = mem_block[A_CLASS_MEMBER].current_size / sizeof(class_member_entry_t);
    int sd_off = mem_block[A_CLASS_DEF].current_size / sizeof(class_def_t);
    int i, j, offset;

    if (prog->num_classes + sd_off > CLASS_NUM_MASK) {
	char buf[512];
	char *end = EndOf(buf);
	char *p;

	p = strput(buf, end, "Too many classes, max is ");
	p = strput_int(p, end, CLASS_NUM_MASK - 1);
	p = strput(p, end, ".\n");
	yyerror(buf);
	return;
    }

    for (i = 0; i < prog->num_classes; i++) {
	str = prog->strings[prog->classes[i].name];
	ihe = find_or_add_ident(str, FOA_GLOBAL_SCOPE);
	if (ihe->dn.class_num == -1) {
	    ihe->dn.class_num = sd_off++;
	    ihe->sem_value++;
	} else {
	    check_class(str, prog, i, ihe->dn.class_num);
	    continue;
	}
	sd = (class_def_t *)allocate_in_mem_block(A_CLASS_DEF, sizeof(class_def_t));
	sd->name = store_prog_string(str);
	sd->size = prog->classes[i].size;
	sd->index = sm_off;
	sm_off += sd->size;
	sme = (class_member_entry_t *)allocate_in_mem_block(A_CLASS_MEMBER, sizeof(class_member_entry_t) * sd->size);
	offset = prog->classes[i].index;
	for (j = 0; j < sd->size; j++) {
	    int oldtype = prog->class_members[offset + j].type;

	    fix_class_type(&oldtype, prog);
	    sme[j].type = oldtype;
	    sme[j].name = store_prog_string(prog->strings[prog->class_members[offset + j].name]);
	}
    }
}

typedef struct ovlwarn_s {
    struct ovlwarn_s *next;
    char *func;
    char *warn;
} ovlwarn_t;

ovlwarn_t *overload_warnings = 0;

static void remove_overload_warnings P1(char *, func) {
    ovlwarn_t **p;
    ovlwarn_t *tmp;

    p = &overload_warnings;
    while (*p) {
	if (!func || (*p)->func == func) {
	    FREE((*p)->warn);
	    tmp = *p;
	    *p = (*p)->next;
	    FREE(tmp);
	} else
	    p = &(*p)->next;
    }
}

static void show_overload_warnings() {
    ovlwarn_t *p, *next;
    p = overload_warnings;
    while (p) {
	yywarn(p->warn);
	FREE(p->warn);
	next = p->next;
	FREE(p);
	p = next;
    }
    overload_warnings = 0;
}

/* Overload the function index with the new definition 
   The idea is to choose a function out of all non-aliases 
   inherited or at the current level.

   non-alias1 ..   non-aliasK -> Final resolution

   The function at oldindex always has the most recent choice.
   The fact that it is at oldindex saves changing ihe.sem_values, 
   among other things.

   The current resolution order is 
   1st inh proto < ... < last inh proto <
   1st non-inh proto < .. < last non-inh proto <
   1st inh defined < .. < last inh defined <
   defined
 */
static void overload_function P6(program_t *, prog, int, index,
				 program_t *, defprog, int, defindex,
				 int, oldindex, int, typemod) {
    int f;
    int oldflags = FUNCTION_FLAGS(oldindex);
    function_t *definition = &defprog->function_table[defindex];
    int newflags = prog->function_flags[index];
    int replace;
    compiler_temp_t *newdef;
    
    if (newflags & FUNC_ALIAS)
	newflags = FUNC_ALIAS | prog->function_flags[newflags & ~FUNC_ALIAS];

    /* check that we aren't overloading a nomask function */
    if ((oldflags & DECL_NOMASK) && 
	!((oldflags|newflags) & (FUNC_NO_CODE))) {
        char buf[256];
	char *end = EndOf(buf);
	char *p;
	
	p = strput(buf, end, "Illegal to redefine 'nomask' function \"");
	p = strput(p, end, definition->name);
	p = strput(p, end, "\"");
	yyerror(buf);
    }

    /* Try to prevent some confusion re: overloading.
     * Warn them about the behavior of inheriting the same function
     * from two branches.
     * 
     * Note that we don't want to scream now, b/c if the function is 
     * overloaded later this becomes irrelevant.
     *
     * Note also that this is real spammy if you inherit the same object
     * twice.  Something should be done about that.
     */
    if ((pragmas & PRAGMA_WARNINGS) &&
	!((oldflags|newflags) & (FUNC_NO_CODE|DECL_PRIVATE|DECL_HIDDEN))
	&& (oldflags & FUNC_INHERITED)
	) {
	/* don't scream if one is private.  Why not?  Because I said so.
	 * private is pretty screwed up anyway.  In the future there
	 * won't be such a clash b/c private won't come up the tree.
	 * This also give the coder a way to shut the compiler up when
	 * you do inherit the same object twice in different branches :)
	 */
	if (!(oldflags & (DECL_PRIVATE|DECL_HIDDEN)) && !(newflags & (DECL_PRIVATE|DECL_HIDDEN))) {
	    char buf[1024];
	    char *end = EndOf(buf);
	    char *p;
	    ovlwarn_t *ow;
	    compiler_temp_t *func_entry = FUNCTION_TEMP(oldindex);
	    program_t *prog2 = INHERIT(func_entry->offset)->prog;
	    program_t *defprog2;

	    defprog2 = func_entry->prog;
	    p = strput(buf, end, definition->name);
	    p = strput(p, end, "() inherited from both /");
	    p = strput(p, end, defprog->name);
	    if (prog != defprog) {
		p = strput(p, end, " (via /");
		p = strput(p, end, prog->name);
		p = strput(p, end, ")");
	    }
	    p = strput(p, end, " and /");
	    p = strput(p, end, defprog2->name);
	    if (prog2 != defprog2) {
		p = strput(p, end, " (via /");
		p = strput(p, end, prog2->name);
		p = strput(p, end, ")");
	    }
	    p = strput(p, end, "; using the definition in /");
	    p = strput(p, end, prog->name);
	    p = strput(p, end, ".");
	    
	    ow = ALLOCATE(ovlwarn_t, TAG_COMPILER, "overload warning");
	    ow->next = overload_warnings;
	    ow->func = definition->name;
	    ow->warn = alloc_cstring(buf, "overload warning");
	    overload_warnings = ow;
	}
    }

    /* A new function also has to be inserted, since this spot will be
     * used when this function is called in an object inheriting us.  Point
     * it at the overloaded function. 
     * No need to set much here because epilog() will fixup this entry
     * later.
     */

    newdef = ALLOCATE(compiler_temp_t, TAG_COMPILER, "overload_function");

    /* The resolution order is given in above comments */
    if (oldflags & FUNC_ALIAS) replace = 1;
    else if (newflags & FUNC_ALIAS) replace = 0;
    else if (!(newflags & (FUNC_PROTOTYPE|FUNC_UNDEFINED))) {
	if (oldflags & (FUNC_PROTOTYPE|FUNC_INHERITED|FUNC_UNDEFINED))
	    replace = 1;
	else replace = 0;
    } else {
	if ((oldflags & (FUNC_PROTOTYPE|FUNC_UNDEFINED)) &&
	    (oldflags & FUNC_INHERITED))
	    replace = 1;
	else replace = 0;
    }

    if (replace) {
	/* We win */
	if (!(oldflags & FUNC_INHERITED)) {
	    /* Mark entry for removal */
	    FUNC(FUNCTION_TEMP(oldindex)->u.index)->address = ADDRESS_MAX;
	}
	*newdef = *FUNCTION_TEMP(oldindex);
	FUNCTION_TEMP(oldindex)->next = newdef;
	newdef = FUNCTION_TEMP(oldindex);
    } else {
	compiler_temp_t *oldnext = FUNCTION_TEMP(oldindex)->next;
	newdef->next = oldnext;
	FUNCTION_TEMP(oldindex)->next = newdef;
    }
  
    newdef->offset = NUM_INHERITS - 1;
    newdef->function_index_offset = index;
    newdef->prog = defprog;
    newdef->u.func = defprog->function_table + defindex;

    f = (newflags & FUNC_MASK) | FUNC_INHERITED;
    
    /* 'private' functions become 'hidden' */
    if (f & DECL_PRIVATE) f = (f & ~DECL_PRIVATE) | DECL_HIDDEN;
    
    f = DECL_MODIFY(typemod, f);
    f &= ~DECL_NOSAVE;

    newdef->flags = f;
}

/*
 * Copy all function definitions from an inherited object. They are added
 * as undefined, so that they can be redefined by a local definition.
 * If they are not redefined, then they will be updated, so that they
 * point to the inherited definition. See epilog(). Types will be copied
 * at that moment (if available).
 *
 * A call to an inherited function will not be
 * done through this entry (because this entry can be replaced by a new
 * definition). If an function defined by inheritance is called, then one
 * special definition will be made at first call.
 */
int copy_functions P2(program_t *, from, int, typemod)
{
    int i, initializer = -1, num_functions;
    ident_hash_elem_t *ihe;
    int num;

    num_functions = from->num_functions_defined + from->last_inherited;
    
    if (from->num_functions_defined && 
	(from->function_table[from->num_functions_defined - 1].name[0] == APPLY___INIT_SPECIAL_CHAR))
        initializer = --num_functions;

    for (i = 0; i < num_functions; i++) {
	program_t *prog = from;
	int index = i;
	function_t *funp;
	int low, mid, high;

	/* Walk up the inheritance tree to the real definition */	
	if (prog->function_flags[index] & FUNC_ALIAS) {
	    index = prog->function_flags[index] & ~FUNC_ALIAS;
	}
	
	while (prog->function_flags[index] & FUNC_INHERITED) {
	    low = 0;
	    high = prog->num_inherited -1;
	    
	    while (high > low) {
		mid = (low + high + 1) >> 1;
		if (prog->inherit[mid].function_index_offset > index)
		    high = mid -1;
		else low = mid;
	    }
	    index -= prog->inherit[low].function_index_offset;
	    prog = prog->inherit[low].prog;
	}

	index -= prog->last_inherited;

	funp = prog->function_table + index;
	
	
	ihe = lookup_ident(funp->name);
	if (ihe && ((num = ihe->dn.function_num)!=-1)) {
	  /* The function has already been defined in this object */
	    overload_function(from, i, prog,index, num, typemod);
	} else {
	    copy_new_function(from, i, prog, index, typemod);
	}
    }
    return initializer;
}

void type_error P2(char *, str, int, type)
{
    static char buff[256];
    char *end = EndOf(buff);
    char *p;

    p = strput(buff, end, str);
    p = strput(p, end, ": \"");
    p = get_type_name(p, end, type);
    p = strput(p, end, "\"");
    yyerror(buff);
}

/*
 * Compare two types, and return true if they are compatible.
 */

/* This one really is t1->t2; it isn't symmetric, since int->void isn't allowed. */
int compatible_types P2(int, t1, int, t2)
{
#ifdef OLD_TYPE_BEHAVIOR
    /* The old version effectively was almost always was true */
    return 1;
#else
    t1 &= ~DECL_MODS;
    t2 &= ~DECL_MODS;
    if (t1 == TYPE_ANY || t2 == TYPE_ANY) return 1;
    if ((t1 == (TYPE_ANY | TYPE_MOD_ARRAY) && (t2 & TYPE_MOD_ARRAY)))
	return 1;
    if ((t2 == (TYPE_ANY | TYPE_MOD_ARRAY) && (t1 & TYPE_MOD_ARRAY)))
	return 1;
    if (t1 & TYPE_MOD_CLASS)
	return t1 == t2;
    if (t1 & TYPE_MOD_ARRAY) {
	if (!(t2 & TYPE_MOD_ARRAY)) return 0;
	return t1 == (TYPE_MOD_ARRAY | TYPE_ANY) ||
	       t2 == (TYPE_MOD_ARRAY | TYPE_ANY) || (t1 == t2);
    } else if (t2 & TYPE_MOD_ARRAY)
	return 0;
    return compatible[t1] & (1 << t2);
#endif
}

/* This one is symmetric.  Used for comparison operators, etc */
int compatible_types2 P2(int, t1, int, t2)
{
#ifdef OLD_TYPE_BEHAVIOR
    /* The old version effectively was almost always was true */
    return 1;
#else
    t1 &= ~DECL_MODS;
    t2 &= ~DECL_MODS;
    if (t1 == TYPE_ANY || t2 == TYPE_ANY) return 1;
    if ((t1 == (TYPE_ANY | TYPE_MOD_ARRAY) && (t2 & TYPE_MOD_ARRAY)))
	return 1;
    if ((t2 == (TYPE_ANY | TYPE_MOD_ARRAY) && (t1 & TYPE_MOD_ARRAY)))
	return 1;
    if (t1 & TYPE_MOD_CLASS)
	return t1 == t2;
    if (t1 & TYPE_MOD_ARRAY) {
	if (!(t2 & TYPE_MOD_ARRAY)) return 0;
	return t1 == (TYPE_MOD_ARRAY | TYPE_ANY) ||
	       t2 == (TYPE_MOD_ARRAY | TYPE_ANY) || (t1 == t2);
    } else if (t2 & TYPE_MOD_ARRAY)
	return 0;
    if (compatible[t1] & (1 << t2))
	return 1;
    return compatible[t2] & (1 << t1);
#endif
}

/*
 * Patch a function definition of an inherited function, to what it really
 * should be.
 * The name of the function can be one of:
 *    object::name
 *    ::name
 * Where 'object' is the name of the superclass.
 *
 * Note: this function is now only used for resolving :: references
 */
static int find_matching_function P3(program_t *, prog, char *, name,
				     parse_node_t *, node) {
    int high = prog->num_functions_defined - 1;
    int low = 0;
    int i, res;

    /* Search our function table */
    while (high >= low) {
	int mid = (high + low)/2;
	char *p = prog->function_table[mid].name;
	
	if (name < p)
	    high = mid-1;
	else if (name > p)
	    low = mid+1;
	else {
	    int ri;
	    int flags;
	    int type;
	    
	    /* Rely on the fact that functions in the table are not inherited
	       or aliased */
	    /* Non-inherited aliased ones are always removed anyway */
	    ri = prog->last_inherited + mid;

	    flags = prog->function_flags[ri];

	    if (flags & (FUNC_UNDEFINED | FUNC_PROTOTYPE)) return 0;
	    if (flags & DECL_PRIVATE) return -1;
	    
	    node->kind = NODE_CALL_2;
	    node->v.number = F_CALL_INHERITED;
	    node->l.number = ri;
	    type = prog->function_table[mid].type;
	    fix_class_type(&type, prog);
	    node->type = type;
	    return 1;
	}
    }
    
    /* Search inherited function tables */
    i = prog->num_inherited;
    while (i--) {
	if ((res = find_matching_function(prog->inherit[i].prog, name, node))) {
	    if ((res == -1) || (prog->inherit[i].type_mod & DECL_PRIVATE))
		return -1;
	    
	    node->l.number += prog->inherit[i].function_index_offset;
	    return 1;
	}
    }
    return 0;
}

int arrange_call_inherited P2(char *, name, parse_node_t *, node)
{
    inherit_t *ip;
    int num_inherits, super_length;
    char *super_name, *p, *real_name = name;
    char *shared_string;
    int ret;
    
    if (real_name[0] == ':') {
	super_name = 0;
	real_name += 2; 	/* There will be exactly two ':' */
	super_length = 0;
    } else if ((p = strchr(real_name, ':'))) {
        super_name = name;
        real_name = p+2;
        super_length = real_name - super_name - 2;
    }

    num_inherits = NUM_INHERITS;
    /* no need to look for it unless its in the shared string table */
    if ((shared_string = findstring(real_name))) {
	ip = (inherit_t *) mem_block[A_INHERITS].block;
	for (; num_inherits > 0; ip++, num_inherits--) {
	    int tmp;
	    
	    if (super_name) {
		int l = SHARED_STRLEN(ip->prog->name);	/* Including .c */

		if (l - 2 < super_length)
		    continue;
		if (strncmp(super_name, ip->prog->name + l - 2 -super_length,
			    super_length) != 0 ||
                    !((l - 2 == super_length) ||
                     ((ip->prog->name + l - 3 - super_length)[0] == '/')))
                    continue;
	    }

	    if ((tmp = find_matching_function(ip->prog, shared_string, node))) {
		if (tmp == -1 || (ip->type_mod & DECL_PRIVATE)) {
		    yyerror("Called function is private.");

		    goto invalid;
		}
		
		ret = node->l.number + ip->function_index_offset;
		node->l.number |= ((ip - (inherit_t *) mem_block[A_INHERITS].block) << 16);
		return ret;
	    }
	}
    }				/* if in shared string table */
    {
	char buff[256];
	char *end = EndOf(buff);
	char *p;
	
	p = strput(buff, end, "No such inherited function ");
	p = strput(p, end, name);
	yyerror(buff);
    }

  invalid:
    node->kind = NODE_CALL_2;
    node->v.number = F_CALL_INHERITED;
    node->l.number = 0;
    node->type = TYPE_ANY;
    
    return -1;
}

/*
 * Define a new function. Note that this function is called at least twice
 * for all function definitions. First as a prototype, then as the real
 * function. Thus, there are tests to avoid generating error messages more
 * than once by looking at (flags & NAME_PROTOTYPE).
 */
/* Warning: returns an index into A_FUNCTIONS, not the full
 * function list
 */
int define_new_function P5(char *, name, int, num_arg, int, num_local,
                           int, flags, int, type)
{
    int oldindex, num, newindex;
    unsigned short argument_start_index;
    ident_hash_elem_t *ihe;
    function_t *funp = 0;
    compiler_temp_t *newfunc;
    
    oldindex = (ihe = lookup_ident(name)) ? ihe->dn.function_num : -1;
    if (oldindex >= 0) {
	int funflags = FUNCTION_FLAGS(oldindex);

	funp = FUNCTION_DEF(oldindex);

	/*
	 * The function was already defined. It may be one of several
	 * reasons:
	 * 
	 * 1.	There has been a prototype.
	 * 2.	There was a function with the same name defined by inheritance.
	 * 3.	This function has been called, but not yet defined.
	 * 4.	The function is doubly defined.
	 * 5.	A "late" prototype has been encountered.
	 */
	if (funflags & FUNC_ALIAS) {
	    fatal("Inconsistent aliasing of functions!\n");
	}
	
	if (!(funflags & (FUNC_INHERITED|FUNC_PROTOTYPE|FUNC_UNDEFINED)) 
	    && !(flags & FUNC_PROTOTYPE)) {
	    char buff[256];
	    char *end = EndOf(buff);
	    char *p;

	    p = strput(buff, end, "Redeclaration of function ");
	    p = strput(p, end, name);
	    p = strput(p, end, ".");
	    yyerror(buff);
	    return -1;
	}
	/*
	 * It was either an undefined but used funtion, or an inherited
	 * function. In both cases, we now consider this to be THE new
	 * definition. It might also have been a prototype to an already
	 * defined function, in which case we simply return.
	 * 
	 * Check arguments only when types are supposed to be tested, and if
	 * this function really has been defined already.
	 * 
	 * 'nomask' functions may not be redefined.
	 */
	if ((funflags & DECL_NOMASK) &&
	    !((flags|funflags) & (FUNC_UNDEFINED|FUNC_PROTOTYPE))) {
	    char buf[256];
	    char *end = EndOf(buf);
	    char *p;
	    
	    p = strput(buf, end, "Illegal to redefine 'nomask' function \"");
	    p = strput(p, end, name);
	    p = strput(p, end, "\"");
	    yyerror(buf);
	}

	/* only check prototypes for matching.  It shouldn't be required that
	   overloading a function must have the same signature */
	if (exact_types && ((flags|funflags) & FUNC_PROTOTYPE)) {
	    int funtype, i;

	    funtype = funp->type;
	    if (FUNCTION_TEMP(oldindex)->prog) {
		fix_class_type(&funtype, FUNCTION_TEMP(oldindex)->prog);
	    }
	    /* This should be changed to catch two prototypes which disagree */
	    if (funtype != TYPE_UNKNOWN) {
		if (funp->num_arg != num_arg 
		    && !((flags|funflags) & FUNC_VARARGS))
		    yywarn("Number of arguments disagrees with previous definition.");
		if (!(funflags & FUNC_STRICT_TYPES))
		    yywarn("Called function not compiled with type testing.");
		
		/* Now check that argument types wasn't changed. */
		if (!compatible_types(type, funtype)) {
		    char buff[512];
		    char *end = EndOf(buff);
		    char *p;
		    
		    if (FUNCTION_TEMP(oldindex)->prog) {
			p = strput(buff, end, "Function inherited from '/");
			p = strput(p, end, FUNCTION_TEMP(oldindex)->prog->name);
			p = strput(p, end, "' does not match ");
		    } else {
			if (funflags & FUNC_PROTOTYPE)
			    p = strput(buff, end, "Previous function prototype ");
			else p = strput(buff, end, "Previous function declaration ");
			p = strput(p, end, "does not match ");
		    }
		    p = strput(p, end, "current function in return type ");
		    p = get_two_types(p, end, funtype, type);

		    yywarn(buff);
		}
		
		for (i = 0; i < num_arg; i++) {
		    /* FIXME: check arg types here */
		    
		}
	    }
	}
	
	/* If it was yet another prototype, then simply return. */
	/* provided the current winning definition is an undefined */
	/* or prototype itself */

	if ((flags & FUNC_PROTOTYPE) &&
	    !(funflags & (FUNC_PROTOTYPE|FUNC_UNDEFINED)))
	    return -1; /* unused for prototypes */

	if (pragmas & PRAGMA_WARNINGS)
	    remove_overload_warnings(funp->name);

	/* If there was already a definition at this level (due to a 
	 * prototype), clear it out.  Don't free the function name, though;
	 * the name is the same, and the ident hash table might be counting
	 * on it to stay allocated.
	 */
	if (FUNCTION_PROG(oldindex) == 0) {
	    num = FUNCTION_TEMP(oldindex)->u.index;
	    funp = FUNC(num);
	} else
	    funp = 0;
    }
    if (!funp) {
	num = mem_block[A_FUNCTIONS].current_size / sizeof(function_t);
	funp = (function_t *)allocate_in_mem_block(A_FUNCTIONS, sizeof(function_t));
	funp->name = make_shared_string(name);
	argument_start_index = INDEX_START_NONE;
	add_to_mem_block(A_ARGUMENT_INDEX, (char *) &argument_start_index,
			 sizeof argument_start_index);
	
    }

    if (oldindex < 0) {
	newindex = add_new_function_entry();
    	ihe = find_or_add_ident(funp->name, FOA_GLOBAL_SCOPE);
	ihe->sem_value++;
	ihe->dn.function_num = newindex;
	newfunc = FUNCTION_TEMP(newindex);
	newfunc->next = (compiler_temp_t *) NULL;
    } else {
	newfunc = ALLOCATE(compiler_temp_t, TAG_TEMPORARY, "define_new_function");
	*newfunc = *FUNCTION_TEMP(oldindex);

	/* We are going to rewrite stuff at oldindex */
	FUNCTION_TEMP(oldindex)->next = newfunc;
	newfunc = FUNCTION_TEMP(oldindex);

	newindex = oldindex;
    }

    newfunc->prog = 0;
    newfunc->u.index = num;

    if (exact_types)
	flags |= FUNC_STRICT_TYPES;
    DEBUG_CHECK(!(flags & DECL_ACCESS), "No access level for function!\n");
    newfunc->flags = flags;



    funp->num_local = num_local;
    funp->num_arg = num_arg;
    funp->type = type;
    funp->address = 0;
#ifdef PROFILE_FUNCTIONS
    funp->calls = 0L;
    funp->self = 0L;
    funp->children = 0L;
#endif

    if (exact_types && num_arg) {
	int i;
	
	if (!(flags & FUNC_PROTOTYPE)) {
	    for (i = 0; i < current_number_of_locals; i++) {
		if (type_of_locals_ptr[locals_ptr[i].runtime_index] & LOCAL_MOD_UNUSED) {
		    char buf[256];
		    char *end = EndOf(buf);
		    char *p;
		    
		    p = strput(buf, end, "Unused local variable '");
		    p = strput(p, end, locals_ptr[i].ihe->name);
		    p = strput(p, end, "'");
		    yywarn(buf);
		    type_of_locals_ptr[locals_ptr[i].runtime_index] &= ~LOCAL_MOD_UNUSED;
		}
	    }
	}
	*((unsigned short *)mem_block[A_ARGUMENT_INDEX].block + num) = 
	    mem_block[A_ARGUMENT_TYPES].current_size / sizeof(unsigned short);
	add_to_mem_block(A_ARGUMENT_TYPES, (char *)type_of_locals_ptr,
			 num_arg * sizeof(*type_of_locals_ptr));
#ifndef SUPPRESS_ARGUMENT_WARNINGS
	if (flags & FUNC_PROTOTYPE) {
	    int i;
	    
	    for (i = 0; i < num_arg; i++) {
		if (*locals_ptr[i].ihe->name)
		    type_of_locals_ptr[i] |= LOCAL_MOD_UNUSED;
	    }
	}
#endif
    }
    return newindex;
}

int define_variable P2(char *, name, int, type)
{
    variable_t *dummy;
    int n;
    ident_hash_elem_t *ihe;

    n = (mem_block[A_VAR_TEMP].current_size / sizeof(variable_t));

    ihe = find_or_add_ident(name, FOA_GLOBAL_SCOPE);
    if (ihe->dn.global_num == -1) {
	ihe->sem_value++;
	ihe->dn.global_num = n;

	if (n >= 256) {
	    yyerror("Too many global variables");
	}
    } else {
	char buf[256];
	char *end = EndOf(buf);
	char *p;
	
	p = strput(buf, end, "Redeclaration of global variable '");
	p = strput(p, end, ihe->name);
	p = strput(p, end, "'.");
	yywarn(buf);
	
	if (VAR_TEMP(ihe->dn.global_num)->type & DECL_NOMASK) {
	    char buf[256];
	    char *end = EndOf(buf);
	    char *p;
	    
	    p = strput(buf, end, "Illegal to redefine 'nomask' variable \"");
	    p = strput(p, end, name);
	    p = strput(p, end, "\"");
	    yyerror(buf);
	}
	/* Okay, the nasty idiots have two variables of the same name in
	   the same object.  This causes headaches for save_object().
	   To keep save_object sane, we need to make one static */
	if (!(VAR_TEMP(ihe->dn.global_num)->type & DECL_NOSAVE))
	    type |= DECL_NOSAVE;
	
	/* hidden variables don't cause variables that are visible to become
	   invisible; we only add them above (in the !hide case) for better
	   error messages */
	if (!(type & DECL_HIDDEN))
	    ihe->dn.global_num = n;
    }

    dummy = (variable_t *)allocate_in_mem_block(A_VAR_TEMP, sizeof(variable_t));
    dummy->name = name;
    dummy->type = type;

    return n;
}

int define_new_variable P2(char *, name, int, type) {
    int n;
    unsigned short *tp;
    char **np;
    
    var_defined = 1;
    name = make_shared_string(name);
    n = define_variable(name, type);
    np = (char **)allocate_in_mem_block(A_VAR_NAME, sizeof(char*));
    *np = name;
    tp = (unsigned short *)allocate_in_mem_block(A_VAR_TYPE, sizeof(unsigned short));
    *tp = type;
    
    return n;
}

parse_node_t *check_refs P3(int, num, parse_node_t *, elist, parse_node_t *, pn) {
    int tmp = num;
    
    elist = elist->r.expr;

    while (elist) {
	if (IS_NODE(elist->v.expr, NODE_UNARY_OP_1, F_MAKE_REF))
	    tmp--;
	elist = elist->r.expr;
    }
    DEBUG_CHECK(tmp < 0, "Oops, found more refs than is possible!\n");
    if (tmp) {
	/* if we didn't find all the refs at the top level of the argument
	 * list, then at least one is buried; i.e. something illegal like
	 * func(x = ref y), func(ref y + 1), etc
	 */
	yyerror("Illegal use of ref");
    }
    if (num) {
	parse_node_t *ret;

	CREATE_UNARY_OP_1(ret, F_KILL_REFS, pn->type, pn, num);
	return ret;
    }
    return pn;
}

/* we know here that x has two modifiers, and hidden isn't one of them */
int decl_fix P1(int, x) {
    int rest = x & ~DECL_ACCESS;

#ifndef SENSIBLE_MODIFIERS
    if (x & DECL_VISIBLE)
	return rest | DECL_VISIBLE;
#endif
    if (x & DECL_PRIVATE)
	return rest | DECL_PRIVATE;

    /* only possibility left is 'public protected' */
    return rest | DECL_PROTECTED;
}

char *compiler_type_names[] = {"unknown", "mixed", "void", "void", 
		               "int", "string", "object", "mapping",
		               "function", "float", "buffer" };

/* This routine has the semantics of strput(); see comments in simulate.c */

char *get_type_modifiers P3(char *, where, char *, end, int, type)
{
#ifdef SENSIBLE_MODIFIERS
    if (type & DECL_HIDDEN)
	where = strput(where, end, "hidden ");
    if (type & DECL_PRIVATE)
	where = strput(where, end, "private ");
    if (type & DECL_PROTECTED)
	where = strput(where, end, "protected ");
    if (type & DECL_NOSAVE)
	where = strput(where, end, "nosave ");
#else
    if (type & DECL_HIDDEN)
	where = strput(where, end, "hidden ");
    if (type & DECL_VISIBLE)
	where = strput(where, end, "public ");
    if (type & DECL_PRIVATE)
	where = strput(where, end, "private ");
    if (type & DECL_PROTECTED)
	where = strput(where, end, "static ");
#endif
    /* no output for public */
    if (type & DECL_NOMASK)
	where = strput(where, end, "nomask ");
    if (type & FUNC_VARARGS)
	where = strput(where, end, "varargs ");

    return where;
}

char *get_type_name P3(char *, where, char *, end, int, type)
{
    int pointer = 0;

    where = get_type_modifiers(where, end, type);
    type &= ~DECL_MODS;
    if (type & TYPE_MOD_ARRAY) {
	pointer = 1;
	type &= ~TYPE_MOD_ARRAY;
    }
    if (type & TYPE_MOD_CLASS) {
	where = strput(where, end, "class ");
	/* we're sometimes called from outside the compiler */
	if (current_file)
	    where = strput(where, end, PROG_STRING(CLASS(type & ~TYPE_MOD_CLASS)->name));
    } else {
	DEBUG_CHECK(type >= sizeof compiler_type_names / sizeof compiler_type_names[0], "Bad type\n");
	where = strput(where, end, compiler_type_names[type]);
    }
    where = strput(where, end, " ");
#ifdef ARRAY_RESERVED_WORD
    if (pointer) {
	/* use just "array" instead of "mixed array" */
	if (type == TYPE_ANY) {
	    where -= strlen(compiler_type_names[type]) + 1;
	}
	where = strput(where, end, "array ");
    }
#else
    if (pointer)
	where = strput(where, end, "* ");
#endif
    return where;
}

#define STRING_HASH(var,str) \
    var = (long)str ^ (long)str >> 16; \
    var = (var ^ var >> 8) & 0xff;

short store_prog_string P1(char *, str)
{
    short i, next, *next_tab, *idxp;
    char **p;
    unsigned char hash, mask, *tagp;

    str = make_shared_string(str);
    STRING_HASH(hash, str);
    idxp = &string_idx[hash];

    /* string_tags is a big bit-array, so find correct bit */
    mask = 1 << (hash & 7);
    tagp = &string_tags[hash >> 3];

    p = (char **)&PROG_STRING(0);
    next_tab = (short *) mem_block[A_STRING_NEXT].block;

    if (*tagp & mask) {
	/* search hash chain to see if it's there */
	for (i = *idxp; i >= 0; i = next_tab[i]) {
	    if (p[i] == str) {
		free_string(str);	/* needed as string is only free'ed
					 * once. */
		((short *) mem_block[A_STRING_REFS].block)[i]++;
		return i;
	    }
	}
	next = *idxp;
    } else {
	*tagp |= mask;
	next = -1;
    }

    /*
     * New string, add to table
     */

    if (freed_string >= 0) {
	/* reuse freed string */
	int top;

	i = freed_string;

	top = mem_block[A_STRINGS].current_size / sizeof str;
	for (freed_string++; freed_string < top; freed_string++) {
	    if (p[freed_string] == 0)
		break;
	}
	if (freed_string >= top)
	    freed_string = -1;
    } else {
	/* grow by one element. */
	allocate_in_mem_block(A_STRINGS, sizeof str);
	allocate_in_mem_block(A_STRING_NEXT, sizeof(short));
	allocate_in_mem_block(A_STRING_REFS, sizeof(short));
	/* test if number of strings isn't too large ? */
	i = mem_block[A_STRINGS].current_size / sizeof str - 1;
    }
    PROG_STRING(i) = str;
    ((short *) mem_block[A_STRING_NEXT].block)[i] = next;
    ((short *) mem_block[A_STRING_REFS].block)[i] = 1;
    *idxp = i;
    return i;
}

void free_prog_string P1(short, num)
{
    short i, prv, *next_tab, top, *idxp;
    char **p, *str;
    unsigned char hash, mask;

    top = mem_block[A_STRINGS].current_size / sizeof(char *) - 1;
    if (num < 0 || num > top) {
	yyerror("free_prog_string: index out of range.\n");
	return;
    }
    if (--((short *) mem_block[A_STRING_REFS].block)[num] >= 1)
	return;

    p = (char **) mem_block[A_STRINGS].block;
    next_tab = (short *) mem_block[A_STRING_NEXT].block;

    str = p[num];
    STRING_HASH(hash, str);
    idxp = &string_idx[hash];

    for (prv = -1, i = *idxp; i != num; prv = i, i = next_tab[i]) {
	if (i == -1) {
	    yyerror("free_prog_string: string not in prog table.\n");
	    return;
	}
    }

    if (prv == -1) {		/* string is head of list */
	*idxp = next_tab[i];
	if (*idxp == -1) {
	    /* clear tag bit since hash chain now empty */
	    mask = 1 << (hash & 7);
	    string_tags[hash >> 3] &= ~mask;
	}
    } else {			/* easy unlink */
	next_tab[prv] = next_tab[i];
    }

    free_string(str);		/* important */
    p[i] = 0;
    if (i != top) {
	if (i < freed_string || freed_string == -1)
	    freed_string = i;
    } else {
	/* shrink table */
	mem_block[A_STRINGS].current_size -= sizeof str;
	mem_block[A_STRING_REFS].current_size -= sizeof(short);
	mem_block[A_STRING_NEXT].current_size -= sizeof(short);
    }
}

int validate_function_call P2(int, f, parse_node_t *, args)
{
    function_t *funp = FUNCTION_DEF(f);
    int funflags = FUNCTION_FLAGS(f);
    int num_arg = ( args ? args->kind : 0 );
    int num_var = 0;
    parse_node_t *pn = args;
    unsigned short *arg_types = 0;
    program_t *prog;
    
    while (pn) {
	if (pn->type & 1) num_var++;
	pn = pn->r.expr;
    }
    
    /* Make sure it isn't private */
    if (funflags & DECL_HIDDEN) {
	char buf[256];
	char *end = EndOf(buf);
	char *p;
	
	p = strput(buf, end, "Illegal to call inherited private function '");
	p = strput(p, end, funp->name);
	p = strput(p, end, "'");
	yyerror(buf);
    }
    
    /*
     * Verify that the function has been defined already.
     */
    if (exact_types) {
	if ((funflags & FUNC_UNDEFINED)) { 
	    char buf[256];
	    char *end = EndOf(buf);
	    char *p;
	    
	    p = strput(buf, end, "Function ");
	    p = strput(p, end, funp->name);
	    p = strput(p, end, " undefined");
	    yyerror(buf);
	}
	/*
	 * Check number of arguments.
	 */
	if (!(funflags & FUNC_VARARGS) &&
	    (funflags & FUNC_STRICT_TYPES)) { 
	    char buff[256];
	    char *end = EndOf(buff);
	    char *p;
	    
	    if (num_var) {
		p = strput(buff, end, "Illegal to pass a variable number of arguments to non-varargs function ");
		p = strput(p, end, funp->name);
		p = strput(p, end, "\n");
		yyerror(buff);
	    } else if (funp->num_arg != num_arg) {
		p = strput(buff, end, "Wrong number of arguments to ");
		p = strput(p, end, funp->name);
		p = strput(p, end, "\n    Expected: ");
		p = strput_int(p, end, funp->num_arg);
		p = strput(p, end, "  Got: ");
		p = strput_int(p, end, num_arg);
		yyerror(buff);
	    }
	}
	/*
	 * Check the argument types.
	 */
	if ((prog = FUNCTION_PROG(f))) {
	    int which = (FUNCTION_TEMP(f)->u.func - prog->function_table);
	    if (prog->type_start) {
		int start = prog->type_start[which];
		if (start != INDEX_START_NONE)
		    arg_types = prog->argument_types + start;
	    }
	} else {
	    int which = FUNCTION_TEMP(f)->u.index;
	    int start = *((unsigned short *)mem_block[A_ARGUMENT_INDEX].block + which);
	    if (start != INDEX_START_NONE)
		arg_types = (unsigned short *)mem_block[A_ARGUMENT_TYPES].block + start;
	}
	
	if (arg_types) {
	    int arg, i, tmp;
	    parse_node_t *enode = args;
	    int fnarg = funp->num_arg;
	    
	    if (funflags & FUNC_TRUE_VARARGS) 
		fnarg--;
	    
	    for (i = 0; (unsigned) i < fnarg && i < num_arg; i++) {
		if (enode->type & 1) break;
		tmp = enode->v.expr->type;

		arg = arg_types[i];
		if (prog)
		    fix_class_type(&arg, prog);
		
		if (!compatible_types(tmp, arg)) {
		    char buff[256];
		    char *end = EndOf(buff);
		    char *p;
		    
		    p = strput(buff, end, "Bad type for argument ");
		    p = strput_int(p, end, i+1);
		    p = strput(p, end, " of ");
		    p = strput(p, end, funp->name);
		    p = strput(p, end, " ");
		    p = get_two_types(p, end, arg_types[i], tmp);
		    yyerror(buff);
		}
		enode = enode->r.expr;
	    }
	}
    }
    return funp->type;
}

parse_node_t *
promote_to_float P1(parse_node_t *, node) {
    parse_node_t *expr;
    if (node->kind == NODE_NUMBER) {
	node->kind = NODE_REAL;
	node->v.real = node->v.number;
	return node;
    }
    expr = new_node();
    expr->kind = NODE_EFUN;
    expr->v.number = predefs[to_float_efun].token;
    expr->type = TYPE_REAL;
    expr->l.number = 1;
    expr->r.expr = new_node_no_line();
    expr->r.expr->kind = 1;
    expr->r.expr->l.expr = expr->r.expr;
    expr->r.expr->type = 0;
    expr->r.expr->v.expr = node;
    expr->r.expr->r.expr = 0;
    return expr;
}

parse_node_t *
promote_to_int P1(parse_node_t *, node) {
    parse_node_t *expr;
    if (node->kind == NODE_REAL) {
	node->kind = NODE_NUMBER;
	node->v.number = node->v.real;
	return node;
    }
    expr = new_node();
    expr->kind = NODE_EFUN;
    expr->v.number = predefs[to_int_efun].token;
    expr->type = TYPE_NUMBER;
    expr->l.number = 1;
    expr->r.expr = new_node_no_line();
    expr->r.expr->kind = 1;
    expr->r.expr->l.expr = expr->r.expr;
    expr->r.expr->type = 0;
    expr->r.expr->v.expr = node;
    expr->r.expr->r.expr = 0;
    return expr;
}

parse_node_t *do_promotions P2(parse_node_t *, node, int, type) {
    if (type == TYPE_REAL) {
	if (node->type == TYPE_NUMBER || node->kind == NODE_NUMBER)
	    return promote_to_float(node);
    }
    if (type == TYPE_NUMBER && node->type == TYPE_REAL)
	return promote_to_int(node);
    return node;
}

/* Take a NODE_CALL, and discard the call, preserving only the args with
   side effects */
parse_node_t *
throw_away_call P1(parse_node_t *, pn) {
    parse_node_t *enode;
    parse_node_t *ret = 0;
    parse_node_t *arg;
    
    enode = pn->r.expr;
    while (enode) {
	arg = insert_pop_value(enode->v.expr);
	if (arg) {
	    /* woops.  Don't lose the side effect. */
	    if (ret) {
		parse_node_t *tmp;
		CREATE_STATEMENTS(tmp, ret, arg);
		ret = tmp;
	    } else {
		ret = arg;
	    }
	}
	enode = enode->r.expr;
    }
    return ret;
}

parse_node_t *
throw_away_mapping P1(parse_node_t *, pn) {
    parse_node_t *enode;
    parse_node_t *ret = 0;
    parse_node_t *arg;
    
    enode = pn->r.expr;
    while (enode) {
	arg = insert_pop_value(enode->v.expr->l.expr);
	if (arg) {
	    /* woops.  Don't lose the side effect. */
	    if (ret) {
		parse_node_t *tmp;
		CREATE_STATEMENTS(tmp, ret, arg);
		ret = tmp;
	    } else {
		ret = arg;
	    }
	}
	arg = insert_pop_value(enode->v.expr->r.expr);
	if (arg) {
	    /* woops.  Don't lose the side effect. */
	    if (ret) {
		parse_node_t *tmp;
		CREATE_STATEMENTS(tmp, ret, arg);
		ret = tmp;
	    } else {
		ret = arg;
	    }
	}
	enode = enode->r.expr;
    }
    return ret;
}

parse_node_t *
validate_efun_call P2(int, f, parse_node_t *, args) {
    int num = args->v.number;
    int min_arg, max_arg, def, *argp;
    int num_var = 0;
    parse_node_t *pn = args->r.expr;
    
    while (pn) {
	if (pn->type & 1) num_var++;
	pn = pn->r.expr;
    }
    
    if (f != -1) {
	/* should this move out of here? */
	switch (predefs[f].token) {
#ifdef F_SIZEOF
	case F_SIZEOF:
	    /* Obscene crap like: sizeof( ({ 1, i++, x + 1, foo() }) )
	     *                    -> i++, foo(), 4
	     */
	    if (!pn && num == 1 && 
		IS_NODE(args->r.expr->v.expr, NODE_CALL, F_AGGREGATE)) {
		parse_node_t *repl, *ret, *node;

		CREATE_NUMBER(node, args->r.expr->v.expr->l.number);
		ret = throw_away_call(args->r.expr->v.expr);
		if (ret) {
		    CREATE_TWO_VALUES(repl, TYPE_NUMBER, ret, node);
		    return repl;
		} else
		    return node;
	    }
#endif
	}

	min_arg = predefs[f].min_args;
	max_arg = predefs[f].max_args;

	def = predefs[f].Default;
	if (!num_var && def != DEFAULT_NONE && num == min_arg -1) {
	    parse_node_t *tmp;
	    tmp = new_node_no_line();
	    tmp->r.expr = 0;
	    tmp->type = 0;
	    args->l.expr->r.expr = tmp;
	    if (def == DEFAULT_THIS_OBJECT) {
		tmp->v.expr = new_node_no_line();
		tmp->v.expr->kind = NODE_EFUN;
		tmp->v.expr->v.number = predefs[this_efun].token;
		tmp->v.expr->l.number = 0;
		tmp->v.expr->type = TYPE_ANY;
		tmp->v.expr->r.expr = 0;
	    } else {
		CREATE_NUMBER(tmp->v.expr, def);
	    }
	    args->v.number++;
	    num++;
	} else if (num_var && max_arg != -1) {
	    char buff[256];
	    char *end = EndOf(buff);
	    char *p;
	    
	    p = strput(buff, end, "Illegal to pass variable number of arguments to non-varargs efun ");
	    p = strput(p, end, predefs[f].word);
	    yyerror(buff);
	    CREATE_ERROR(args);
	    return args;
	} else if ((num - num_var) < min_arg) {
	    char buff[256];
	    char *end = EndOf(buff);
	    char *p;
	    
	    p = strput(buff, end, "Too few arguments to ");
	    p = strput(p, end, predefs[f].word);
	    CREATE_ERROR(args);
	    yyerror(buff);
	    return args;
	} else if (num > max_arg && max_arg != -1) {
	    char buff[256];
	    char *end = EndOf(buff);
	    char *p;
	    
	    p = strput(buff, end, "Too many arguments to ");
	    p = strput(p, end, predefs[f].word);
	    yyerror(buff);
	    CREATE_ERROR(args);
	    return args;
	}
	if (max_arg != -1 && exact_types) {
	    /*
	     * Now check all types of arguments to efuns.
	     */
	    int i, argn, tmp;
	    char buff[256];
	    char *end = EndOf(buff);
	    char *p;
	    parse_node_t *enode = args;
	    argp = &efun_arg_types[predefs[f].arg_index];
	    
	    for (argn = 0; argn < num; argn++) {
		enode = enode->r.expr;
		if (enode->type & 1) break;
		/* this can happen for default args */
		if (!enode->v.expr) break;
		tmp = enode->v.expr->type;
		for (i=0; !compatible_types(tmp, argp[i]) && argp[i] != 0; i++)
		    ;

		if (argp[i] == 0) {
		    p = strput(buff, end, "Bad argument ");
		    p = strput_int(p, end, argn + 1);
		    p = strput(p, end, " to efun ");
		    p = strput(p, end, predefs[f].word);
		    p = strput(p, end, "()");
		    yyerror(buff);
		} else {
		    /* check for (int) -> (float) promotion */
		    if (tmp == TYPE_NUMBER && argp[i] == TYPE_REAL) {
			for (i++; argp[i] && argp[i] != TYPE_NUMBER; i++)
			    ;
			if (!argp[i])
			    enode->v.expr = promote_to_float(enode->v.expr);
		    }
		    else if (tmp == TYPE_REAL && argp[i] == TYPE_NUMBER) {
			for (i++; argp[i] && argp[i] != TYPE_REAL; i++)
			    ;
			if (!argp[i])
			    enode->v.expr = promote_to_int(enode->v.expr);
		    }
		}
		while (argp[i] != 0)
		    i++;
		argp += i + 1;
	    }
	}
	args->l.number = num;
	args->v.number = predefs[f].token;
	args->type = predefs[f].ret_type;
	if (args->type == TYPE_NOVALUE) {
	    args->v.number += NOVALUE_USED_FLAG;
	    args->type = TYPE_VOID;
	}
	args->kind = NODE_EFUN;
    } else {
	CREATE_ERROR(args);
    }
    return args;
}

void yyerror P1(char *, str)
{
    extern int num_parse_error;

    function_context.num_parameters = -1;
    if (num_parse_error > 5) {
	lex_fatal = 1;
	return;
    }
    smart_log(current_file, current_line, str, 0);
#ifdef PACKAGE_MUDLIB_STATS
    add_errors_for_file (current_file, 1);
#endif
    num_parse_error++;
}

void yywarn P1(char *, str) {
    if (!(pragmas & PRAGMA_WARNINGS)) return;
    
    smart_log(current_file, current_line, str, 1);
}

/*
 * Compile an LPC file.
 */
program_t *
compile_file P2(int, f, char *, name) {
    int yyparse PROT((void));
    static int guard = 0;
    program_t *prog;
    
    /* The parser isn't reentrant.  On a few occasions (compile
     * errors, valid_override) LPC code is called during compilation,
     * causing the possibility of arriving here again.
     */
    if (guard) {
	error("Object cannot be loaded during compilation.\n");
    }
    guard = 1;
    
    prolog(f, name);
    yyparse();
    prog = epilog();

    guard = 0;
    return prog;
}

int get_id_number() {
    static int current_id_number = 1;
    return current_id_number++;
}

INLINE_STATIC void copy_in P2(int, which, char **, start) {
    char *block;
    int size;

    size = mem_block[which].current_size;
    if (!size) return;

    block = mem_block[which].block;
    memcpy(*start, block, size);

    *start += align(size);
}

static int compare_funcs P2(unsigned short *, x, unsigned short *, y) {
    char *n1 = FUNC(*x)->name;
    char *n2 = FUNC(*y)->name;
    int sp1, sp2;
    
    /* make sure #global_init# stays last; also shuffle empty entries to
     * the end so we can delete them easily.
     */
    if (n1[0] == '#')
	sp1 = 1;
    else if (FUNC(*x)->address == ADDRESS_MAX)
	sp1 = 2;
    else
	sp1 = 0;
    
    if (n2[0] == '#')
	sp2 = 1;
    else if (FUNC(*y)->address == ADDRESS_MAX)
	sp2 = 2;
    else
	sp2 = 0;
    
    if (sp1 < sp2)
	return -1;
    if (sp1 > sp2)
	return 1;
    
    if (n1 < n2)
	return -1;
    if (n1 > n2)
	return 1;

    return 0;
}

static void handle_functions() {
    int num_func, total_func;
    int i;
    compiler_temp_t *cur_def;
    int new_index, num_def;
    int final_index;
    inherit_t *inheritp;
    
    /* Pass one: Sort the compiler functions first                        */
    
    num_func = total_func = mem_block[A_FUNCTIONS].current_size / sizeof (function_t);
    if (num_func) {
        func_index_map = CALLOCATE(num_func, unsigned short, 
				   TAG_TEMPORARY, "handle_functions");
	comp_sorted_funcs = CALLOCATE(num_func, unsigned short,
				      TAG_TEMPORARY, "handle_functions");
	
 	i = num_func;
	while (i--) func_index_map[i] = i;
	
	quickSort(func_index_map, num_func, sizeof(unsigned short), 
		  compare_funcs);
	
	i = num_func;
	while (i--)
	    comp_sorted_funcs[func_index_map[i]] = i;
	
	while (num_func && 
	       FUNC(func_index_map[num_func-1])->address == ADDRESS_MAX)
	    num_func--;
    }	
    
    if (NUM_INHERITS) {
        program_t *inherited_prog;
	
	inheritp = INHERIT(NUM_INHERITS - 1);
	
	inherited_prog = inheritp->prog;
	
	comp_last_inherited = inheritp->function_index_offset
	    + inherited_prog->last_inherited +
	    inherited_prog->num_functions_defined;
	
	if (inherited_prog->num_functions_defined &&
	    inherited_prog->function_table[inherited_prog->num_functions_defined -1].name[0] == '#') comp_last_inherited--;
    } else comp_last_inherited = 0;
    
    /* Pass one: We allocate space for the comp_def_index_map             */
    /*           and prog_flags, and fill them out in sequential order    */
    
    num_def = mem_block[A_FUNCTION_DEFS].current_size/sizeof(compiler_temp_t);
    if (num_def) {
	comp_def_index_map = CALLOCATE(num_def, unsigned short,
				       TAG_TEMPORARY, "handle functions");
	prog_flags = CALLOCATE(comp_last_inherited + total_func, unsigned short,
			       TAG_TEMPORARY, "handle_functions");
	
	for (i = 0; i < num_def; i++) {
	    cur_def = FUNCTION_TEMP(i);
	    
	    if (cur_def->flags & FUNC_INHERITED) {
		final_index = INHERIT(cur_def->offset)->function_index_offset
		    + cur_def->function_index_offset;
	    } else {
		final_index = comp_last_inherited + 
			      comp_sorted_funcs[cur_def->u.index];
	    }
	    if (cur_def->flags & FUNC_ALIAS) {
		fatal("Aliasing difficulties!\n");
		exit(1);
 	    }
	    
	    comp_def_index_map[i] = final_index;
	    prog_flags[final_index] = cur_def->flags;
	    
	    while ((cur_def = cur_def->next)) {
		if (cur_def->flags & FUNC_INHERITED) {
		    new_index = INHERIT(cur_def->offset)->function_index_offset
			+ cur_def->function_index_offset;
		} else {
		    new_index = comp_last_inherited + 
				comp_sorted_funcs[cur_def->u.index];
		}	    
		/* We aren't worried about repeating new_indices */
		/* because it's all the same value               */
		/* except the case where new_index is actually final_index */
		
		if (new_index != final_index) {
		    prog_flags[new_index] = FUNC_ALIAS | final_index;
		} 
	    }
	}
    }

    if (total_func)     
	FREE((char *) comp_sorted_funcs);
}		    
 		    
/*
 * The program has been compiled. Prepare a 'program_t' to be returned.
 */
static program_t *epilog PROT((void)) {
    int size, i, lnsz, lnoff;
    char *p;
    int num_func;
    ident_hash_elem_t *ihe;
    program_t *prog;
    compiler_temp_t *fundefp, *nextdefp;
    
    if (num_parse_error > 0 || inherit_file) {
	/* don't print these; they can be wrong, since we didn't parse the
	   entire file */
	if (pragmas & PRAGMA_WARNINGS)
	    remove_overload_warnings(0);
	clean_parser();
	end_new_file();
	free_string(current_file);
	current_file = 0;
	return 0;
    }

    if (pragmas & PRAGMA_WARNINGS)
	show_overload_warnings();
    
    /*
     * Define the #global_init# function, but only if there was any code
     * to initialize.
     */
    UPDATE_PROGRAM_SIZE;

    if (comp_trees[TREE_INIT]) {
	parse_node_t *pn, *newnode;
  	int fun;
  	/* end the __INIT function */
  	CREATE_RETURN(pn, 0);
	newnode = comp_trees[TREE_INIT];
	CREATE_TWO_VALUES(comp_trees[TREE_INIT], 0,
			  newnode, pn);
 	fun = define_new_function(APPLY___INIT, 0, 0,
				  DECL_HIDDEN | FUNC_STRICT_TYPES, TYPE_VOID);
	pn = new_node_no_line();
	pn->kind = NODE_FUNCTION;
	pn->v.number = fun;
	pn->l.number = 0;
	pn->r.expr = comp_trees[TREE_INIT];
	comp_trees[TREE_INIT] = pn;
    }

    /* Do stuff with functions first */
    /* calculate comp_last_inherited also */
    handle_functions();

    /* generate the trees */

    current_tree = TREE_MAIN;
    generate(comp_trees[TREE_MAIN]);

    current_tree = TREE_INIT;
    generate(comp_trees[TREE_INIT]);

    current_tree = TREE_MAIN;

    generate_final_program(0);
    UPDATE_PROGRAM_SIZE;

    if (mem_block[0].current_size > ADDRESS_MAX) {
	yyerror("Program too large");
	clean_parser();
	end_new_file();
	free_string(current_file);
	current_file = 0;
	return 0;
    }

    generate_final_program(1);

    free_tree();

    size = align(sizeof (program_t));

    /* delete argument information if we're not saving it */
    if (!(pragmas & PRAGMA_SAVE_TYPES))
	mem_block[A_ARGUMENT_TYPES].current_size = 0;
    if (!(mem_block[A_ARGUMENT_TYPES].current_size))
	mem_block[A_ARGUMENT_INDEX].current_size = 0;

    for (i=0; i<NUMPAREAS; i++)
	if (i != A_LINENUMBERS && i != A_FILE_INFO && 
	    i != A_FUNCTION_DEFS && i != A_FUNCTIONS &&
	    i != A_ARGUMENT_INDEX)
	    size += align(mem_block[i].current_size);
    
    num_func = mem_block[A_FUNCTIONS].current_size/sizeof(function_t);

    while (num_func && 
	   FUNC(func_index_map[num_func - 1])->address == ADDRESS_MAX) 
	num_func--;

    size += align((num_func * sizeof(function_t)));   /* A_FUNCTIONS */

    /* function flags */
    size += align(((comp_last_inherited + num_func) * sizeof(unsigned short)));

    /* A_ARGUMENT_INDEX */
    if (mem_block[A_ARGUMENT_INDEX].current_size)
	size += align(num_func * sizeof(unsigned short)); 

    p = (char *)DXALLOC(size, TAG_PROGRAM, "epilog: 1");
    prog = (program_t *)p;
    *prog = NULL_program;
    prog->total_size = size;
    prog->ref = 0;
    prog->func_ref = 0;
    ihe = lookup_ident("heart_beat");
    if (ihe && ihe->dn.function_num != -1) {
	prog->heart_beat = comp_def_index_map[ihe->dn.function_num] + 1;
	if (prog_flags && prog_flags[prog->heart_beat-1] & 
	    (FUNC_PROTOTYPE|FUNC_UNDEFINED))
	    prog->heart_beat = 0;
    } else
	prog->heart_beat = 0;
    prog->name = current_file;

    current_file = 0;

    total_num_prog_blocks++;
    total_prog_block_size += size;

    prog->line_swap_index = -1;

    /* Format is now:
     * <short total size> <short line_info_offset> <file info> <line info>
     */
    lnoff = 2 + (mem_block[A_FILE_INFO].current_size / sizeof(short));
    lnsz = lnoff * sizeof(short) + mem_block[A_LINENUMBERS].current_size;

    prog->file_info = (unsigned short *) DXALLOC(lnsz, TAG_LINENUMBERS
						    , "epilog");

    prog->file_info[0] = (unsigned short)lnsz;
    prog->file_info[1] = (unsigned short)lnoff;

    memcpy(((char*)&prog->file_info[2]),
	   mem_block[A_FILE_INFO].block,
	   mem_block[A_FILE_INFO].current_size);

    prog->line_info = (unsigned char *)(&prog->file_info[lnoff]);
    memcpy(((char*)&prog->file_info[lnoff]),
	   mem_block[A_LINENUMBERS].block,
	   mem_block[A_LINENUMBERS].current_size);

    p += align(sizeof(program_t));
    prog->program = p;
    prog->program_size = mem_block[A_PROGRAM].current_size;
    copy_in(A_PROGRAM, &p);


    /* copy the functions in */
    prog->last_inherited = comp_last_inherited;
    prog->num_functions_defined = num_func;

    prog->function_table = (function_t *)p;
    for (i = 0; i < num_func; i++)
	prog->function_table[i] = *FUNC(func_index_map[i]);

    p += align(sizeof(function_t) * num_func);


    prog->function_flags = (unsigned short *)p;
    if (prog_flags) {
	memcpy(p,prog_flags, 
	       (comp_last_inherited + num_func) * sizeof(unsigned short));
	FREE((char *)prog_flags);
    }
    p += align(sizeof(unsigned short) * (comp_last_inherited + num_func));

    if (mem_block[A_ARGUMENT_INDEX].current_size) {
	unsigned short *dest;

	prog->argument_types = (unsigned short *)p;
	copy_in(A_ARGUMENT_TYPES, &p);

	dest = prog->type_start = (unsigned short *)p;
	i = num_func;
	while (i--)
	    dest[i] = *((unsigned short *) mem_block[A_ARGUMENT_INDEX].block
			+ func_index_map[i]);
	p += align(num_func * sizeof(unsigned short));
    } else {
	prog->argument_types = 0;
	prog->type_start = 0;
    }
    
    prog->classes = (class_def_t *)p;
    prog->num_classes = mem_block[A_CLASS_DEF].current_size /
	  sizeof (class_def_t);
    copy_in(A_CLASS_DEF, &p);

    prog->class_members = (class_member_entry_t *)p;
    copy_in(A_CLASS_MEMBER, &p);

    prog->strings = (char **)p;
    prog->num_strings = mem_block[A_STRINGS].current_size /
	  sizeof (char *);
    copy_in(A_STRINGS, &p);

    prog->num_variables_defined = mem_block[A_VAR_NAME].current_size /
	sizeof (char *);
    prog->num_variables_total = mem_block[A_VAR_TEMP].current_size /
	  sizeof (variable_t);

    prog->variable_table = (char **)p;
    copy_in(A_VAR_NAME, &p);
    prog->variable_types = (unsigned short *)p;
    copy_in(A_VAR_TYPE, &p);

    prog->num_inherited = mem_block[A_INHERITS].current_size /
	  sizeof (inherit_t);
    if (prog->num_inherited) {
	prog->inherit = (inherit_t *)p;
	copy_in(A_INHERITS, &p);
    } else
	prog->inherit = 0;

#ifdef DEBUG
    if (p - (char *)prog != size) {
	fprintf(stderr, "Program size miscalculated for /%s.\n", prog->name);
	fprintf(stderr, "is: %i, expected: %i\n", p-(char *)prog, (int) size);
    }
#endif
#ifdef BINARIES
    if ((pragmas & PRAGMA_SAVE_BINARY)
#ifdef LPC_TO_C
    || compile_to_c
#endif
    ) {
	save_binary(prog, &mem_block[A_INCLUDES], &mem_block[A_PATCH]);
    }
#endif

    swap_line_numbers(prog); /* do this after saving binary */

    for (i = 0; i < mem_block[A_FUNCTION_DEFS].current_size / sizeof(*fundefp); i++) {
	fundefp = FUNCTION_TEMP(i)->next;
	while (fundefp) {
	    nextdefp = fundefp->next;
	    FREE((char *) fundefp);
	    fundefp = nextdefp;
	}
    }

    for (i=0; i<NUMAREAS; i++)
	FREE((char *)mem_block[i].block);
    if (comp_def_index_map) FREE((char *) comp_def_index_map);
    if (func_index_map) FREE((char *) func_index_map);
    
    /*  marion
	Do referencing here - avoid multiple referencing when an object
	inherits more than one object and one of the inherited is already
	loaded and not the last inherited
    */
    reference_prog (prog, "epilog");
    for (i = 0; (unsigned)i < prog->num_inherited; i++) {
	reference_prog (prog->inherit[i].prog, "inheritance");
    }
    release_tree();
    uninitialize_parser();
    scratch_destroy();
    clean_up_locals();
    free_unused_identifiers();
    end_new_file();

    return prog;
}

/*
 * Initialize the environment that the compiler needs.
 */
static void prolog P2(int, f, char *, name) {
    int i;

    function_context.num_parameters = -1;
    num_parse_error = 0;
    global_modifiers = 0;
    var_defined = 0;
    
    /* Initialize memory blocks where the result of the compilation
     * will be stored.
     */
    for (i=0; i < NUMAREAS; i++) {
	mem_block[i].block = DXALLOC(START_BLOCK_SIZE, TAG_COMPILER, "prolog: 2");
	mem_block[i].current_size = 0;
	mem_block[i].max_size = START_BLOCK_SIZE;
    }
    for (i = 0; i < NUMTREES; i++) {
        comp_trees[i] = 0;
    }
    prog_flags = 0;
    func_index_map = 0;
    comp_def_index_map = 0;

    memset(string_tags, 0, sizeof(string_tags));
    freed_string = -1;
    initialize_parser();

    current_file = make_shared_string(name);
    current_file_id = add_program_file(name, 1);

    /*
     * if we've got a simul_efun object and we're not reloading it, make a copy
     * of its class definitions in the object we're compiling now.
     */
    if (simul_efun_ob && *simul_efun_ob->name)
	copy_structures(simul_efun_ob->prog);

    start_new_file(f);
}

/*
 * The program has errors, clean things up.
 */
static void clean_parser() {
    int i, n;
    function_t *funp;
    compiler_temp_t *fundefp, *nextdefp;

    /*
     * Free function stuff.
     */
    for (i = 0; i < mem_block[A_FUNCTIONS].current_size / sizeof(function_t); i++) {
	funp = FUNC(i);
	if (funp->name) free_string(funp->name);
    }

    for (i = 0; i < mem_block[A_FUNCTION_DEFS].current_size / 
	     sizeof(compiler_temp_t); i++) {
	fundefp = FUNCTION_TEMP(i)->next;
	while (fundefp) {
	    nextdefp = fundefp->next;
	    FREE((char *) fundefp);
	    fundefp = nextdefp;
	}
    }

    n = mem_block[A_STRINGS].current_size / sizeof(char *);
    for (i = 0; i < n; i++) {
	free_string(*((char **)mem_block[A_STRINGS].block + i));
    }
    n = mem_block[A_VAR_NAME].current_size / sizeof(char *);
    for (i = 0; i < n; i++) {
	free_string(*((char **)mem_block[A_VAR_NAME].block + i));
    }

    prog = 0;
    for (i=0; i<NUMAREAS; i++)
	FREE(mem_block[i].block);
    if (comp_def_index_map) FREE((char *) comp_def_index_map);
    if (func_index_map) FREE((char *) func_index_map);
    if (prog_flags) FREE((char *) prog_flags);

    /* don't need the parse trees any more */
    release_tree();
    uninitialize_parser();
    clean_up_locals();
    scratch_destroy();
    free_unused_identifiers();
}

char *
the_file_name P1(char *, name)
{
    char *tmp;
    int len;

    len = strlen(name);
    if (len < 3) {
	return string_copy(name, "the_file_name");
    }
    tmp = new_string(len - 1, "the_file_name");
    if (!tmp) {
	return string_copy(name, "the_file_name");
    }
    tmp[0] = '/';
    strncpy(tmp + 1, name, len - 2);
    tmp[len - 1] = '\0';
    return tmp;
}

static int case_compare P2(parse_node_t **, c1, parse_node_t **, c2) {
    if ((*c1)->kind == NODE_DEFAULT)
	return -1;
    if ((*c2)->kind == NODE_DEFAULT)
	return 1;

    if ((*c1)->r.number > (*c2)->r.number) return 1;
    if ((*c1)->r.number < (*c2)->r.number) return -1;
    return 0;
}

static int string_case_compare P2(parse_node_t **, c1, parse_node_t **, c2) {
    int i1, i2;
    char *p1, *p2;
    
    if ((*c1)->kind == NODE_DEFAULT)
	return -1;
    if ((*c2)->kind == NODE_DEFAULT)
	return 1;

    i1 = (*c1)->r.number;
    i2 = (*c2)->r.number;
    p1 = (i1 ? PROG_STRING(i1) : 0);
    p2 = (i2 ? PROG_STRING(i2) : 0);
    
    return (p1 - p2);
}

void prepare_cases P2(parse_node_t *, pn, int, start) {
    parse_node_t **ce_start, **ce_end, **ce;
    int end, last_key, this_key;
    int direct = 1;
    
    ce_start = (parse_node_t **)&mem_block[A_CASES].block[start];
    end = mem_block[A_CASES].current_size;
    ce_end = (parse_node_t **)&mem_block[A_CASES].block[end];

    if (ce_start == ce_end) {
	/* no cases */
	pn->v.expr = 0;
	mem_block[A_CASES].current_size = start;
	return;
    }

    if (pn->kind == NODE_SWITCH_STRINGS)
	quickSort((char *)ce_start, ce_end - ce_start, sizeof(parse_node_t *),
		  string_case_compare);
    else
	quickSort((char *)ce_start, ce_end - ce_start, sizeof(parse_node_t *),
		  case_compare);

    ce = ce_start;
    if ((*ce)->kind == NODE_DEFAULT) {
	if (ce + 1 == ce_end) {
	    /* only a default */
	    pn->v.expr = *ce;
	    (*ce)->l.expr = 0;
	    mem_block[A_CASES].current_size = start;
	    return;
	}
	ce++;
	(*(ce-1))->l.expr = *ce;
    }
    if ((*ce)->v.expr) {
	last_key = (*ce)->v.expr->r.number;
	direct = 0;
    } else
	last_key = (*ce)->r.number;
    ce++;
    while (ce < ce_end) {
	this_key = (*ce)->r.number;
	if (pn->kind == NODE_SWITCH_RANGES && this_key <= last_key) {
	    char buf[1024];
	    char *end = EndOf(buf);
	    char *p;
	    char *f1, *f2;
	    int fi1, fi2;
	    int l1, l2;

	    /* make sure line numbers exist for the cases */
	    save_file_info(current_file_id, current_line - current_line_saved);
	    current_line_saved = current_line;

	    translate_absolute_line((*ce)->line, 
				    (unsigned short *)mem_block[A_FILE_INFO].block,
				    &fi1, &l1);
	    translate_absolute_line((*(ce-1))->line, 
				    (unsigned short *)mem_block[A_FILE_INFO].block,
				    &fi2, &l2);
	    f1 = PROG_STRING(fi1 - 1);
	    f2 = PROG_STRING(fi2 - 1);

	    p = strput(buf, end, "Overlapping cases: ");
	    if (f1) {
		p = strput(p, end, f1);
		p = strput(p, end, ":");
	    } else
		p = strput(p, end, "line ");
	    p = strput_int(p, end, l1);
	    p = strput(p, end, " and ");
	    if (f2) {
		p = strput(p, end, f2);
		p = strput(p, end, ":");
	    } else
		p = strput(p, end, "line ");
	    p = strput_int(p, end, l2);
	    p = strput(p, end, ".");
	    yyerror(buf);
	}
	(*(ce-1))->l.expr = *ce;
	if ((*ce)->v.expr) {
	    last_key = (*ce)->v.expr->r.number;
	    direct = 0;
	} else {
	    if (last_key + 1 != this_key) direct = 0;
	    last_key = this_key;
	}
	ce++;
    }
    (*(ce_end-1))->l.expr = 0;
    if (direct && pn->kind == NODE_SWITCH_NUMBERS)
	pn->kind = NODE_SWITCH_DIRECT;
    pn->v.expr = *(ce_start);
    mem_block[A_CASES].current_size = start;
}

void
save_file_info P2(int, file_id, int, lines) {
    short fi[2];

    fi[0] = lines;
    fi[1] = file_id;
    add_to_mem_block(A_FILE_INFO, (char *)&fi[0], sizeof(fi));
}

int
add_program_file P2(char *, name, int, top) {
    if (!top)
	add_to_mem_block(A_INCLUDES, name, strlen(name)+1);
    return store_prog_string(name) + 1;
}