| // Copyright 2017-2020 The Verible Authors. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| #include "verilog/analysis/checkers/port_name_suffix_rule.h" |
| |
| #include <map> |
| #include <set> |
| #include <string> |
| #include <vector> |
| |
| #include "absl/strings/str_split.h" |
| #include "absl/strings/string_view.h" |
| #include "common/analysis/lint_rule_status.h" |
| #include "common/analysis/matcher/bound_symbol_manager.h" |
| #include "common/analysis/matcher/matcher.h" |
| #include "common/text/symbol.h" |
| #include "common/text/syntax_tree_context.h" |
| #include "common/text/token_info.h" |
| #include "common/util/logging.h" |
| #include "verilog/CST/port.h" |
| #include "verilog/CST/verilog_matchers.h" |
| #include "verilog/analysis/descriptions.h" |
| #include "verilog/analysis/lint_rule_registry.h" |
| |
| namespace verilog { |
| namespace analysis { |
| |
| using verible::LintRuleStatus; |
| using verible::LintViolation; |
| using verible::Symbol; |
| using verible::SyntaxTreeContext; |
| using verible::TokenInfo; |
| using verible::matcher::Matcher; |
| |
| // Register PortNameSuffixRule. |
| VERILOG_REGISTER_LINT_RULE(PortNameSuffixRule); |
| |
| static constexpr absl::string_view kMessageIn = |
| "input port names must end with _i, _ni or _pi"; |
| static constexpr absl::string_view kMessageOut = |
| "output port names must end with _o, _no, or _po"; |
| static constexpr absl::string_view kMessageInOut = |
| "inout port names must end with _io, _nio or _pio"; |
| |
| const LintRuleDescriptor &PortNameSuffixRule::GetDescriptor() { |
| static const LintRuleDescriptor d{ |
| .name = "port-name-suffix", |
| .topic = "suffixes-for-signals-and-types", |
| .desc = |
| "Check that port names end with _i for inputs, _o for outputs and " |
| "_io for inouts. " |
| "Alternatively, for active-low signals use _n[io], for differential " |
| "pairs use _n[io] and _p[io].", |
| }; |
| return d; |
| } |
| |
| static const Matcher &PortMatcher() { |
| static const Matcher matcher(NodekPortDeclaration()); |
| return matcher; |
| } |
| |
| void PortNameSuffixRule::Violation(absl::string_view direction, |
| const TokenInfo &token, |
| const SyntaxTreeContext &context) { |
| if (direction == "input") { |
| violations_.insert(LintViolation(token, kMessageIn, context)); |
| } else if (direction == "output") { |
| violations_.insert(LintViolation(token, kMessageOut, context)); |
| } else if (direction == "inout") { |
| violations_.insert(LintViolation(token, kMessageInOut, context)); |
| } |
| } |
| |
| bool PortNameSuffixRule::IsSuffixCorrect(absl::string_view suffix, |
| absl::string_view direction) { |
| static const std::map<absl::string_view, std::set<absl::string_view>> |
| suffixes = {{"input", {"i", "ni", "pi"}}, |
| {"output", {"o", "no", "po"}}, |
| {"inout", {"io", "nio", "pio"}}}; |
| |
| // At this point it is guaranteed that the direction will be set to |
| // one of the expected values (used as keys in the map above). |
| // Therefore checking the suffix like this is safe |
| return suffixes.at(direction).count(suffix) == 1; |
| } |
| |
| void PortNameSuffixRule::HandleSymbol(const Symbol &symbol, |
| const SyntaxTreeContext &context) { |
| constexpr absl::string_view implicit_direction = "input"; |
| verible::matcher::BoundSymbolManager manager; |
| if (PortMatcher().Matches(symbol, &manager)) { |
| const auto *identifier_leaf = GetIdentifierFromPortDeclaration(symbol); |
| const auto *direction_leaf = GetDirectionFromPortDeclaration(symbol); |
| const auto token = identifier_leaf->get(); |
| const auto direction = |
| direction_leaf ? direction_leaf->get().text() : implicit_direction; |
| const auto name = ABSL_DIE_IF_NULL(identifier_leaf)->get().text(); |
| |
| // Check if there is any suffix |
| std::vector<std::string> name_parts = |
| absl::StrSplit(name, '_', absl::SkipEmpty()); |
| |
| if (name_parts.size() < 2) { |
| // No suffix at all |
| Violation(direction, token, context); |
| } |
| |
| if (!IsSuffixCorrect(name_parts.back(), direction)) { |
| Violation(direction, token, context); |
| } |
| } |
| } |
| |
| LintRuleStatus PortNameSuffixRule::Report() const { |
| return LintRuleStatus(violations_, GetDescriptor()); |
| } |
| |
| } // namespace analysis |
| } // namespace verilog |