// -*- mode: C++; c-file-style: "cc-mode" -*- //************************************************************************* // DESCRIPTION: Verilator: Large 4-state numbers // // Code available from: https://verilator.org // //************************************************************************* // // Copyright 2003-2020 by Wilson Snyder. This program is free software; you // can redistribute it and/or modify it under the terms of either the GNU // Lesser General Public License Version 3 or the Perl Artistic License // Version 2.0. // SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0 // //************************************************************************* #include "config_build.h" #include "verilatedos.h" #include "V3Global.h" #include "V3Number.h" #include "V3Ast.h" #include #include #include #include #include #define MAX_SPRINTF_DOUBLE_SIZE 100 // Maximum characters with a sprintf %e/%f/%g (probably < 30) // Number operations build output in-place so can't call e.g. foo.opX(foo) #define NUM_ASSERT_OP_ARGS1(arg1) \ UASSERT((this != &(arg1)), \ "Number operation called with same source and dest"); #define NUM_ASSERT_OP_ARGS2(arg1, arg2) \ UASSERT((this != &(arg1) && this != &(arg2)), \ "Number operation called with same source and dest"); #define NUM_ASSERT_OP_ARGS3(arg1, arg2, arg3) \ UASSERT((this != &(arg1) && this != &(arg2) && this != &(arg3)), \ "Number operation called with same source and dest"); #define NUM_ASSERT_LOGIC_ARGS1(arg1) \ UASSERT((!(arg1).isDouble() && !(arg1).isString()), \ "Number operation called with non-logic (double or string) argument: '"<<(arg1)<<'"'); #define NUM_ASSERT_LOGIC_ARGS2(arg1, arg2) \ NUM_ASSERT_LOGIC_ARGS1(arg1); NUM_ASSERT_LOGIC_ARGS1(arg2); #define NUM_ASSERT_STRING_ARGS1(arg1) \ UASSERT((arg1).isString(), \ "Number operation called with non-string argument: '"<<(arg1)<<'"'); #define NUM_ASSERT_STRING_ARGS2(arg1, arg2) \ NUM_ASSERT_STRING_ARGS1(arg1); NUM_ASSERT_STRING_ARGS1(arg2); #define NUM_ASSERT_DOUBLE_ARGS1(arg1) \ UASSERT((arg1).isDouble(), \ "Number operation called with non-double argument: '"<<(arg1)<<'"'); #define NUM_ASSERT_DOUBLE_ARGS2(arg1, arg2) \ NUM_ASSERT_DOUBLE_ARGS1(arg1); NUM_ASSERT_DOUBLE_ARGS1(arg2); //====================================================================== // Errors void V3Number::v3errorEnd(std::ostringstream& str) const { std::ostringstream nsstr; nsstr<v3errorEnd(nsstr); } else { m_fileline->v3errorEnd(nsstr); } } //====================================================================== // Read class functions // CREATION V3Number::V3Number(VerilogStringLiteral, AstNode* nodep, const string& str) { // Create a number using a verilog string as the value, thus 8 bits per character. // cppcheck bug - doesn't see init() resets these // cppcheck: Member variable 'm_sized/m_width' is not initialized in the constructor init(nodep, str.length()*8); m_fromString = true; for (unsigned pos=0; pos v3Global.opt.maxNumWidth()) { // atoi might convert large number to negative, so can't tell which v3error("Unsupported: Width of number exceeds implementation limit: " << sourcep << " (IEEE 1800-2017 6.9.1)"); width(v3Global.opt.maxNumWidth(), true); } else { width(atoi(widthn.c_str()), true); } } } else { unbased = true; base = 'd'; } for (int i = 0; i < words(); ++i) m_value[i] = m_valueX[i] = 0; // Special SystemVerilog unsized constructs if (base == '0') { setBit(0, 0); width(1, false); // So we extend it m_autoExtend = true; } else if (base == '1') { setBit(0, 1); width(1, false); // So we extend it m_autoExtend = true; } else if (tolower(base) == 'z') { setBit(0, 'z'); width(1, false); // So we extend it m_autoExtend = true; } else if (tolower(base) == 'x') { setBit(0, 'x'); width(1, false); // So we extend it m_autoExtend = true; } // Otherwise... else if (!m_sized) { width(32, false); // Says IEEE 1800-2012 5.7.1 if (unbased) isSigned(true); // Also says the spec. } // Ignore leading blanks while (*value_startp=='_' || isspace(*value_startp)) value_startp++; if (!*value_startp && !m_autoExtend) { v3error("Number is missing value digits: "<opAdd(product, addend); if (product.bitsValue(width(), 4)) { // Overflowed static int warned = 0; v3error("Too many digits for "< 1) { v3error("Mixing X/Z/? with digits not legal in decimal constant: "<= value_startp; cp--) { if (*cp!='_' && *cp!='0' && obit>=width()) { v3error("Too many digits for "<fileline(); } //====================================================================== // Global int V3Number::log2b(uint32_t num) { // See also opCLog2 for (int bit=31; bit>0; bit--) if (num & (VL_ULL(1)<>VL_ULL(32)) & VL_ULL(0xffffffff); opCleanThis(); return *this; } V3Number& V3Number::setLong(uint32_t value) { for (int i=0; ifileline(), vformat); } string V3Number::displayed(FileLine* fl, const string& vformat) const { string::const_iterator pos = vformat.begin(); UASSERT(pos != vformat.end() && pos[0]=='%', "$display-like function with non format argument "<<*this); ++pos; bool left = false; if (pos[0] == '-') { left = true; ++pos; } string fmtsize; for (; pos != vformat.end() && (isdigit(pos[0]) || pos[0]=='.'); ++pos) { fmtsize += pos[0]; } string str; char code = tolower(pos[0]); switch (code) { case 'b': { int bit = width()-1; if (fmtsize == "0") while (bit && bitIs0(bit)) bit--; for (; bit>=0; bit--) { if (bitIs0(bit)) str+='0'; else if (bitIs1(bit)) str+='1'; else if (bitIsZ(bit)) str+='z'; else str+='x'; } return str; } case 'o': { int bit = width()-1; if (fmtsize == "0") while (bit && bitIs0(bit)) bit--; while ((bit%3)!=2) bit++; for (; bit>0; bit -= 3) { int v = bitsValue(bit-2, 3); str += static_cast('0' + v); } return str; } case 'h': case 'x': { int bit = width()-1; if (fmtsize == "0") while (bit && bitIs0(bit)) bit--; while ((bit%4)!=3) bit++; for (; bit>0; bit -= 4) { int v = bitsValue(bit-3, 4); if (v>=10) str += static_cast('a' + v - 10); else str += static_cast('0' + v); } return str; } case 'c': { if (width()>8) fl->v3warn(WIDTH, "$display-like format of %c format of > 8 bit value"); unsigned int v = bitsValue(0, 8); char strc[2]; strc[0] = v&0xff; strc[1] = '\0'; str = strc; return str; } case 's': { // Spec says always drop leading zeros, this isn't quite right, we space pad. int bit = this->width()-1; bool start = true; while ((bit%8)!=7) bit++; for (; bit>=0; bit -= 8) { int v = bitsValue(bit-7, 8); if (!start || v) { str += static_cast((v==0) ? ' ' : v); start = false; // Drop leading 0s } else { if (fmtsize != "0") str += ' '; } } size_t fmtsizen = static_cast(atoi(fmtsize.c_str())); str = displayPad(fmtsizen, ' ', left, str); return str; } case '~': // Signed decimal case 't': // Time case 'd': { // Unsigned decimal bool issigned = (code == '~'); if (fmtsize == "") { double mantissabits = this->width() - (issigned?1:0); double maxval = pow(2.0, mantissabits); double dchars = log10(maxval)+1.0; if (issigned) dchars++; // space for sign fmtsize = cvtToStr(int(dchars)); } if (issigned) { if (width() > 64) { str = toDecimalS(); } else { str = cvtToStr(toSQuad()); } } else { if (width() > 64) { str = toDecimalU(); } else { str = cvtToStr(toUQuad()); } } bool zeropad = fmtsize.length()>0 && fmtsize[0]=='0'; // fmtsize might have changed since we parsed the %fmtsize size_t fmtsizen = static_cast(atoi(fmtsize.c_str())); str = displayPad(fmtsizen, (zeropad ? '0' : ' '), left, str); return str; } case 'e': case 'f': case 'g': case '^': { // Realtime char tmp[MAX_SPRINTF_DOUBLE_SIZE]; sprintf(tmp, vformat.c_str(), toDouble()); return tmp; } // 'l' // Library - converted to text by V3LinkResolve // 'p' // Packed - converted to another code by V3Width case 'u': { // Packed 2-state for (int i=0; i((m_value[i] >> 0) & 0xff); str += static_cast((m_value[i] >> 8) & 0xff); str += static_cast((m_value[i] >> 16) & 0xff); str += static_cast((m_value[i] >> 24) & 0xff); } return str; } case 'z': { // Packed 4-state for (int i=0; i((m_value[i] >> 0) & 0xff); str += static_cast((m_value[i] >> 8) & 0xff); str += static_cast((m_value[i] >> 16) & 0xff); str += static_cast((m_value[i] >> 24) & 0xff); str += static_cast((m_valueX[i] >> 0) & 0xff); str += static_cast((m_valueX[i] >> 8) & 0xff); str += static_cast((m_valueX[i] >> 16) & 0xff); str += static_cast((m_valueX[i] >> 24) & 0xff); } return str; } case 'v': { // Strength int bit = width()-1; for (; bit>=0; bit--) { if (bitIs0(bit)) str += "St0 "; // Yes, always a space even for bit 0 else if (bitIs1(bit)) str += "St1 "; else if (bitIsZ(bit)) str += "StZ "; else str += "StX"; } return str; } case '@': { // Packed string size_t fmtsizen = static_cast(atoi(fmtsize.c_str())); str = displayPad(fmtsizen, ' ', left, toString()); return str; } default: fl->v3fatalSrc("Unknown $display-like format code for number: %"<= 0 && bitIs0(from_bit); --from_bit); // Double-dabble algorithm for (; from_bit >= 0; --from_bit) { // Any digits >= 5 need an add 3 (via tmp) for (int nibble_bit = 0; nibble_bit < maxdecwidth; nibble_bit += 4) { if (bcd.bitsValue(nibble_bit, 4) >= 5) { tmp2.setAllBits0(); tmp2.setBit(nibble_bit, 1); // Add 3, decomposed as two bits tmp2.setBit(nibble_bit+1, 1); tmp.opAssign(bcd); bcd.opAdd(tmp, tmp2); } } // Shift; bcd = bcd << 1 tmp.opAssign(bcd); bcd.opShiftL(tmp, V3Number(this, 32, 1)); // bcd[0] = this[from_bit] if (bitIs1(from_bit)) bcd.setBit(0, 1); } string output; int lsb = (maxdecwidth-1) & ~3; for (; lsb>0; lsb-=4) { // Skip leading zeros if (bcd.bitsValue(lsb, 4)) break; } for (; lsb>=0; lsb-=4) { output += ('0' + bcd.bitsValue(lsb, 4)); // 0..9 } return output; } //====================================================================== // ACCESSORS - as numbers uint32_t V3Number::toUInt() const { UASSERT(!isFourState(), "toUInt with 4-state "<<*this); // We allow wide numbers that represent values <= 32 bits for (int i=1; i(extended); } else { // Where we use this (widths, etc) and care about signedness, // we can reasonably assume the MSB isn't set on unsigned numbers. return static_cast(toUInt()); } } vluint64_t V3Number::toUQuad() const { UASSERT(!isFourState(), "toUQuad with 4-state "<<*this); // We allow wide numbers that represent values <= 64 bits if (isDouble()) return static_cast(toDouble()); for (int i=2; i(toUInt())); return ((static_cast(m_value[1])<(m_value[0]))); } vlsint64_t V3Number::toSQuad() const { if (isDouble()) return static_cast(toDouble()); vluint64_t v = toUQuad(); vluint64_t signExtend = (-(v & (VL_ULL(1)<<(width()-1)))); vluint64_t extended = v | signExtend; return static_cast(extended); } string V3Number::toString() const { UASSERT(!isFourState(), "toString with 4-state "<<*this); // Spec says always drop leading zeros, this isn't quite right, we space pad. if (isString()) return m_stringVal; int bit = this->width()-1; bool start = true; while ((bit%8)!=7) bit++; string str; for (; bit>=0; bit -= 8) { int v = bitsValue(bit-7, 8); if (!start || v) { str += static_cast((v==0) ? ' ' : v); start = false; // Drop leading 0s } } return str; } uint32_t V3Number::toHash() const { return m_value[0]; } uint32_t V3Number::edataWord(int eword) const { UASSERT(!isFourState(), "edataWord with 4-state "<<*this); return m_value[eword]; } uint8_t V3Number::dataByte(int byte) const { return (edataWord(byte / (VL_EDATASIZE / 8)) >> ((byte * 8) % VL_EDATASIZE)) & 0xff; } bool V3Number::isEqZero() const { for (int i=0; iwidth(), rhs.width()); bit++) { if (this->bitIs1(bit) && rhs.bitIs0(bit)) { return 1; } if (rhs.bitIs1(bit) && this->bitIs0(bit)) { return 0; } if (this->bitIsXZ(bit)) { return 0; } if (rhs.bitIsXZ(bit)) { return 0; } } return 0; } bool V3Number::isLtXZ(const V3Number& rhs) const { // Include X/Z in comparisons for sort ordering for (int bit=0; bit < std::max(this->width(), rhs.width()); bit++) { if (this->bitIs1(bit) && rhs.bitIs0(bit)) { return 1; } if (rhs.bitIs1(bit) && this->bitIs0(bit)) { return 0; } if (this->bitIsXZ(bit)) { return 1; } if (rhs.bitIsXZ(bit)) { return 0; } } return 0; } int V3Number::widthMin() const { for (int bit=width()-1; bit>0; bit--) { if (!bitIs0(bit)) return bit+1; } return 1; // one bit even if number is == 0 } uint32_t V3Number::countOnes() const { int n = 0; for (int bit=0; bitwidth(); bit++) { if (bitIs1(bit)) n++; } return n; } uint32_t V3Number::mostSetBitP1() const { for (int bit=this->width()-1; bit>=0; bit--) { if (bitIs1(bit)) return bit+1; } return 0; } //====================================================================== V3Number& V3Number::opBitsNonX(const V3Number& lhs) { // 0/1->1, X/Z->0 // op i, L(lhs) bit return NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_LOGIC_ARGS1(lhs); setZero(); for (int bit=0; bitwidth(); bit++) { if (lhs.bitIs0(bit) || lhs.bitIs1(bit)) { setBit(bit, 1); } } return *this; } V3Number& V3Number::opBitsOne(const V3Number& lhs) { // 1->1, 0/X/Z->0 // op i, L(lhs) bit return NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_LOGIC_ARGS1(lhs); setZero(); for (int bit=0; bitwidth(); bit++) { if (lhs.bitIs1(bit)) { setBit(bit, 1); } } return *this; } V3Number& V3Number::opBitsXZ(const V3Number& lhs) { // 0/1->1, X/Z->0 // op i, L(lhs) bit return NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_LOGIC_ARGS1(lhs); setZero(); for (int bit=0; bitwidth(); bit++) { if (lhs.bitIsXZ(bit)) { setBit(bit, 1); } } return *this; } V3Number& V3Number::opBitsZ(const V3Number& lhs) { // 0/1->1, X/Z->0 // op i, L(lhs) bit return NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_LOGIC_ARGS1(lhs); setZero(); for (int bit=0; bitwidth(); bit++) { if (lhs.bitIsZ(bit)) { setBit(bit, 1); } } return *this; } V3Number& V3Number::opBitsNonZ(const V3Number& lhs) { // 0/1->1, X/Z->0 // op i, L(lhs) bit return NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_LOGIC_ARGS1(lhs); setZero(); for (int bit=0; bitwidth(); bit++) { if (!lhs.bitIsZ(bit)) { setBit(bit, 1); } } return *this; } //====================================================================== // Operators - Simple per-bit logical ops V3Number& V3Number::opRedOr(const V3Number& lhs) { // op i, 1 bit return NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_LOGIC_ARGS1(lhs); char outc = 0; for (int bit=0; bit=0; bit--) { if (lhs.bitIs1(bit)) { setLong(bit+adjust); return *this; } } setZero(); return *this; } V3Number& V3Number::opLogNot(const V3Number& lhs) { NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_LOGIC_ARGS1(lhs); // op i, 1 bit return char outc = 1; for (int bit=0; bitwidth(); bit++) { if (lhs.bitIs0(bit)) { setBit(bit, 1); } else if (lhs.bitIsXZ(bit)) { setBit(bit,'x'); } } return *this; } V3Number& V3Number::opAnd(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); // i op j, max(L(lhs),L(rhs)) bit return, careful need to X/Z extend. setZero(); for (int bit=0; bitwidth(); bit++) { if (lhs.bitIs1(bit) && rhs.bitIs1(bit)) { setBit(bit, 1); } else if (lhs.bitIs0(bit) || rhs.bitIs0(bit)) ; // 0 else { setBit(bit,'x'); } } return *this; } V3Number& V3Number::opOr(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); // i op j, max(L(lhs),L(rhs)) bit return, careful need to X/Z extend. setZero(); for (int bit=0; bitwidth(); bit++) { if (lhs.bitIs1(bit) || rhs.bitIs1(bit)) { setBit(bit, 1); } else if (lhs.bitIs0(bit) && rhs.bitIs0(bit)) ; // 0 else { setBit(bit,'x'); } } return *this; } V3Number& V3Number::opChangeXor(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); // 32 bit result opEq(lhs, rhs); return *this; } V3Number& V3Number::opXor(const V3Number& lhs, const V3Number& rhs) { // i op j, max(L(lhs),L(rhs)) bit return, careful need to X/Z extend. NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); setZero(); for (int bit=0; bitwidth(); bit++) { if (lhs.bitIs1(bit) && rhs.bitIs0(bit)) { setBit(bit, 1); } else if (lhs.bitIs0(bit) && rhs.bitIs1(bit)) { setBit(bit, 1); } else if (lhs.bitIsXZ(bit) && rhs.bitIsXZ(bit)) { setBit(bit,'x'); } // else zero } return *this; } V3Number& V3Number::opXnor(const V3Number& lhs, const V3Number& rhs) { // i op j, max(L(lhs),L(rhs)) bit return, careful need to X/Z extend. NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); setZero(); for (int bit=0; bitwidth(); bit++) { if (lhs.bitIs1(bit) && rhs.bitIs1(bit)) { setBit(bit, 1); } else if (lhs.bitIs0(bit) && rhs.bitIs0(bit)) { setBit(bit, 1); } else if (lhs.bitIsXZ(bit) && rhs.bitIsXZ(bit)) { setBit(bit,'x'); } // else zero } return *this; } V3Number& V3Number::opConcat(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); setZero(); // See also error in V3Width if (!lhs.sized() || !rhs.sized()) { v3warn(WIDTHCONCAT, "Unsized numbers/parameters not allowed in concatenations."); } int obit = 0; for (int bit=0; bit 8192) { v3warn(WIDTHCONCAT, "More than a 8k bit replication is probably wrong: "<(lhs.width())); for (int istart=0; istart(v)); } V3Number& V3Number::opPutcN(const V3Number& lhs, const V3Number& rhs, const V3Number& ths) { NUM_ASSERT_OP_ARGS3(lhs, rhs, ths); NUM_ASSERT_STRING_ARGS1(lhs); string lstring = lhs.toString(); const vlsint32_t i = rhs.toSInt(); const vlsint32_t c = ths.toSInt() & 0xFF; // 6.16.2:str.putc(i, c) does not change the value when i < 0 || i >= str.len() || c == 0 // when evaluating the second condition, i must be positive. if (0 <= i && static_cast(i) < lstring.length() && c != 0) lstring[i] = c; return setString(lstring); } V3Number& V3Number::opGetcN(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_STRING_ARGS1(lhs); const string lstring = lhs.toString(); const vlsint32_t i = rhs.toSInt(); vlsint32_t v = 0; // 6.16.3:str.getc(i) returns 0 if i < 0 || i >= str.len() // when evaluating the second condition, i must be positive. if (0 <= i && static_cast(i) < lstring.length()) v = lstring[i]; return setLong(v); } V3Number& V3Number::opSubstrN(const V3Number& lhs, const V3Number& rhs, const V3Number& ths) { NUM_ASSERT_OP_ARGS3(lhs, rhs, ths); NUM_ASSERT_STRING_ARGS1(lhs); const string lstring = lhs.toString(); const vlsint32_t i = rhs.toSInt(); const vlsint32_t j = ths.toSInt(); // 6.16.8:str.substr(i, j) returns an empty string when i < 0 || j < i || j >= str.len() // when evaluating the third condition, j must be positive because 0 <= i <= j is guaranteed by // the former two conditions. if (i < 0 || j < i || static_cast(j) >= lstring.length()) return setString(""); // The second parameter of std::string::substr(i, n) is length, not position as SystemVerilog. return setString(lstring.substr(i, j - i + 1)); } V3Number& V3Number::opCompareNN(const V3Number& lhs, const V3Number& rhs, bool ignoreCase) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_STRING_ARGS2(lhs, rhs); // SystemVerilog Language Standard does not allow a string variable to contain '\0'. // So C functions such as strcmp() can correctly compare strings. int result; string lstring = lhs.toString(); string rstring = rhs.toString(); if (ignoreCase) { result = VL_STRCASECMP(lstring.c_str(), rstring.c_str()); } else { result = std::strcmp(lstring.c_str(), rstring.c_str()); } return setLongS(result); } V3Number& V3Number::opEq(const V3Number& lhs, const V3Number& rhs) { // i op j, 1 bit return, max(L(lhs),L(rhs)) calculation, careful need to X/Z extend. NUM_ASSERT_OP_ARGS2(lhs, rhs); if (lhs.isString()) return opEqN(lhs, rhs); if (lhs.isDouble()) return opEqD(lhs, rhs); char outc = 1; for (int bit=0; bit < std::max(lhs.width(), rhs.width()); bit++) { if (lhs.bitIs1(bit) && rhs.bitIs0(bit)) { outc = 0; goto last; } if (lhs.bitIs0(bit) && rhs.bitIs1(bit)) { outc = 0; goto last; } if (lhs.bitIsXZ(bit)) { outc='x'; } if (rhs.bitIsXZ(bit)) { outc='x'; } } last: return setSingleBits(outc); } V3Number& V3Number::opNeq(const V3Number& lhs, const V3Number& rhs) { // i op j, 1 bit return, max(L(lhs),L(rhs)) calculation, careful need to X/Z extend. NUM_ASSERT_OP_ARGS2(lhs, rhs); if (lhs.isString()) return opNeqN(lhs, rhs); if (lhs.isDouble()) return opNeqD(lhs, rhs); char outc = 0; for (int bit=0; bit < std::max(lhs.width(), rhs.width()); bit++) { if (lhs.bitIs1(bit) && rhs.bitIs0(bit)) { outc = 1; goto last; } if (lhs.bitIs0(bit) && rhs.bitIs1(bit)) { outc = 1; goto last; } if (lhs.bitIsXZ(bit)) { outc='x'; } if (rhs.bitIsXZ(bit)) { outc='x'; } } last: return setSingleBits(outc); } bool V3Number::isCaseEq(const V3Number& rhs) const { // i op j, 1 bit return, max(L(lhs),L(rhs)) calculation, careful need to X/Z extend. if (isString()) return toString() == rhs.toString(); if (isDouble()) return toDouble() == rhs.toDouble(); if (this->width() != rhs.width()) return false; for (int bit=0; bit < std::max(this->width(), rhs.width()); bit++) { if (this->bitIs(bit) != rhs.bitIs(bit)) { return false; } } return true; } V3Number& V3Number::opCaseEq(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); return setSingleBits(lhs.isCaseEq(rhs) ? 1:0); } V3Number& V3Number::opCaseNeq(const V3Number& lhs, const V3Number& rhs) { // i op j, 1 bit return, max(L(lhs),L(rhs)) calculation, careful need to X/Z extend. NUM_ASSERT_OP_ARGS2(lhs, rhs); char outc = 0; if (lhs.isString()) return opNeqN(lhs, rhs); else if (lhs.isDouble()) return opNeqD(lhs, rhs); for (int bit=0; bit < std::max(lhs.width(), rhs.width()); bit++) { if (lhs.bitIs(bit) != rhs.bitIs(bit)) { outc=1; goto last; } } last: return setSingleBits(outc); } V3Number& V3Number::opWildEq(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); char outc = 1; for (int bit=0; bit < std::max(lhs.width(), rhs.width()); bit++) { if (!rhs.bitIsXZ(bit) && lhs.bitIs(bit) != rhs.bitIs(bit)) { outc = 0; goto last; } if (lhs.bitIsXZ(bit)) outc='x'; } last: return setSingleBits(outc); } V3Number& V3Number::opWildNeq(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); char outc = 0; for (int bit=0; bit < std::max(lhs.width(), rhs.width()); bit++) { if (!rhs.bitIsXZ(bit) && lhs.bitIs(bit) != rhs.bitIs(bit)) { outc=1; goto last; } if (lhs.bitIsXZ(bit)) outc='x'; } last: return setSingleBits(outc); } V3Number& V3Number::opGt(const V3Number& lhs, const V3Number& rhs) { // i op j, 1 bit return, max(L(lhs),L(rhs)) calculation, careful need to X/Z extend. NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); char outc = 0; for (int bit=0; bit < std::max(lhs.width(), rhs.width()); bit++) { if (lhs.bitIs1(bit) && rhs.bitIs0(bit)) { outc=1; } if (rhs.bitIs1(bit) && lhs.bitIs0(bit)) { outc = 0; } if (lhs.bitIsXZ(bit)) { outc='x'; } if (rhs.bitIsXZ(bit)) { outc='x'; } } return setSingleBits(outc); } V3Number& V3Number::opGtS(const V3Number& lhs, const V3Number& rhs) { // i op j, 1 bit return, max(L(lhs),L(rhs)) calculation, careful need to X/Z extend. NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); char outc = 0; { int mbit = std::max(lhs.width()-1, rhs.width()-1); if (lhs.bitIsXZ(mbit)) { outc='x'; } else if (rhs.bitIsXZ(mbit)) { outc='x'; } else if (lhs.bitIs0(mbit) && rhs.bitIs1Extend(mbit)) { outc = 1; } // + > - else if (lhs.bitIs1Extend(mbit) && rhs.bitIs0(mbit)) { outc = 0; } // - !> + else { // both positive or negative, normal > for (int bit=0; bit < std::max(lhs.width()-1, rhs.width()-1); bit++) { if (lhs.bitIs1Extend(bit) && rhs.bitIs0(bit)) { outc = 1; } if (rhs.bitIs1Extend(bit) && lhs.bitIs0(bit)) { outc = 0; } if (lhs.bitIsXZ(bit)) { outc = 'x'; } if (rhs.bitIsXZ(bit)) { outc = 'x'; } } } } return setSingleBits(outc); } V3Number& V3Number::opGte(const V3Number& lhs, const V3Number& rhs) { // i op j, 1 bit return, max(L(lhs),L(rhs)) calculation, careful need to X/Z extend. NUM_ASSERT_OP_ARGS2(lhs, rhs); V3Number& eq = opEq(lhs, rhs); if (eq.isNeqZero()) return eq; // Return true return opGt(lhs, rhs); } V3Number& V3Number::opGteS(const V3Number& lhs, const V3Number& rhs) { // i op j, 1 bit return, max(L(lhs),L(rhs)) calculation, careful need to X/Z extend. NUM_ASSERT_OP_ARGS2(lhs, rhs); V3Number& eq = opEq(lhs, rhs); if (eq.isNeqZero()) return eq; // Return true return opGtS(lhs, rhs); } V3Number& V3Number::opLt(const V3Number& lhs, const V3Number& rhs) { return opGt(rhs, lhs); } V3Number& V3Number::opLte(const V3Number& lhs, const V3Number& rhs) { return opGte(rhs, lhs); } V3Number& V3Number::opLtS(const V3Number& lhs, const V3Number& rhs) { return opGtS(rhs, lhs); } V3Number& V3Number::opLteS(const V3Number& lhs, const V3Number& rhs) { return opGteS(rhs, lhs); } V3Number& V3Number::opRotR(const V3Number& lhs, const V3Number& rhs) { // L(lhs) bit return NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); if (rhs.isFourState()) return setAllBitsX(); setZero(); uint32_t rhsval = rhs.toUInt(); for (int bit=0; bitwidth(); bit++) { setBit(bit, lhs.bitIs((bit + rhsval) % this->width())); } return *this; } V3Number& V3Number::opRotL(const V3Number& lhs, const V3Number& rhs) { // L(lhs) bit return NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); if (rhs.isFourState()) return setAllBitsX(); setZero(); uint32_t rhsval = rhs.toUInt(); for (int bit=0; bitwidth(); bit++) { if (bit >= static_cast(rhsval)) { setBit(bit, lhs.bitIs((bit - rhsval) % this->width())); } } return *this; } V3Number& V3Number::opShiftR(const V3Number& lhs, const V3Number& rhs) { // L(lhs) bit return NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); if (rhs.isFourState()) return setAllBitsX(); setZero(); for (int bit=32; bit(lhs.width())) { for (int bit=0; bitwidth(); bit++) { setBit(bit, lhs.bitIs(bit + rhsval)); } } return *this; } V3Number& V3Number::opShiftRS(const V3Number& lhs, const V3Number& rhs, uint32_t lbits) { // L(lhs) bit return // The spec says a unsigned >>> still acts as a normal >>. // We presume it is signed; as that's V3Width's job to convert to opShiftR NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); if (rhs.isFourState()) return setAllBitsX(); setZero(); for (int bit=32; bitwidth(); sbit++) { setBit(sbit, lhs.bitIs(lbits - 1)); // 0/1/X/Z } if (rhs.bitIs1(lbits-1)) setAllBits1(); // -1 else 0 return *this; // shift of over 2^32 must be -1/0 } uint32_t rhsval = rhs.toUInt(); if (rhsval < static_cast(lhs.width())) { for (int bit=0; bitwidth(); bit++) { setBit(bit, lhs.bitIsExtend(bit + rhsval, lbits)); } } else { for (int bit=0; bitwidth(); bit++) { setBit(bit, lhs.bitIs(lbits-1)); // 0/1/X/Z } } return *this; } V3Number& V3Number::opShiftL(const V3Number& lhs, const V3Number& rhs) { // L(lhs) bit return NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); if (rhs.isFourState()) return setAllBitsX(); setZero(); for (int bit=32; bitwidth(); bit++) { if (bit >= static_cast(rhsval)) { setBit(bit, lhs.bitIs(bit - rhsval)); } } return *this; } //====================================================================== // Ops - Arithmetic V3Number& V3Number::opAbsS(const V3Number& lhs) { // op i, L(lhs) bit return NUM_ASSERT_OP_ARGS1(lhs); if (lhs.isFourState()) return setAllBitsX(); if (lhs.isNegative()) { return opNegate(lhs); } else { return opAssign(lhs); } } V3Number& V3Number::opNegate(const V3Number& lhs) { // op i, L(lhs) bit return NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_LOGIC_ARGS1(lhs); if (lhs.isFourState()) return setAllBitsX(); V3Number notlhs (&lhs, width()); notlhs.opNot(lhs); V3Number one (&lhs, width(), 1); opAdd(notlhs, one); return *this; } V3Number& V3Number::opAdd(const V3Number& lhs, const V3Number& rhs) { // i op j, max(L(lhs),L(rhs)) bit return, if any 4-state, 4-state return NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); if (lhs.isFourState() || rhs.isFourState()) return setAllBitsX(); setZero(); // Addem int carry=0; for (int bit=0; bitwidth(); bit++) { int sum = ((lhs.bitIs1(bit)?1:0) + (rhs.bitIs1(bit)?1:0) + carry); if (sum & 1) { setBit(bit, 1); } carry = (sum >= 2); } return *this; } V3Number& V3Number::opSub(const V3Number& lhs, const V3Number& rhs) { // i op j, max(L(lhs),L(rhs)) bit return, if any 4-state, 4-state return NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); if (lhs.isFourState() || rhs.isFourState()) return setAllBitsX(); V3Number negrhs (&rhs, rhs.width()); negrhs.opNegate(rhs); return opAdd(lhs, negrhs); } V3Number& V3Number::opMul(const V3Number& lhs, const V3Number& rhs) { // i op j, max(L(lhs),L(rhs)) bit return, if any 4-state, 4-state return NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); if (lhs.isFourState() || rhs.isFourState()) return setAllBitsX(); setZero(); if (width() <= 64) { setQuad(lhs.toUQuad() * rhs.toUQuad()); opCleanThis(); // Mult produces extra bits in result } else { for (int lword=0; lword(lhs.m_value[lword]) * static_cast(rhs.m_value[rword]); for (int qword=lword+rword; qwordwords(); qword++) { mul += static_cast(m_value[qword]); m_value[qword] = (mul & VL_ULL(0xffffffff)); mul = (mul >> VL_ULL(32)) & VL_ULL(0xffffffff); } } } opCleanThis(); // Mult produces extra bits in result } return *this; } V3Number& V3Number::opMulS(const V3Number& lhs, const V3Number& rhs) { // Signed multiply NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); if (lhs.isFourState() || rhs.isFourState()) return setAllBitsX(); V3Number lhsNoSign = lhs; if (lhs.isNegative()) lhsNoSign.opNegate(lhs); V3Number rhsNoSign = rhs; if (rhs.isNegative()) rhsNoSign.opNegate(rhs); V3Number qNoSign = opMul(lhsNoSign, rhsNoSign); if ((lhs.isNegative() && !rhs.isNegative()) || (!lhs.isNegative() && rhs.isNegative())) { opNegate(qNoSign); } else { opAssign(qNoSign); } return *this; } V3Number& V3Number::opDiv(const V3Number& lhs, const V3Number& rhs) { // i op j, max(L(lhs),L(rhs)) bit return, if any 4-state, 4-state return NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_LOGIC_ARGS2(lhs, rhs); //UINFO(9, "opdiv "<>divs-start "<divs-mid "<= 0; j--) { vluint64_t unw64 = ((k<(lhs.m_value[j])); m_value[j] = unw64 / static_cast(rhs.m_value[0]); k = unw64 - (static_cast(m_value[j]) * static_cast(rhs.m_value[0])); } UINFO(9, " opmoddiv-1w "<> 32 won't mask the value for (int i = vw-1; i>0; i--) { vn[i] = (rhs.m_value[i] << s) | (shift_mask & (rhs.m_value[i-1] >> (32-s))); } vn[0] = rhs.m_value[0] << s; // Copy and shift dividend by same amount; may set new upper word if (s) un[uw] = lhs.m_value[uw-1] >> (32-s); else un[uw] = 0; for (int i=uw-1; i>0; i--) { un[i] = (lhs.m_value[i] << s) | (shift_mask & (lhs.m_value[i-1] >> (32-s))); } un[0] = lhs.m_value[0] << s; //printf(" un="); for (int i=5; i>=0; i--) printf(" %08x",un[i]); printf("\n"); //printf(" vn="); for (int i=5; i>=0; i--) printf(" %08x",vn[i]); printf("\n"); //printf(" mv="); for (int i=5; i>=0; i--) printf(" %08x",m_value[i]); printf("\n"); // Main loop for (int j = uw - vw; j >= 0; j--) { // Estimate vluint64_t unw64 = (static_cast(un[j+vw])<(un[j+vw-1])); vluint64_t qhat = unw64 / static_cast(vn[vw-1]); vluint64_t rhat = unw64 - qhat*static_cast(vn[vw-1]); again: if (qhat >= VL_ULL(0x100000000) || ((qhat*vn[vw-2]) > ((rhat<> VL_ULL(32)) - (t >> VL_ULL(32)); } t = un[j+vw] - k; un[j+vw] = t; this->m_value[j] = qhat; // Save quotient digit if (t < 0) { // Over subtracted; correct by adding back this->m_value[j]--; k = 0; for (int i=0; i(un[i+j]) + static_cast(vn[i]) + k; un[i+j] = t; k = t >> VL_ULL(32); } un[j+vw] = un[j+vw] + k; } } //printf(" un="); for (int i=5; i>=0; i--) printf(" %08x",un[i]); printf("\n"); //printf(" vn="); for (int i=5; i>=0; i--) printf(" %08x",vn[i]); printf("\n"); //printf(" mv="); for (int i=5; i>=0; i--) printf(" %08x",m_value[i]); printf("\n"); if (is_modulus) { // modulus // Need to reverse normalization on copy to output for (int i=0; i> s) | (shift_mask & (un[i+1] << (32-s))); } for (int i=vw; iwidth(); bit++) { setBit(bit, ignoreXZ ? lhs.bitIs1(bit) : lhs.bitIs(bit)); } } } return *this; } V3Number& V3Number::opExtendS(const V3Number& lhs, uint32_t lbits) { // Note may be a width change during the sign extension NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_LOGIC_ARGS1(lhs); setZero(); for (int bit=0; bit < width(); bit++) { char extendWith = lhs.bitIsExtend(bit, lbits); setBit(bit, extendWith); } return *this; } V3Number& V3Number::opExtendXZ(const V3Number& lhs, uint32_t lbits) { // Note may be a width change during the X/Z extension NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_LOGIC_ARGS1(lhs); setZero(); for (int bit=0; bit < width(); bit++) { char extendWith = lhs.bitIsExtend(bit, lbits); if (extendWith == '1' || extendWith == 1) extendWith = 0; setBit(bit, lhs.bitIsExtend(bit, lbits)); } return *this; } V3Number& V3Number::opClean(const V3Number& lhs, uint32_t bits) { return opSel(lhs, bits-1, 0); } void V3Number::opCleanThis(bool warnOnTruncation) { // Clean MSB of number NUM_ASSERT_LOGIC_ARGS1(*this); uint32_t newValueMsb = m_value[words()-1] & hiWordMask(); uint32_t newValueXMsb = m_valueX[words()-1] & hiWordMask(); if (warnOnTruncation && (newValueMsb != m_value[words()-1] || newValueXMsb != m_valueX[words()-1])) { // Displaying in decimal avoids hiWordMask truncation v3warn(WIDTH, "Value too large for "<width(); bit++) { if (ibit>=0 && ibit(msbval)) { setBit(bit, lhs.bitIs(ibit)); } else { setBit(bit, 'x'); } ++ibit; } //UINFO(0,"RANGE "<width(); bit++) { if (if0s.bitIs1(bit) && if1s.bitIs1(bit)) { setBit(bit, 1); } else if (if0s.bitIs0(bit) && if1s.bitIs0(bit)) { setBit(bit, 0); } else setBit(bit,'x'); } } return *this; } //====================================================================== // Ops - Floating point V3Number& V3Number::opIToRD(const V3Number& lhs) { NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_LOGIC_ARGS1(lhs); // IEEE says we ignore x/z in real conversions V3Number noxz(lhs); noxz.opAssignNonXZ(lhs); return setDouble(noxz.toSInt()); } V3Number& V3Number::opRToIS(const V3Number& lhs) { NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_DOUBLE_ARGS1(lhs); double v = VL_TRUNC(lhs.toDouble()); vlsint32_t i = static_cast(v); // C converts from double to vlsint32 return setLongS(i); } V3Number& V3Number::opRToIRoundS(const V3Number& lhs) { NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_DOUBLE_ARGS1(lhs); double v = VL_ROUND(lhs.toDouble()); setZero(); union { double d; vluint64_t q; } u; u.d = v; if (u.d == 0.0) {} int exp = static_cast((u.q >> VL_ULL(52)) & VL_MASK_Q(11)) - 1023; int lsb = exp - 52; vluint64_t mantissa = (u.q & VL_MASK_Q(52)) | (VL_ULL(1) << 52); if (v != 0) { // IEEE format: [63]=sign [62:52]=exp+1023 [51:0]=mantissa // This does not need to support subnormals as they are sub-integral for (int bit = 0; bit <= 52; ++bit) { if (mantissa & (VL_ULL(1) << bit)) { int outbit = bit + lsb; if (outbit >= 0) setBit(outbit, 1); } } if (v < 0) { V3Number noSign = *this; opNegate(noSign); } } return *this; } V3Number& V3Number::opRealToBits(const V3Number& lhs) { NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_DOUBLE_ARGS1(lhs); // Conveniently our internal format is identical so we can copy bits... if (lhs.width()!=64 || this->width()!=64) { v3fatalSrc("Real operation on wrong sized number"); } opAssign(lhs); m_double = false; return *this; } V3Number& V3Number::opBitsToRealD(const V3Number& lhs) { NUM_ASSERT_OP_ARGS1(lhs); // Conveniently our internal format is identical so we can copy bits... if (lhs.width()!=64 || this->width()!=64) { v3fatalSrc("Real operation on wrong sized number"); } opAssign(lhs); m_double = true; return *this; } V3Number& V3Number::opNegateD(const V3Number& lhs) { NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_DOUBLE_ARGS1(lhs); return setDouble(- lhs.toDouble()); } V3Number& V3Number::opAddD(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_DOUBLE_ARGS2(lhs, rhs); return setDouble(lhs.toDouble() + rhs.toDouble()); } V3Number& V3Number::opSubD(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_DOUBLE_ARGS2(lhs, rhs); return setDouble(lhs.toDouble() - rhs.toDouble()); } V3Number& V3Number::opMulD(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_DOUBLE_ARGS2(lhs, rhs); return setDouble(lhs.toDouble() * rhs.toDouble()); } V3Number& V3Number::opDivD(const V3Number& lhs, const V3Number& rhs) { // On exceptions, we just generate 'inf' through floating point // IEEE says it's implementation defined what happens NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_DOUBLE_ARGS2(lhs, rhs); return setDouble(lhs.toDouble() / rhs.toDouble()); } V3Number& V3Number::opPowD(const V3Number& lhs, const V3Number& rhs) { // On exceptions, we just generate 'inf' through floating point // IEEE says it's implementation defined what happens NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_DOUBLE_ARGS2(lhs, rhs); return setDouble(pow(lhs.toDouble(), rhs.toDouble())); } V3Number& V3Number::opEqD(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_DOUBLE_ARGS2(lhs, rhs); return setSingleBits(lhs.toDouble() == rhs.toDouble()); } V3Number& V3Number::opNeqD(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_DOUBLE_ARGS2(lhs, rhs); return setSingleBits(lhs.toDouble() != rhs.toDouble()); } V3Number& V3Number::opGtD(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_DOUBLE_ARGS2(lhs, rhs); return setSingleBits(lhs.toDouble() > rhs.toDouble()); } V3Number& V3Number::opGteD(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_DOUBLE_ARGS2(lhs, rhs); return setSingleBits(lhs.toDouble() >= rhs.toDouble()); } V3Number& V3Number::opLtD(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_DOUBLE_ARGS2(lhs, rhs); return setSingleBits(lhs.toDouble() < rhs.toDouble()); } V3Number& V3Number::opLteD(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_DOUBLE_ARGS2(lhs, rhs); return setSingleBits(lhs.toDouble() <= rhs.toDouble()); } //====================================================================== // Ops - String V3Number& V3Number::opConcatN(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_STRING_ARGS2(lhs, rhs); return setString(lhs.toString() + rhs.toString()); } V3Number& V3Number::opReplN(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_STRING_ARGS1(lhs); NUM_ASSERT_LOGIC_ARGS1(rhs); return opReplN(lhs, rhs.toUInt()); } V3Number& V3Number::opReplN(const V3Number& lhs, uint32_t rhsval) { NUM_ASSERT_OP_ARGS1(lhs); NUM_ASSERT_STRING_ARGS1(lhs); string out; out.reserve(lhs.toString().length() * rhsval); for (unsigned times=0; times rhs.toString()); } V3Number& V3Number::opGteN(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_STRING_ARGS2(lhs, rhs); return setSingleBits(lhs.toString() >= rhs.toString()); } V3Number& V3Number::opLtN(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_STRING_ARGS2(lhs, rhs); return setSingleBits(lhs.toString() < rhs.toString()); } V3Number& V3Number::opLteN(const V3Number& lhs, const V3Number& rhs) { NUM_ASSERT_OP_ARGS2(lhs, rhs); NUM_ASSERT_STRING_ARGS2(lhs, rhs); return setSingleBits(lhs.toString() <= rhs.toString()); }