X-Git-Url: https://gerrit.o-ran-sc.org/r/gitweb?a=blobdiff_plain;ds=sidebyside;f=libasn1compiler%2Fasn1c_out.h;fp=libasn1compiler%2Fasn1c_out.h;h=e999fc9d2ee88e90a315ca10df3c836229749a11;hb=70ee6fc793ec8e828067a3569849b6c216054497;hp=0000000000000000000000000000000000000000;hpb=59f84608ec15c016958a6e0e0ddd813f376c0925;p=com%2Fasn1c.git diff --git a/libasn1compiler/asn1c_out.h b/libasn1compiler/asn1c_out.h new file mode 100644 index 0000000..e999fc9 --- /dev/null +++ b/libasn1compiler/asn1c_out.h @@ -0,0 +1,162 @@ +#ifndef ASN1_COMPILED_OUTPUT_H +#define ASN1_COMPILED_OUTPUT_H + +/* + * An elementary chunk of target language text. + */ +typedef struct out_chunk { + char *buf; + int len; + + TQ_ENTRY(struct out_chunk) next; +} out_chunk_t; + +typedef struct compiler_streams { + enum { + OT_IGNORE, /* Ignore this output */ + OT_INCLUDES, /* #include files */ + OT_DEPS, /* Dependencies (other than #includes) */ + OT_FWD_DECLS, /* Forward declarations */ + OT_FWD_DEFS, /* Forward definitions */ + OT_TYPE_DECLS, /* Type declarations */ + OT_FUNC_DECLS, /* Function declarations */ + OT_POST_INCLUDE,/* #include after type definition */ + OT_IOC_TABLES, /* Information Object Class tables */ + OT_CTABLES, /* Constraint tables */ + OT_CODE, /* Some code */ + OT_CTDEFS, /* Constraint definitions */ + OT_STAT_DEFS, /* Static definitions */ + OT_MAX + } target; + + struct compiler_stream_destination_s { + TQ_HEAD(out_chunk_t) chunks; + int indent_level; + int indented; + } destination[OT_MAX]; +} compiler_streams_t; + +static char *_compiler_stream2str[] __attribute__ ((unused)) + = { "IGNORE", "INCLUDES", "DEPS", "FWD-DECLS", "FWD-DEFS", "TYPE-DECLS", "FUNC-DECLS", "POST-INCLUDE", "IOC-TABLES", "CTABLES", "CODE", "CTDEFS", "STAT-DEFS" }; + +int asn1c_compiled_output(arg_t *arg, const char *file, int lineno, + const char *func, const char *fmt, ...) + __attribute__((format(printf, 5, 6))); + + +/***************************************************************** + * Useful macros for invoking asn1c_compiled_output() and friends. + */ + +/* Redirect output to a different stream. */ +#define REDIR(foo) do { arg->target->target = foo; } while(0) +#define INDENT_LEVEL \ + arg->target->destination[arg->target->target].indent_level +#define INDENT(val) INDENT_LEVEL += (val) +#define INDENTED(code) do { \ + INDENT(+1); \ + do { code; } while(0); \ + INDENT(-1); \ + } while(0) + +#define EMBED(ev) \ + do { \ + arg->embed++; \ + INDENTED(arg_t _tmp = *arg; _tmp.expr = ev; \ + _tmp.default_cb(&_tmp, NULL);); \ + arg->embed--; \ + assert(arg->target->target == OT_TYPE_DECLS \ + || arg->target->target == OT_FWD_DEFS); \ + } while(0) + +#define EMBED_WITH_IOCT(ev, ioc) \ + do { \ + arg->embed++; \ + INDENTED(arg_t _tmp = *arg; _tmp.expr = ev; \ + _tmp.default_cb(&_tmp, ((ioc).ioct ? &ioc : 0));); \ + arg->embed--; \ + assert(arg->target->target == OT_TYPE_DECLS \ + || arg->target->target == OT_FWD_DEFS); \ + } while(0) + +/* Output a piece of text into a default stream */ +#define OUT(fmt, args...) \ + asn1c_compiled_output(arg, __FILE__, __LINE__, __func__, fmt, ##args) +#define OUT_NOINDENT(fmt, args...) do { \ + int _saved_indent = INDENT_LEVEL; \ + INDENT_LEVEL = 0; \ + OUT(fmt, ##args); \ + INDENT_LEVEL = _saved_indent; \ +} while(0) +#define OUT_DEBUG(fmt, args...) do { \ + if(arg->flags & A1C_DEBUG) OUT(fmt, ##args); \ + } while(0) + +/* Generate #include line */ +#define GEN_INCLUDE_STD(typename) do { \ + if((arg->flags & A1C_INCLUDES_QUOTED)) { \ + GEN_INCLUDE("\"" typename ".h\""); \ + } else { \ + GEN_INCLUDE("<" typename ".h>"); \ + } } while(0) +#define GEN_INCLUDE(filename) \ + GEN_POS_INCLUDE(OT_INCLUDES, filename) +#define GEN_POSTINCLUDE(filename) \ + GEN_POS_INCLUDE(OT_POST_INCLUDE, filename) +#define GEN_POS_INCLUDE(pos, filename) do { \ + int saved_target = arg->target->target; \ + if(!filename) break; \ + REDIR(pos); \ + OUT_NOINDENT("#include %s\n", filename); \ + REDIR(saved_target); \ +} while(0) +#define GEN_POS_INCLUDE_BASE(pos, expr) do { \ + asn1p_expr_t *rhs_pspecs = expr->rhs_pspecs; \ + expr->rhs_pspecs = (asn1p_expr_t *)0; \ + int saved_target = arg->target->target; \ + REDIR(pos); \ + OUT_NOINDENT("#include %s\n", \ + asn1c_type_name(arg, expr, TNF_INCLUDE)); \ + expr->rhs_pspecs = rhs_pspecs; \ + REDIR(saved_target); \ +} while(0) + +/* Generate ASN.1 type declaration */ +#define GEN_DECLARE(type_name, expr) do { \ + int saved_target = arg->target->target; \ + REDIR(OT_FUNC_DECLS); \ + OUT_NOINDENT("extern asn_TYPE_descriptor_t " \ + "asn_DEF_%s;\n", MKID(expr)); \ + if (expr->_type_referenced) { \ + OUT_NOINDENT("extern asn_%s_specifics_t " \ + "asn_SPC_%s_specs_%d;\n", type_name, \ + MKID(expr), expr->_type_unique_index); \ + if(expr_elements_count(arg, expr)) \ + OUT_NOINDENT("extern asn_TYPE_member_t " \ + "asn_MBR_%s_%d[%d];\n", \ + MKID(expr), expr->_type_unique_index, \ + expr_elements_count(arg, expr)); \ + } \ + REDIR(saved_target); \ +} while(0) + +/* + * Format LONG_MIN according to C90 rules. + */ +#define OINT(iv) \ + do { \ + if(iv == (-2147483647L - 1)) \ + OUT("(-2147483647L - 1)"); \ + else \ + OUT("%s", asn1p_itoa(iv)); \ + } while(0) + +#define OINTS(iv) \ + do { \ + if(iv == (-2147483647L - 1)) \ + OUT("(-2147483647L - 1)"); \ + else \ + OUT("%s%s", (iv >= 0) ? " " : "", asn1p_itoa(iv)); \ + } while(0) + +#endif /* ASN1_COMPILED_OUTPUT_H */