Skip to content

Commit c61103a

Browse files
grebecopybara-github
authored andcommitted
Remove codegen specialization for priority selects.
We previously specialized the codegen of priority selects when the selector was never zero. The specialization would call `$error()` if the selector was zero and propagate `X`, the idea being that this form might synthesize to something smaller. Unfortunately, this specialization causes more headaches than it is worth. The `$error()` call isn't a real assertion, so it could spuriously fire. We also generally prefer to avoid propagating X explicitly. Benchmarks showed that this specialization didn't give a net benefit on our corpus, so it seems better to avoid the headaches by removing it. Fixes #1481. PiperOrigin-RevId: 742428399
1 parent ff35a02 commit c61103a

11 files changed

+9
-457
lines changed

xls/codegen/combinational_generator_test.cc

-83
Original file line numberDiff line numberDiff line change
@@ -513,47 +513,6 @@ top fn main(p: bits[2], x: bits[16], y: bits[16], d: bits[16]) -> bits[16] {
513513
IsOkAndHolds(Value(UBits(0xf1ef, 16))));
514514
}
515515

516-
TEST_P(CombinationalGeneratorTest, PrioritySelectSelectorNeverZero) {
517-
std::string text = R"(
518-
package PrioritySelectSelectorNeverZero
519-
520-
top fn main(p: bits[2], x: bits[16], y: bits[16], d: bits[16]) -> bits[16] {
521-
literal.1: bits[2] = literal(value=1)
522-
or.2: bits[2] = or(p, literal.1)
523-
ret priority_sel.3: bits[16] = priority_sel(or.2, cases=[x, y], default=d)
524-
}
525-
)";
526-
XLS_ASSERT_OK_AND_ASSIGN(std::unique_ptr<Package> package,
527-
Parser::ParsePackage(text));
528-
529-
std::optional<FunctionBase*> top = package->GetTop();
530-
ASSERT_TRUE(top.has_value());
531-
XLS_ASSERT_OK_AND_ASSIGN(
532-
auto result, GenerateCombinationalModule(top.value(), codegen_options()));
533-
534-
ExpectVerilogEqualToGoldenFile(GoldenFilePath(kTestName, kTestdataPath),
535-
result.verilog_text);
536-
537-
ModuleSimulator simulator =
538-
NewModuleSimulator(result.verilog_text, result.signature);
539-
absl::flat_hash_map<std::string, Value> args = {
540-
{"x", Value(UBits(0x00ff, 16))},
541-
{"y", Value(UBits(0xf0f0, 16))},
542-
{"d", Value(UBits(0xff00, 16))}};
543-
args["p"] = Value(UBits(0b00, 2));
544-
EXPECT_THAT(simulator.RunFunction(args),
545-
IsOkAndHolds(Value(UBits(0x00ff, 16))));
546-
args["p"] = Value(UBits(0b01, 2));
547-
EXPECT_THAT(simulator.RunFunction(args),
548-
IsOkAndHolds(Value(UBits(0x00ff, 16))));
549-
args["p"] = Value(UBits(0b10, 2));
550-
EXPECT_THAT(simulator.RunFunction(args),
551-
IsOkAndHolds(Value(UBits(0x00ff, 16))));
552-
args["p"] = Value(UBits(0b11, 2));
553-
EXPECT_THAT(simulator.RunFunction(args),
554-
IsOkAndHolds(Value(UBits(0x00ff, 16))));
555-
}
556-
557516
TEST_P(CombinationalGeneratorTest, PrioritySelectWithAndWithoutDefault) {
558517
// Tests that the different specialized ways of codegen'ing priority selects
559518
// are applied in the right context, e.g. we don't make a no-default priority
@@ -734,48 +693,6 @@ top fn main(p: bits[2], x_short: bits[16][2], x_long: bits[16][4], y_short: bits
734693
IsOkAndHolds(Value::Tuple({x_short_value, x_long_value})));
735694
}
736695

