//
// aegis - project change supervisor
// Copyright (C) 1991-1994, 1998, 1999, 2001-2008 Peter Miller
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation; either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see
// .
//
#include // for assert
#include
#include
type_list::~type_list()
{
}
type_list::type_list(const nstring &a_name, type *a_type) :
type(a_name),
subtype(a_type)
{
}
void
type_list::gen_include()
const
{
assert(subtype);
indent_putchar('\n');
indent_printf("#ifndef %s_DEF\n", def_name().c_str());
indent_printf("#define %s_DEF\n", def_name().c_str());
indent_printf("struct %s_ty\n", def_name().c_str());
indent_printf("{\n");
indent_printf("size_t length;\n");
indent_printf("size_t maximum;\n");
subtype->gen_include_declarator("list", 1);
indent_printf("};\n");
indent_printf("#endif // %s_DEF\n", def_name().c_str());
indent_putchar('\n');
indent_printf("extern meta_type %s_type;\n", def_name().c_str());
indent_putchar('\n');
indent_printf
(
"%s_ty *%s_copy(%s_ty *);\n",
def_name().c_str(),
def_name().c_str(),
def_name().c_str()
);
indent_printf
(
"%s_ty *%s_clone(%s_ty *);\n",
def_name().c_str(),
def_name().c_str(),
def_name().c_str()
);
indent_printf
(
"void %s_write(const output::pointer &fp, const char *name, "
"%s_ty *value);\n",
def_name().c_str(),
def_name().c_str()
);
indent_printf
(
"void %s_write_xml(const output::pointer &fp, const char *name, "
"%s_ty *value);\n",
def_name().c_str(),
def_name().c_str()
);
indent_printf("#ifdef DEBUG\n");
indent_printf
(
"void %s_trace_real(const char *name, const %s_ty *value);\n",
def_name().c_str(),
def_name().c_str()
);
indent_printf
(
"#define %s_trace(x) ((void)(trace_pretest_ && (trace_where_, "
"%s_trace_real(trace_stringize(x), x), 0)))\n",
def_name().c_str(),
def_name().c_str()
);
indent_printf("#else\n");
indent_printf("#define %s_trace(x)\n", def_name().c_str());
indent_printf("#endif\n");
}
void
type_list::gen_include_declarator(const nstring &variable_name,
bool is_a_list) const
{
assert(subtype);
const char *deref = (is_a_list ? "*" : "");
indent_printf
(
"%s_ty %s*%s;\n",
def_name().c_str(),
deref,
variable_name.c_str()
);
}
void
type_list::gen_code()
const
{
assert(subtype);
static bool lincl_done;
if (!lincl_done)
{
indent_putchar('\n');
indent_printf("#include \n");
lincl_done = true;
}
indent_putchar('\n');
indent_printf("void\n");
indent_printf
(
"%s_write(const output::pointer &fp, const char *name, "
"%s_ty *this_thing)\n",
def_name().c_str(),
def_name().c_str()
);
indent_printf("{\n");
indent_printf("size_t j;\n");
indent_putchar('\n');
indent_printf("if (!this_thing)\n");
indent_more();
indent_printf("return;\n");
indent_less();
indent_printf
(
"trace((\"%s_write(name = \\\"%%s\\\", this_thing = %%08lX)\\n"
"{\\n\", name, (long)this_thing));\n",
def_name().c_str()
);
indent_printf("if (name)\n");
indent_printf("{\n");
indent_printf("fp->fputs(name);\n");
indent_printf("fp->fputs(\" =\\n\");\n");
indent_printf("}\n");
indent_printf("assert(this_thing->length <= this_thing->maximum);\n");
indent_printf("assert(!this_thing->list == !this_thing->maximum);\n");
indent_printf("fp->fputs(\"[\\n\");\n");
indent_printf("for (j = 0; j < this_thing->length; ++j)\n");
indent_printf("{\n");
subtype->gen_code_declarator("list[j]", 1, ATTRIBUTE_SHOW_IF_DEFAULT);
indent_printf("fp->fputs(\",\\n\");\n");
indent_printf("}\n");
indent_printf("fp->fputs(\"]\");\n");
indent_printf("if (name)\n");
indent_more();
indent_printf("fp->fputs(\";\\n\");\n");
indent_less();
indent_printf("trace((\"}\\n\"));\n");
indent_printf("}\n");
indent_putchar('\n');
indent_printf("void\n");
indent_printf
(
"%s_write_xml(const output::pointer &fp, const char *name, "
"%s_ty *this_thing)\n",
def_name().c_str(),
def_name().c_str()
);
indent_printf("{\n");
indent_printf("size_t j;\n");
indent_putchar('\n');
indent_printf("if (!this_thing)\n");
indent_more();
indent_printf("return;\n");
indent_less();
indent_printf
(
"trace((\"%s_write_xml(name = \\\"%%s\\\", this_thing = %%08lX)\\n"
"{\\n\", name, (long)this_thing));\n",
def_name().c_str()
);
indent_printf("assert(name);\n");
indent_printf("fp->fputc('<');\n");
indent_printf("fp->fputs(name);\n");
indent_printf("fp->fputs(\">\\n\");\n");
indent_printf("assert(this_thing->length <= this_thing->maximum);\n");
indent_printf("assert(!this_thing->list == !this_thing->maximum);\n");
indent_printf("for (j = 0; j < this_thing->length; ++j)\n");
indent_printf("{\n");
subtype->gen_code_call_xml
(
subtype->def_name(),
"list[j]",
ATTRIBUTE_SHOW_IF_DEFAULT
);
indent_printf("}\n");
indent_printf("fp->fputs(\"\");\n");
indent_printf("fp->fputs(name);\n");
indent_printf("fp->fputs(\">\\n\");\n");
indent_printf("trace((\"}\\n\"));\n");
indent_printf("}\n");
indent_putchar('\n');
indent_printf("static void *\n");
indent_printf("%s_alloc(void)\n", def_name().c_str());
indent_printf("{\n");
indent_printf("%s_ty *result;\n\n", def_name().c_str());
indent_printf("trace((\"%s_alloc()\\n{\\n\"));\n",
def_name().c_str());
indent_printf
(
"result = (%s_ty *)mem_alloc(sizeof(%s_ty));\n",
def_name().c_str(),
def_name().c_str()
);
indent_printf("result->list = 0;\n");
indent_printf("result->length = 0;\n");
indent_printf("result->maximum = 0;\n");
indent_printf("trace((\"return %%08lX;\\n\", (long)result));\n");
indent_printf("trace((\"}\\n\"));\n");
indent_printf("return result;\n");
indent_printf("}\n");
indent_putchar('\n');
indent_printf("static void\n");
indent_printf("%s_free(void *that)\n", def_name().c_str());
indent_printf("{\n");
indent_printf("%s_ty *this_thing;\n", def_name().c_str());
indent_printf("size_t j;\n");
indent_putchar('\n');
indent_printf("this_thing = (%s_ty *)that;\n", def_name().c_str());
indent_printf("if (!this_thing)\n");
indent_more();
indent_printf("return;\n");
indent_less();
indent_printf
(
"trace((\"%s_free(this_thing = %%08lX)\\n{\\n\", (long)this_thing));\n",
def_name().c_str()
);
indent_printf("assert(this_thing->length <= this_thing->maximum);\n");
indent_printf("assert(!this_thing->list == !this_thing->maximum);\n");
indent_printf("for (j = 0; j < this_thing->length; ++j)\n");
indent_more();
subtype->gen_free_declarator("list[j]", 1);
indent_less();
indent_printf("delete [] this_thing->list;\n");
indent_printf("mem_free(this_thing);\n");
indent_printf("trace((\"}\\n\"));\n");
indent_printf("}\n");
indent_putchar('\n');
indent_printf("static void *\n");
indent_printf
(
"%s_parse(void *that, meta_type **type_pp)\n",
def_name().c_str()
);
indent_printf("{\n");
indent_printf("%s_ty *this_thing;\n", def_name().c_str());
indent_printf("void *addr;\n");
indent_putchar('\n');
indent_printf("this_thing = (%s_ty *)that;\n", def_name().c_str());
indent_printf
(
"trace((\"%s_parse(this_thing = %%08lX, type_pp = %%08lX)\\n"
"{\\n\", (long)this_thing, (long)type_pp));\n",
def_name().c_str()
);
indent_printf("assert(this_thing->length <= this_thing->maximum);\n");
indent_printf("assert(!this_thing->list == !this_thing->maximum);\n");
indent_printf("*type_pp = &%s_type;\n", subtype->def_name().c_str());
indent_printf("trace_pointer(*type_pp);\n");
indent_printf("if (this_thing->length >= this_thing->maximum)\n");
indent_printf("{\n");
indent_printf("this_thing->maximum = this_thing->maximum * 2 + 16;\n");
indent_printf
(
"%s *new_list = new %s [this_thing->maximum];\n",
subtype->c_name().c_str(),
subtype->c_name().c_str()
);
indent_printf("for (size_t j = 0; j < this_thing->length; ++j)\n");
indent_more();
indent_printf("new_list[j] = this_thing->list[j];\n");
indent_less();
indent_printf("delete [] this_thing->list;\n");
indent_printf("this_thing->list = new_list;\n");
indent_printf("}\n");
indent_printf("addr = &this_thing->list[this_thing->length++];\n");
indent_printf("trace((\"return %%08lX;\\n\", (long)addr));\n");
indent_printf("trace((\"}\\n\"));\n");
indent_printf("return addr;\n");
indent_printf("}\n");
indent_putchar('\n');
indent_printf("static rpt_value::pointer\n");
indent_printf("%s_convert(void *that)\n", def_name().c_str());
indent_printf("{\n");
indent_printf("%s_ty *this_thing;\n", def_name().c_str());
indent_printf("size_t j;\n");
indent_printf("rpt_value::pointer vp;\n");
indent_putchar('\n');
indent_printf("this_thing = *(%s_ty **)that;\n",
def_name().c_str());
indent_printf("if (!this_thing)\n");
indent_more();
indent_printf("return rpt_value::pointer();\n");
indent_less();
indent_printf
(
"trace((\"%s_convert(this_thing = %%08lX)\\n{\\n\", "
"(long)this_thing));\n",
def_name().c_str()
);
indent_printf("assert(this_thing->length <= this_thing->maximum);\n");
indent_printf("assert(!this_thing->list == !this_thing->maximum);\n");
indent_printf("rpt_value_list *p = new rpt_value_list();\n");
indent_printf("rpt_value::pointer result(p);\n");
indent_printf("for (j = 0; j < this_thing->length; ++j)\n");
indent_printf("{\n");
indent_printf
(
"vp = %s_type.convert(&this_thing->list[j]);\n",
subtype->def_name().c_str()
);
indent_printf("assert(vp);\n");
indent_printf("p->append(vp);\n");
indent_printf("}\n");
indent_printf("trace((\"}\\n\"));\n");
indent_printf("trace((\"return %%08lX;\\n\", (long)result.get()));\n");
indent_printf("return result;\n");
indent_printf("}\n");
indent_putchar('\n');
indent_printf("%s_ty *\n", def_name().c_str());
indent_printf
(
"%s_copy(%s_ty *from)\n",
def_name().c_str(),
def_name().c_str()
);
indent_printf("{\n");
indent_printf("size_t j;\n");
indent_printf("%s_ty *result;\n", def_name().c_str());
indent_putchar('\n');
indent_printf("if (!from)\n");
indent_more();
indent_printf("return 0;\n");
indent_less();
indent_printf
(
"trace((\"%s_copy(from = %%08lX)\\n{\\n\", (long)from));\n",
def_name().c_str()
);
indent_printf
(
"result = (%s_ty *)%s_alloc();\n",
def_name().c_str(),
def_name().c_str()
);
indent_printf("assert(from->length <= from->maximum);\n");
indent_printf("assert(!from->list == !from->maximum);\n");
indent_printf("for (j = 0; j < from->length; ++j)\n");
indent_printf("{\n");
indent_printf("%s mp;\n", subtype->c_name().c_str());
indent_printf("%s *mpp;\n", subtype->c_name().c_str());
indent_printf("meta_type *bogus;\n\n");
indent_printf("mp = from->list[j];\n");
indent_printf
(
"mpp = (%s*)%s_parse(result, &bogus);\n",
subtype->c_name().c_str(),
def_name().c_str()
);
indent_printf("*mpp = %s_copy(mp);\n", subtype->def_name().c_str());
indent_printf("}\n");
indent_printf("trace((\"return %%8.8lX\\n\", (long)result));\n");
indent_printf("trace((\"}\\n\"));\n");
indent_printf("return result;\n");
indent_printf("}\n");
indent_putchar('\n');
indent_printf("%s_ty *\n", def_name().c_str());
indent_printf
(
"%s_clone(%s_ty *from)\n",
def_name().c_str(),
def_name().c_str()
);
indent_printf("{\n");
indent_printf("return %s_copy(from);\n", def_name().c_str());
indent_printf("}\n");
indent_printf("\n#ifdef DEBUG\n\n");
indent_printf("void\n");
indent_printf
(
"%s_trace_real(const char *name, const %s_ty *value)\n",
def_name().c_str(),
def_name().c_str()
);
indent_printf("{\n");
indent_printf("if (name && *name)\n");
indent_printf("{\n");
indent_printf("trace_printf(\"%%s = \", name);\n");
indent_printf("}\n");
indent_printf("if (!value)\n");
indent_printf("{\n");
indent_printf("trace_printf(\"NULL\");\n");
indent_printf("}\n");
indent_printf("else\n");
indent_printf("{\n");
indent_printf("trace_printf(\"[\\n\");\n");
indent_printf
(
"trace_printf(\"// length = %%ld\\n\", (long)value->length);\n"
);
indent_printf
(
"trace_printf(\"// maximum = %%ld\\n\", (long)value->maximum);\n"
);
indent_printf("assert(value->length <= value->maximum);\n");
indent_printf("assert(!value->list == !value->maximum);\n");
indent_printf("for (size_t j = 0; j < value->length; ++j)\n");
indent_printf("{\n");
indent_printf("%s mp = value->list[j];\n", subtype->c_name().c_str());
subtype->gen_code_trace("", "mp");
indent_printf("}\n");
indent_printf("trace_printf(\"]\");\n");
indent_printf("}\n");
indent_printf("trace_printf((name && *name) ? \";\\n\" : \",\\n\");\n");
indent_printf("}\n");
indent_printf("\n#endif // DEBUG\n");
indent_putchar('\n');
indent_printf("meta_type %s_type =\n", def_name().c_str());
indent_printf("{\n");
indent_printf("\"%s\",\n", def_name().c_str());
indent_printf("%s_alloc,\n", def_name().c_str());
indent_printf("%s_free,\n", def_name().c_str());
indent_printf("0, // enum_parse\n");
indent_printf("%s_parse,\n", def_name().c_str());
indent_printf("0, // struct_parse\n");
indent_printf("0, // fuzzy\n");
indent_printf("%s_convert,\n", def_name().c_str());
indent_printf("generic_struct_is_set,\n");
indent_printf("};\n");
}
void
type_list::gen_code_declarator(const nstring &variable_name, bool is_a_list,
int) const
{
assert(subtype);
indent_printf("%s_write(fp, ", def_name().c_str());
if (is_a_list)
indent_printf("\"\"");
else
indent_printf("\"%s\"", variable_name.c_str());
indent_printf(", this_thing->%s);\n", variable_name.c_str());
}
void
type_list::gen_code_call_xml(const nstring &form_name,
const nstring &member_name, int) const
{
assert(subtype);
indent_printf
(
"%s_write_xml(fp, \"%s\", this_thing->%s);\n",
def_name().c_str(),
form_name.c_str(),
member_name.c_str()
);
}
void
type_list::gen_code_copy(const nstring &member_name)
const
{
assert(subtype);
indent_printf
(
"result->%s = %s_clone(this_thing->%s);\n",
member_name.c_str(),
def_name().c_str(),
member_name.c_str()
);
}
void
type_list::gen_free_declarator(const nstring &variable_name, bool)
const
{
assert(subtype);
if (is_in_include_file())
{
indent_printf
(
"%s_type.free(this_thing->%s);\n",
def_name().c_str(),
variable_name.c_str()
);
}
else
{
indent_printf
(
"%s_free(this_thing->%s);\n",
def_name().c_str(),
variable_name.c_str()
);
}
}
void
type_list::in_include_file()
{
assert(subtype);
type::in_include_file();
subtype->in_include_file();
}
nstring
type_list::c_name_inner()
const
{
return (def_name() + "_ty *");
}
bool
type_list::has_a_mask()
const
{
return false;
}
void
type_list::gen_code_trace(const nstring &vname, const nstring &value)
const
{
indent_printf
(
"%s_trace_real(\"%s\", %s);\n",
def_name().c_str(),
vname.c_str(),
value.c_str()
);
}