mirror of
https://github.com/YosysHQ/yosys
synced 2025-04-05 17:14:08 +00:00
Add "nowrshmsk" attribute, fix shift-and-mask bit slice write for signed offset, fixes #1990
Signed-off-by: Claire Wolf <claire@symbioticeda.com>
This commit is contained in:
parent
ca3fc3c882
commit
bbbce0d1c5
|
@ -281,6 +281,9 @@ Verilog Attributes and non-standard features
|
||||||
temporary variable within an always block. This is mostly used internally
|
temporary variable within an always block. This is mostly used internally
|
||||||
by Yosys to synthesize Verilog functions and access arrays.
|
by Yosys to synthesize Verilog functions and access arrays.
|
||||||
|
|
||||||
|
- The ``nowrshmsk`` attribute on a register prohibits the generation of
|
||||||
|
shift-and-mask type circuits for writing to bit slices of that register.
|
||||||
|
|
||||||
- The ``onehot`` attribute on wires mark them as one-hot state register. This
|
- The ``onehot`` attribute on wires mark them as one-hot state register. This
|
||||||
is used for example for memory port sharing and set by the fsm_map pass.
|
is used for example for memory port sharing and set by the fsm_map pass.
|
||||||
|
|
||||||
|
|
|
@ -110,6 +110,8 @@ std::string AST::type2str(AstNodeType type)
|
||||||
X(AST_SHIFT_RIGHT)
|
X(AST_SHIFT_RIGHT)
|
||||||
X(AST_SHIFT_SLEFT)
|
X(AST_SHIFT_SLEFT)
|
||||||
X(AST_SHIFT_SRIGHT)
|
X(AST_SHIFT_SRIGHT)
|
||||||
|
X(AST_SHIFTX)
|
||||||
|
X(AST_SHIFT)
|
||||||
X(AST_LT)
|
X(AST_LT)
|
||||||
X(AST_LE)
|
X(AST_LE)
|
||||||
X(AST_EQ)
|
X(AST_EQ)
|
||||||
|
@ -628,6 +630,8 @@ void AstNode::dumpVlog(FILE *f, std::string indent) const
|
||||||
if (0) { case AST_SHIFT_RIGHT: txt = ">>"; }
|
if (0) { case AST_SHIFT_RIGHT: txt = ">>"; }
|
||||||
if (0) { case AST_SHIFT_SLEFT: txt = "<<<"; }
|
if (0) { case AST_SHIFT_SLEFT: txt = "<<<"; }
|
||||||
if (0) { case AST_SHIFT_SRIGHT: txt = ">>>"; }
|
if (0) { case AST_SHIFT_SRIGHT: txt = ">>>"; }
|
||||||
|
if (0) { case AST_SHIFTX: txt = "@shiftx@"; }
|
||||||
|
if (0) { case AST_SHIFT: txt = "@shift@"; }
|
||||||
if (0) { case AST_LT: txt = "<"; }
|
if (0) { case AST_LT: txt = "<"; }
|
||||||
if (0) { case AST_LE: txt = "<="; }
|
if (0) { case AST_LE: txt = "<="; }
|
||||||
if (0) { case AST_EQ: txt = "=="; }
|
if (0) { case AST_EQ: txt = "=="; }
|
||||||
|
|
|
@ -91,6 +91,8 @@ namespace AST
|
||||||
AST_SHIFT_RIGHT,
|
AST_SHIFT_RIGHT,
|
||||||
AST_SHIFT_SLEFT,
|
AST_SHIFT_SLEFT,
|
||||||
AST_SHIFT_SRIGHT,
|
AST_SHIFT_SRIGHT,
|
||||||
|
AST_SHIFTX,
|
||||||
|
AST_SHIFT,
|
||||||
AST_LT,
|
AST_LT,
|
||||||
AST_LE,
|
AST_LE,
|
||||||
AST_EQ,
|
AST_EQ,
|
||||||
|
|
|
@ -856,6 +856,8 @@ void AstNode::detectSignWidthWorker(int &width_hint, bool &sign_hint, bool *foun
|
||||||
case AST_SHIFT_RIGHT:
|
case AST_SHIFT_RIGHT:
|
||||||
case AST_SHIFT_SLEFT:
|
case AST_SHIFT_SLEFT:
|
||||||
case AST_SHIFT_SRIGHT:
|
case AST_SHIFT_SRIGHT:
|
||||||
|
case AST_SHIFTX:
|
||||||
|
case AST_SHIFT:
|
||||||
case AST_POW:
|
case AST_POW:
|
||||||
children[0]->detectSignWidthWorker(width_hint, sign_hint, found_real);
|
children[0]->detectSignWidthWorker(width_hint, sign_hint, found_real);
|
||||||
break;
|
break;
|
||||||
|
@ -1356,6 +1358,8 @@ RTLIL::SigSpec AstNode::genRTLIL(int width_hint, bool sign_hint)
|
||||||
if (0) { case AST_SHIFT_RIGHT: type_name = ID($shr); }
|
if (0) { case AST_SHIFT_RIGHT: type_name = ID($shr); }
|
||||||
if (0) { case AST_SHIFT_SLEFT: type_name = ID($sshl); }
|
if (0) { case AST_SHIFT_SLEFT: type_name = ID($sshl); }
|
||||||
if (0) { case AST_SHIFT_SRIGHT: type_name = ID($sshr); }
|
if (0) { case AST_SHIFT_SRIGHT: type_name = ID($sshr); }
|
||||||
|
if (0) { case AST_SHIFTX: type_name = ID($shiftx); }
|
||||||
|
if (0) { case AST_SHIFT: type_name = ID($shift); }
|
||||||
{
|
{
|
||||||
if (width_hint < 0)
|
if (width_hint < 0)
|
||||||
detectSignWidth(width_hint, sign_hint);
|
detectSignWidth(width_hint, sign_hint);
|
||||||
|
|
|
@ -1786,7 +1786,18 @@ bool AstNode::simplify(bool const_fold, bool at_zero, bool in_lvalue, int stage,
|
||||||
result_width = abs(int(left_at_zero_ast->integer - right_at_zero_ast->integer)) + 1;
|
result_width = abs(int(left_at_zero_ast->integer - right_at_zero_ast->integer)) + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (0)
|
bool use_case_method = false;
|
||||||
|
|
||||||
|
if (children[0]->id2ast->attributes.count(ID::nowrshmsk)) {
|
||||||
|
AstNode *node = children[0]->id2ast->attributes.at(ID::nowrshmsk);
|
||||||
|
while (node->simplify(true, false, false, stage, -1, false, false)) { }
|
||||||
|
if (node->type != AST_CONSTANT)
|
||||||
|
log_file_error(filename, location.first_line, "Non-constant value for `nowrshmsk' attribute on `%s'!\n", children[0]->id2ast->str.c_str());
|
||||||
|
if (node->asAttrConst().as_bool())
|
||||||
|
use_case_method = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (use_case_method)
|
||||||
{
|
{
|
||||||
// big case block
|
// big case block
|
||||||
|
|
||||||
|
@ -1794,10 +1805,10 @@ bool AstNode::simplify(bool const_fold, bool at_zero, bool in_lvalue, int stage,
|
||||||
newNode = new AstNode(AST_CASE, shift_expr);
|
newNode = new AstNode(AST_CASE, shift_expr);
|
||||||
for (int i = 0; i < source_width; i++) {
|
for (int i = 0; i < source_width; i++) {
|
||||||
int start_bit = children[0]->id2ast->range_right + i;
|
int start_bit = children[0]->id2ast->range_right + i;
|
||||||
|
int end_bit = std::min(start_bit+result_width,source_width) - 1;
|
||||||
AstNode *cond = new AstNode(AST_COND, mkconst_int(start_bit, true));
|
AstNode *cond = new AstNode(AST_COND, mkconst_int(start_bit, true));
|
||||||
AstNode *lvalue = children[0]->clone();
|
AstNode *lvalue = children[0]->clone();
|
||||||
lvalue->delete_children();
|
lvalue->delete_children();
|
||||||
int end_bit = std::min(start_bit+result_width,source_width) - 1;
|
|
||||||
lvalue->children.push_back(new AstNode(AST_RANGE,
|
lvalue->children.push_back(new AstNode(AST_RANGE,
|
||||||
mkconst_int(end_bit, true), mkconst_int(start_bit, true)));
|
mkconst_int(end_bit, true), mkconst_int(start_bit, true)));
|
||||||
cond->children.push_back(new AstNode(AST_BLOCK, new AstNode(type, lvalue, children[1]->clone())));
|
cond->children.push_back(new AstNode(AST_BLOCK, new AstNode(type, lvalue, children[1]->clone())));
|
||||||
|
@ -1844,11 +1855,36 @@ bool AstNode::simplify(bool const_fold, bool at_zero, bool in_lvalue, int stage,
|
||||||
|
|
||||||
AstNode *shamt = shift_expr;
|
AstNode *shamt = shift_expr;
|
||||||
|
|
||||||
newNode->children.push_back(new AstNode(AST_ASSIGN_EQ, ref_mask->clone(),
|
int start_bit = children[0]->id2ast->range_right;
|
||||||
new AstNode(AST_SHIFT_LEFT, mkconst_bits(std::vector<RTLIL::State>(result_width, State::S1), false), shamt->clone())));
|
bool use_shift = shamt->is_signed;
|
||||||
newNode->children.push_back(new AstNode(AST_ASSIGN_EQ, ref_data->clone(),
|
|
||||||
new AstNode(AST_SHIFT_LEFT, new AstNode(AST_BIT_AND, mkconst_bits(std::vector<RTLIL::State>(result_width, State::S1), false), children[1]->clone()), shamt)));
|
if (start_bit != 0) {
|
||||||
newNode->children.push_back(new AstNode(type, lvalue, new AstNode(AST_BIT_OR, new AstNode(AST_BIT_AND, old_data, new AstNode(AST_BIT_NOT, ref_mask)), ref_data)));
|
shamt = new AstNode(AST_SUB, shamt, mkconst_int(start_bit, true));
|
||||||
|
use_shift = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
AstNode *t;
|
||||||
|
|
||||||
|
t = mkconst_bits(std::vector<RTLIL::State>(result_width, State::S1), false);
|
||||||
|
if (use_shift)
|
||||||
|
t = new AstNode(AST_SHIFT, t, new AstNode(AST_NEG, shamt->clone()));
|
||||||
|
else
|
||||||
|
t = new AstNode(AST_SHIFT_LEFT, t, shamt->clone());
|
||||||
|
t = new AstNode(AST_ASSIGN_EQ, ref_mask->clone(), t);
|
||||||
|
newNode->children.push_back(t);
|
||||||
|
|
||||||
|
t = new AstNode(AST_BIT_AND, mkconst_bits(std::vector<RTLIL::State>(result_width, State::S1), false), children[1]->clone());
|
||||||
|
if (use_shift)
|
||||||
|
t = new AstNode(AST_SHIFT, t, new AstNode(AST_NEG, shamt));
|
||||||
|
else
|
||||||
|
t = new AstNode(AST_SHIFT_LEFT, t, shamt);
|
||||||
|
t = new AstNode(AST_ASSIGN_EQ, ref_data->clone(), t);
|
||||||
|
newNode->children.push_back(t);
|
||||||
|
|
||||||
|
t = new AstNode(AST_BIT_AND, old_data, new AstNode(AST_BIT_NOT, ref_mask));
|
||||||
|
t = new AstNode(AST_BIT_OR, t, ref_data);
|
||||||
|
t = new AstNode(type, lvalue, t);
|
||||||
|
newNode->children.push_back(t);
|
||||||
}
|
}
|
||||||
|
|
||||||
goto apply_newNode;
|
goto apply_newNode;
|
||||||
|
|
|
@ -123,6 +123,7 @@ X(nomem2init)
|
||||||
X(nomem2reg)
|
X(nomem2reg)
|
||||||
X(nomeminit)
|
X(nomeminit)
|
||||||
X(nosync)
|
X(nosync)
|
||||||
|
X(nowrshmsk)
|
||||||
X(O)
|
X(O)
|
||||||
X(OFFSET)
|
X(OFFSET)
|
||||||
X(onehot)
|
X(onehot)
|
||||||
|
|
Loading…
Reference in a new issue