diff --git a/verilog/analysis/checkers/uvm_macro_semicolon_rule.cc b/verilog/analysis/checkers/uvm_macro_semicolon_rule.cc index 3e7bc9d1c..379dcb268 100644 --- a/verilog/analysis/checkers/uvm_macro_semicolon_rule.cc +++ b/verilog/analysis/checkers/uvm_macro_semicolon_rule.cc @@ -55,11 +55,21 @@ static std::string FormatReason(const verible::TokenInfo ¯o_id) { // Returns true if leaf is a macro and matches `uvm_ static bool IsUvmMacroId(const verible::SyntaxTreeLeaf &leaf) { + const absl::string_view text = leaf.get().text(); + const bool starts_with_uvm = absl::StartsWithIgnoreCase(text, "`uvm_"); + if (leaf.Tag().tag == verilog_tokentype::MacroCallId || - leaf.Tag().tag == verilog_tokentype::MacroIdItem || - leaf.Tag().tag == verilog_tokentype::MacroIdentifier) { - return absl::StartsWithIgnoreCase(leaf.get().text(), "`uvm_"); + leaf.Tag().tag == verilog_tokentype::MacroIdItem) { + return starts_with_uvm; + } + + const bool ends_with_end = absl::EndsWithIgnoreCase(text, "_end"); + // We don't want to complain about macros like: + // `UVM_DEFAULT_TIMEOUT, UVM_MAX_STREAMBITS, ... + if (leaf.Tag().tag == verilog_tokentype::MacroIdentifier) { + return starts_with_uvm && ends_with_end; } + return false; } diff --git a/verilog/analysis/checkers/uvm_macro_semicolon_rule_test.cc b/verilog/analysis/checkers/uvm_macro_semicolon_rule_test.cc index db9f69464..1ac39f425 100644 --- a/verilog/analysis/checkers/uvm_macro_semicolon_rule_test.cc +++ b/verilog/analysis/checkers/uvm_macro_semicolon_rule_test.cc @@ -43,6 +43,15 @@ TEST(UvmMacroSemicolonRuleTest, BaseTests) { RunLintTestCases(kTestCases); } +TEST(UvmMacroSemicolonRuleTest, NoFalsePositivesTest) { + const std::initializer_list kTestCases = { + {"module m;\nint k = `UVM_DEFAULT_TIMEOUT; endmodule\n"}, + {"module m;\nbit [63:0] k = `UVM_REG_ADDR_WIDTH'(0); endmodule\n"}, + }; + + RunLintTestCases(kTestCases); +} + TEST(UvmMacroSemicolonRuleTest, AcceptedUvmMacroCallTests) { const std::initializer_list kTestCases = { // Function/Task scope