/* * yosys -- Yosys Open SYnthesis Suite * * Copyright (C) 2012 Claire Xenia Wolf * * Permission to use, copy, modify, and/or distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. * */ #include "libparse.h" #include #include #include #include #include #include #ifdef FILTERLIB #undef log_assert void log_assert(bool cond) { if (!cond) fprintf(stderr, "Unspecified assertion failed\n"); } void warn(std::string str) { std::cerr << str; } #else #include "kernel/log.h" void warn(std::string str) { Yosys::log_formatted_warning("", str); } #endif using namespace Yosys; #ifndef FILTERLIB LibertyAstCache LibertyAstCache::instance; std::shared_ptr LibertyAstCache::cached_ast(const std::string &fname) { auto it = cached.find(fname); if (it == cached.end()) return nullptr; if (verbose) log("Using cached data for liberty file `%s'\n", fname); return it->second; } void LibertyAstCache::parsed_ast(const std::string &fname, const std::shared_ptr &ast) { auto it = cache_path.find(fname); bool should_cache = it == cache_path.end() ? cache_by_default : it->second; if (!should_cache) return; if (verbose) log("Caching data for liberty file `%s'\n", fname); cached.emplace(fname, ast); } #endif bool LibertyInputStream::extend_buffer_once() { if (eof) return false; // To support unget we leave the last already read character in the buffer if (buf_pos > 1) { size_t move_pos = buf_pos - 1; memmove(buffer.data(), buffer.data() + move_pos, buf_end - move_pos); buf_pos -= move_pos; buf_end -= move_pos; } const size_t chunk_size = 4096; if (buffer.size() < buf_end + chunk_size) { buffer.resize(buf_end + chunk_size); } size_t read_size = f.rdbuf()->sgetn((char *)buffer.data() + buf_end, chunk_size); buf_end += read_size; if (read_size < chunk_size) eof = true; return read_size != 0; } bool LibertyInputStream::extend_buffer_at_least(size_t size) { while (buffered_size() < size) { if (!extend_buffer_once()) return false; } return true; } int LibertyInputStream::get_cold() { if (buf_pos == buf_end) { if (!extend_buffer_at_least()) return EOF; } int c = buffer[buf_pos]; buf_pos += 1; return c; } int LibertyInputStream::peek_cold(size_t offset) { if (buf_pos + offset >= buf_end) { if (!extend_buffer_at_least(offset + 1)) return EOF; } return buffer[buf_pos + offset]; } LibertyAst::~LibertyAst() { for (auto child : children) delete child; children.clear(); } const LibertyAst *LibertyAst::find(std::string name) const { for (auto child : children) if (child->id == name) return child; return NULL; } void LibertyAst::dump(FILE *f, sieve &blacklist, sieve &whitelist, std::string indent, std::string path, bool path_ok) const { if (whitelist.count(path + "/*") > 0) path_ok = true; path += "/" + id; if (blacklist.count(id) > 0 || blacklist.count(path) > 0) return; if (whitelist.size() > 0 && whitelist.count(id) == 0 && whitelist.count(path) == 0 && !path_ok) { fprintf(stderr, "Automatically added to blacklist: %s\n", path.c_str()); blacklist.insert(id); return; } fprintf(f, "%s%s", indent.c_str(), id.c_str()); if (!args.empty() || !children.empty()) { fprintf(f, "("); for (size_t i = 0; i < args.size(); i++) fprintf(f, "%s%s", i > 0 ? ", " : "", args[i].c_str()); fprintf(f, ")"); } if (!value.empty()) fprintf(f, " : %s", value.c_str()); if (!children.empty()) { fprintf(f, " {\n"); for (size_t i = 0; i < children.size(); i++) children[i]->dump(f, blacklist, whitelist, indent + " ", path, path_ok); fprintf(f, "%s}\n", indent.c_str()); } else fprintf(f, " ;\n"); } // binary operators excluding ' ' bool LibertyExpression::char_is_nice_binop(char c) { return c == '*' || c == '&' || c == '^' || c == '+' || c == '|'; } bool LibertyExpression::is_binop() { return kind == AND || kind == OR || kind == XOR; } // https://matklad.github.io/2020/04/13/simple-but-powerful-pratt-parsing.html LibertyExpression LibertyExpression::parse(Lexer &s, int min_prio) { if (s.empty()) return LibertyExpression{}; char c = s.peek(); auto lhs = LibertyExpression{}; while (isspace(c)) { if (s.empty()) return lhs; s.next(); c = s.peek(); } if (isalpha(c)) { // pin lhs.kind = Kind::PIN; lhs.name = s.pin(); } else if (c == '(') { // parens s.next(); lhs = parse(s); if (s.peek() != ')') { std::stringstream ss; ss << "expected ')' instead of " << s.peek() << " while parsing Liberty expression '" << s.full_expr() << "'\n"; warn(ss.str()); return lhs; } s.next(); } else if (c == '!') { // prefix NOT s.next(); lhs.kind = Kind::NOT; lhs.children.push_back(parse(s, 7)); } else { std::stringstream ss; ss << "unrecognised character " << c << " while parsing Liberty expression " << s.full_expr() << "\n"; warn(ss.str()); return lhs; } while (true) { if (s.empty()) break; c = s.peek(); if (c == '\'') { // postfix NOT if (min_prio > 7) break; s.next(); auto n = LibertyExpression{}; n.kind = Kind::NOT; n.children.push_back(lhs); lhs = std::move(n); continue; } else if (c == '^') { // infix XOR if (min_prio > 5) break; s.next(); auto rhs = parse(s, 6); auto n = LibertyExpression{}; n.kind = Kind::XOR; n.children.push_back(lhs); n.children.push_back(rhs); lhs = std::move(n); continue; } else if (c == '&' || c == '*' || isspace(c)) { // infix AND if (min_prio > 3) break; if (isspace(c)) { // Rewind past this space and any further spaces while (isspace(c)) { if (s.empty()) return lhs; s.next(); c = s.peek(); } if (char_is_nice_binop(c)) { // We found a real binop, so this space wasn't an AND // and we just discard it as meaningless whitespace continue; } } else { // Rewind past this op s.next(); } auto rhs = parse(s, 4); if (rhs.kind == EMPTY) continue; auto n = LibertyExpression{}; n.kind = Kind::AND; n.children.push_back(lhs); n.children.push_back(rhs); lhs = std::move(n); continue; } else if (c == '+' || c == '|') { // infix OR if (min_prio > 1) break; s.next(); auto rhs = parse(s, 2); auto n = LibertyExpression{}; n.kind = Kind::OR; n.children.push_back(lhs); n.children.push_back(rhs); lhs = std::move(n); continue; } break; } return lhs; } void LibertyExpression::get_pin_names(std::unordered_set& names) { if (kind == Kind::PIN) { names.insert(name); } else { for (auto& child : children) child.get_pin_names(names); } } bool LibertyExpression::eval(std::unordered_map& values) { bool result = false; switch (kind) { case Kind::AND: result = true; for (auto& child : children) result &= child.eval(values); return result; case Kind::OR: result = false; for (auto& child : children) result |= child.eval(values); return result; case Kind::NOT: log_assert(children.size() == 1); return !children[0].eval(values); case Kind::XOR: result = false; for (auto& child : children) result ^= child.eval(values); return result; case Kind::PIN: return values.at(name); case Kind::EMPTY: log_assert(false); } return false; } std::string LibertyExpression::sexpr_str(int indent) { std::string prefix; switch (kind) { case AND: prefix += "(and "; break; case OR: prefix += "(or "; break; case NOT: prefix += "(not "; break; case XOR: prefix += "(xor "; break; case PIN: prefix += "(pin \"" + name + "\""; break; case EMPTY: prefix += "("; break; default: log_assert(false); } size_t add_indent = prefix.length(); bool first = true; for (auto child : children) { if (!first) { prefix += "\n" + std::string(indent + add_indent, ' '); } prefix += child.sexpr_str(indent + add_indent); first = false; } prefix += ")"; return prefix; } std::string LibertyExpression::vlog_str() { std::string prefix; if (kind != PIN) prefix += "("; if (is_binop()) { log_assert(children.size() == 2); prefix += children[0].vlog_str(); switch (kind) { case AND: prefix += "&"; break; case OR: prefix += "|"; break; case XOR: prefix += "^"; break; default: log_assert(false); } prefix += children[1].vlog_str(); } else { switch (kind) { case NOT: log_assert(children.size() == 1); prefix += "~"; prefix += children[0].vlog_str(); break; case PIN: prefix += name; break; default: log_assert(false); } } if (kind != PIN) prefix += ")"; return prefix; } int LibertyParser::lexer_inner(std::string &str) { int c; // eat whitespace do { c = f.get(); } while (c == ' ' || c == '\t' || c == '\r'); // search for identifiers, numbers, plus or minus. if (('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z') || ('0' <= c && c <= '9') || c == '_' || c == '-' || c == '+' || c == '.') { f.unget(); size_t i = 1; while (true) { c = f.peek(i); if (('a' <= c && c <= 'z') || ('A' <= c && c <= 'Z') || ('0' <= c && c <= '9') || c == '_' || c == '-' || c == '+' || c == '.') i += 1; else break; } str.clear(); str.append(f.buffered_data(), f.buffered_data() + i); f.consume(i); if (str == "+" || str == "-") { /* Single operator is not an identifier */ return str[0]; } else { return 'v'; } } // if it wasn't an identifer, number of array range, // maybe it's a string? if (c == '"') { f.consume(1); size_t i = 0; while (true) { c = f.peek(i); line += (c == '\n'); if (c != '"' && c != EOF) i += 1; else break; } str.clear(); f.unget(); str.append(f.buffered_data(), f.buffered_data() + i + 1); // Usage in filterlib is expected to retain quotes // but yosys expects to get unquoted #ifdef FILTERLIB str = "\"" + str + "\""; #endif f.consume(i + 2); return 'v'; } // if it wasn't a string, perhaps it's a comment or a forward slash? if (c == '/') { c = f.get(); if (c == '*') { // start of '/*' block comment int last_c = 0; while (c > 0 && (last_c != '*' || c != '/')) { last_c = c; c = f.get(); if (c == '\n') line++; } return lexer(str); } else if (c == '/') { // start of '//' line comment while (c > 0 && c != '\n') c = f.get(); line++; return lexer(str); } f.unget(); return '/'; // a single '/' charater. } // check for a backslash if (c == '\\') { c = f.get(); if (c == '\r') c = f.get(); if (c == '\n') { line++; return lexer(str); } f.unget(); return '\\'; } // check for a new line if (c == '\n') { line++; return 'n'; } // anything else, such as ';' will get passed // through as literal items. return c; } int LibertyParser::lexer(std::string &str) { int ret = lexer_inner(str); // if (ret >= 32 && ret < 255) { // fprintf(stdout, "LEX: ret >>%c<<\n", ret); // } else if (ret == 'v') { // fprintf(stdout, "LEX: ret v str %s\n", str.c_str()); // } else { // fprintf(stdout, "LEX: ret %d\n", ret); // } return ret; } void LibertyParser::report_unexpected_token(int tok) { std::string eReport; switch(tok) { case 'n': error("Unexpected newline."); break; case '[': case ']': case '}': case '{': case '\"': case ':': eReport = "Unexpected '"; eReport += static_cast(tok); eReport += "'."; error(eReport); break; case EOF: error("Unexpected end of file"); break; default: eReport = "Unexpected token: "; eReport += static_cast(tok); error(eReport); } } // FIXME: the AST needs to be extended to store // these vector ranges. void LibertyParser::parse_vector_range(int tok) { // parse vector range [A] or [A:B] std::string arg; tok = lexer(arg); if (tok != 'v') { // expected a vector array index error("Expected a number."); } else { // fixme: check for number A } tok = lexer(arg); // optionally check for : in case of [A:B] // if it isn't we just expect ']' // as we have [A] if (tok == ':') { tok = lexer(arg); if (tok != 'v') { // expected a vector array index error("Expected a number."); } else { // fixme: check for number B tok = lexer(arg); } } // expect a closing bracket of array range if (tok != ']') { error("Expected ']' on array range."); } } // Consume into out_str any string-ish tokens, seperated with spaces // to cope with abuse of the underdefined spec by real world PDKs // enabled by proprietary implementations. // Sorry. int LibertyParser::consume_wrecked_str(int tok, std::string& out_str) { std::string str = ""; while (tok != ';' && tok != EOF && tok != 'n') { out_str += " "; if (tok == 'v') out_str += str; else out_str += tok; tok = lexer(str); } if (tok == EOF) error("wrecked string EOF"); return tok; } LibertyAst *LibertyParser::parse(bool top_level) { std::string str; int tok = lexer(str); // there are liberty files in the wild that // have superfluous ';' at the end of // a { ... }. We simply ignore a ';' here. // and get to the next statement. while ((tok == 'n') || (tok == ';')) tok = lexer(str); if (tok == EOF) { if (top_level) return NULL; report_unexpected_token(tok); } if (tok == '}') return NULL; if (tok != 'v') { report_unexpected_token(tok); } LibertyAst *ast = new LibertyAst; ast->id = str; while (1) { tok = lexer(str); // allow both ';' and new lines to // terminate a statement. if ((tok == ';') || (tok == 'n')) break; if (tok == ':' && ast->value.empty()) { tok = lexer(ast->value); if (tok == 'v') { tok = lexer(str); if (tok == '[') { parse_vector_range(tok); tok = lexer(str); } else { // Hack for when an expression string is unquoted // std::cout << "consume_wrecked_str from :\n"; // std::cout << " weh " << str << "\n"; tok = consume_wrecked_str(tok, ast->value); } } else if (tok == '(') { // Hack for when an expression string is unquoted and starts with // parentheses // tok = ''; // ast->value = "("; // std::cout << "consume_wrecked_str from (\n"; tok = consume_wrecked_str(tok, ast->value); } while (tok == '+' || tok == '-' || tok == '*' || tok == '/' || tok == '!') { ast->value += tok; tok = lexer(str); if (tok != 'v') error("one"); ast->value += str; tok = lexer(str); } // In a liberty file, all key : value pairs should end in ';' // However, there are some liberty files in the wild that // just have a newline. We'll be kind and accept a newline // instead of the ';' too.. if ((tok == ';') || (tok == 'n')) break; else error("two"); continue; } if (tok == '(') { while (1) { std::string arg; tok = lexer(arg); if (tok == ',') continue; if (tok == ')') break; if (tok == '[') { parse_vector_range(tok); continue; } if (tok == 'n') continue; if (tok != 'v') { report_unexpected_token(tok); } ast->args.push_back(arg); } continue; } if (tok == '{') { bool terminated = false; while (1) { LibertyAst *child = parse(false); if (child == NULL) { terminated = true; break; } ast->children.push_back(child); } if (!terminated) { report_unexpected_token(EOF); } break; } report_unexpected_token(tok); } return ast; } #ifndef FILTERLIB void LibertyParser::error() const { log_error("Syntax error in liberty file on line %d.\n", line); } void LibertyParser::error(const std::string &str) const { std::stringstream ss; ss << "Syntax error in liberty file on line " << line << ".\n"; ss << " " << str << "\n"; log_error("%s", ss.str()); } #else YS_ATTRIBUTE(weak) void LibertyParser::error() const { fprintf(stderr, "Syntax error in liberty file on line %d.\n", line); exit(1); } YS_ATTRIBUTE(weak) void LibertyParser::error(const std::string &str) const { std::stringstream ss; ss << "Syntax error in liberty file on line " << line << ".\n"; ss << " " << str << "\n"; printf("%s", ss.str().c_str()); exit(1); } /**** BEGIN: http://svn.clairexen.net/tools/trunk/examples/check.h ****/ #define CHECK_NV(result, check) \ do { \ auto _R = (result); \ if (!(_R check)) { \ fprintf(stderr, "Error from '%s' (%ld %s) in %s:%d.\n", \ #result, (long int)_R, #check, __FILE__, __LINE__); \ abort(); \ } \ } while(0) #define CHECK_COND(result) \ do { \ if (!(result)) { \ fprintf(stderr, "Error from '%s' in %s:%d.\n", \ #result, __FILE__, __LINE__); \ abort(); \ } \ } while(0) /**** END: http://svn.clairexen.net/tools/trunk/examples/check.h ****/ const LibertyAst *find_non_null(const LibertyAst *node, const char *name) { const LibertyAst *ret = node->find(name); if (ret == NULL) fprintf(stderr, "Error: expected to find `%s' node.\n", name); return ret; } std::string func2vl(std::string str) { auto helper = LibertyExpression::Lexer(str); return LibertyExpression::parse(helper).vlog_str(); } void event2vl(const LibertyAst *ast, std::string &edge, std::string &expr) { edge.clear(); expr.clear(); if (ast != NULL) { expr = func2vl(ast->value); if (expr.size() > 0 && expr[0] == '~') edge = "negedge " + expr.substr(1); else edge = "posedge " + expr; } } void clear_preset_var(std::string var, std::string type) { if (type.find('L') != std::string::npos) { printf(" %s <= 0;\n", var.c_str()); return; } if (type.find('H') != std::string::npos) { printf(" %s <= 1;\n", var.c_str()); return; } if (type.find('T') != std::string::npos) { printf(" %s <= ~%s;\n", var.c_str(), var.c_str()); return; } if (type.find('X') != std::string::npos) { printf(" %s <= 'bx;\n", var.c_str()); return; } } void gen_verilogsim_cell(const LibertyAst *ast) { if (ast->find("statetable") != NULL) return; CHECK_NV(ast->args.size(), == 1); printf("module %s (", ast->args[0].c_str()); bool first = true; for (auto child : ast->children) { if (child->id != "pin") continue; CHECK_NV(child->args.size(), == 1); printf("%s%s", first ? "" : ", ", child->args[0].c_str()); first = false; } printf(");\n"); for (auto child : ast->children) { if (child->id != "ff" && child->id != "latch") continue; printf(" reg "); first = true; for (auto arg : child->args) { printf("%s%s", first ? "" : ", ", arg.c_str()); first = false; } printf(";\n"); } for (auto child : ast->children) { if (child->id != "pin") continue; CHECK_NV(child->args.size(), == 1); const LibertyAst *dir = find_non_null(child, "direction"); const LibertyAst *func = child->find("function"); printf(" %s %s;\n", dir->value.c_str(), child->args[0].c_str()); if (func != NULL) printf(" assign %s = %s; // %s\n", child->args[0].c_str(), func2vl(func->value).c_str(), func->value.c_str()); } for (auto child : ast->children) { if (child->id != "ff" || child->args.size() != 2) continue; std::string iq_var = child->args[0]; std::string iqn_var = child->args[1]; std::string clock_edge, clock_expr; event2vl(child->find("clocked_on"), clock_edge, clock_expr); std::string clear_edge, clear_expr; event2vl(child->find("clear"), clear_edge, clear_expr); std::string preset_edge, preset_expr; event2vl(child->find("preset"), preset_edge, preset_expr); std::string edge = ""; if (!clock_edge.empty()) edge += (edge.empty() ? "" : ", ") + clock_edge; if (!clear_edge.empty()) edge += (edge.empty() ? "" : ", ") + clear_edge; if (!preset_edge.empty()) edge += (edge.empty() ? "" : ", ") + preset_edge; if (edge.empty()) continue; printf(" always @(%s) begin\n", edge.c_str()); const char *else_prefix = ""; if (!clear_expr.empty() && !preset_expr.empty()) { printf(" %sif ((%s) && (%s)) begin\n", else_prefix, clear_expr.c_str(), preset_expr.c_str()); clear_preset_var(iq_var, find_non_null(child, "clear_preset_var1")->value); clear_preset_var(iqn_var, find_non_null(child, "clear_preset_var2")->value); printf(" end\n"); else_prefix = "else "; } if (!clear_expr.empty()) { printf(" %sif (%s) begin\n", else_prefix, clear_expr.c_str()); printf(" %s <= 0;\n", iq_var.c_str()); printf(" %s <= 1;\n", iqn_var.c_str()); printf(" end\n"); else_prefix = "else "; } if (!preset_expr.empty()) { printf(" %sif (%s) begin\n", else_prefix, preset_expr.c_str()); printf(" %s <= 1;\n", iq_var.c_str()); printf(" %s <= 0;\n", iqn_var.c_str()); printf(" end\n"); else_prefix = "else "; } if (*else_prefix) printf(" %sbegin\n", else_prefix); std::string expr = find_non_null(child, "next_state")->value; printf(" // %s\n", expr.c_str()); printf(" %s <= %s;\n", iq_var.c_str(), func2vl(expr).c_str()); printf(" %s <= ~(%s);\n", iqn_var.c_str(), func2vl(expr).c_str()); if (*else_prefix) printf(" end\n"); printf(" end\n"); } for (auto child : ast->children) { if (child->id != "latch" || child->args.size() != 2) continue; std::string iq_var = child->args[0]; std::string iqn_var = child->args[1]; std::string enable_edge, enable_expr; event2vl(child->find("enable"), enable_edge, enable_expr); std::string clear_edge, clear_expr; event2vl(child->find("clear"), clear_edge, clear_expr); std::string preset_edge, preset_expr; event2vl(child->find("preset"), preset_edge, preset_expr); printf(" always @* begin\n"); const char *else_prefix = ""; if (!clear_expr.empty() && !preset_expr.empty()) { printf(" %sif ((%s) && (%s)) begin\n", else_prefix, clear_expr.c_str(), preset_expr.c_str()); clear_preset_var(iq_var, find_non_null(child, "clear_preset_var1")->value); clear_preset_var(iqn_var, find_non_null(child, "clear_preset_var2")->value); printf(" end\n"); else_prefix = "else "; } if (!clear_expr.empty()) { printf(" %sif (%s) begin\n", else_prefix, clear_expr.c_str()); printf(" %s <= 0;\n", iq_var.c_str()); printf(" %s <= 1;\n", iqn_var.c_str()); printf(" end\n"); else_prefix = "else "; } if (!preset_expr.empty()) { printf(" %sif (%s) begin\n", else_prefix, preset_expr.c_str()); printf(" %s <= 1;\n", iq_var.c_str()); printf(" %s <= 0;\n", iqn_var.c_str()); printf(" end\n"); else_prefix = "else "; } if (!enable_expr.empty()) { printf(" %sif (%s) begin\n", else_prefix, enable_expr.c_str()); std::string expr = find_non_null(child, "data_in")->value; printf(" %s <= %s;\n", iq_var.c_str(), func2vl(expr).c_str()); printf(" %s <= ~(%s);\n", iqn_var.c_str(), func2vl(expr).c_str()); printf(" end\n"); else_prefix = "else "; } printf(" end\n"); } printf("endmodule\n"); } void gen_verilogsim(const LibertyAst *ast) { CHECK_COND(ast->id == "library"); for (auto child : ast->children) if (child->id == "cell" && !child->find("dont_use")) gen_verilogsim_cell(child); } void usage() { fprintf(stderr, "Usage: filterlib [rules-file [liberty-file]]\n"); fprintf(stderr, " or: filterlib -verilogsim [liberty-file]\n"); exit(1); } int main(int argc, char **argv) { bool flag_verilogsim = false; std::set whitelist, blacklist; if (argc > 3) usage(); if (argc > 1) { if (!strcmp(argv[1], "-verilogsim")) flag_verilogsim = true; if (!strcmp(argv[1], "-") || !strcmp(argv[1], "-verilogsim")) { whitelist.insert("/library"); whitelist.insert("/library/cell"); whitelist.insert("/library/cell/area"); whitelist.insert("/library/cell/cell_footprint"); whitelist.insert("/library/cell/dont_touch"); whitelist.insert("/library/cell/dont_use"); whitelist.insert("/library/cell/ff"); whitelist.insert("/library/cell/ff/*"); whitelist.insert("/library/cell/latch"); whitelist.insert("/library/cell/latch/*"); whitelist.insert("/library/cell/pin"); whitelist.insert("/library/cell/pin/clock"); whitelist.insert("/library/cell/pin/direction"); whitelist.insert("/library/cell/pin/driver_type"); whitelist.insert("/library/cell/pin/function"); whitelist.insert("/library/cell/pin_opposite"); whitelist.insert("/library/cell/pin/state_function"); whitelist.insert("/library/cell/pin/three_state"); whitelist.insert("/library/cell/statetable"); whitelist.insert("/library/cell/statetable/*"); } else { FILE *f = fopen(argv[1], "r"); if (f == NULL) { fprintf(stderr, "Can't open rules file `%s'.\n", argv[1]); usage(); } char buffer[1024]; while (fgets(buffer, 1024, f) != NULL) { char mode = 0; std::string id; for (char *p = buffer; *p; p++) { if (*p == '-' || *p == '+') { if (mode != 0) goto syntax_error; mode = *p; continue; } if (*p == ' ' || *p == '\t' || *p == '\r' || *p == '\n' || *p == '#') { if (!id.empty()) { if (mode == '-') blacklist.insert(id); else if (mode == '+') whitelist.insert(id); else goto syntax_error; } id.clear(); if (*p == '#') break; continue; } id += *p; continue; syntax_error: fprintf(stderr, "Syntax error in rules file:\n%s", buffer); exit(1); } } } } std::istream *f = &std::cin; if (argc == 3) { std::ifstream *ff = new std::ifstream; ff->open(argv[2]); if (ff->fail()) { delete ff; fprintf(stderr, "Can't open liberty file `%s'.\n", argv[2]); usage(); } f = ff; } LibertyParser parser(*f); if (parser.ast) { if (flag_verilogsim) gen_verilogsim(parser.ast); else parser.ast->dump(stdout, blacklist, whitelist); } if (argc == 3) delete f; return 0; } #endif