737-
TEST_P(CombinationalGeneratorTest, PrioritySelectArraySelectorNeverZero) {
738-
std::string text = R"(
739-
package PrioritySelectArraySelectorNeverZero
740-
741-
top fn main(p: bits[2], x: bits[16][4], y: bits[16][4], d: bits[16][4]) -> bits[16][4] {
742-
743-
literal.1: bits[2] = literal(value=1)
744-
or.2: bits[2] = or(p, literal.1)
745-
ret priority_sel.3: bits[16][4] = priority_sel(or.2, cases=[x, y], default=d)
746-
}
747-
)";
748-
XLS_ASSERT_OK_AND_ASSIGN(std::unique_ptr<Package> package,
749-
Parser::ParsePackage(text));
750-
751-
std::optional<FunctionBase*> top = package->GetTop();
752-
ASSERT_TRUE(top.has_value());
753-
XLS_ASSERT_OK_AND_ASSIGN(
754-
auto result, GenerateCombinationalModule(top.value(), codegen_options()));
755-
756-
ExpectVerilogEqualToGoldenFile(GoldenFilePath(kTestName, kTestdataPath),
757-
result.verilog_text);
758-
759-
ModuleSimulator simulator =
760-
NewModuleSimulator(result.verilog_text, result.signature);
761-
XLS_ASSERT_OK_AND_ASSIGN(
762-
Value x_value, Value::UBitsArray({0x00ff, 0x00ff, 0x00ff, 0x00ff}, 16));
763-
XLS_ASSERT_OK_AND_ASSIGN(
764-
Value y_value, Value::UBitsArray({0xf0f0, 0xf0f0, 0xf0f0, 0xf0f0}, 16));
765-
XLS_ASSERT_OK_AND_ASSIGN(
766-
Value d_value, Value::UBitsArray({0xff00, 0xff00, 0xff00, 0xff00}, 16));
767-
absl::flat_hash_map<std::string, Value> args = {
768-
{"x", x_value}, {"y", y_value}, {"d", d_value}};
769-
args["p"] = Value(UBits(0b00, 2));
770-
EXPECT_THAT(simulator.RunFunction(args), IsOkAndHolds(x_value));
771-
args["p"] = Value(UBits(0b01, 2));
772-
EXPECT_THAT(simulator.RunFunction(args), IsOkAndHolds(x_value));
773-
args["p"] = Value(UBits(0b10, 2));
774-
EXPECT_THAT(simulator.RunFunction(args), IsOkAndHolds(x_value));
775-
args["p"] = Value(UBits(0b11, 2));
776-
EXPECT_THAT(simulator.RunFunction(args), IsOkAndHolds(x_value));
777-
}
778-
779696
TEST_P(CombinationalGeneratorTest, PrioritySelectArrayOneHot) {
780697
std::string text = R"(
781698
package PrioritySelectArrayOneHot

xls/codegen/module_builder.cc

+7-53
Original file line numberDiff line numberDiff line change
@@ -166,7 +166,7 @@ absl::StatusOr<ArrayAssignmentPattern*> ValueToArrayAssignmentPattern(
166166
absl::StatusOr<VerilogFunction*> DefinePrioritySelectFunction(
167167
Node* selector, Type* tpe, int64_t num_cases, const SourceInfo& loc,
168168
std::string_view function_name, ModuleSection* section,
169-
SelectorProperties selector_properties, const CodegenOptions& options) {
169+
const CodegenOptions& options) {
170170
VerilogFile* file = section->file();
171171

172172
VerilogFunction* func = section->Add<VerilogFunction>(
@@ -228,32 +228,8 @@ absl::StatusOr<VerilogFunction*> DefinePrioritySelectFunction(
228228
// Verilog doesn't support 'X, so use the less desirable 16'dxxxx format.
229229
x_literal = file->Make<XSentinel>(loc, tpe->GetFlatBitCount());
230230
}
231-
if (selector_properties.never_zero) {
232-
// If the selector cannot be zero, throw an error if we see zero.
233-
// We still explicitly propagate X (like the default case below) for
234-
// synthesis.
235-
// TODO: github/xls#1481 - this should really be an assert (or assume)
236-
// predicated on selector being valid, but it's a bit tricky to do. For now,
237-
// it seems better to have an overactive error rather than silently
238-
// propagate X.
239-
Expression* error_message =
240-
file->Make<QuotedString>(loc, "Zero selector not allowed.");
241-
StatementBlock* case_block = case_statement->AddCaseArm(zero_label);
242-
auto [macro_name, polarity] =
243-
MacroNameAndPolarity(options.simulation_macro_name());
244-
MacroStatementBlock* ifdef_block =
245-
case_block
246-
->Add<StatementConditionalDirective>(loc, polarity, macro_name)
247-
->consequent();
248-
ifdef_block->Add<SystemTaskCall>(
249-
loc, "error", std::initializer_list<Expression*>{error_message});
250-
case_block->Add<Comment>(loc, "Never taken, propagate X");
251-
case_block->Add<BlockingAssignment>(loc, func->return_value_ref(),
252-
x_literal);
253-
} else {
254-
case_statement->AddCaseArm(zero_label)
255-
->Add<BlockingAssignment>(loc, func->return_value_ref(), default_value);
256-
}
231+
case_statement->AddCaseArm(zero_label)
232+
->Add<BlockingAssignment>(loc, func->return_value_ref(), default_value);
257233

258234
// Add a default case that propagates X.
259235
StatementBlock* case_block = case_statement->AddCaseArm(DefaultSentinel());
@@ -798,15 +774,6 @@ absl::Status ModuleBuilder::EmitArrayCopyAndUpdate(
798774
return absl::OkStatus();
799775
}
800776

801-
absl::StatusOr<SelectorProperties> ModuleBuilder::GetSelectorProperties(
802-
Node* selector) {
803-
XLS_RET_CHECK(selector->GetType()->IsBits());
804-
XLS_RETURN_IF_ERROR(
805-
query_engine_.Populate(selector->function_base()).status());
806-
return SelectorProperties{.never_zero =
807-
query_engine_.AtLeastOneBitTrue(selector)};
808-
}
809-
810777
// We emit asserts if at least one of the following is true:
811778
// (1) We are targeting SystemVerilog. In this case, we have a default
812779
// assertion format.
@@ -1138,16 +1105,13 @@ absl::StatusOr<LogicRef*> ModuleBuilder::EmitAsAssignment(
11381105
if (node_functions_.contains(function_name)) {
11391106
func = node_functions_.at(function_name);
11401107
} else {
1141-
XLS_ASSIGN_OR_RETURN(
1142-
SelectorProperties selector_properties,
1143-
GetSelectorProperties(node->As<PrioritySelect>()->selector()));
11441108
XLS_ASSIGN_OR_RETURN(
11451109
func,
11461110
DefinePrioritySelectFunction(
11471111
node->As<PrioritySelect>()->selector(), /*tpe=*/element_type,
11481112
/*num_cases=*/node->As<PrioritySelect>()->cases().size(),
11491113
/*loc=*/node->loc(), function_name, functions_section_,
1150-
selector_properties, options_));
1114+
options_));
11511115
node_functions_[function_name] = func;
11521116
}
11531117
auto priority_sel_element = [&](absl::Span<Expression* const> inputs) {
@@ -1550,16 +1514,9 @@ absl::StatusOr<std::string> ModuleBuilder::VerilogFunctionName(Node* node) {
15501514
"%s_w%d_%db_%db", OpToString(node->op()), node->BitCountOrDie(),
15511515
node->operand(1)->BitCountOrDie(), node->operand(2)->BitCountOrDie());
15521516
case Op::kPrioritySel: {
1553-
XLS_ASSIGN_OR_RETURN(
1554-
SelectorProperties selector_properties,
1555-
GetSelectorProperties(node->As<PrioritySelect>()->selector()));
1556-
std::string_view never_zero_str;
1557-
if (selector_properties.never_zero) {
1558-
never_zero_str = "_snz"; // selector never zero
1559-
}
1560-
return absl::StrFormat("%s_%db_%dway%s", OpToString(node->op()),
1517+
return absl::StrFormat("%s_%db_%dway", OpToString(node->op()),
15611518
node->GetType()->GetFlatBitCount(),
1562-
node->operand(0)->BitCountOrDie(), never_zero_str);
1519+
node->operand(0)->BitCountOrDie());
15631520
}
15641521
case Op::kSDiv:
15651522
case Op::kUDiv:
@@ -2103,16 +2060,13 @@ absl::StatusOr<VerilogFunction*> ModuleBuilder::DefineFunction(Node* node) {
21032060
function_name, functions_section_);
21042061
break;
21052062
case Op::kPrioritySel: {
2106-
XLS_ASSIGN_OR_RETURN(
2107-
SelectorProperties selector_properties,
2108-
GetSelectorProperties(node->As<PrioritySelect>()->selector()));
21092063
XLS_ASSIGN_OR_RETURN(
21102064
func,
21112065
DefinePrioritySelectFunction(
21122066
node->As<PrioritySelect>()->selector(), /*tpe=*/node->GetType(),
21132067
/*num_cases=*/node->As<PrioritySelect>()->cases().size(),
21142068
/*loc=*/node->loc(), function_name, functions_section_,
2115-
selector_properties, options_));
2069+
options_));
21162070
break;
21172071
}
21182072
case Op::kUDiv:

xls/codegen/module_builder_test.cc

-31
Original file line numberDiff line numberDiff line change
@@ -567,37 +567,6 @@ TEST_P(ModuleBuilderTest, TwoWayPrioritySelectAsFunction) {
567567
file.Emit());
568568
}
569569

570-
TEST_P(ModuleBuilderTest, NeverZeroPrioritySelectAsFunction) {
571-
VerilogFile file = NewVerilogFile();
572-
Package package(TestBaseName());
573-
FunctionBuilder fb(TestBaseName(), &package);
574-
Type* u3 = package.GetBitsType(3);
575-
Type* u32 = package.GetBitsType(32);
576-
BValue s_param = fb.Param("s", u3);
577-
BValue x_param = fb.Param("x", u32);
578-
BValue y_param = fb.Param("y", u32);
579-
BValue z_param = fb.Param("z", u32);
580-
BValue d_param = fb.Param("d", u32);
581-
BValue selector = fb.Or(s_param, fb.Literal(Bits::PowerOfTwo(2, 3)));
582-
BValue priority_select =
583-
fb.PrioritySelect(selector, {x_param, y_param, z_param}, d_param);
584-
XLS_ASSERT_OK(fb.Build());
585-
586-
ModuleBuilder mb(TestBaseName(), &file, codegen_options());
587-
XLS_ASSERT_OK_AND_ASSIGN(LogicRef * s, mb.AddInputPort("s", u3));
588-
XLS_ASSERT_OK_AND_ASSIGN(LogicRef * x, mb.AddInputPort("x", u32));
589-
XLS_ASSERT_OK_AND_ASSIGN(LogicRef * y, mb.AddInputPort("y", u32));
590-
XLS_ASSERT_OK_AND_ASSIGN(LogicRef * z, mb.AddInputPort("z", u32));
591-
XLS_ASSERT_OK_AND_ASSIGN(LogicRef * d, mb.AddInputPort("d", u32));
592-
593-
XLS_ASSERT_OK(mb.EmitAsAssignment("priority_select", priority_select.node(),
594-
{s, x, y, z, d})
595-
.status());
596-
597-
ExpectVerilogEqualToGoldenFile(GoldenFilePath(kTestName, kTestdataPath),
598-
file.Emit());
599-
}
600-
601570
TEST_P(ModuleBuilderTest, SimilarPrioritySelects) {
602571
VerilogFile file = NewVerilogFile();
603572
Package package(TestBaseName());

xls/codegen/testdata/combinational_generator_test_PrioritySelectArraySelectorNeverZero.svtxt

-52
This file was deleted.

xls/codegen/testdata/combinational_generator_test_PrioritySelectArraySelectorNeverZero.vtxt

-52
This file was deleted.

xls/codegen/testdata/combinational_generator_test_PrioritySelectSelectorNeverZero.svtxt

-33
This file was deleted.

0 commit comments

Comments
 (0)