📄 gen.c
字号:
/* * Copyright (c) 1997 - 2005 Kungliga Tekniska H鰃skolan * (Royal Institute of Technology, Stockholm, Sweden). * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * 3. Neither the name of the Institute nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``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 INSTITUTE OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */#include "gen_locl.h"RCSID("$Id: gen.c 22429 2008-01-13 10:25:50Z lha $");FILE *headerfile, *codefile, *logfile;#define STEM "asn1"static const char *orig_filename;static char *header;static const char *headerbase = STEM;/* * list of all IMPORTs */struct import { const char *module; struct import *next;};static struct import *imports = NULL;voidadd_import (const char *module){ struct import *tmp = emalloc (sizeof(*tmp)); tmp->module = module; tmp->next = imports; imports = tmp; fprintf (headerfile, "#include <%s_asn1.h>\n", module);}const char *get_filename (void){ return orig_filename;}voidinit_generate (const char *filename, const char *base){ char *fn; orig_filename = filename; if (base != NULL) { headerbase = strdup(base); if (headerbase == NULL) errx(1, "strdup"); } asprintf(&header, "%s.h", headerbase); if (header == NULL) errx(1, "malloc"); headerfile = fopen (header, "w"); if (headerfile == NULL) err (1, "open %s", header); fprintf (headerfile, "/* Generated from %s */\n" "/* Do not edit */\n\n", filename); fprintf (headerfile, "#ifndef __%s_h__\n" "#define __%s_h__\n\n", headerbase, headerbase); fprintf (headerfile, "#include <stddef.h>\n" "#include <time.h>\n\n"); fprintf (headerfile, "#ifndef __asn1_common_definitions__\n" "#define __asn1_common_definitions__\n\n"); fprintf (headerfile, "typedef struct heim_integer {\n" " size_t length;\n" " void *data;\n" " int negative;\n" "} heim_integer;\n\n"); fprintf (headerfile, "typedef struct heim_octet_string {\n" " size_t length;\n" " void *data;\n" "} heim_octet_string;\n\n"); fprintf (headerfile, "typedef char *heim_general_string;\n\n" ); fprintf (headerfile, "typedef char *heim_utf8_string;\n\n" ); fprintf (headerfile, "typedef char *heim_printable_string;\n\n" ); fprintf (headerfile, "typedef char *heim_ia5_string;\n\n" ); fprintf (headerfile, "typedef struct heim_bmp_string {\n" " size_t length;\n" " uint16_t *data;\n" "} heim_bmp_string;\n\n"); fprintf (headerfile, "typedef struct heim_universal_string {\n" " size_t length;\n" " uint32_t *data;\n" "} heim_universal_string;\n\n"); fprintf (headerfile, "typedef char *heim_visible_string;\n\n" ); fprintf (headerfile, "typedef struct heim_oid {\n" " size_t length;\n" " unsigned *components;\n" "} heim_oid;\n\n"); fprintf (headerfile, "typedef struct heim_bit_string {\n" " size_t length;\n" " void *data;\n" "} heim_bit_string;\n\n"); fprintf (headerfile, "typedef struct heim_octet_string heim_any;\n" "typedef struct heim_octet_string heim_any_set;\n\n"); fputs("#define ASN1_MALLOC_ENCODE(T, B, BL, S, L, R) \\\n" " do { \\\n" " (BL) = length_##T((S)); \\\n" " (B) = malloc((BL)); \\\n" " if((B) == NULL) { \\\n" " (R) = ENOMEM; \\\n" " } else { \\\n" " (R) = encode_##T(((unsigned char*)(B)) + (BL) - 1, (BL), \\\n" " (S), (L)); \\\n" " if((R) != 0) { \\\n" " free((B)); \\\n" " (B) = NULL; \\\n" " } \\\n" " } \\\n" " } while (0)\n\n", headerfile); fprintf (headerfile, "struct units;\n\n"); fprintf (headerfile, "#endif\n\n"); asprintf(&fn, "%s_files", base); if (fn == NULL) errx(1, "malloc"); logfile = fopen(fn, "w"); if (logfile == NULL) err (1, "open %s", fn);}voidclose_generate (void){ fprintf (headerfile, "#endif /* __%s_h__ */\n", headerbase); fclose (headerfile); fprintf (logfile, "\n"); fclose (logfile);}voidgen_assign_defval(const char *var, struct value *val){ switch(val->type) { case stringvalue: fprintf(codefile, "if((%s = strdup(\"%s\")) == NULL)\nreturn ENOMEM;\n", var, val->u.stringvalue); break; case integervalue: fprintf(codefile, "%s = %d;\n", var, val->u.integervalue); break; case booleanvalue: if(val->u.booleanvalue) fprintf(codefile, "%s = TRUE;\n", var); else fprintf(codefile, "%s = FALSE;\n", var); break; default: abort(); }}voidgen_compare_defval(const char *var, struct value *val){ switch(val->type) { case stringvalue: fprintf(codefile, "if(strcmp(%s, \"%s\") != 0)\n", var, val->u.stringvalue); break; case integervalue: fprintf(codefile, "if(%s != %d)\n", var, val->u.integervalue); break; case booleanvalue: if(val->u.booleanvalue) fprintf(codefile, "if(!%s)\n", var); else fprintf(codefile, "if(%s)\n", var); break; default: abort(); }}static voidgenerate_header_of_codefile(const char *name){ char *filename; if (codefile != NULL) abort(); asprintf (&filename, "%s_%s.x", STEM, name); if (filename == NULL) errx(1, "malloc"); codefile = fopen (filename, "w"); if (codefile == NULL) err (1, "fopen %s", filename); fprintf(logfile, "%s ", filename); free(filename); fprintf (codefile, "/* Generated from %s */\n" "/* Do not edit */\n\n" "#include <stdio.h>\n" "#include <stdlib.h>\n" "#include <time.h>\n" "#include <string.h>\n" "#include <errno.h>\n" "#include <limits.h>\n" "#include <krb5-types.h>\n", orig_filename); fprintf (codefile, "#include <%s.h>\n", headerbase); fprintf (codefile, "#include <asn1_err.h>\n" "#include <der.h>\n" "#include <parse_units.h>\n\n");}static voidclose_codefile(void){ if (codefile == NULL) abort(); fclose(codefile); codefile = NULL;}voidgenerate_constant (const Symbol *s){ switch(s->value->type) { case booleanvalue: break; case integervalue: fprintf (headerfile, "enum { %s = %d };\n\n", s->gen_name, s->value->u.integervalue); break; case nullvalue: break; case stringvalue: break; case objectidentifiervalue: { struct objid *o, **list; int i, len; generate_header_of_codefile(s->gen_name); len = 0; for (o = s->value->u.objectidentifiervalue; o != NULL; o = o->next) len++; list = emalloc(sizeof(*list) * len); i = 0; for (o = s->value->u.objectidentifiervalue; o != NULL; o = o->next) list[i++] = o; fprintf (headerfile, "/* OBJECT IDENTIFIER %s ::= { ", s->name); for (i = len - 1 ; i >= 0; i--) { o = list[i]; fprintf(headerfile, "%s(%d) ", o->label ? o->label : "label-less", o->value); } fprintf (headerfile, "} */\n"); fprintf (headerfile, "const heim_oid *oid_%s(void);\n\n", s->gen_name); fprintf (codefile, "static unsigned oid_%s_variable_num[%d] = {", s->gen_name, len); for (i = len - 1 ; i >= 0; i--) { fprintf(codefile, "%d%s ", list[i]->value, i > 0 ? "," : ""); } fprintf(codefile, "};\n"); fprintf (codefile, "static const heim_oid oid_%s_variable = " "{ %d, oid_%s_variable_num };\n\n", s->gen_name, len, s->gen_name); fprintf (codefile, "const heim_oid *oid_%s(void)\n" "{\n" "return &oid_%s_variable;\n" "}\n\n", s->gen_name, s->gen_name); close_codefile(); break; } default: abort(); }}static voidspace(int level){ while(level-- > 0) fprintf(headerfile, " ");}static const char *last_member_p(struct member *m){ struct member *n = ASN1_TAILQ_NEXT(m, members); if (n == NULL) return ""; if (n->ellipsis && ASN1_TAILQ_NEXT(n, members) == NULL) return ""; return ",";}static struct member *have_ellipsis(Type *t){ struct member *m; ASN1_TAILQ_FOREACH(m, t->members, members) { if (m->ellipsis) return m; } return NULL;}static voiddefine_asn1 (int level, Type *t){ switch (t->type) { case TType: fprintf (headerfile, "%s", t->symbol->name); break; case TInteger: if(t->members == NULL) { fprintf (headerfile, "INTEGER"); if (t->range) fprintf (headerfile, " (%d..%d)", t->range->min, t->range->max); } else { Member *m; fprintf (headerfile, "INTEGER {\n"); ASN1_TAILQ_FOREACH(m, t->members, members) { space (level + 1); fprintf(headerfile, "%s(%d)%s\n", m->gen_name, m->val, last_member_p(m)); } space(level); fprintf (headerfile, "}"); }
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -