FreeCalypso > hg > ueda-linux
view ueda/sverp/vparse.c @ 115:a7276a03289d
M4 library: new metric QFP footprints
author | Mychaela Falconia <falcon@freecalypso.org> |
---|---|
date | Fri, 12 Jun 2020 07:25:04 +0000 |
parents | 7b4f78fcca08 |
children |
line wrap: on
line source
/* * Parser for the minimal subset of Verilog we grok */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <strings.h> #include "struct.h" #include "lexer.h" extern struct module_def *glob_module_list; extern char *parser_filename; extern FILE *parser_readF; extern int parser_lineno; extern char parser_read_word[]; extern int parser_read_number; extern int pushback_token; static struct module_def *curmod; static struct module_def_subinst *subinst; static struct connect_entry *curconn, **subinst_nextconn; /* internal (to the parser) structure for passing range info */ struct range_info { int range_given; int range_msb; int range_lsb; }; static int capture_range(spec) struct range_info *spec; { register int t; t = get_token(); if (t != '[') { pushback_token = t; bzero(spec, sizeof(struct range_info)); return(0); } spec->range_given = 1; t = get_token(); if (t != NUMBER) parse_error("expected number after '['"); spec->range_msb = parser_read_number; t = get_token(); if (t == ']') { spec->range_lsb = spec->range_msb; return(1); } if (t != ':') parse_error("expected ':' or ']' after MSB number"); t = get_token(); if (t != NUMBER) parse_error("expected number after ':'"); spec->range_lsb = parser_read_number; t = get_token(); if (t != ']') parse_error("expected ']' after LSB number"); return(1); } static void create_module() { register struct module_def *mod, **modp; char *buf; for (modp = &glob_module_list; mod = *modp; modp = &mod->next) { if (!strcmp(mod->name, parser_read_word)) parse_error("duplicate module name"); } buf = malloc(sizeof(struct module_def) + strlen(parser_read_word) + 1); if (!buf) { perror("malloc"); exit(1); } mod = (struct module_def *) buf; bzero(mod, sizeof(struct module_def)); buf += sizeof(struct module_def); strcpy(buf, parser_read_word); mod->name = buf; *modp = mod; curmod = mod; } static void add_port() { register struct module_net_def *n, **np; char *buf; for (np = &curmod->nets; n = *np; np = &n->next) { if (!strcmp(n->name, parser_read_word)) parse_error("duplicate port name in module line"); } buf = malloc(sizeof(struct module_net_def) + strlen(parser_read_word) + 1); if (!buf) { perror("malloc"); exit(1); } n = (struct module_net_def *) buf; bzero(n, sizeof(struct module_net_def)); buf += sizeof(struct module_net_def); strcpy(buf, parser_read_word); n->name = buf; n->is_port = 1; *np = n; curmod->nports++; } static void process_port_list() { register int t; for (;;) { t = get_token(); if (t == ')') return; if (t != WORD) parse_error("expected port name in module line"); add_port(); t = get_token(); if (t == ')') return; if (t != ',') parse_error("expected ',' or ')' in module line"); } } static void netdef_set_businfo(n, range) register struct module_net_def *n; register struct range_info *range; { if (!range->range_given) { n->is_bus = 0; n->bus_width = 1; return; } if (range->range_msb == range->range_lsb) parse_error("bus declaration with MSB==LSB is meaningless"); n->is_bus = 1; n->bus_msb = range->range_msb; n->bus_lsb = range->range_lsb; if (n->bus_msb > n->bus_lsb) n->bus_width = n->bus_msb - n->bus_lsb + 1; else n->bus_width = n->bus_lsb - n->bus_msb + 1; } static void complete_port_def(range) struct range_info *range; { register struct module_net_def *n; for (n = curmod->nets; n; n = n->next) if (!strcmp(n->name, parser_read_word)) break; if (!n) { fprintf(stderr, "%s line %d: no port named %s\n", parser_filename, parser_lineno, parser_read_word); exit(1); } if (n->def_complete) { fprintf(stderr, "%s line %d: inout decl for %s is a redefinition\n", parser_filename, parser_lineno, parser_read_word); exit(1); } netdef_set_businfo(n, range); n->def_complete = 1; } static void add_wire(range) struct range_info *range; { register struct module_net_def *n, **np; char *buf; for (np = &curmod->nets; n = *np; np = &n->next) if (!strcmp(n->name, parser_read_word)) { fprintf(stderr, "%s line %d: wire %s is a redefinition\n", parser_filename, parser_lineno, parser_read_word); exit(1); } buf = malloc(sizeof(struct module_net_def) + strlen(parser_read_word) + 1); if (!buf) { perror("malloc"); exit(1); } n = (struct module_net_def *) buf; bzero(n, sizeof(struct module_net_def)); buf += sizeof(struct module_net_def); strcpy(buf, parser_read_word); n->name = buf; *np = n; netdef_set_businfo(n, range); n->def_complete = 1; } static void process_netdef_line(is_port) { struct range_info range; register int t; capture_range(&range); for (;;) { t = get_token(); if (t == ';') return; if (t != WORD) parse_error("expected ident on wire def line"); if (is_port) complete_port_def(&range); else add_wire(&range); t = get_token(); if (t == ';') return; if (t != ',') parse_error("expected ',' or ';' in wire def line"); } } static void create_subinst(submod_name, inst_name) char *submod_name, *inst_name; { register struct module_def_subinst *s, **sp; char *buf; for (sp = &curmod->subinst; s = *sp; sp = &s->next) if (!strcmp(s->inst_name, inst_name)) parse_error("duplicate subinstance name"); buf = malloc(sizeof(struct module_def_subinst) + strlen(submod_name) + strlen(inst_name) + 2); if (!buf) { perror("malloc"); exit(1); } s = (struct module_def_subinst *) buf; bzero(s, sizeof(struct module_def_subinst)); buf += sizeof(struct module_def_subinst); strcpy(buf, submod_name); s->submod_name = buf; buf += strlen(submod_name) + 1; strcpy(buf, inst_name); s->inst_name = buf; *sp = s; subinst = s; subinst_nextconn = &s->connections; } static void alloc_connect_entry(downport) char *downport; { struct connect_entry *s; int len; char *buf; if (downport) len = sizeof(struct connect_entry) + strlen(downport) + 1; else len = sizeof(struct connect_entry); s = (struct connect_entry *) malloc(len); if (!s) { perror("malloc"); exit(1); } bzero(s, sizeof(struct connect_entry)); if (downport) { buf = (char *)(s + 1); strcpy(buf, downport); s->down_portname = buf; } curconn = s; *subinst_nextconn = s; subinst_nextconn = &s->next; } static void connect_upper() { struct range_info range; register struct module_net_def *n; for (n = curmod->nets; n; n = n->next) if (!strcmp(n->name, parser_read_word)) break; if (!n) { fprintf(stderr, "%s line %d: no net named %s\n", parser_filename, parser_lineno, parser_read_word); exit(1); } curconn->up_netdef = n; if (!capture_range(&range)) { curconn->up_offset = 0; curconn->up_width = n->bus_width; return; } if (!n->is_bus) { fprintf(stderr, "%s line %d: net %s is not a bus, range spec is meaningless\n", parser_filename, parser_lineno, n->name); exit(1); } if (n->bus_msb > n->bus_lsb) { if (range.range_msb < range.range_lsb) { error_reversed: fprintf(stderr, "%s line %d: reversed range on bus %s\n", parser_filename, parser_lineno, n->name); exit(1); } if (range.range_msb > n->bus_msb) { error_outofrange: fprintf(stderr, "%s line %d: subrange of bus %s to connect exceeds full bus range\n", parser_filename, parser_lineno, n->name); exit(1); } if (range.range_lsb < n->bus_lsb) goto error_outofrange; curconn->up_offset = n->bus_msb - range.range_msb; curconn->up_width = range.range_msb - range.range_lsb + 1; } else { if (range.range_msb > range.range_lsb) goto error_reversed; if (range.range_msb < n->bus_msb) goto error_outofrange; if (range.range_lsb > n->bus_lsb) goto error_outofrange; curconn->up_offset = range.range_msb - n->bus_msb; curconn->up_width = range.range_lsb - range.range_msb + 1; } } static void connect_by_name() { struct range_info range; register int t; for (;;) { t = get_token(); if (t == ')') return; if (t != '.') parse_error("expected '.' for downward port"); t = get_token(); if (t != WORD) parse_error("expected downward port name after '.'"); alloc_connect_entry(parser_read_word); curconn->src_lineno = parser_lineno; if (capture_range(&range)) { curconn->down_range_given = 1; curconn->down_range_msb = range.range_msb; curconn->down_range_lsb = range.range_lsb; } t = get_token(); if (t != '(') parse_error("expected '(' for connect-by-name"); t = get_token(); if (t != ')') { if (t != WORD) parse_error( "expected in-module net name to connect"); connect_upper(); t = get_token(); if (t != ')') parse_error("expected ')' for connect-by-name"); } t = get_token(); if (t == ')') return; if (t != ',') parse_error("expected ',' or ')' in connection list"); } } static void connect_by_order() { register int t; for (;;) { t = get_token(); if (t == ')') return; if (t != WORD) parse_error("expected in-module net name to connect"); alloc_connect_entry(0); curconn->src_lineno = parser_lineno; connect_upper(); subinst->connect_by_order++; t = get_token(); if (t == ')') return; if (t != ',') parse_error("expected ',' or ')' in connection list"); } } static void process_subinst() { char submod_name[MAXWORD+1]; register int t; strcpy(submod_name, parser_read_word); t = get_token(); if (t != WORD) parse_error("expected instance name after submodule name"); create_subinst(submod_name, parser_read_word); t = get_token(); if (t != '(') parse_error("expected '(' after <module> <instance>"); t = get_token(); pushback_token = t; if (t == '.') connect_by_name(); else connect_by_order(); t = get_token(); if (t != ';') parse_error("expected ';' at the end of instantiation"); } static void preen_module_nets() { register struct module_net_def *n; register int idx; idx = 0; for (n = curmod->nets; n; n = n->next) { if (!n->def_complete) { fprintf(stderr, "error: definition of port %s in module %s is incomplete\n", n->name, curmod->name); exit(1); } n->array_index = idx; idx += n->bus_width; if (n->is_port) curmod->nwires_ports += n->bus_width; else curmod->nwires_internal += n->bus_width; } } read_verilog_file(filename_arg) char *filename_arg; { register int t; parser_filename = filename_arg; parser_readF = fopen(parser_filename, "r"); if (!parser_readF) { perror(parser_filename); exit(1); } parser_lineno = 1; pushback_token = 0; t = get_token(); if (t != WORD || strcmp(parser_read_word, "module")) parse_error("first token is expected to be \"module\""); t = get_token(); if (t != WORD) parse_error("module name expected after \"module\""); create_module(); t = get_token(); if (t != '(') parse_error("expected '(' after module <modname>"); process_port_list(); t = get_token(); if (t != ';') parse_error("expected ';' at the end of module line"); for (;;) { t = get_token(); if (t != WORD) parse_error( "expected word token at the beginning of module item"); if (!strcmp(parser_read_word, "endmodule")) break; if (!strcmp(parser_read_word, "input") || !strcmp(parser_read_word, "output") || !strcmp(parser_read_word, "inout")) process_netdef_line(1); else if (!strcmp(parser_read_word, "wire")) process_netdef_line(0); else process_subinst(); } fclose(parser_readF); preen_module_nets(); return(0); }