|  | /* | 
|  | *  yosys -- Yosys Open SYnthesis Suite | 
|  | * | 
|  | *  Copyright (C) 2012  Claire Xenia Wolf <claire@yosyshq.com> | 
|  | * | 
|  | *  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. | 
|  | * | 
|  | *  --- | 
|  | * | 
|  | *  The Verilog frontend. | 
|  | * | 
|  | *  This frontend is using the AST frontend library (see frontends/ast/). | 
|  | *  Thus this frontend does not generate RTLIL code directly but creates an | 
|  | *  AST directly from the Verilog parse tree and then passes this AST to | 
|  | *  the AST frontend library. | 
|  | * | 
|  | *  --- | 
|  | * | 
|  | *  This file contains an ad-hoc parser for Verilog constants. The Verilog | 
|  | *  lexer does only recognize a constant but does not actually split it to its | 
|  | *  components. I.e. it just passes the Verilog code for the constant to the | 
|  | *  bison parser. The parser then uses the function const2ast() from this file | 
|  | *  to create an AST node for the constant. | 
|  | * | 
|  | *  --- | 
|  | * | 
|  | *  The file has been adapted for use in Yosys SystemVerilog Plugin. | 
|  | * | 
|  | */ | 
|  |  | 
|  | #include "const2ast.h" | 
|  | #include "frontends/ast/ast.h" | 
|  | #include "kernel/log.h" | 
|  |  | 
|  | #include <string> | 
|  | #include <cmath> | 
|  | #include <vector> | 
|  |  | 
|  | using namespace Yosys; | 
|  | using namespace Yosys::AST; | 
|  |  | 
|  | // divide an arbitrary length decimal number by two and return the rest | 
|  | static int my_decimal_div_by_two(std::vector<uint8_t> &digits) | 
|  | { | 
|  | int carry = 0; | 
|  | for (size_t i = 0; i < digits.size(); i++) { | 
|  | if (digits[i] >= 10) | 
|  | log_file_error(current_filename, get_line_num(), "Invalid use of [a-fxz?] in decimal constant.\n"); | 
|  | digits[i] += carry * 10; | 
|  | carry = digits[i] % 2; | 
|  | digits[i] /= 2; | 
|  | } | 
|  | while (!digits.empty() && !digits.front()) | 
|  | digits.erase(digits.begin()); | 
|  | return carry; | 
|  | } | 
|  |  | 
|  | // find the number of significant bits in a binary number (not including the sign bit) | 
|  | static int my_ilog2(int x) | 
|  | { | 
|  | int ret = 0; | 
|  | while (x != 0 && x != -1) { | 
|  | x = x >> 1; | 
|  | ret++; | 
|  | } | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | // parse a binary, decimal, hexadecimal or octal number with support for special bits ('x', 'z' and '?') | 
|  | static void my_strtobin(std::vector<RTLIL::State> &data, const char *str, int len_in_bits, int base, char case_type, bool is_unsized) | 
|  | { | 
|  | // all digits in string (MSB at index 0) | 
|  | std::vector<uint8_t> digits; | 
|  |  | 
|  | while (*str) { | 
|  | if ('0' <= *str && *str <= '9') | 
|  | digits.push_back(*str - '0'); | 
|  | else if ('a' <= *str && *str <= 'f') | 
|  | digits.push_back(10 + *str - 'a'); | 
|  | else if ('A' <= *str && *str <= 'F') | 
|  | digits.push_back(10 + *str - 'A'); | 
|  | else if (*str == 'x' || *str == 'X') | 
|  | digits.push_back(0xf0); | 
|  | else if (*str == 'z' || *str == 'Z' || *str == '?') | 
|  | digits.push_back(0xf1); | 
|  | str++; | 
|  | } | 
|  |  | 
|  | if (base == 10 && GetSize(digits) == 1 && digits.front() >= 0xf0) | 
|  | base = 2; | 
|  |  | 
|  | data.clear(); | 
|  |  | 
|  | if (base == 10) { | 
|  | while (!digits.empty()) | 
|  | data.push_back(my_decimal_div_by_two(digits) ? State::S1 : State::S0); | 
|  | } else { | 
|  | int bits_per_digit = my_ilog2(base-1); | 
|  | for (auto it = digits.rbegin(), e = digits.rend(); it != e; it++) { | 
|  | if (*it > (base-1) && *it < 0xf0) | 
|  | log_file_error(current_filename, get_line_num(), "Digit larger than %d used in in base-%d constant.\n", | 
|  | base-1, base); | 
|  | for (int i = 0; i < bits_per_digit; i++) { | 
|  | int bitmask = 1 << i; | 
|  | if (*it == 0xf0) | 
|  | data.push_back(case_type == 'x' ? RTLIL::Sa : RTLIL::Sx); | 
|  | else if (*it == 0xf1) | 
|  | data.push_back(case_type == 'x' || case_type == 'z' ? RTLIL::Sa : RTLIL::Sz); | 
|  | else | 
|  | data.push_back((*it & bitmask) ? State::S1 : State::S0); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | int len = GetSize(data); | 
|  | RTLIL::State msb = data.empty() ? State::S0 : data.back(); | 
|  |  | 
|  | if (len_in_bits < 0) { | 
|  | if (len < 32) | 
|  | data.resize(32, msb == State::S0 || msb == State::S1 ? RTLIL::S0 : msb); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (is_unsized && (len > len_in_bits)) | 
|  | log_file_error(current_filename, get_line_num(), "Unsized constant must have width of 1 bit, but have %d bits!\n", len); | 
|  |  | 
|  | for (len = len - 1; len >= 0; len--) | 
|  | if (data[len] == State::S1) | 
|  | break; | 
|  | if (msb == State::S0 || msb == State::S1) { | 
|  | len += 1; | 
|  | data.resize(len_in_bits, State::S0); | 
|  | } else { | 
|  | len += 2; | 
|  | data.resize(len_in_bits, msb); | 
|  | } | 
|  |  | 
|  | if (len_in_bits == 0) | 
|  | log_file_error(current_filename, get_line_num(), "Illegal integer constant size of zero (IEEE 1800-2012, 5.7).\n"); | 
|  |  | 
|  | if (len > len_in_bits) | 
|  | log_warning("Literal has a width of %d bit, but value requires %d bit. (%s:%d)\n", | 
|  | len_in_bits, len, current_filename.c_str(), get_line_num()); | 
|  | } | 
|  |  | 
|  | // convert the Verilog code for a constant to an AST node | 
|  | AstNode *systemverilog_plugin::const2ast(std::string code, char case_type, bool warn_z) | 
|  | { | 
|  | if (warn_z) { | 
|  | AstNode *ret = const2ast(code, case_type); | 
|  | if (ret != nullptr && std::find(ret->bits.begin(), ret->bits.end(), RTLIL::State::Sz) != ret->bits.end()) | 
|  | log_warning("Yosys has only limited support for tri-state logic at the moment. (%s:%d)\n", | 
|  | current_filename.c_str(), get_line_num()); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | const char *str = code.c_str(); | 
|  |  | 
|  | // Strings | 
|  | if (*str == '"') { | 
|  | int len = strlen(str) - 2; | 
|  | std::vector<RTLIL::State> data; | 
|  | data.reserve(len * 8); | 
|  | for (int i = 0; i < len; i++) { | 
|  | unsigned char ch = str[len - i]; | 
|  | for (int j = 0; j < 8; j++) { | 
|  | data.push_back((ch & 1) ? State::S1 : State::S0); | 
|  | ch = ch >> 1; | 
|  | } | 
|  | } | 
|  | AstNode *ast = AstNode::mkconst_bits(data, false); | 
|  | ast->str = code; | 
|  | return ast; | 
|  | } | 
|  |  | 
|  | for (size_t i = 0; i < code.size(); i++) | 
|  | if (code[i] == '_' || code[i] == ' ' || code[i] == '\t' || code[i] == '\r' || code[i] == '\n') | 
|  | code.erase(code.begin()+(i--)); | 
|  | str = code.c_str(); | 
|  |  | 
|  | char *endptr; | 
|  | long len_in_bits = strtol(str, &endptr, 10); | 
|  |  | 
|  | // Simple base-10 integer | 
|  | if (*endptr == 0) { | 
|  | std::vector<RTLIL::State> data; | 
|  | my_strtobin(data, str, -1, 10, case_type, false); | 
|  | if (data.back() == State::S1) | 
|  | data.push_back(State::S0); | 
|  | return AstNode::mkconst_bits(data, true); | 
|  | } | 
|  |  | 
|  | // unsized constant | 
|  | if (str == endptr) | 
|  | len_in_bits = -1; | 
|  |  | 
|  | // The "<bits>'[sS]?[bodhBODH]<digits>" syntax | 
|  | if (*endptr == '\'') | 
|  | { | 
|  | std::vector<RTLIL::State> data; | 
|  | bool is_signed = false; | 
|  | bool is_unsized = len_in_bits < 0; | 
|  | if (*(endptr+1) == 's' || *(endptr+1) == 'S') { | 
|  | is_signed = true; | 
|  | endptr++; | 
|  | } | 
|  | switch (*(endptr+1)) | 
|  | { | 
|  | case 'b': | 
|  | case 'B': | 
|  | my_strtobin(data, endptr+2, len_in_bits, 2, case_type, is_unsized); | 
|  | break; | 
|  | case 'o': | 
|  | case 'O': | 
|  | my_strtobin(data, endptr+2, len_in_bits, 8, case_type, is_unsized); | 
|  | break; | 
|  | case 'd': | 
|  | case 'D': | 
|  | my_strtobin(data, endptr+2, len_in_bits, 10, case_type, is_unsized); | 
|  | break; | 
|  | case 'h': | 
|  | case 'H': | 
|  | my_strtobin(data, endptr+2, len_in_bits, 16, case_type, is_unsized); | 
|  | break; | 
|  | default: | 
|  | char next_char = char(tolower(*(endptr+1))); | 
|  | if (next_char == '0' || next_char == '1' || next_char == 'x' || next_char == 'z') { | 
|  | is_unsized = true; | 
|  | my_strtobin(data, endptr+1, 1, 2, case_type, is_unsized); | 
|  | } else { | 
|  | return NULL; | 
|  | } | 
|  | } | 
|  | if (len_in_bits < 0) { | 
|  | if (is_signed && data.back() == State::S1) | 
|  | data.push_back(State::S0); | 
|  | } | 
|  | return AstNode::mkconst_bits(data, is_signed, is_unsized); | 
|  | } | 
|  |  | 
|  | return NULL; | 
|  | } |