| // Copyright 2023 Google LLC |
| // |
| // 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 |
| // |
| // https://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. |
| |
| #ifndef MPACT_SIM_DECODER_BIN_FORMAT_VISITOR_H_ |
| #define MPACT_SIM_DECODER_BIN_FORMAT_VISITOR_H_ |
| |
| #include <cstdint> |
| #include <deque> |
| #include <list> |
| #include <memory> |
| #include <string> |
| #include <tuple> |
| #include <utility> |
| #include <vector> |
| |
| #include "absl/container/flat_hash_map.h" |
| #include "absl/container/flat_hash_set.h" |
| #include "absl/status/status.h" |
| #include "absl/strings/string_view.h" |
| #include "antlr4-runtime/ParserRuleContext.h" |
| #include "mpact/sim/decoder/BinFormatLexer.h" |
| #include "mpact/sim/decoder/antlr_parser_wrapper.h" |
| #include "mpact/sim/decoder/bin_encoding_info.h" |
| #include "mpact/sim/decoder/bin_format_contexts.h" |
| #include "mpact/sim/decoder/decoder_error_listener.h" |
| #include "re2/re2.h" |
| |
| namespace mpact { |
| namespace sim { |
| namespace decoder { |
| namespace bin_format { |
| |
| enum class ConstraintType : int { kEq = 0, kNe, kLt, kLe, kGt, kGe }; |
| |
| // This struct holds information about a range assignment in an instruction |
| // generator. |
| struct RangeAssignmentInfo { |
| std::vector<std::string> range_names; |
| std::list<RE2> range_regexes; |
| std::vector<std::vector<std::string>> range_values; |
| }; |
| |
| struct BinaryNum { |
| int64_t value; |
| int width; |
| }; |
| |
| struct BitRange { |
| int first; |
| int last; |
| }; |
| |
| class Format; |
| class Overlay; |
| class InstructionEncoding; |
| class InstructionGroup; |
| |
| using ::mpact::sim::decoder::bin_format::generated::BinFormatLexer; |
| |
| using BinFmtAntlrParserWrapper = |
| decoder::AntlrParserWrapper<BinFormatParser, BinFormatLexer>; |
| |
| class BinFormatVisitor { |
| public: |
| struct StringPair { |
| std::string h_output; |
| std::string cc_output; |
| }; |
| struct StringTriple { |
| std::string h_output; |
| std::string cc_output; |
| std::string types_output; |
| }; |
| |
| BinFormatVisitor(); |
| ~BinFormatVisitor(); |
| |
| // Entry point for processing a source_stream input, generating any output |
| // files in the given directory. Returns OK if no errors were encountered. |
| absl::Status Process(const std::vector<std::string>& file_names, |
| const std::string& decoder_name, |
| absl::string_view prefix, |
| const std::vector<std::string>& include_roots, |
| absl::string_view directory); |
| |
| private: |
| // Check the encodings to make sure there aren't any obvious errors. |
| void PerformEncodingChecks(BinEncodingInfo* encoding); |
| // Called to generate and emit code for the decoder according to the parsed |
| // input file. |
| StringTriple EmitDecoderCode(BinEncodingInfo* encoding); |
| StringTriple EmitDecoderFilePrefix(const std::string& dot_h_name, |
| const std::string& types_dot_h_name, |
| BinEncodingInfo* encoding_info) const; |
| // Called to generate and emit code for the decoder according to the parsed |
| // input file. |
| std::tuple<std::string, std::string> EmitEncoderCode( |
| BinEncodingInfo* encoding); |
| std::tuple<std::string, std::string> EmitEncoderFilePrefix( |
| const std::string& dot_h_name, const std::string& enum_h_name, |
| const std::string& types_dot_h_name, |
| BinEncodingInfo* encoding_info) const; |
| // Generate the file suffixes (namespace closing etc.) |
| StringTriple EmitFileSuffix(const std::string& dot_h_name, |
| const std::string& types_dot_h_name, |
| BinEncodingInfo* encoding_info); |
| // Utility methods to parse certain nodes. |
| BinaryNum ParseBinaryNum(TerminalNode* node); |
| BitRange GetBitIndexRange(BitIndexRangeCtx* ctx); |
| int ConvertToInt(NumberCtx* ctx); |
| // Methods that visit the nodes of the parse tree. |
| std::unique_ptr<BinEncodingInfo> ProcessTopLevel( |
| const std::string& decoder_name); |
| void PreProcessDeclarations(DeclarationListCtx* ctx); |
| void VisitDeclarations(DeclarationListCtx* ctx, |
| BinEncodingInfo* encoding_info); |
| void VisitFormatDef(FormatDefCtx* ctx, BinEncodingInfo* encoding_info); |
| void VisitFieldDef(FieldDefCtx* ctx, Format* format, |
| BinEncodingInfo* encoding_info); |
| void VisitIncludeFile(IncludeFileCtx* ctx); |
| void ParseIncludeFile(antlr4::ParserRuleContext* ctx, |
| const std::string& file_name, |
| const std::vector<std::string>& dirs); |
| void VisitOverlayDef(OverlayDefCtx* ctx, Format* format); |
| void VisitOverlayBitField(BitFieldCtx* ctx, Overlay* overlay); |
| InstructionGroup* VisitInstructionGroupDef(InstructionGroupDefCtx* ctx, |
| BinEncodingInfo* encoding_info); |
| std::unique_ptr<BinEncodingInfo> VisitDecoderDef(DecoderDefCtx* ctx); |
| void VisitInstructionDef(InstructionDefCtx* ctx, |
| InstructionGroup* inst_group); |
| void ProcessInstructionDefGenerator(InstructionDefCtx* ctx, |
| InstructionGroup* inst_group); |
| std::string GenerateInstructionDefList( |
| const std::vector<RangeAssignmentInfo*>& range_info_vec, int index, |
| const std::string& template_str_in) const; |
| void VisitConstraint(Format* format, FieldConstraintCtx* ctx, |
| InstructionEncoding* inst_encoding); |
| InstructionGroup* VisitInstructionGroupNameList( |
| const std::string& group_name, GroupNameListCtx* ctx, |
| absl::flat_hash_set<std::string>& group_name_set, |
| BinEncodingInfo* encoding_info); |
| void ProcessSpecializations(BinEncodingInfo* encoding_info); |
| |
| // Accessors. |
| decoder::DecoderErrorListener* error_listener() const { |
| return error_listener_.get(); |
| } |
| void set_error_listener( |
| std::unique_ptr<decoder::DecoderErrorListener> listener) { |
| error_listener_ = std::move(listener); |
| } |
| |
| int current_file_index_ = 0; |
| // Vector of file names. |
| std::vector<std::string> file_names_; |
| // Map from context pointer to file index. |
| absl::flat_hash_map<const antlr4::ParserRuleContext*, int> context_file_map_; |
| // This stores a vector of include file root directories. |
| std::vector<std::string> include_dir_vec_; |
| // Keep track of files that are included in case there is recursive includes. |
| std::deque<std::string> include_file_stack_; |
| // Error listening object passed to the parser. |
| std::unique_ptr<decoder::DecoderErrorListener> error_listener_ = nullptr; |
| std::string decoder_name_; |
| // Maps from identifiers to declaration contexts. |
| absl::flat_hash_map<std::string, FormatDefCtx*> format_decl_map_; |
| absl::flat_hash_map<std::string, InstructionGroupDefCtx*> group_decl_map_; |
| absl::flat_hash_map<std::string, DecoderDefCtx*> decoder_decl_map_; |
| // AntlrParserWrapper vector. |
| std::vector<BinFmtAntlrParserWrapper*> antlr_parser_wrappers_; |
| // Map from comparator string to constraint type. |
| absl::flat_hash_map<std::string, ConstraintType> constraint_string_to_type_; |
| // Set of include files marked as once. |
| absl::flat_hash_set<std::string> once_include_files_; |
| // Specializations to process after all instructions have been processed. |
| std::vector<InstructionDefCtx*> specializations_; |
| }; |
| |
| } // namespace bin_format |
| } // namespace decoder |
| } // namespace sim |
| } // namespace mpact |
| |
| #endif // MPACT_SIM_DECODER_BIN_FORMAT_VISITOR_H_ |