diff --git a/.prettierignore b/.prettierignore new file mode 100644 index 000000000..10ae89d45 --- /dev/null +++ b/.prettierignore @@ -0,0 +1,11 @@ +gen +.home +.singularity +node_modules +.asciidoctor +.bundle +.git +.github +.vscode +docs/ruby +ext diff --git a/.prettierrc b/.prettierrc new file mode 100644 index 000000000..0967ef424 --- /dev/null +++ b/.prettierrc @@ -0,0 +1 @@ +{} diff --git a/.rubocop.yml b/.rubocop.yml index ca2d2a8fd..3d27e703b 100644 --- a/.rubocop.yml +++ b/.rubocop.yml @@ -40,6 +40,10 @@ Metrics/AbcSize: Metrics/CyclomaticComplexity: Enabled: false +Metrics/ParameterLists: + Enabled: true + CountKeywordArgs: false + Metrics/PerceivedComplexity: Enabled: false diff --git a/Gemfile b/Gemfile index eb360e885..9509ba661 100644 --- a/Gemfile +++ b/Gemfile @@ -4,25 +4,27 @@ ruby "3.2.3" source "https://rubygems.org" +gem "activesupport" gem "asciidoctor-diagram", "~> 2.2" gem "asciidoctor-pdf" gem "base64" gem "bigdecimal" gem "json_schemer", "~> 1.0" gem "minitest" -gem "ruby-progressbar", "~> 1.13" gem "pygments.rb" gem "rake", "~> 13.0" gem "rouge" +gem "ruby-progressbar", "~> 1.13" gem "treetop", "1.6.12" gem "ttfunk", "1.7" # needed to avoid having asciidoctor-pdf dependencies pulling in a buggy version of ttunk (1.8) gem "webrick" gem "yard" group :development do - gem "solargraph" - gem 'rubocop-minitest' - gem 'ruby-prof' - gem "ruby-prof-flamegraph" + gem "debug" gem "rdbg" + gem "rubocop-minitest" + gem "ruby-prof" + gem "ruby-prof-flamegraph" + gem "solargraph" end diff --git a/Gemfile.lock b/Gemfile.lock index 0a897582c..9177f572a 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -2,6 +2,19 @@ GEM remote: https://rubygems.org/ specs: Ascii85 (1.1.1) + activesupport (8.0.0) + base64 + benchmark (>= 0.3) + bigdecimal + concurrent-ruby (~> 1.0, >= 1.3.1) + connection_pool (>= 2.2.5) + drb + i18n (>= 1.6, < 2) + logger (>= 1.4.2) + minitest (>= 5.1) + securerandom (>= 0.3) + tzinfo (~> 2.0, >= 2.0.5) + uri (>= 0.13.1) addressable (2.8.7) public_suffix (>= 2.0.2, < 7.0) afm (0.2.2) @@ -31,18 +44,21 @@ GEM benchmark (0.3.0) bigdecimal (3.1.8) concurrent-ruby (1.3.3) + connection_pool (2.4.1) css_parser (1.17.1) addressable - date (3.4.1) debug (1.9.2) irb (~> 1.10) reline (>= 0.3.8) diff-lcs (1.5.1) + drb (2.2.1) e2mmap (0.1.0) hana (1.3.7) hashery (2.1.2) - io-console (0.8.0) - irb (1.14.2) + i18n (1.14.6) + concurrent-ruby (~> 1.0) + io-console (0.7.2) + irb (1.14.1) rdoc (>= 4.0.0) reline (>= 0.4.2) jaro_winkler (1.6.0) @@ -56,6 +72,7 @@ GEM kramdown-parser-gfm (1.1.0) kramdown (~> 2.0) language_server-protocol (3.17.0.3) + logger (1.6.2) matrix (0.4.2) minitest (5.24.1) nokogiri (1.16.5-aarch64-linux) @@ -89,8 +106,7 @@ GEM prawn-templates (0.1.2) pdf-reader (~> 2.0) prawn (~> 2.2) - psych (5.2.1) - date + psych (5.2.0) stringio public_suffix (6.0.0) pygments.rb (3.0.0) @@ -100,10 +116,10 @@ GEM rbs (2.8.4) rdbg (0.1.0) debug (>= 1.2.2) - rdoc (6.9.1) + rdoc (6.8.1) psych (>= 4.0.0) regexp_parser (2.9.2) - reline (0.6.0) + reline (0.5.11) io-console (~> 0.5) reverse_markdown (2.1.1) nokogiri @@ -131,6 +147,7 @@ GEM ruby-prof (~> 0.13) ruby-progressbar (1.13.0) ruby-rc4 (0.1.5) + securerandom (0.4.0) simpleidn (0.2.3) solargraph (0.50.0) backport (~> 1.2) @@ -155,7 +172,10 @@ GEM treetop (1.6.12) polyglot (~> 0.3) ttfunk (1.7.0) + tzinfo (2.0.6) + concurrent-ruby (~> 1.0) unicode-display_width (2.5.0) + uri (1.0.2) webrick (1.8.1) yard (0.9.36) @@ -164,10 +184,12 @@ PLATFORMS x86_64-linux-gnu DEPENDENCIES + activesupport asciidoctor-diagram (~> 2.2) asciidoctor-pdf base64 bigdecimal + debug json_schemer (~> 1.0) minitest pygments.rb diff --git a/Rakefile b/Rakefile index 84461cca2..4e754773c 100644 --- a/Rakefile +++ b/Rakefile @@ -9,7 +9,7 @@ require "ruby-progressbar" require "yard" require "minitest/test_task" -require_relative $root / "lib" / "validate" +require_relative $root / "lib" / "architecture" directory "#{$root}/.stamps" @@ -19,22 +19,50 @@ end directory "#{$root}/.stamps" -file "#{$root}/.stamps/dev_gems" => "#{$root}/.stamps" do - Dir.chdir($root) do - sh "bundle config set --local with development" - sh "bundle install" - FileUtils.touch "#{$root}/.stamps/dev_gems" - end +def cfg_arch_for(config_name) + Rake::Task["#{$root}/.stamps/resolve-#{config_name}.stamp"].invoke + + @cfg_archs ||= {} + return @cfg_archs[config_name] if @cfg_archs.key?(config_name) + + @cfg_archs[config_name] = + ConfiguredArchitecture.new( + config_name, + $root / "gen" / "resolved_arch" / config_name, + overlay_path: $root / "cfgs" / config_name / "arch_overlay" + ) end namespace :gen do desc "Generate documentation for the ruby tooling" task tool_doc: "#{$root}/.stamps/dev_gems" do Dir.chdir($root) do - sh "bundle exec yard doc --yardopts arch_def.yardopts" + sh "bundle exec yard doc --yardopts cfg_arch.yardopts" sh "bundle exec yard doc --yardopts idl.yardopts" end end + + desc "Resolve the standard in arch/, and write it to resolved_arch/" + task "resolved_arch" do + sh "#{$root}/.home/.venv/bin/python3 lib/yaml_resolver.py resolve arch resolved_arch" + end +end + +# rule to generate standard for any configurations with an overlay +rule %r{#{$root}/.stamps/resolve-.+\.stamp} => proc { |tname| + cfg_name = File.basename(tname, ".stamp").sub("resolve-", "") + arch_files = Dir.glob("#{$root}/arch/**/*.yaml") + overlay_files = Dir.glob("#{$root}/cfgs/#{cfg_name}/arch_overlay/**/*.yaml") + [ + "#{$root}/.stamps", + "#{$root}/lib/yaml_resolver.py" + ] + arch_files + overlay_files +} do |t| + cfg_name = File.basename(t.name, ".stamp").sub("resolve-", "") + sh "#{$root}/.home/.venv/bin/python3 lib/yaml_resolver.py merge arch cfgs/#{cfg_name}/arch_overlay gen/arch/#{cfg_name}" + sh "#{$root}/.home/.venv/bin/python3 lib/yaml_resolver.py resolve gen/arch/#{cfg_name} gen/resolved_arch/#{cfg_name}" + + FileUtils.touch t.name end namespace :serve do @@ -75,8 +103,7 @@ end desc "Clean up all generated files" task :clean do - FileUtils.rm_rf $root / "gen" - FileUtils.rm_rf $root / ".stamps" + warn "Don't run clean using Rake. Run `./do clean` (alias for `./bin/clean`) instead." end namespace :test do @@ -88,30 +115,23 @@ namespace :test do end puts "All instruction encodings pass basic sanity tests" end - task schema: "gen:arch" do - validator = Validator.instance + task schema: "gen:resolved_arch" do puts "Checking arch files against schema.." - arch_files = Dir.glob("#{$root}/arch/**/*.yaml") - progressbar = ProgressBar.create(total: arch_files.size) - arch_files.each do |f| - progressbar.increment - validator.validate(f) - end - Rake::Task["test:insts"].invoke + Architecture.new("#{$root}/resolved_arch").validate(show_progress: true) puts "All files validate against their schema" end - task idl_model: ["gen:arch", "#{$root}/.stamps/arch-gen-_32.stamp", "#{$root}/.stamps/arch-gen-_64.stamp"] do + task idl: ["gen:resolved_arch", "#{$root}/.stamps/resolve-rv32.stamp", "#{$root}/.stamps/resolve-rv64.stamp"] do print "Parsing IDL code for RV32..." - arch_def_32 = arch_def_for("_32") + cfg_arch32 = cfg_arch_for("rv32") puts "done" - arch_def_32.type_check + cfg_arch32.type_check print "Parsing IDL code for RV64..." - arch_def_64 = arch_def_for("_64") + cfg_arch64 = cfg_arch_for("rv64") puts "done" - arch_def_64.type_check + cfg_arch64.type_check puts "All IDL passed type checking" end @@ -121,7 +141,7 @@ def insert_warning(str, from) # insert a warning on the second line lines = str.lines first_line = lines.shift - lines.unshift(first_line, "\n# WARNING: This file is auto-generated from #{Pathname.new(from).relative_path_from($root)}\n\n").join("") + lines.unshift(first_line, "\n# WARNING: This file is auto-generated from #{Pathname.new(from).relative_path_from($root)}").join("") end private :insert_warning @@ -130,7 +150,6 @@ private :insert_warning "#{$root}/arch/csr/Zihpm/mhpmcounterN.layout", __FILE__ ] do |t| - puts "Generating #{Pathname.new(t.name).relative_path_from($root)}" erb = ERB.new(File.read($root / "arch/csr/Zihpm/mhpmcounterN.layout"), trim_mode: "-") erb.filename = "#{$root}/arch/csr/Zihpm/mhpmcounterN.layout" File.write(t.name, insert_warning(erb.result(binding), t.prerequisites.first)) @@ -139,7 +158,6 @@ private :insert_warning "#{$root}/arch/csr/Zihpm/mhpmcounterNh.layout", __FILE__ ] do |t| - puts "Generating #{Pathname.new(t.name).relative_path_from($root)}" erb = ERB.new(File.read($root / "arch/csr/Zihpm/mhpmcounterNh.layout"), trim_mode: "-") erb.filename = "#{$root}/arch/csr/Zihpm/mhpmcounterNh.layout" File.write(t.name, insert_warning(erb.result(binding), t.prerequisites.first)) @@ -148,7 +166,6 @@ private :insert_warning "#{$root}/arch/csr/Zihpm/mhpmeventN.layout", __FILE__ ] do |t| - puts "Generating #{Pathname.new(t.name).relative_path_from($root)}" erb = ERB.new(File.read($root / "arch/csr/Zihpm/mhpmeventN.layout"), trim_mode: "-") erb.filename = "#{$root}/arch/csr/Zihpm/mhpmeventN.layout" File.write(t.name, insert_warning(erb.result(binding), t.prerequisites.first)) @@ -157,7 +174,6 @@ private :insert_warning "#{$root}/arch/csr/Zihpm/mhpmeventNh.layout", __FILE__ ] do |t| - puts "Generating #{Pathname.new(t.name).relative_path_from($root)}" erb = ERB.new(File.read($root / "arch/csr/Zihpm/mhpmeventNh.layout"), trim_mode: "-") erb.filename = "#{$root}/arch/csr/Zihpm/mhpmeventNh.layout" File.write(t.name, insert_warning(erb.result(binding), t.prerequisites.first)) @@ -166,7 +182,6 @@ private :insert_warning "#{$root}/arch/csr/Zihpm/hpmcounterN.layout", __FILE__ ] do |t| - puts "Generating #{Pathname.new(t.name).relative_path_from($root)}" erb = ERB.new(File.read($root / "arch/csr/Zihpm/hpmcounterN.layout"), trim_mode: "-") erb.filename = "#{$root}/arch/csr/Zihpm/hpmcounterN.layout" File.write(t.name, insert_warning(erb.result(binding), t.prerequisites.first)) @@ -175,7 +190,6 @@ private :insert_warning "#{$root}/arch/csr/Zihpm/hpmcounterNh.layout", __FILE__ ] do |t| - puts "Generating #{Pathname.new(t.name).relative_path_from($root)}" erb = ERB.new(File.read($root / "arch/csr/Zihpm/hpmcounterNh.layout"), trim_mode: "-") erb.filename = "#{$root}/arch/csr/Zihpm/hpmcounterNh.layout" File.write(t.name, insert_warning(erb.result(binding), t.prerequisites.first)) @@ -187,7 +201,6 @@ end "#{$root}/arch/csr/I/pmpaddrN.layout", __FILE__ ] do |t| - puts "Generating #{Pathname.new(t.name).relative_path_from($root)}" erb = ERB.new(File.read($root / "arch/csr/I/pmpaddrN.layout"), trim_mode: "-") erb.filename = "#{$root}/arch/csr/I/pmpaddrN.layout" File.write(t.name, insert_warning(erb.result(binding), t.prerequisites.first)) @@ -199,7 +212,6 @@ end "#{$root}/arch/csr/I/pmpcfgN.layout", __FILE__ ] do |t| - puts "Generating #{Pathname.new(t.name).relative_path_from($root)}" erb = ERB.new(File.read($root / "arch/csr/I/pmpcfgN.layout"), trim_mode: "-") erb.filename = "#{$root}/arch/csr/I/pmpcfgN.layout" File.write(t.name, insert_warning(erb.result(binding), t.prerequisites.first)) @@ -210,7 +222,6 @@ file "#{$root}/arch/csr/I/mcounteren.yaml" => [ "#{$root}/arch/csr/I/mcounteren.layout", __FILE__ ] do |t| - puts "Generating #{Pathname.new(t.name).relative_path_from($root)}" erb = ERB.new(File.read($root / "arch/csr/I/mcounteren.layout"), trim_mode: "-") erb.filename = "#{$root}/arch/csr/I/mcounteren.layout" File.write(t.name, insert_warning(erb.result(binding), t.prerequisites.first)) @@ -220,7 +231,6 @@ file "#{$root}/arch/csr/S/scounteren.yaml" => [ "#{$root}/arch/csr/S/scounteren.layout", __FILE__ ] do |t| - puts "Generating #{Pathname.new(t.name).relative_path_from($root)}" erb = ERB.new(File.read($root / "arch/csr/S/scounteren.layout"), trim_mode: "-") erb.filename = "#{$root}/arch/csr/S/scounteren.layout" File.write(t.name, insert_warning(erb.result(binding), t.prerequisites.first)) @@ -230,7 +240,6 @@ file "#{$root}/arch/csr/H/hcounteren.yaml" => [ "#{$root}/arch/csr/H/hcounteren.layout", __FILE__ ] do |t| - puts "Generating #{Pathname.new(t.name).relative_path_from($root)}" erb = ERB.new(File.read($root / "arch/csr/H/hcounteren.layout"), trim_mode: "-") erb.filename = "#{$root}/arch/csr/H/hcounteren.layout" File.write(t.name, insert_warning(erb.result(binding), t.prerequisites.first)) @@ -240,7 +249,6 @@ file "#{$root}/arch/csr/Zicntr/mcountinhibit.yaml" => [ "#{$root}/arch/csr/Zicntr/mcountinhibit.layout", __FILE__ ] do |t| - puts "Generating #{Pathname.new(t.name).relative_path_from($root)}" erb = ERB.new(File.read($root / "arch/csr/Zicntr/mcountinhibit.layout"), trim_mode: "-") erb.filename = "#{$root}/arch/csr/Zicntr/mcountinhibit.layout" File.write(t.name, insert_warning(erb.result(binding), t.prerequisites.first)) @@ -284,7 +292,7 @@ namespace :test do Rake::Task["test:idl_compiler"].invoke Rake::Task["test:lib"].invoke Rake::Task["test:schema"].invoke - Rake::Task["test:idl_model"].invoke + Rake::Task["test:idl"].invoke end desc <<~DESC diff --git a/arch/README.adoc b/arch/README.adoc index 185100f3b..5f80b2692 100644 --- a/arch/README.adoc +++ b/arch/README.adoc @@ -15,10 +15,16 @@ A few challenging issues that arise include: To tame this challenge, this specification generator takes the following approach: - * The generic architecture (in the `arch/` folder) is described in a way that covers all implementation options. As much as possible, the architecture is defined in a structured way that can be easily parsed using any programming language with a https://en.wikipedia.org/wiki/YAML[YAML] library. Complex behavior (_e.g._, instruction operation) and some configuration-dependent metadata is specified in an architecture definition language (xref:prose/idl.adoc[IDL]) that can formally define the architecture in arbitrary ways. + * The RVI standard architecture (in the `arch/` folder) is described in a way that covers all implementation options. As much as possible, the architecture is defined in a structured way that can be easily parsed using any programming language with a https://en.wikipedia.org/wiki/YAML[YAML] library. Complex behavior (_e.g._, instruction operation) and some configuration-dependent metadata is specified in an architecture definition language (xref:prose/idl.adoc[IDL]) that can formally define the architecture in arbitrary ways. * An implementation of RISC-V is specified as a _configuration_ containing all unnamed parameters and list of named supported extensions (in the `cfgs/` folder). * A tool, included in this repository, can generate an implmentation-specific specification by applying the configuration to the generic spec. Behaviors that are not relevant are left out, and behaviors that are affected by multiple extensions are merged into a single description. +Three standard configs are present: + + * `_`: A completely _unconfigured_ configuration, i.e., where there are no restrictions and no known parameter values. + * `rv32`: A configuration where only `MXLEN` is known to be 32, i.e., the RV32 ISA. + * `rv64`: A configuration where only `MXLEN` is known to be 64, i.e., the RV64 ISA. + The architecture is specified in a series of https://en.wikipedia.org/wiki/YAML[YAML] files for _Extensions_, _Instructions_, and _Control and Status Registers (CSRs)_. Each extension/instruction/CSR has its own file. @@ -27,48 +33,61 @@ Each extension/instruction/CSR has its own file. [ditaa] .... -+----------------------------------------------------------------------------------------------+ -| Config (cfgs/NAME) | -| +-------------------------+ +-------------------------+ +---------------------------------+ | -| | Implementation Options | | Extension List | | Implementation Overlay | | -| | (cfgs/NAME/params.yaml) | | (cfgs/NAME/params.yaml) | | (cfgs/NAME/arch_overlay/*.yaml) | | -| +-------------------------+ +-------------------------+ +---------------------------------+ | -+------------------------|-----|--------------------------------|------------------------------+ - | | | - +----------------+ v v v - |{s} Generic | /----------\ /----------\ - | Arch Spec |-->| Render | | Render | - | (arch) | | (ERB) : | (ERB) : - +----------------+ \----------/ \----------/ - | | - v v - +-----------------------------+ +-----------------------------------+ - | {s} Redered Arch Spec | | {s} Redered Overlay | - | (gen/NAME/.rendered_arch) | | (gen/NAME/.rendered_overlay_arch) | - +-----------------------------+ +------------------------------------+ - | | - v | - /-------------------------\ | - | Overlay |<-------------------+ - | (gen/NAME/.merged_arch) : - \-------------------------/ - | - | - /-----------\ - | Normalize | - \-----------/ - | - v - +-----------------------------+ - | {s} Implementation-specific | - | Archiecture Spec | - | (gen/NAME/arch) | - +-----------------------------+ ++--------------------------------------------------------------------------+ +| Config (cfgs/NAME) | +| +------------------------------+ +-----------------------------------+ | +| | Config (extensions & params) | | Implementation Overlay [optional] | | +| | (cfgs/NAME/cfg.yaml) | | (cfgs/NAME/arch_overlay/*.yaml) | | +| +------------------------------+ +-----------------------------------+ | ++--------|------------------------------------------------------|----------+ + | | + | +----------------+ v + | |{s} Standard | /--------------------\ + | | Arch Spec |--->| JSON Merge Patch | + | | (arch) | | RFC 7386 | + | +----------------+ \--------------------/ + | | + | v + | +--------------------------+ + | | {s} Merged Arch Spec | + | | (gen/arch/NAME/*.yaml) | + | +--------------------------+ + | | + | v + | /---------------------\ + | | Resolve Operators | + | | ($inherits/$remove) | + | \---------------------/ + | | + | v + | +---------------------------------+ + | | {s} Implementation-specific | + | | Archiecture Spec | + | | (gen/resolved_arch/NAME/*.yaml) | + | +---------------------------------+ + | | + +------|-------------------------------------------------|---------+ + | | Ruby Interface | | + | /--------\ | | + | | Config | | | + | \--------/ /---------------\ | | + | | +-| Architecture |<----------------------+ | + | | | \---------------/ | + | | | | + | v v | + | /------------------------\ | + | | ConfiguredArchitecture | | + | \------------------------/ | + +------------------------------------------------------------------+ .... -The normalization step transitively determines all extensions/instructions/csrs that are implemented -(_e.g._, because some extensions have implied dependencies) and ensures that missing fields are -filled in with their defaults. +The final architecture files go through two transformation steps. First, config overlay files, if present, +are merged on top of the standard architecture files according to JSON Merge Patch. This step is where +any custom behavior can be added to the specification. Second, the (possibly) merged files are _resolved_ +by implementing the database operators `$inherits` and `$remove`. + +The `ConfiguredArchitecture` class represents a configured view of the database, and is the basis +of most queries about a config. == Data Format diff --git a/arch/certificate_model/MC100.yaml b/arch/certificate_model/MC100.yaml index 4fdad7097..b3dba5128 100644 --- a/arch/certificate_model/MC100.yaml +++ b/arch/certificate_model/MC100.yaml @@ -16,7 +16,7 @@ base: 32 revision_history: - revision: "0.7.0" - date: 2024-07-29 + date: "2024-07-29" changes: - First version after moving non-microcontroller content in this document to a new document called "RISC-V CRDs (Certification Requirement Documents)" @@ -27,7 +27,7 @@ revision_history: - Added requirements for WFI instruction - Added requirements related to msip memory-mapped register - revision: "0.6.0" - date: 2024-07-11 + date: "2024-07-11" changes: - Supporting multiple MC versions to support customers wanting to certify existing microcontrollers not using the latest version of ratified standards. - Changed versioning scheme to use major.minor.patch instead of 3-digit major & minor. @@ -42,28 +42,28 @@ revision_history: - Added more options for interrupts - Moved non-microcontroller content in this document to a new document called "RISC-V Certification Plans" - revision: "0.5.0" - date: 2024-06-03 + date: "2024-06-03" changes: - Renamed to "RISC-V Microcontroller Certification Plan" based on Jason's recommendation - Added mvendorid, marchid, mimpid, and mhardid read-only priv CSRs because Allen pointed out these are mandatory in M-mode v1.13 (probably older versions too, haven't looked yet). - Added table showing mapping of MC versions to associated RISC-V specifications - revision: "0.4.0" - date: 2024-06-03 + date: "2024-06-03" changes: - Added M-mode instruction requirements - Made Zicntr MANDATORY due to very low cost for implementations to support (in the spirit of minimizing options). - Removed OPT-CNTR-PREC since minstret and mcycle must be a full 64 bits to be standard-compliant. - revision: "0.3.0" - date: 2024-05-25 + date: "2024-05-25" changes: - Includes Zicntr as OPTIONAL and then has only 32-bit counters for instret and cycle. - revision: "0.2.0" - date: 2024-05-20 + date: "2024-05-20" changes: - Very early draft - revision: "0.1.0" - date: 2024-05-16 + date: "2024-05-16" changes: - Initial version diff --git a/arch/certificate_model/MockCertificateModel.yaml b/arch/certificate_model/MockCertificateModel.yaml index 951c892ee..86b2601ec 100644 --- a/arch/certificate_model/MockCertificateModel.yaml +++ b/arch/certificate_model/MockCertificateModel.yaml @@ -17,11 +17,11 @@ versions: revision_history: - revision: "0.1.0" - date: 2024-10-04 + date: "2024-10-04" changes: - Created to test CRDs - revision: "0.2.0" - date: 2024-10-05 + date: "2024-10-05" changes: - Also created to test CRDs @@ -40,11 +40,10 @@ debug_manual_revision: "0.13.2" # XXX - Remove version information since specifying priv/unpriv ISA manual should imply this. extensions: $inherits: - - "profile_release/MockProfileRelease.yaml#/MockProfileRelease/profiles/MP-U-64/extensions" - - "profile_release/MockProfileRelease.yaml#/MockProfileRelease/profiles/MP-S-64/extensions" + - "profile/MP-S-64.yaml#/extensions" I: note: Just added this note to I extension - MockExt: + Xmock: presence: mandatory parameters: MOCK_ENUM_2_INTS: {} diff --git a/arch/common/inst_variable_types.yaml b/arch/common/inst_variable_types.yaml deleted file mode 100644 index 7824ae5ac..000000000 --- a/arch/common/inst_variable_types.yaml +++ /dev/null @@ -1,4 +0,0 @@ -# yaml-language-server: $schema=../../schemas/inst_variable_metadatas.json ---- -itype_imm: - location: 31-20 diff --git a/arch/csr/H/hcounteren.yaml b/arch/csr/H/hcounteren.yaml index 46b2f2f1b..07900bd61 100644 --- a/arch/csr/H/hcounteren.yaml +++ b/arch/csr/H/hcounteren.yaml @@ -1,7 +1,6 @@ # yaml-language-server: $schema=../../../schemas/csr_schema.json # WARNING: This file is auto-generated from arch/csr/H/hcounteren.layout - $schema: csr_schema.json# kind: csr name: hcounteren diff --git a/arch/csr/I/mcounteren.layout b/arch/csr/I/mcounteren.layout index 0a53745e4..c573cc1ab 100644 --- a/arch/csr/I/mcounteren.layout +++ b/arch/csr/I/mcounteren.layout @@ -80,8 +80,6 @@ description: | <%%- end -%> . <%%- end -%> - - definedBy: U # actually, defined by RV64, but must implement U-mode for this CSR to exist fields: CY: diff --git a/arch/csr/I/mcounteren.yaml b/arch/csr/I/mcounteren.yaml index 23c4a4865..7430451a3 100644 --- a/arch/csr/I/mcounteren.yaml +++ b/arch/csr/I/mcounteren.yaml @@ -1,7 +1,6 @@ # yaml-language-server: $schema=../../schemas/csr_schema.json # WARNING: This file is auto-generated from arch/csr/I/mcounteren.layout - $schema: csr_schema.json# kind: csr name: mcounteren @@ -82,7 +81,6 @@ description: | <%- end -%> . <%- end -%> - definedBy: U # actually, defined by RV64, but must implement U-mode for this CSR to exist fields: CY: diff --git a/arch/csr/I/pmpcfgN.layout b/arch/csr/I/pmpcfgN.layout index 571f75097..dfe6b22a5 100644 --- a/arch/csr/I/pmpcfgN.layout +++ b/arch/csr/I/pmpcfgN.layout @@ -22,39 +22,39 @@ fields: base: 64 # upper half doesn't exist in RV32 <%- end -%> description: | - *PMP configuration for entry <%= pmpcfg_num*4 + i %>* + *PMP configuration for entry <%= pmpcfg_num*4 + i %>* - The bits are as follows: + The bits are as follows: - [separator="!",%autowidth] - !=== - ! Name ! Location ! Description + [separator="!",%autowidth] + !=== + ! Name ! Location ! Description - h! L ! <%= ((i+1)*8)-1 %> ! Locks the entry from further modification. Additionally, when set, PMP checks also apply to M-mode for the entry. - h! - ! <%= ((i+1)*8)-2 %>:<%= ((i+1)*8)-3 %> ! _Reserved_ Writes shall be ignored. - h! A ! <%= ((i+1)*8)-4 %>:<%= ((i+1)*8)-5 %> - a! Address matching mode. One of: + h! L ! <%= ((i+1)*8)-1 %> ! Locks the entry from further modification. Additionally, when set, PMP checks also apply to M-mode for the entry. + h! - ! <%= ((i+1)*8)-2 %>:<%= ((i+1)*8)-3 %> ! _Reserved_ Writes shall be ignored. + h! A ! <%= ((i+1)*8)-4 %>:<%= ((i+1)*8)-5 %> + a! Address matching mode. One of: - [when="PMP_GRANULARITY < 2"] - * *OFF* (0) - Null region (disabled) - * *TOR* (1) - Top of range - * *NA4* (2) - Naturally aligned four-byte region - * *NAPOT* (3) - Natrually aligned power of two + [when="PMP_GRANULARITY < 2"] + * *OFF* (0) - Null region (disabled) + * *TOR* (1) - Top of range + * *NA4* (2) - Naturally aligned four-byte region + * *NAPOT* (3) - Natrually aligned power of two - [when="PMP_GRANULARITY >= 2"] - * *OFF* (0) - Null region (disabled) - * *TOR* (1) - Top of range - * *NAPOT* (3) - Natrually aligned power of two + [when="PMP_GRANULARITY >= 2"] + * *OFF* (0) - Null region (disabled) + * *TOR* (1) - Top of range + * *NAPOT* (3) - Natrually aligned power of two - [when="PMP_GRANULARITY >= 2"] - Naturally aligned four-byte region, *NA4* (2), is not valid (not needed when the PMP granularity is larger than 4 bytes). + [when="PMP_GRANULARITY >= 2"] + Naturally aligned four-byte region, *NA4* (2), is not valid (not needed when the PMP granularity is larger than 4 bytes). - h! X ! <%= ((i)*8)+2 %> ! When clear, instruction fetchs cause an `Access Fault` for the matching region and privilege mode. - h! W ! <%= ((i)*8)+1 %> ! When clear, stores and AMOs cause an `Access Fault` for the matching region and privilege mode. - h! R ! <%= ((i)*8)+0 %> ! When clear, loads cause an `Access Fault` for the matching region and privilege mode. - !=== + h! X ! <%= ((i)*8)+2 %> ! When clear, instruction fetchs cause an `Access Fault` for the matching region and privilege mode. + h! W ! <%= ((i)*8)+1 %> ! When clear, stores and AMOs cause an `Access Fault` for the matching region and privilege mode. + h! R ! <%= ((i)*8)+0 %> ! When clear, loads cause an `Access Fault` for the matching region and privilege mode. + !=== - The combination of R = 0, W = 1 is reserved. + The combination of R = 0, W = 1 is reserved. type(): | if (NUM_PMP_ENTRIES > <%= pmpcfg_num*4 + i %>) { return CsrFieldType::RWR; diff --git a/arch/csr/S/scounteren.yaml b/arch/csr/S/scounteren.yaml index 395ec77d8..107962021 100644 --- a/arch/csr/S/scounteren.yaml +++ b/arch/csr/S/scounteren.yaml @@ -1,7 +1,6 @@ # yaml-language-server: $schema=../../schemas/csr_schema.json # WARNING: This file is auto-generated from arch/csr/S/scounteren.layout - $schema: csr_schema.json# kind: csr name: scounteren diff --git a/arch/csr/Zicntr/mcountinhibit.layout b/arch/csr/Zicntr/mcountinhibit.layout index 549b2061e..257ab8586 100644 --- a/arch/csr/Zicntr/mcountinhibit.layout +++ b/arch/csr/Zicntr/mcountinhibit.layout @@ -41,8 +41,8 @@ description: | definedBy: anyOf: - - name: Sm - - name: Smhpm + - name: Sm + - name: Smhpm fields: CY: location: 0 diff --git a/arch/csr/Zicntr/mcountinhibit.yaml b/arch/csr/Zicntr/mcountinhibit.yaml index 685b45357..1a06640b7 100644 --- a/arch/csr/Zicntr/mcountinhibit.yaml +++ b/arch/csr/Zicntr/mcountinhibit.yaml @@ -1,7 +1,6 @@ # yaml-language-server: $schema=../../../schemas/csr_schema.json # WARNING: This file is auto-generated from arch/csr/Zicntr/mcountinhibit.layout - $schema: csr_schema.json# kind: csr name: mcountinhibit diff --git a/arch/csr/Zihpm/mhpmcounterN.layout b/arch/csr/Zihpm/mhpmcounterN.layout index 8b6fd187d..6ceb9cfcb 100644 --- a/arch/csr/Zihpm/mhpmcounterN.layout +++ b/arch/csr/Zihpm/mhpmcounterN.layout @@ -40,8 +40,8 @@ fields: [when="HPM_COUNTER_EN[<%= hpm_num %>] == false"] Unimplemented performance counter. Must be read-only 0 (access does not cause trap). - type(): 'return (HPM_COUNTER_EN[<%= hpm_num %>]) ? CsrFieldType::RWH : CsrFieldType::RO;' - reset_value(): 'return (HPM_COUNTER_EN[<%= hpm_num %>]) ? UNDEFINED_LEGAL : 0;' + type(): "return (HPM_COUNTER_EN[<%= hpm_num %>]) ? CsrFieldType::RWH : CsrFieldType::RO;" + reset_value(): "return (HPM_COUNTER_EN[<%= hpm_num %>]) ? UNDEFINED_LEGAL : 0;" sw_read(): | if (HPM_COUNTER_EN[<%= hpm_num %>]) { return read_hpm_counter(<%= hpm_num %>); diff --git a/arch/csr/Zihpm/mhpmevent10.yaml b/arch/csr/Zihpm/mhpmevent10.yaml index 52359704c..5c5aa0319 100644 --- a/arch/csr/Zihpm/mhpmevent10.yaml +++ b/arch/csr/Zihpm/mhpmevent10.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent11.yaml b/arch/csr/Zihpm/mhpmevent11.yaml index f07693ca2..ab7ef79ba 100644 --- a/arch/csr/Zihpm/mhpmevent11.yaml +++ b/arch/csr/Zihpm/mhpmevent11.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent12.yaml b/arch/csr/Zihpm/mhpmevent12.yaml index 4233c540b..d9723cf65 100644 --- a/arch/csr/Zihpm/mhpmevent12.yaml +++ b/arch/csr/Zihpm/mhpmevent12.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent13.yaml b/arch/csr/Zihpm/mhpmevent13.yaml index b6691e172..abbad38d8 100644 --- a/arch/csr/Zihpm/mhpmevent13.yaml +++ b/arch/csr/Zihpm/mhpmevent13.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent14.yaml b/arch/csr/Zihpm/mhpmevent14.yaml index ed534b841..13e26ba6b 100644 --- a/arch/csr/Zihpm/mhpmevent14.yaml +++ b/arch/csr/Zihpm/mhpmevent14.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent15.yaml b/arch/csr/Zihpm/mhpmevent15.yaml index 2140c2a97..aac17d83c 100644 --- a/arch/csr/Zihpm/mhpmevent15.yaml +++ b/arch/csr/Zihpm/mhpmevent15.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent16.yaml b/arch/csr/Zihpm/mhpmevent16.yaml index 7b041c5e8..bae1d8402 100644 --- a/arch/csr/Zihpm/mhpmevent16.yaml +++ b/arch/csr/Zihpm/mhpmevent16.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent17.yaml b/arch/csr/Zihpm/mhpmevent17.yaml index 698dde435..402981d78 100644 --- a/arch/csr/Zihpm/mhpmevent17.yaml +++ b/arch/csr/Zihpm/mhpmevent17.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent18.yaml b/arch/csr/Zihpm/mhpmevent18.yaml index c6f2dfc40..102925471 100644 --- a/arch/csr/Zihpm/mhpmevent18.yaml +++ b/arch/csr/Zihpm/mhpmevent18.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent19.yaml b/arch/csr/Zihpm/mhpmevent19.yaml index 39dd37ef6..68bbb178f 100644 --- a/arch/csr/Zihpm/mhpmevent19.yaml +++ b/arch/csr/Zihpm/mhpmevent19.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent20.yaml b/arch/csr/Zihpm/mhpmevent20.yaml index 2c6645a79..718d2aaab 100644 --- a/arch/csr/Zihpm/mhpmevent20.yaml +++ b/arch/csr/Zihpm/mhpmevent20.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent21.yaml b/arch/csr/Zihpm/mhpmevent21.yaml index 257aec1e1..bbe47e8fb 100644 --- a/arch/csr/Zihpm/mhpmevent21.yaml +++ b/arch/csr/Zihpm/mhpmevent21.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent22.yaml b/arch/csr/Zihpm/mhpmevent22.yaml index 4731ba4e7..9109a7e26 100644 --- a/arch/csr/Zihpm/mhpmevent22.yaml +++ b/arch/csr/Zihpm/mhpmevent22.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent23.yaml b/arch/csr/Zihpm/mhpmevent23.yaml index cc78b7c59..f07456c81 100644 --- a/arch/csr/Zihpm/mhpmevent23.yaml +++ b/arch/csr/Zihpm/mhpmevent23.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent24.yaml b/arch/csr/Zihpm/mhpmevent24.yaml index cd86cfa8f..63bbfe972 100644 --- a/arch/csr/Zihpm/mhpmevent24.yaml +++ b/arch/csr/Zihpm/mhpmevent24.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent25.yaml b/arch/csr/Zihpm/mhpmevent25.yaml index 79ce58cbd..e6451315d 100644 --- a/arch/csr/Zihpm/mhpmevent25.yaml +++ b/arch/csr/Zihpm/mhpmevent25.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent26.yaml b/arch/csr/Zihpm/mhpmevent26.yaml index 3537d02c3..952d80903 100644 --- a/arch/csr/Zihpm/mhpmevent26.yaml +++ b/arch/csr/Zihpm/mhpmevent26.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent27.yaml b/arch/csr/Zihpm/mhpmevent27.yaml index 339b1afc4..48036d557 100644 --- a/arch/csr/Zihpm/mhpmevent27.yaml +++ b/arch/csr/Zihpm/mhpmevent27.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent28.yaml b/arch/csr/Zihpm/mhpmevent28.yaml index c38cdcab0..39fe839ad 100644 --- a/arch/csr/Zihpm/mhpmevent28.yaml +++ b/arch/csr/Zihpm/mhpmevent28.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent29.yaml b/arch/csr/Zihpm/mhpmevent29.yaml index 393ec1fd5..add075a81 100644 --- a/arch/csr/Zihpm/mhpmevent29.yaml +++ b/arch/csr/Zihpm/mhpmevent29.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent3.yaml b/arch/csr/Zihpm/mhpmevent3.yaml index 64006dee4..dc92768d8 100644 --- a/arch/csr/Zihpm/mhpmevent3.yaml +++ b/arch/csr/Zihpm/mhpmevent3.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent30.yaml b/arch/csr/Zihpm/mhpmevent30.yaml index 17a7b1a94..f7a09d917 100644 --- a/arch/csr/Zihpm/mhpmevent30.yaml +++ b/arch/csr/Zihpm/mhpmevent30.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent31.yaml b/arch/csr/Zihpm/mhpmevent31.yaml index 281363e27..aa66ff87c 100644 --- a/arch/csr/Zihpm/mhpmevent31.yaml +++ b/arch/csr/Zihpm/mhpmevent31.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent4.yaml b/arch/csr/Zihpm/mhpmevent4.yaml index 4c42d7c2d..d4880a960 100644 --- a/arch/csr/Zihpm/mhpmevent4.yaml +++ b/arch/csr/Zihpm/mhpmevent4.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent5.yaml b/arch/csr/Zihpm/mhpmevent5.yaml index a5dcd5c67..d373fe3bf 100644 --- a/arch/csr/Zihpm/mhpmevent5.yaml +++ b/arch/csr/Zihpm/mhpmevent5.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent6.yaml b/arch/csr/Zihpm/mhpmevent6.yaml index a9bdf1c39..0ab1ee66b 100644 --- a/arch/csr/Zihpm/mhpmevent6.yaml +++ b/arch/csr/Zihpm/mhpmevent6.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent7.yaml b/arch/csr/Zihpm/mhpmevent7.yaml index 2cc53fd9b..15c5c3ca5 100644 --- a/arch/csr/Zihpm/mhpmevent7.yaml +++ b/arch/csr/Zihpm/mhpmevent7.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent8.yaml b/arch/csr/Zihpm/mhpmevent8.yaml index 769eb3ab1..cc9d7e4f8 100644 --- a/arch/csr/Zihpm/mhpmevent8.yaml +++ b/arch/csr/Zihpm/mhpmevent8.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/Zihpm/mhpmevent9.yaml b/arch/csr/Zihpm/mhpmevent9.yaml index 1674e0c64..b0deeed6e 100644 --- a/arch/csr/Zihpm/mhpmevent9.yaml +++ b/arch/csr/Zihpm/mhpmevent9.yaml @@ -1,6 +1,4 @@ -# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout - -# yaml-language-server: $schema=../../../schemas/csr_schema.json +# WARNING: This file is auto-generated from arch/csr/Zihpm/mhpmeventN.layout# yaml-language-server: $schema=../../../schemas/csr_schema.json $schema: csr_schema.json# kind: csr diff --git a/arch/csr/vstvec.yaml b/arch/csr/vstvec.yaml index a74254e64..4357ac589 100644 --- a/arch/csr/vstvec.yaml +++ b/arch/csr/vstvec.yaml @@ -39,8 +39,10 @@ fields: When Vectored, asynchronous interrupts jump to (`vstvec.BASE` << 2 + `vscause.CAUSE`*4) while synchronous exceptions continue to jump to (`vstvec.BASE` << 2). type: RW-R sw_write(csr_value): | - if (csr_value.MODE == 0 || csr_value.MODE == 1) { - return csr_value.MODE; + if (VSTVEC_MODE_DIRECT && csr_value.MODE == 0) { + return 0; + } else if (VSTVEC_MODE_VECTORED && csr_value.MODE == 1) { + return 1; } else { return UNDEFINED_LEGAL_DETERMINISTIC; } diff --git a/arch/ext/C.yaml b/arch/ext/C.yaml index 6020da884..bc1e427c1 100644 --- a/arch/ext/C.yaml +++ b/arch/ext/C.yaml @@ -12,7 +12,7 @@ doc_license: name: Creative Commons Attribution 4.0 International License url: https://creativecommons.org/licenses/by/4.0/ versions: - - version: "2.2.0" + - version: "2.0.0" state: ratified ratification_date: 2019-12 description: | diff --git a/arch/ext/H.yaml b/arch/ext/H.yaml index e63b9863b..0c73ea648 100644 --- a/arch/ext/H.yaml +++ b/arch/ext/H.yaml @@ -311,6 +311,14 @@ params: When false `vstval` is written with 0 when an `IllegalInstruction` exception occurs. schema: type: boolean + REPORT_GPA_IN_HTVAL_ON_GUEST_PAGE_FAULT: + description: | + When true, `htval` is written with the Guest Physical Address, shifted right by 2, that + caused a `GuestPageFault` exception. + + When false, `htval` is written with0 when a `GuestPageFault` exception occurs. + schema: + type: boolean HCOUNTENABLE_EN: description: | Indicates which counters can delegated via `hcounteren` @@ -637,3 +645,15 @@ params: schema: type: boolean default: true + VSTVEC_MODE_DIRECT: + description: | + Whether or not `vstvec.MODE` supports Direct (0). + schema: + type: boolean + extra_validation: assert STVEC_MODE_DIRECT || STVEC_MODE_VECTORED + VSTVEC_MODE_VECTORED: + description: | + Whether or not `stvec.MODE` supports Vectored (1). + schema: + type: boolean + extra_validation: assert STVEC_MODE_DIRECT || STVEC_MODE_VECTORED diff --git a/arch/ext/Q.yaml b/arch/ext/Q.yaml new file mode 100644 index 000000000..0f57f5717 --- /dev/null +++ b/arch/ext/Q.yaml @@ -0,0 +1,20 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Q +long_name: Quad-Precision Floating-Point +description: | + 128-bit quad-precision binary floating-point instructions compliant with the IEEE 754-2008 + arithmetic standard. + `Q` depends on the double-precision floating-point extension `D`. + With `Q`, the floating-point registers are extended to hold either a single, double, or quad-precision + floating-point value (FLEN=128). + The NaN-boxing is extended recursively to allow a single-precision value to be NaN-boxed inside + a double-precision value which is itself NaN-boxed inside a quad-precision value. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + requires: D diff --git a/arch/ext/Sdext.yaml b/arch/ext/Sdext.yaml new file mode 100644 index 000000000..8cadccac1 --- /dev/null +++ b/arch/ext/Sdext.yaml @@ -0,0 +1,13 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Sdext +long_name: Debug +description: | + Hart-visible portion of the debug spec. +type: privileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Shgatpa.yaml b/arch/ext/Shgatpa.yaml new file mode 100644 index 000000000..346ed0023 --- /dev/null +++ b/arch/ext/Shgatpa.yaml @@ -0,0 +1,34 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Shgatpa +long_name: hgtap profile requirements +description: | + For each supported virtual memory scheme SvNN supported in + `satp`, the corresponding hgatp SvNNx4 mode must be supported. The + `hgatp` mode Bare must also be supported. + + [NOTE] + This extension was ratified with the RVA22 profiles. +type: privileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + param_constraints: + SV32X4_TRANSLATION: + extra_validation: | + (SV32X4_TRANSLATION && ext?(:Sv32)) || (!SV32X4_TRANSLATION && !ext?(:Sv32)) + SV39X4_TRANSLATION: + extra_validation: | + (SV39X4_TRANSLATION && ext?(:Sv39)) || (!SV39X4_TRANSLATION && !ext?(:Sv39)) + SV48X4_TRANSLATION: + extra_validation: | + (SV48X4_TRANSLATION && ext?(:Sv48)) || (!SV48X4_TRANSLATION && !ext?(:Sv48)) + SV57X4_TRANSLATION: + extra_validation: | + (SV57X4_TRANSLATION && ext?(:Sv57)) || (!SV57X4_TRANSLATION && !ext?(:Sv57)) + GSTAGE_MODE_BARE: + schema: + const: true diff --git a/arch/ext/Shtvala.yaml b/arch/ext/Shtvala.yaml new file mode 100644 index 000000000..db95a2ab3 --- /dev/null +++ b/arch/ext/Shtvala.yaml @@ -0,0 +1,25 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Shtvala +long_name: htval profile requirements +description: | + htval must be written with the faulting virtual address + for load, store, and instruction page-fault, access-fault, and + misaligned exceptions, and for breakpoint exceptions other than + those caused by execution of the `ebreak` or `c.ebreak` instructions. + For virtual-instruction and illegal-instruction exceptions, htval must be written with the + faulting instruction. + + [NOTE] + This extension was ratified with the RVA22 profiles. +type: privileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + param_constraints: + REPORT_GPA_IN_HTVAL_ON_GUEST_PAGE_FAULT: + schema: + const: true diff --git a/arch/ext/Shvstvala.yaml b/arch/ext/Shvstvala.yaml new file mode 100644 index 000000000..e01f9f1fc --- /dev/null +++ b/arch/ext/Shvstvala.yaml @@ -0,0 +1,55 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Shvstvala +long_name: vstval profile requirements +description: | + vstval must be written with the faulting virtual address + for load, store, and instruction page-fault, access-fault, and + misaligned exceptions, and for breakpoint exceptions other than + those caused by execution of the `ebreak` or `c.ebreak` instructions. + For virtual-instruction and illegal-instruction exceptions, vstval must be written with the + faulting instruction. + + [NOTE] + This extension was ratified with the RVA22 profiles. +type: privileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + param_constraints: + REPORT_VA_IN_VSTVAL_ON_BREAKPOINT: + schema: + const: true + REPORT_VA_IN_VSTVAL_ON_LOAD_MISALIGNED: + schema: + const: true + REPORT_VA_IN_VSTVAL_ON_STORE_AMO_MISALIGNED: + schema: + const: true + REPORT_VA_IN_VSTVAL_ON_INSTRUCTION_MISALIGNED: + schema: + const: true + REPORT_VA_IN_VSTVAL_ON_LOAD_ACCESS_FAULT: + schema: + const: true + REPORT_VA_IN_VSTVAL_ON_STORE_AMO_ACCESS_FAULT: + schema: + const: true + REPORT_VA_IN_VSTVAL_ON_INSTRUCTION_ACCESS_FAULT: + schema: + const: true + REPORT_VA_IN_VSTVAL_ON_LOAD_PAGE_FAULT: + schema: + const: true + REPORT_VA_IN_VSTVAL_ON_STORE_AMO_PAGE_FAULT: + schema: + const: true + REPORT_VA_IN_VSTVAL_ON_INSTRUCTION_PAGE_FAULT: + schema: + const: true + REPORT_ENCODING_IN_VSTVAL_ON_ILLEGAL_INSTRUCTION: + schema: + const: true diff --git a/arch/ext/Shvstvecd.yaml b/arch/ext/Shvstvecd.yaml new file mode 100644 index 000000000..04fa927fd --- /dev/null +++ b/arch/ext/Shvstvecd.yaml @@ -0,0 +1,22 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Shvstvecd +long_name: vstvec profile requirements +description: | + `vstvec.MODE` must be capable of holding the value 0 (Direct). + When `vstvec.MODE`=Direct, `vstvec.BASE` must be capable of holding + any valid four-byte-aligned address. + + [NOTE] + This extension was ratified with the RVA22 profiles. +type: privileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + param_constraints: + VSTVEC_MODE_DIRECT: + schema: + const: true diff --git a/arch/ext/Smdbltrp.yaml b/arch/ext/Smdbltrp.yaml new file mode 100644 index 000000000..0f1b1227b --- /dev/null +++ b/arch/ext/Smdbltrp.yaml @@ -0,0 +1,22 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Smdbltrp +long_name: Double trap +description: | + The `Smdbltrp` extension addresses a double trap in M-mode. + When the `Smrnmi` extension is implemented, it enables invocation of the RNMI handler on a + double trap in M-mode to handle the critical error. + If the `Smrnmi` extension is not implemented or if a double trap occurs during the RNMI + handler’s execution, this extension helps transition the hart to a critical error state and + enables signaling the critical error to the platform. + + To improve error diagnosis and resolution, this extension supports debugging harts in a critical + error state. The extension introduces a mechanism to enter Debug Mode instead of asserting a + critical-error signal to the platform when the hart is in a critical error state. +type: privileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Smrnmi.yaml b/arch/ext/Smrnmi.yaml new file mode 100644 index 000000000..51c13b81a --- /dev/null +++ b/arch/ext/Smrnmi.yaml @@ -0,0 +1,21 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Smrnmi +long_name: Resumable Non-Maskable Interrupts +description: | + The base machine-level architecture supports only unresumable non-maskable interrupts (UNMIs), + where the NMI jumps to a handler in machine mode, overwriting the current `mepc` and `mcause` + register values. + If the hart had been executing machine-mode code in a trap handler, the previous values in `mepc` + and `mcause` would not be recoverable and so execution is not generally resumable. + + The `Smrnmi` extension adds support for resumable non-maskable interrupts (RNMIs) to RISC-V. + The extension adds four new CSRs (`mnepc`, `mncause`, `mnstatus`, and `mnscratch`) to hold the + interrupted state, and one new instruction, `mnret`, to resume from the RNMI handler. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Ssstateen.yaml b/arch/ext/Ssstateen.yaml new file mode 100644 index 000000000..f63d5e823 --- /dev/null +++ b/arch/ext/Ssstateen.yaml @@ -0,0 +1,28 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Ssstateen +long_name: Supervisor-mode view of the state-enable extension +description: | + Supervisor-mode view of the state-enable extension. The + supervisor-mode (`sstateen0-3`) and hypervisor-mode (`hstateen0-3`) + state-enable registers must be provided. + + NOTE: The Smstateen extension specification is an M-mode extension as + it includes M-mode features, but the supervisor-mode visible + components of the extension are named as the Ssstateen extension. Only + Ssstateen is mandated in the RVA22S64 profile when the hypervisor + extension is implemented. These registers are not mandated or + supported options without the hypervisor extension, as there are no + RVA22S64 supported options with relevant state to control in the + absence of the hypervisor extension. + + [NOTE] + This extension was ratified with the RVA22 profiles. +type: privileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + # TODO: add param_constraints diff --git a/arch/ext/Ssu64xl.yaml b/arch/ext/Ssu64xl.yaml new file mode 100644 index 000000000..621ac189e --- /dev/null +++ b/arch/ext/Ssu64xl.yaml @@ -0,0 +1,20 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Ssu64xl +long_name: 64-bit UXLEN +description: | + `sstatus.UXL` must be capable of holding the value 2 (i.e., UXLEN=64 must be supported). + + [NOTE] + This extension is defined by RVA22. +type: privileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + param_constraints: + UXLEN: + schema: + enum: [64, 3264] diff --git a/arch/ext/MockExt.yaml b/arch/ext/Xmock.yaml similarity index 99% rename from arch/ext/MockExt.yaml rename to arch/ext/Xmock.yaml index 594bfbd4a..db1c0e0b7 100644 --- a/arch/ext/MockExt.yaml +++ b/arch/ext/Xmock.yaml @@ -2,7 +2,7 @@ $schema: "ext_schema.json#" kind: extension -name: MockExt +name: Xmock type: privileged long_name: Mock Extension (for testing database) description: This is just for testing diff --git a/arch/ext/Zabha.yaml b/arch/ext/Zabha.yaml new file mode 100644 index 000000000..62ebb54ad --- /dev/null +++ b/arch/ext/Zabha.yaml @@ -0,0 +1,14 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zabha +long_name: Byte and Halfword Atomic Memory Operations +description: | + Adds byte and halfword atomic memory operations to the RISC-V Unprivileged ISA. + type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + requires: Zaamo diff --git a/arch/ext/Zacas.yaml b/arch/ext/Zacas.yaml new file mode 100644 index 000000000..96764df7c --- /dev/null +++ b/arch/ext/Zacas.yaml @@ -0,0 +1,14 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zacas +long_name: Atomic Compare-and-Swap (CAS) Instructions +description: | + Adds Word/Doubleword/Quadword compare-and-swap instructions. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + requires: Zaamo diff --git a/arch/ext/Zalasr.yaml b/arch/ext/Zalasr.yaml new file mode 100644 index 000000000..d0df726b7 --- /dev/null +++ b/arch/ext/Zalasr.yaml @@ -0,0 +1,13 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zalasr +long_name: Atomic, Load-Acquire Store-Release +description: | + load-acquire and store-release instructions. +type: unprivileged +versions: + - version: "0.3.5" + state: development + ratification_date: null diff --git a/arch/ext/Zawrs.yaml b/arch/ext/Zawrs.yaml new file mode 100644 index 000000000..26bf97203 --- /dev/null +++ b/arch/ext/Zawrs.yaml @@ -0,0 +1,22 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zawrs +long_name: Wait-on-Reservation-Set Instructions +description: | + The `Zawrs` extension defines a pair of instructions to be used in polling loops that allows a + core to enter a low-power state and wait on a store to a memory location. + Waiting for a memory location to be updated is a common pattern in many use cases such as: + + * Contenders for a lock waiting for the lock variable to be updated. + * Consumers waiting on the tail of an empty queue for the producer to queue work/data. + The producer may be code executing on a RISC-V hart, an accelerator device, an external I/O agent. + * Code waiting on a flag to be set in memory indicative of an event occurring. + For example, software on a RISC-V hart may wait on a "done" flag to be set in memory by an + accelerator device indicating completion of a job previously submitted to the device. +type: unprivileged +versions: + - version: "1.1.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zbkb.yaml b/arch/ext/Zbkb.yaml new file mode 100644 index 000000000..a0a606d4d --- /dev/null +++ b/arch/ext/Zbkb.yaml @@ -0,0 +1,12 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zbkb +long_name: Bit-manipulation for Cryptography +description: This extension contains instructions essential for implementing common operations in cryptographic workloads. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zbkc.yaml b/arch/ext/Zbkc.yaml new file mode 100644 index 000000000..8913375ea --- /dev/null +++ b/arch/ext/Zbkc.yaml @@ -0,0 +1,17 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zbkc +long_name: Carry-less multiplication for Cryptography +description: | + Carry-less multiplication is the multiplication in the polynomial ring over GF(2). + This is a critical operation in some cryptographic workloads, particularly the AES-GCM + authenticated encryption scheme. + This extension provides only the instructions needed to efficiently implement the GHASH operation, + which is part of this workload. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zbkx.yaml b/arch/ext/Zbkx.yaml new file mode 100644 index 000000000..07b373bce --- /dev/null +++ b/arch/ext/Zbkx.yaml @@ -0,0 +1,18 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zbkx +long_name: Crossbar permutations +description: | + Adds instructions to implement a "lookup table" for 4 and 8 bit elements inside the general purpose + registers. + + These instructions are useful for expressing N-bit to N-bit boolean operations, and implementing + cryptographic code with secret dependent memory accesses (particularly SBoxes) such that the + execution latency does not depend on the (secret) data being operated on. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zca.yaml b/arch/ext/Zca.yaml new file mode 100644 index 000000000..7dbcb29c6 --- /dev/null +++ b/arch/ext/Zca.yaml @@ -0,0 +1,43 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zca +long_name: C instructions excluding floating-point +description: | + The Zca extension is added as way to refer to instructions in the `C` extension that do not + include the floating-point loads and stores. + + Therefore it excludes all 16-bit floating point loads and stores: + `c.flw`, `c.flwsp`, `c.fsw`, `c.fswsp`, `c.fld`, `c.fldsp`, `c.fsd`, `c.fsdsp`. + + [NOTE] + The 'C' extension only includes `F`/`D` instructions when `D` and `F` are also specified. + +type: unprivileged +company: + name: RISC-V International + url: https://riscv.org +versions: + - version: "1.0.0" + state: ratified + ratification_date: 2023-04 + repositories: + - url: https://github.com/riscv/riscv-code-size-reduction + branch: main + contributors: + - name: Tariq Kurd + - name: Ibrahim Abu Kharmeh + - name: Torbjørn Viem Ness + - name: Matteo Perotti + - name: Nidal Faour + - name: Bill Traynor + - name: Rafael Sene + - name: Xinlong Wu + - name: sinan + - name: Jeremy Bennett + - name: Heda Chen + - name: Alasdair Armstrong + - name: Graeme Smecher + - name: Nicolas Brunie + - name: Jiawei diff --git a/arch/ext/Zce.yaml b/arch/ext/Zce.yaml new file mode 100644 index 000000000..442018759 --- /dev/null +++ b/arch/ext/Zce.yaml @@ -0,0 +1,75 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zce +long_name: Compressed instructions for microcontrollers +description: | + The Zce extension is intended to be used for microcontrollers, and includes all relevant Zc + extensions. + + * Specifying `Zce` on RV32 without `F` includes `Zca`, `Zcb`, `Zcmp`, `Zcmt` + * Specifying `Zce` on RV32 with `F` includes `Zca`, `Zcb`, `Zcmp`, `Zcmt` and `Zcf` + * Specifying `Zce` on RV64 always includes `Zca`, `Zcb`, `Zcmp`, `Zcmt` + * `Zcf` doesn’t exist for RV64 + + Therefore common ISA strings can be updated as follows to include the relevant Zc extensions, for + example: + + * RV32IMC becomes RV32IM_Zce + * RV32IMCF becomes RV32IMF_Zce + +type: unprivileged +company: + name: RISC-V International + url: https://riscv.org +versions: + - version: "1.0.0" + state: ratified + ratification_date: 2023-04 + repositories: + - url: https://github.com/riscv/riscv-code-size-reduction + branch: main + contributors: + - name: Tariq Kurd + - name: Ibrahim Abu Kharmeh + - name: Torbjørn Viem Ness + - name: Matteo Perotti + - name: Nidal Faour + - name: Bill Traynor + - name: Rafael Sene + - name: Xinlong Wu + - name: sinan + - name: Jeremy Bennett + - name: Heda Chen + - name: Alasdair Armstrong + - name: Graeme Smecher + - name: Nicolas Brunie + - name: Jiawei + implies: + - [Zca, "1.0.0"] + - [Zcb, "1.0.0"] + - [Zcmp, "1.0.0"] + - [Zcmt, "1.0.0"] + + # TODO: this implication is conditional!!! (see description) + # So it should look something like this: + + # if: + # allOf: + # param: + # XLEN: 32 + # extensions: + # - F + # then: + # - [Zca, "1.0.0"] + # - [Zcb, "1.0.0"] + # - [Zcmp, "1.0.0"] + # - [Zcmt, "1.0.0"] + # - [Zf, "1.0.0"] + # else: + # # TODO: this implication is conditional!!! (see description) + # - [Zca, "1.0.0"] + # - [Zcb, "1.0.0"] + # - [Zcmp, "1.0.0"] + # - [Zcmt, "1.0.0"] diff --git a/arch/ext/Zfa.yaml b/arch/ext/Zfa.yaml new file mode 100644 index 000000000..dece4a87b --- /dev/null +++ b/arch/ext/Zfa.yaml @@ -0,0 +1,19 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zfa +long_name: Extension for Additional Floating-Point Instructions +description: | + `Zfa` adds instructions for immediate loads, IEEE 754-2019 minimum and maximum operations, + round-to-integer operations, and quiet floating-point comparisons. + For RV32D, the `Zfa` extension also adds instructions to transfer double-precision floating-point + values to and from integer registers, and for RV64Q, it adds analogous instructions for + quad-precision floating-point values. + The `Zfa` extension depends on the `F` extension. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + requires: F diff --git a/arch/ext/Zfbfmin.yaml b/arch/ext/Zfbfmin.yaml new file mode 100644 index 000000000..82fb7bd6d --- /dev/null +++ b/arch/ext/Zfbfmin.yaml @@ -0,0 +1,23 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zfbfmin +long_name: Scalar BF16 Converts +description: | + The minimal set of instructions needed to enable scalar support of the BF16 format. + It enables BF16 as an interchange format as it provides conversion between BF16 values and + FP32 values. + + This extension depends upon the single-precision floating-point extension `F`, + and the `flh`, `fsh`, `fmv.x.h`, and `fmv.h.x` instructions as defined in the `Zfh` extension. + +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + requires: + allOf: + - name: F + - name: Zfh diff --git a/arch/ext/Zfh.yaml b/arch/ext/Zfh.yaml new file mode 100644 index 000000000..6e931bfc6 --- /dev/null +++ b/arch/ext/Zfh.yaml @@ -0,0 +1,18 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zfh +long_name: Half-precision floating point +description: | + 16-bit half-precision binary floating-point instructions compliant with the IEEE 754-2008 + arithmetic standard. + The `Zfh` extension depends on the single-precision floating-point extension, `F`. + The NaN-boxing scheme is extended to allow a half-precision value to be NaN-boxed inside a + single-precision value (which may be recursively NaN-boxed inside a double- or quad-precision + value when the D or Q extension is present). +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zicbom.yaml b/arch/ext/Zicbom.yaml index f874010f7..c040ce139 100644 --- a/arch/ext/Zicbom.yaml +++ b/arch/ext/Zicbom.yaml @@ -7,7 +7,7 @@ long_name: Cache block management instructions description: Cache block management instructions type: unprivileged versions: - - version: "1.0.1-b34ea8a" + - version: "1.0.0" state: ratified ratification_date: 2022-05 params: diff --git a/arch/ext/Zicbop.yaml b/arch/ext/Zicbop.yaml index 50fa8e862..c25d22c90 100644 --- a/arch/ext/Zicbop.yaml +++ b/arch/ext/Zicbop.yaml @@ -7,7 +7,7 @@ long_name: Cache block prefetch description: Cache block prefetch instruction type: unprivileged versions: - - version: 1.0.1-b34ea8a + - version: "1.0.0" state: ratified ratification_date: 2022-05 params: diff --git a/arch/ext/Zicboz.yaml b/arch/ext/Zicboz.yaml index c3ce2ab69..a6d4c9c17 100644 --- a/arch/ext/Zicboz.yaml +++ b/arch/ext/Zicboz.yaml @@ -7,7 +7,7 @@ long_name: Cache block zero instruction description: Cache block zero instruction type: unprivileged versions: - - version: 1.0.1-b34ea8a + - version: "1.0.0" state: ratified ratification_date: 2022-05 params: diff --git a/arch/ext/Zicond.yaml b/arch/ext/Zicond.yaml new file mode 100644 index 000000000..e69fef548 --- /dev/null +++ b/arch/ext/Zicond.yaml @@ -0,0 +1,21 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zicond +long_name: Integer Conditional Operations +description: | + The "Conditional" operations extension provides a simple solution that provides most of the + benefit and all of the flexibility one would desire to support conditional arithmetic and + conditional-select/move operations, while remaining true to the RISC-V design philosophy. + The instructions follow the format for R-type instructions with 3 operands (_i.e._, 2 source + operands and 1 destination operand). + + Using these instructions, branchless sequences can be implemented (typically in two-instruction + sequences) without the need for instruction fusion, special provisions during the decoding of + architectural instructions, or other microarchitectural provisions. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zimop.yaml b/arch/ext/Zimop.yaml new file mode 100644 index 000000000..f67973291 --- /dev/null +++ b/arch/ext/Zimop.yaml @@ -0,0 +1,16 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zimop +long_name: May-be Operations +description: | + Instructions that may be operations (MOPs). + MOPs are initially defined to simply write zero to x[rd], but are designed to be redefined by + later extensions to perform some other action. + The Zimop extension defines an encoding space for 40 MOPs. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zk.yaml b/arch/ext/Zk.yaml new file mode 100644 index 000000000..6ffeaf47c --- /dev/null +++ b/arch/ext/Zk.yaml @@ -0,0 +1,22 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zk +long_name: Standard Cryptography Extensions +description: | + This extension is shorthand for the following set of other extensions: + + * `Zkn` + * `Zkr` + * `Zkt` + +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + implies: + - ["Zkn", "1.0.0"] + - ["Zkr", "1.0.0"] + - ["Zkt", "1.0.0"] diff --git a/arch/ext/Zkn.yaml b/arch/ext/Zkn.yaml new file mode 100644 index 000000000..1ec5a3b46 --- /dev/null +++ b/arch/ext/Zkn.yaml @@ -0,0 +1,28 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zkn +long_name: NIST Algorithm Suite +description: | + This extension is shorthand for the following set of other extensions: + + * `Zbkb` + * `Zbkc` + * `Zbkx` + * `Zkne` + * `Zknd` + * `Zknh` + +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + implies: + - ["Zbkb", "1.0.0"] + - ["Zbkc", "1.0.0"] + - ["Zbkx", "1.0.0"] + - ["Zkne", "1.0.0"] + - ["Zknd", "1.0.0"] + - ["Zknh", "1.0.0"] diff --git a/arch/ext/Zknd.yaml b/arch/ext/Zknd.yaml new file mode 100644 index 000000000..413d02f91 --- /dev/null +++ b/arch/ext/Zknd.yaml @@ -0,0 +1,13 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zknd +long_name: "NIST Suite: AES Decryption" +description: | + Instructions for accelerating the decryption and key-schedule functions of the AES block cipher. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zkne.yaml b/arch/ext/Zkne.yaml new file mode 100644 index 000000000..d0ec03c6c --- /dev/null +++ b/arch/ext/Zkne.yaml @@ -0,0 +1,13 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zkne +long_name: "NIST Suite: AES Encryption" +description: | + Instructions for accelerating the encryption and key-schedule functions of the AES block cipher. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zknh.yaml b/arch/ext/Zknh.yaml new file mode 100644 index 000000000..92f64a174 --- /dev/null +++ b/arch/ext/Zknh.yaml @@ -0,0 +1,13 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zknh +long_name: "NIST Suite: Hash Function Instructions" +description: | + Instructions for accelerating the SHA2 family of cryptographic hash functions. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zkr.yaml b/arch/ext/Zkr.yaml new file mode 100644 index 000000000..35e7fc709 --- /dev/null +++ b/arch/ext/Zkr.yaml @@ -0,0 +1,14 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zkr +long_name: Entropy Source +description: | + Defines the `seed` CSR. + This CSR provides up to 16 physical entropy bits that can be used to seed cryptographic random bit generators. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zks.yaml b/arch/ext/Zks.yaml new file mode 100644 index 000000000..33f6c6917 --- /dev/null +++ b/arch/ext/Zks.yaml @@ -0,0 +1,27 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zks +long_name: ShangMi Algorithm Suite +description: | + This extension is shorthand for the following set of other extensions: + + * `Zbkb` + * `Zbkc` + * `Zbkx` + * `Zksed` + * `Zksh` + +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + implies: + - ["Zbkb", "1.0.0"] + - ["Zbkc", "1.0.0"] + - ["Zbkx", "1.0.0"] + - ["Zkne", "1.0.0"] + - ["Zknd", "1.0.0"] + - ["Zknh", "1.0.0"] diff --git a/arch/ext/Zksed.yaml b/arch/ext/Zksed.yaml new file mode 100644 index 000000000..d0e862020 --- /dev/null +++ b/arch/ext/Zksed.yaml @@ -0,0 +1,14 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zksed +long_name: "ShangMi Suite: SM4 Block Cipher Instructions" +description: | + Instructions for accelerating the SM4 Block Cipher. Note that unlike AES, this cipher uses the + same core operation for encryption and decryption, hence there is only one extension for it. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zksh.yaml b/arch/ext/Zksh.yaml new file mode 100644 index 000000000..983a1b415 --- /dev/null +++ b/arch/ext/Zksh.yaml @@ -0,0 +1,13 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zksh +long_name: "ShangMi Suite: SM3 Hash Function Instructions" +description: | + Instructions for accelerating the SM3 hash function. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zvbb.yaml b/arch/ext/Zvbb.yaml new file mode 100644 index 000000000..3ceccfe1e --- /dev/null +++ b/arch/ext/Zvbb.yaml @@ -0,0 +1,14 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvbb +long_name: Vector Basic Bit-manipulation +description: | + Vector basic bit-manipulation instructions. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + implies: [Zvkb, "1.0.0"] diff --git a/arch/ext/Zvbc.yaml b/arch/ext/Zvbc.yaml new file mode 100644 index 000000000..e9d001d01 --- /dev/null +++ b/arch/ext/Zvbc.yaml @@ -0,0 +1,16 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvbc +long_name: Vector Carryless Multiplication +description: | + General purpose carryless multiplication instructions which are commonly used in cryptography and + hashing (e.g., Elliptic curve cryptography, GHASH, CRC). + + These instructions are only defined for SEW=64. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zvfbfmin.yaml b/arch/ext/Zvfbfmin.yaml new file mode 100644 index 000000000..a1eba800e --- /dev/null +++ b/arch/ext/Zvfbfmin.yaml @@ -0,0 +1,21 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvfbfmin +long_name: Vector BF16 Converts +description: | + This extension provides the minimal set of instructions needed to enable vector support of the + BF16 format. + It enables BF16 as an interchange format as it provides conversion between BF16 values and FP32 values. + + This extension depends upon either the `V` extension or the `Zve32f` embedded vector extension. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + requires: + anyOf: + - V + - Zve32f diff --git a/arch/ext/Zvfbfwma.yaml b/arch/ext/Zvfbfwma.yaml new file mode 100644 index 000000000..ad062ba54 --- /dev/null +++ b/arch/ext/Zvfbfwma.yaml @@ -0,0 +1,19 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvfbfwma +long_name: Vector BF16 widening mul-add +description: | + This extension provides a vector widening BF16 mul-add instruction that accumulates into FP32. + + This extension depends upon the `Zvfbfmin` extension and the `Zfbfmin` extension. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + requires: + allOf: + - Zvfbfmin + - Zfbfmin diff --git a/arch/ext/Zvkb.yaml b/arch/ext/Zvkb.yaml new file mode 100644 index 000000000..8809074e0 --- /dev/null +++ b/arch/ext/Zvkb.yaml @@ -0,0 +1,13 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvkb +long_name: Vector Cryptography Bit-manipulation +description: | + Vector bit-manipulation instructions that are essential for implementing common cryptographic workloads securely & efficiently. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zvkg.yaml b/arch/ext/Zvkg.yaml new file mode 100644 index 000000000..042e78acc --- /dev/null +++ b/arch/ext/Zvkg.yaml @@ -0,0 +1,23 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvkg +long_name: Vector GCM/GMAC +description: | + Instructions to enable the efficient implementation of GHASHH which is used in Galois/Counter Mode + (GCM) and Galois Message Authentication Code (GMAC). + + All of these instructions work on 128-bit element groups comprised of four 32-bit elements. + + To help avoid side-channel timing attacks, these instructions shall be implemented with data-independent timing. + + The number of element groups to be processed is vl/EGS. + vl must be set to the number of SEW=32 elements to be processed and therefore must be a multiple of EGS=4. + + Likewise, vstart must be a multiple of EGS=4. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zvkn.yaml b/arch/ext/Zvkn.yaml new file mode 100644 index 000000000..13eec434b --- /dev/null +++ b/arch/ext/Zvkn.yaml @@ -0,0 +1,23 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvkn +long_name: NIST Algorithm Suite +description: | + This extension is shorthand for the following set of other extensions: + + * `Zvkned` + * `Zvknhb` + * `Zvkb` + * `Zvkt` +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + implies: + - [Zvkned, "1.0.0"] + - [Zvknhb, "1.0.0"] + - [Zvkb, "1.0.0"] + - [Zvkt, "1.0.0"] diff --git a/arch/ext/Zvkned.yaml b/arch/ext/Zvkned.yaml new file mode 100644 index 000000000..4b626a265 --- /dev/null +++ b/arch/ext/Zvkned.yaml @@ -0,0 +1,19 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvkned +long_name: "NIST Suite: Vector AES Block Cipher" +description: | + Instructions for accelerating encryption, decryption and key-schedule functions of the AES block + cipher as defined in Federal Information Processing Standards Publication 197. + + All of these instructions work on 128-bit element groups comprised of four 32-bit elements. + + To help avoid side-channel timing attacks, these instructions shall be implemented with data-independent timing. + +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zvknha.yaml b/arch/ext/Zvknha.yaml new file mode 100644 index 000000000..818af8e44 --- /dev/null +++ b/arch/ext/Zvknha.yaml @@ -0,0 +1,14 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvknha +long_name: "NIST Suite: Vector SHA-2 Secure Hash (SHA-256)" +description: | + Instructions for accelerating 256-bit SHA-2 as defined in FIPS PUB 180-4 Secure Hash Standard (SHS) + +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zvknhb.yaml b/arch/ext/Zvknhb.yaml new file mode 100644 index 000000000..7ca550d95 --- /dev/null +++ b/arch/ext/Zvknhb.yaml @@ -0,0 +1,15 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvknhb +long_name: "NIST Suite: Vector SHA-2 Secure Hash (SHA-256 + SHA-512)" +description: | + Instructions for accelerating 256-bit/512-bit SHA-2 as defined in FIPS PUB 180-4 Secure Hash Standard (SHS) + +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + implies: [Zvknha, "1.0.0"] diff --git a/arch/ext/Zvks.yaml b/arch/ext/Zvks.yaml new file mode 100644 index 000000000..4d1d31013 --- /dev/null +++ b/arch/ext/Zvks.yaml @@ -0,0 +1,23 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvks +long_name: ShangMi Algorithm Suite +description: | + This extension is shorthand for the following set of other extensions: + + * `Zvksed` + * `Zvksh` + * `Zvkb` + * `Zvkt` +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null + implies: + - [Zvksed, "1.0.0"] + - [Zvksh, "1.0.0"] + - [Zvkb, "1.0.0"] + - [Zvkt, "1.0.0"] diff --git a/arch/ext/Zvksed.yaml b/arch/ext/Zvksed.yaml new file mode 100644 index 000000000..546ec7c23 --- /dev/null +++ b/arch/ext/Zvksed.yaml @@ -0,0 +1,21 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvksed +long_name: "ShangMi Suite: SM4 Block Cipher" +description: | + Instructions for accelerating encryption, decryption and key-schedule functions of the SM4 block cipher. + + The SM4 block cipher is specified in 32907-2016: {SM4} Block Cipher Algorithm (GB/T 32907-2016: SM4 Block Cipher Algorithm, 2016) + + There are other various sources available that describe the SM4 block cipher. While not the final version of the standard, RFC 8998 ShangMi (SM) Cipher Suites for TLS 1.3 is useful and easy to access. + + All of these instructions work on 128-bit element groups comprised of four 32-bit elements. + + To help avoid side-channel timing attacks, these instructions shall be implemented with data-independent timing. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zvksh.yaml b/arch/ext/Zvksh.yaml new file mode 100644 index 000000000..7ce13b0c8 --- /dev/null +++ b/arch/ext/Zvksh.yaml @@ -0,0 +1,17 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvksh +long_name: "ShangMi Suite: SM3 Secure Hash" +description: | + Instructions for accelerating functions of the SM3 Hash Function. + + All of these instructions work on 256-bit element groups comprised of eight 32-bit elements. + + To help avoid side-channel timing attacks, these instructions shall be implemented with data-independent timing. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/ext/Zvkt.yaml b/arch/ext/Zvkt.yaml new file mode 100644 index 000000000..409cefed2 --- /dev/null +++ b/arch/ext/Zvkt.yaml @@ -0,0 +1,25 @@ +# yaml-language-server: $schema=../../schemas/ext_schema.json + +$schema: "ext_schema.json#" +kind: extension +name: Zvkt +long_name: Vector Data-Independent Execution Latency +description: | + The Zvkt extension requires all implemented instructions from the following list to be executed + with data-independent execution latency. + + Data-independent execution latency (DIEL) applies to all data operands of an instruction, even + those that are not a part of the body or that are inactive. + However, DIEL does not apply to other values such as vl, vtype, and the mask (when used to control + execution of a masked vector instruction). + Also, DIEL does not apply to constant values specified in the instruction encoding such as the + use of the zero register (x0), and, in the case of immediate forms of an instruction, the values + in the immediate fields (i.e., imm, and uimm). + + In some cases --- which are explicitly specified in the lists below --- operands that are used as + control rather than data are exempt from DIEL. +type: unprivileged +versions: + - version: "1.0.0" + state: ratified + ratification_date: null diff --git a/arch/inst/F/fleq.s.yaml b/arch/inst/F/fleq.s.yaml index 2d40e3d28..72e648b89 100644 --- a/arch/inst/F/fleq.s.yaml +++ b/arch/inst/F/fleq.s.yaml @@ -6,7 +6,7 @@ name: fleq.s long_name: No synopsis available. description: | No description available. -definedBy: F, Zfa +definedBy: { allOf: [F, Zfa] } assembly: xd, fs1, fs2 encoding: match: 1010000----------100-----1010011 diff --git a/arch/inst/F/fli.s.yaml b/arch/inst/F/fli.s.yaml index 4ddd2390e..2d4258920 100644 --- a/arch/inst/F/fli.s.yaml +++ b/arch/inst/F/fli.s.yaml @@ -6,7 +6,7 @@ name: fli.s long_name: No synopsis available. description: | No description available. -definedBy: F, Zfa +definedBy: { allOf: [F, Zfa] } assembly: fd, fs1 encoding: match: 111100000001-----000-----1010011 diff --git a/arch/inst/F/fltq.s.yaml b/arch/inst/F/fltq.s.yaml index a06e604e7..099bd8231 100644 --- a/arch/inst/F/fltq.s.yaml +++ b/arch/inst/F/fltq.s.yaml @@ -6,7 +6,7 @@ name: fltq.s long_name: No synopsis available. description: | No description available. -definedBy: F, Zfa +definedBy: { allOf: [F, Zfa] } assembly: xd, fs1, fs2 encoding: match: 1010000----------101-----1010011 diff --git a/arch/inst/F/fmaxm.s.yaml b/arch/inst/F/fmaxm.s.yaml index db023f712..d57459ac4 100644 --- a/arch/inst/F/fmaxm.s.yaml +++ b/arch/inst/F/fmaxm.s.yaml @@ -6,7 +6,7 @@ name: fmaxm.s long_name: No synopsis available. description: | No description available. -definedBy: F, Zfa +definedBy: { allOf: [F, Zfa] } assembly: xd, xs1, xs2 encoding: match: 0010100----------011-----1010011 diff --git a/arch/inst/F/fminm.s.yaml b/arch/inst/F/fminm.s.yaml index aa8d00371..7ae32aab1 100644 --- a/arch/inst/F/fminm.s.yaml +++ b/arch/inst/F/fminm.s.yaml @@ -6,7 +6,7 @@ name: fminm.s long_name: No synopsis available. description: | No description available. -definedBy: F, Zfa +definedBy: { allOf: [F, Zfa] } assembly: fd, fs1, fs2 encoding: match: 0010100----------010-----1010011 diff --git a/arch/inst/F/fround.s.yaml b/arch/inst/F/fround.s.yaml index 34db217b9..5eb132bef 100644 --- a/arch/inst/F/fround.s.yaml +++ b/arch/inst/F/fround.s.yaml @@ -6,7 +6,7 @@ name: fround.s long_name: No synopsis available. description: | No description available. -definedBy: F, Zfa +definedBy: { allOf: [F, Zfa] } assembly: fd, xs1, rm encoding: match: 010000000100-------------1010011 diff --git a/arch/inst/F/froundnx.s.yaml b/arch/inst/F/froundnx.s.yaml index 8ff644feb..e87cf9737 100644 --- a/arch/inst/F/froundnx.s.yaml +++ b/arch/inst/F/froundnx.s.yaml @@ -6,7 +6,7 @@ name: froundnx.s long_name: No synopsis available. description: | No description available. -definedBy: F, Zfa +definedBy: { allOf: [F, Zfa] } assembly: fd, rs1, rm encoding: match: 010000000101-------------1010011 diff --git a/arch/inst/I/addi.yaml b/arch/inst/I/addi.yaml index f7c6e5bce..829a853ab 100644 --- a/arch/inst/I/addi.yaml +++ b/arch/inst/I/addi.yaml @@ -11,7 +11,7 @@ encoding: match: -----------------000-----0010011 variables: - name: imm - $inherits: common/inst_variable_types.yaml#/itype_imm + location: 31-20 - name: rs1 location: 19-15 - name: rd diff --git a/arch/inst/V/vadd.vi.yaml b/arch/inst/V/vadd.vi.yaml index c12f08144..bbfd787ba 100644 --- a/arch/inst/V/vadd.vi.yaml +++ b/arch/inst/V/vadd.vi.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vadd.vv.yaml b/arch/inst/V/vadd.vv.yaml index a280ed6b4..b44b40a67 100644 --- a/arch/inst/V/vadd.vv.yaml +++ b/arch/inst/V/vadd.vv.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vadd.vx.yaml b/arch/inst/V/vadd.vx.yaml index 61127518b..8496de5db 100644 --- a/arch/inst/V/vadd.vx.yaml +++ b/arch/inst/V/vadd.vx.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vrsub.vi.yaml b/arch/inst/V/vrsub.vi.yaml index b5b332738..7ce1c46e2 100644 --- a/arch/inst/V/vrsub.vi.yaml +++ b/arch/inst/V/vrsub.vi.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vrsub.vx.yaml b/arch/inst/V/vrsub.vx.yaml index 8891c4031..026dd276f 100644 --- a/arch/inst/V/vrsub.vx.yaml +++ b/arch/inst/V/vrsub.vx.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vsub.vv.yaml b/arch/inst/V/vsub.vv.yaml index 090a04356..683967701 100644 --- a/arch/inst/V/vsub.vv.yaml +++ b/arch/inst/V/vsub.vv.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vsub.vx.yaml b/arch/inst/V/vsub.vx.yaml index 14f33d6ae..a638234fd 100644 --- a/arch/inst/V/vsub.vx.yaml +++ b/arch/inst/V/vsub.vx.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwadd.vv.yaml b/arch/inst/V/vwadd.vv.yaml index c1e5223c2..2a01b899e 100644 --- a/arch/inst/V/vwadd.vv.yaml +++ b/arch/inst/V/vwadd.vv.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwadd.vx.yaml b/arch/inst/V/vwadd.vx.yaml index 1c6d1ea1c..61dce8135 100644 --- a/arch/inst/V/vwadd.vx.yaml +++ b/arch/inst/V/vwadd.vx.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwadd.wv.yaml b/arch/inst/V/vwadd.wv.yaml index 9480da43a..c865337a3 100644 --- a/arch/inst/V/vwadd.wv.yaml +++ b/arch/inst/V/vwadd.wv.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwadd.wx.yaml b/arch/inst/V/vwadd.wx.yaml index 3d7c24bf3..5f091e84a 100644 --- a/arch/inst/V/vwadd.wx.yaml +++ b/arch/inst/V/vwadd.wx.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwaddu.vv.yaml b/arch/inst/V/vwaddu.vv.yaml index 66235b1d7..d6544a18f 100644 --- a/arch/inst/V/vwaddu.vv.yaml +++ b/arch/inst/V/vwaddu.vv.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwaddu.vx.yaml b/arch/inst/V/vwaddu.vx.yaml index 8764aaa6a..0f6206d80 100644 --- a/arch/inst/V/vwaddu.vx.yaml +++ b/arch/inst/V/vwaddu.vx.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwaddu.wv.yaml b/arch/inst/V/vwaddu.wv.yaml index 4b3396bb4..70d252b66 100644 --- a/arch/inst/V/vwaddu.wv.yaml +++ b/arch/inst/V/vwaddu.wv.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwaddu.wx.yaml b/arch/inst/V/vwaddu.wx.yaml index 8d614a193..399998730 100644 --- a/arch/inst/V/vwaddu.wx.yaml +++ b/arch/inst/V/vwaddu.wx.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwsub.vv.yaml b/arch/inst/V/vwsub.vv.yaml index ebeb4d80d..f5280b310 100644 --- a/arch/inst/V/vwsub.vv.yaml +++ b/arch/inst/V/vwsub.vv.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwsub.vx.yaml b/arch/inst/V/vwsub.vx.yaml index 0998ef195..1c79543c6 100644 --- a/arch/inst/V/vwsub.vx.yaml +++ b/arch/inst/V/vwsub.vx.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwsub.wv.yaml b/arch/inst/V/vwsub.wv.yaml index 644ea2292..6d088c073 100644 --- a/arch/inst/V/vwsub.wv.yaml +++ b/arch/inst/V/vwsub.wv.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwsub.wx.yaml b/arch/inst/V/vwsub.wx.yaml index e7a6707cd..264cff457 100644 --- a/arch/inst/V/vwsub.wx.yaml +++ b/arch/inst/V/vwsub.wx.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwsubu.vv.yaml b/arch/inst/V/vwsubu.vv.yaml index 1fec77e82..b85ca3f40 100644 --- a/arch/inst/V/vwsubu.vv.yaml +++ b/arch/inst/V/vwsubu.vv.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwsubu.vx.yaml b/arch/inst/V/vwsubu.vx.yaml index dbdb5dd94..f4759e065 100644 --- a/arch/inst/V/vwsubu.vx.yaml +++ b/arch/inst/V/vwsubu.vx.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwsubu.wv.yaml b/arch/inst/V/vwsubu.wv.yaml index a5f210c2c..8426ec2c6 100644 --- a/arch/inst/V/vwsubu.wv.yaml +++ b/arch/inst/V/vwsubu.wv.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/V/vwsubu.wx.yaml b/arch/inst/V/vwsubu.wx.yaml index d881e60a6..4a1cfa22e 100644 --- a/arch/inst/V/vwsubu.wx.yaml +++ b/arch/inst/V/vwsubu.wx.yaml @@ -24,7 +24,7 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | sail(): | diff --git a/arch/inst/Zbp/gorci.yaml b/arch/inst/Zbp/gorci.yaml deleted file mode 100644 index c74ab2a88..000000000 --- a/arch/inst/Zbp/gorci.yaml +++ /dev/null @@ -1,28 +0,0 @@ -# yaml-language-server: $schema=../../../schemas/inst_schema.json - -$schema: inst_schema.json# -kind: instruction -name: gorci -long_name: No synopsis available. -description: | - No description available. -definedBy: - anyOf: [B, Zbp] -assembly: xd, xs1, shamt -encoding: - match: 001010-----------101-----0010011 - variables: - - name: shamt - location: 25-20 - - name: rs1 - location: 19-15 - - name: rd - location: 11-7 -access: - s: always - u: always - vs: always - vu: always -data_independent_timing: false -base: 64 -operation(): | diff --git a/arch/inst/Zbp/grevi.yaml b/arch/inst/Zbp/grevi.yaml deleted file mode 100644 index 2fb9fbac8..000000000 --- a/arch/inst/Zbp/grevi.yaml +++ /dev/null @@ -1,28 +0,0 @@ -# yaml-language-server: $schema=../../../schemas/inst_schema.json - -$schema: inst_schema.json# -kind: instruction -name: grevi -long_name: No synopsis available. -description: | - No description available. -definedBy: - anyOf: [B, Zbp] -assembly: xd, xs1, shamt -encoding: - match: 011010-----------101-----0010011 - variables: - - name: shamt - location: 25-20 - - name: rs1 - location: 19-15 - - name: rd - location: 11-7 -access: - s: always - u: always - vs: always - vu: always -data_independent_timing: false -base: 64 -operation(): | diff --git a/arch/inst/Zbp/shfli.yaml b/arch/inst/Zbp/shfli.yaml deleted file mode 100644 index ec21d0f74..000000000 --- a/arch/inst/Zbp/shfli.yaml +++ /dev/null @@ -1,28 +0,0 @@ -# yaml-language-server: $schema=../../../schemas/inst_schema.json - -$schema: inst_schema.json# -kind: instruction -name: shfli -long_name: No synopsis available. -description: | - No description available. -definedBy: - anyOf: [B, Zbp] -assembly: xd, xs1, shamt -encoding: - match: 0000100----------001-----0010011 - variables: - - name: shamt - location: 24-20 - - name: rs1 - location: 19-15 - - name: rd - location: 11-7 -access: - s: always - u: always - vs: always - vu: always -data_independent_timing: false -base: 64 -operation(): | diff --git a/arch/inst/Zbp/unshfli.yaml b/arch/inst/Zbp/unshfli.yaml deleted file mode 100644 index 7eab8ae10..000000000 --- a/arch/inst/Zbp/unshfli.yaml +++ /dev/null @@ -1,28 +0,0 @@ -# yaml-language-server: $schema=../../../schemas/inst_schema.json - -$schema: inst_schema.json# -kind: instruction -name: unshfli -long_name: No synopsis available. -description: | - No description available. -definedBy: - anyOf: [B, Zbp] -assembly: xd, xs1, shamt -encoding: - match: 0000100----------101-----0010011 - variables: - - name: shamt - location: 24-20 - - name: rs1 - location: 19-15 - - name: rd - location: 11-7 -access: - s: always - u: always - vs: always - vu: always -data_independent_timing: false -base: 64 -operation(): | diff --git a/arch/inst/Zbp/xperm16.yaml b/arch/inst/Zbp/xperm16.yaml deleted file mode 100644 index 2efc25548..000000000 --- a/arch/inst/Zbp/xperm16.yaml +++ /dev/null @@ -1,27 +0,0 @@ -# yaml-language-server: $schema=../../../schemas/inst_schema.json - -$schema: inst_schema.json# -kind: instruction -name: xperm16 -long_name: No synopsis available. -description: | - No description available. -definedBy: - anyOf: [B, Zbp] -assembly: xd, xs1, xs2 -encoding: - match: 0010100----------110-----0110011 - variables: - - name: rs2 - location: 24-20 - - name: rs1 - location: 19-15 - - name: rd - location: 11-7 -access: - s: always - u: always - vs: always - vu: always -data_independent_timing: false -operation(): | diff --git a/arch/inst/Zbp/xperm32.yaml b/arch/inst/Zbp/xperm32.yaml deleted file mode 100644 index 9832c3aa0..000000000 --- a/arch/inst/Zbp/xperm32.yaml +++ /dev/null @@ -1,28 +0,0 @@ -# yaml-language-server: $schema=../../../schemas/inst_schema.json - -$schema: inst_schema.json# -kind: instruction -name: xperm32 -long_name: No synopsis available. -description: | - No description available. -definedBy: - anyOf: [B, Zbp] -assembly: xd, xs1, xs2 -encoding: - match: 0010100----------000-----0110011 - variables: - - name: rs2 - location: 24-20 - - name: rs1 - location: 19-15 - - name: rd - location: 11-7 -access: - s: always - u: always - vs: always - vu: always -data_independent_timing: false -base: 64 -operation(): | diff --git a/arch/inst/Zvbb/vandn.vv.yaml b/arch/inst/Zvbb/vandn.vv.yaml index ecf6c738a..44aee7472 100644 --- a/arch/inst/Zvbb/vandn.vv.yaml +++ b/arch/inst/Zvbb/vandn.vv.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vandn.vx.yaml b/arch/inst/Zvbb/vandn.vx.yaml index d3f99128b..494371e1d 100644 --- a/arch/inst/Zvbb/vandn.vx.yaml +++ b/arch/inst/Zvbb/vandn.vx.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vbrev.v.yaml b/arch/inst/Zvbb/vbrev.v.yaml index d6545977d..b37df247c 100644 --- a/arch/inst/Zvbb/vbrev.v.yaml +++ b/arch/inst/Zvbb/vbrev.v.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vbrev8.v.yaml b/arch/inst/Zvbb/vbrev8.v.yaml index e148793f4..dc0896bae 100644 --- a/arch/inst/Zvbb/vbrev8.v.yaml +++ b/arch/inst/Zvbb/vbrev8.v.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vclz.v.yaml b/arch/inst/Zvbb/vclz.v.yaml index b8f0efd51..9dfec9f7c 100644 --- a/arch/inst/Zvbb/vclz.v.yaml +++ b/arch/inst/Zvbb/vclz.v.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vcpop.v.yaml b/arch/inst/Zvbb/vcpop.v.yaml index 051a4ae1e..ce91a11f1 100644 --- a/arch/inst/Zvbb/vcpop.v.yaml +++ b/arch/inst/Zvbb/vcpop.v.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vctz.v.yaml b/arch/inst/Zvbb/vctz.v.yaml index 6890a24fa..af27a7493 100644 --- a/arch/inst/Zvbb/vctz.v.yaml +++ b/arch/inst/Zvbb/vctz.v.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vrev8.v.yaml b/arch/inst/Zvbb/vrev8.v.yaml index 54620007e..409d295ff 100644 --- a/arch/inst/Zvbb/vrev8.v.yaml +++ b/arch/inst/Zvbb/vrev8.v.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vrol.vv.yaml b/arch/inst/Zvbb/vrol.vv.yaml index 414334ac9..8b074c067 100644 --- a/arch/inst/Zvbb/vrol.vv.yaml +++ b/arch/inst/Zvbb/vrol.vv.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vrol.vx.yaml b/arch/inst/Zvbb/vrol.vx.yaml index 0563a4ac4..7a1ac9f9b 100644 --- a/arch/inst/Zvbb/vrol.vx.yaml +++ b/arch/inst/Zvbb/vrol.vx.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vror.vi.yaml b/arch/inst/Zvbb/vror.vi.yaml index e3ee0a6f3..113a5a85e 100644 --- a/arch/inst/Zvbb/vror.vi.yaml +++ b/arch/inst/Zvbb/vror.vi.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vror.vv.yaml b/arch/inst/Zvbb/vror.vv.yaml index ac2f5aa2e..41b8d4074 100644 --- a/arch/inst/Zvbb/vror.vv.yaml +++ b/arch/inst/Zvbb/vror.vv.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vror.vx.yaml b/arch/inst/Zvbb/vror.vx.yaml index 44871c290..85c7425f1 100644 --- a/arch/inst/Zvbb/vror.vx.yaml +++ b/arch/inst/Zvbb/vror.vx.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vwsll.vi.yaml b/arch/inst/Zvbb/vwsll.vi.yaml index 00dbd0ba2..ae3468bc7 100644 --- a/arch/inst/Zvbb/vwsll.vi.yaml +++ b/arch/inst/Zvbb/vwsll.vi.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vwsll.vv.yaml b/arch/inst/Zvbb/vwsll.vv.yaml index 0f2b4d1dd..bf5c1ffad 100644 --- a/arch/inst/Zvbb/vwsll.vv.yaml +++ b/arch/inst/Zvbb/vwsll.vv.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbb/vwsll.vx.yaml b/arch/inst/Zvbb/vwsll.vx.yaml index 91c02eb5b..f270898f0 100644 --- a/arch/inst/Zvbb/vwsll.vx.yaml +++ b/arch/inst/Zvbb/vwsll.vx.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbc/vclmul.vv.yaml b/arch/inst/Zvbc/vclmul.vv.yaml index 261456a1a..3e8170ec6 100644 --- a/arch/inst/Zvbc/vclmul.vv.yaml +++ b/arch/inst/Zvbc/vclmul.vv.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbc/vclmul.vx.yaml b/arch/inst/Zvbc/vclmul.vx.yaml index e90c8b3f6..8432653a0 100644 --- a/arch/inst/Zvbc/vclmul.vx.yaml +++ b/arch/inst/Zvbc/vclmul.vx.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbc/vclmulh.vv.yaml b/arch/inst/Zvbc/vclmulh.vv.yaml index 7511606c5..43515fc67 100644 --- a/arch/inst/Zvbc/vclmulh.vv.yaml +++ b/arch/inst/Zvbc/vclmulh.vv.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvbc/vclmulh.vx.yaml b/arch/inst/Zvbc/vclmulh.vx.yaml index 21b7b865e..023a959d0 100644 --- a/arch/inst/Zvbc/vclmulh.vx.yaml +++ b/arch/inst/Zvbc/vclmulh.vx.yaml @@ -25,5 +25,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkg/vghsh.vv.yaml b/arch/inst/Zvkg/vghsh.vv.yaml index a4f20bd16..c5854d614 100644 --- a/arch/inst/Zvkg/vghsh.vv.yaml +++ b/arch/inst/Zvkg/vghsh.vv.yaml @@ -22,5 +22,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkg/vgmul.vv.yaml b/arch/inst/Zvkg/vgmul.vv.yaml index 713f4c957..d458b4b43 100644 --- a/arch/inst/Zvkg/vgmul.vv.yaml +++ b/arch/inst/Zvkg/vgmul.vv.yaml @@ -20,5 +20,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vaesdf.vs.yaml b/arch/inst/Zvkn/vaesdf.vs.yaml index dfab3452f..6741428ed 100644 --- a/arch/inst/Zvkn/vaesdf.vs.yaml +++ b/arch/inst/Zvkn/vaesdf.vs.yaml @@ -21,5 +21,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vaesdf.vv.yaml b/arch/inst/Zvkn/vaesdf.vv.yaml index 943a7652c..5f2b460ff 100644 --- a/arch/inst/Zvkn/vaesdf.vv.yaml +++ b/arch/inst/Zvkn/vaesdf.vv.yaml @@ -21,5 +21,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vaesdm.vs.yaml b/arch/inst/Zvkn/vaesdm.vs.yaml index 986d97d46..27c9e740e 100644 --- a/arch/inst/Zvkn/vaesdm.vs.yaml +++ b/arch/inst/Zvkn/vaesdm.vs.yaml @@ -21,5 +21,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vaesdm.vv.yaml b/arch/inst/Zvkn/vaesdm.vv.yaml index 2585b72b0..209a1b947 100644 --- a/arch/inst/Zvkn/vaesdm.vv.yaml +++ b/arch/inst/Zvkn/vaesdm.vv.yaml @@ -21,5 +21,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vaesef.vs.yaml b/arch/inst/Zvkn/vaesef.vs.yaml index 490d8b6e1..0ad2f9490 100644 --- a/arch/inst/Zvkn/vaesef.vs.yaml +++ b/arch/inst/Zvkn/vaesef.vs.yaml @@ -21,5 +21,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vaesef.vv.yaml b/arch/inst/Zvkn/vaesef.vv.yaml index 3bf5f12c8..283bc428c 100644 --- a/arch/inst/Zvkn/vaesef.vv.yaml +++ b/arch/inst/Zvkn/vaesef.vv.yaml @@ -21,5 +21,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vaesem.vs.yaml b/arch/inst/Zvkn/vaesem.vs.yaml index 0a8ea43b3..ec1a73382 100644 --- a/arch/inst/Zvkn/vaesem.vs.yaml +++ b/arch/inst/Zvkn/vaesem.vs.yaml @@ -21,5 +21,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vaesem.vv.yaml b/arch/inst/Zvkn/vaesem.vv.yaml index e515036e4..866d20bdb 100644 --- a/arch/inst/Zvkn/vaesem.vv.yaml +++ b/arch/inst/Zvkn/vaesem.vv.yaml @@ -21,5 +21,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vaeskf1.vi.yaml b/arch/inst/Zvkn/vaeskf1.vi.yaml index e8554288b..89ca6562d 100644 --- a/arch/inst/Zvkn/vaeskf1.vi.yaml +++ b/arch/inst/Zvkn/vaeskf1.vi.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vaeskf2.vi.yaml b/arch/inst/Zvkn/vaeskf2.vi.yaml index 6c2a56603..ac12aef4f 100644 --- a/arch/inst/Zvkn/vaeskf2.vi.yaml +++ b/arch/inst/Zvkn/vaeskf2.vi.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vaesz.vs.yaml b/arch/inst/Zvkn/vaesz.vs.yaml index 8d422dd7c..a518fab1f 100644 --- a/arch/inst/Zvkn/vaesz.vs.yaml +++ b/arch/inst/Zvkn/vaesz.vs.yaml @@ -3,7 +3,7 @@ $schema: "inst_schema.json#" kind: instruction name: vaesz.vs -long_name: No synopsis available. +long_name: Vector AES round zero description: | No description available. definedBy: @@ -21,5 +21,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vsha2ch.vv.yaml b/arch/inst/Zvkn/vsha2ch.vv.yaml index 47b3bdf00..d57cf09b2 100644 --- a/arch/inst/Zvkn/vsha2ch.vv.yaml +++ b/arch/inst/Zvkn/vsha2ch.vv.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vsha2cl.vv.yaml b/arch/inst/Zvkn/vsha2cl.vv.yaml index 4d66f7692..46c07b102 100644 --- a/arch/inst/Zvkn/vsha2cl.vv.yaml +++ b/arch/inst/Zvkn/vsha2cl.vv.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvkn/vsha2ms.vv.yaml b/arch/inst/Zvkn/vsha2ms.vv.yaml index d7e30cfd7..c63f13abb 100644 --- a/arch/inst/Zvkn/vsha2ms.vv.yaml +++ b/arch/inst/Zvkn/vsha2ms.vv.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvks/vsm3c.vi.yaml b/arch/inst/Zvks/vsm3c.vi.yaml index a053310cd..98ed236f7 100644 --- a/arch/inst/Zvks/vsm3c.vi.yaml +++ b/arch/inst/Zvks/vsm3c.vi.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvks/vsm3me.vv.yaml b/arch/inst/Zvks/vsm3me.vv.yaml index eb092b573..90d885d4b 100644 --- a/arch/inst/Zvks/vsm3me.vv.yaml +++ b/arch/inst/Zvks/vsm3me.vv.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvks/vsm4k.vi.yaml b/arch/inst/Zvks/vsm4k.vi.yaml index cb04d00b0..b1b897326 100644 --- a/arch/inst/Zvks/vsm4k.vi.yaml +++ b/arch/inst/Zvks/vsm4k.vi.yaml @@ -23,5 +23,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvks/vsm4r.vs.yaml b/arch/inst/Zvks/vsm4r.vs.yaml index 968903254..2df7c266d 100644 --- a/arch/inst/Zvks/vsm4r.vs.yaml +++ b/arch/inst/Zvks/vsm4r.vs.yaml @@ -21,5 +21,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/inst/Zvks/vsm4r.vv.yaml b/arch/inst/Zvks/vsm4r.vv.yaml index ec2507b6b..eb67ea2e1 100644 --- a/arch/inst/Zvks/vsm4r.vv.yaml +++ b/arch/inst/Zvks/vsm4r.vv.yaml @@ -21,5 +21,5 @@ access: u: always vs: always vu: always -data_independent_timing: false +data_independent_timing: true operation(): | diff --git a/arch/manual/isa/isa.yaml b/arch/manual/isa.yaml similarity index 74% rename from arch/manual/isa/isa.yaml rename to arch/manual/isa.yaml index 7d573628b..85a85c92e 100644 --- a/arch/manual/isa/isa.yaml +++ b/arch/manual/isa.yaml @@ -1,3 +1,7 @@ +# yaml-language-server: $schema=../../schemas/manual_schema.json + +$schema: manual_schema.json# +kind: manual name: isa marketing_name: RISC-V ISA Manual url: https://github.com/riscv/riscv-isa-manual diff --git a/arch/manual/isa/20240411/contents.yaml b/arch/manual_version/isa/20240411/isa_20240411.yaml similarity index 60% rename from arch/manual/isa/20240411/contents.yaml rename to arch/manual_version/isa/20240411/isa_20240411.yaml index b28f4ce72..ca0408516 100644 --- a/arch/manual/isa/20240411/contents.yaml +++ b/arch/manual_version/isa/20240411/isa_20240411.yaml @@ -1,11 +1,13 @@ # yaml-language-server: $schema=../../../../schemas/manual_version_schema.json --- -manual: isa +$schema: manual_version_schema.json# +kind: manual version +manual: { $ref: manual/isa.yaml# } version: "4.0.0-pre" -name: "20240411" +name: "isa_20240411" marketing_version: "20240411" state: "development" -url: https://github.com/riscv/riscv-isa-manual/releases/tag/20240411 +url: https://github.com/riscv/releases/tag/20240411 uses_isa_manual: true isa_manual_tree: "tags/20240411" volumes: @@ -18,51 +20,51 @@ volumes: name: SiFive, Inc. url: https://www.sifive.com/ chapters: - - riscv-isa-manual/src/colophon.adoc - - riscv-isa-manual/src/intro.adoc - - riscv-isa-manual/src/rv32.adoc - - riscv-isa-manual/src/rv32e.adoc - - riscv-isa-manual/src/rv64.adoc - - riscv-isa-manual/src/rv128.adoc - - riscv-isa-manual/src/zifencei.adoc - - riscv-isa-manual/src/zicsr.adoc - - riscv-isa-manual/src/counters.adoc - - riscv-isa-manual/src/zihintntl.adoc - - riscv-isa-manual/src/zihintpause.adoc - - riscv-isa-manual/src/zimop.adoc - - riscv-isa-manual/src/zicond.adoc - - riscv-isa-manual/src/m-st-ext.adoc - - riscv-isa-manual/src/a-st-ext.adoc - - riscv-isa-manual/src/zawrs.adoc - - riscv-isa-manual/src/zacas.adoc - - riscv-isa-manual/src/rvwmo.adoc - - riscv-isa-manual/src/ztso-st-ext.adoc - - riscv-isa-manual/src/cmo.adoc - - riscv-isa-manual/src/f-st-ext.adoc - - riscv-isa-manual/src/d-st-ext.adoc - - riscv-isa-manual/src/q-st-ext.adoc - - riscv-isa-manual/src/zfh.adoc - - riscv-isa-manual/src/zfa.adoc - - riscv-isa-manual/src/zfinx.adoc - - riscv-isa-manual/src/c-st-ext.adoc - - riscv-isa-manual/src/zc.adoc - - riscv-isa-manual/src/b-st-ext.adoc - - riscv-isa-manual/src/j-st-ext.adoc - - riscv-isa-manual/src/p-st-ext.adoc - - riscv-isa-manual/src/v-st-ext.adoc - - riscv-isa-manual/src/scalar-crypto.adoc - - riscv-isa-manual/src/vector-crypto.adoc - - riscv-isa-manual/src/rv-32-64g.adoc - - riscv-isa-manual/src/extending.adoc - - riscv-isa-manual/src/naming.adoc - - riscv-isa-manual/src/history.adoc - - riscv-isa-manual/src/mm-eplan.adoc - - riscv-isa-manual/src/mm-formal.adoc + - src/colophon.adoc + - src/intro.adoc + - src/rv32.adoc + - src/rv32e.adoc + - src/rv64.adoc + - src/rv128.adoc + - src/zifencei.adoc + - src/zicsr.adoc + - src/counters.adoc + - src/zihintntl.adoc + - src/zihintpause.adoc + - src/zimop.adoc + - src/zicond.adoc + - src/m-st-ext.adoc + - src/a-st-ext.adoc + - src/zawrs.adoc + - src/zacas.adoc + - src/rvwmo.adoc + - src/ztso-st-ext.adoc + - src/cmo.adoc + - src/f-st-ext.adoc + - src/d-st-ext.adoc + - src/q-st-ext.adoc + - src/zfh.adoc + - src/zfa.adoc + - src/zfinx.adoc + - src/c-st-ext.adoc + - src/zc.adoc + - src/b-st-ext.adoc + - src/j-st-ext.adoc + - src/p-st-ext.adoc + - src/v-st-ext.adoc + - src/scalar-crypto.adoc + - src/vector-crypto.adoc + - src/rv-32-64g.adoc + - src/extending.adoc + - src/naming.adoc + - src/history.adoc + - src/mm-eplan.adoc + - src/mm-formal.adoc #Appendices for Vector - - riscv-isa-manual/src/vector-examples.adoc - - riscv-isa-manual/src/calling-convention.adoc + - src/vector-examples.adoc + - src/calling-convention.adoc #/End of Vector appendices - - riscv-isa-manual/src/index.adoc + - src/index.adoc extensions: - [I, "2.1.0"] - [U, "1.12.0"] @@ -132,9 +134,6 @@ volumes: - [Zvfh, "1.0.0"] - [Zvknha, "1.0.0"] - [Zvknhb, "1.0.0"] - - [Zbkb, "1.0.0"] - - [Zbkc, "1.0.0"] - - [Zbkx, "1.0.0"] - [Zknd, "1.0.0"] - [Zkne, "1.0.0"] - [Zknh, "1.0.0"] @@ -177,23 +176,23 @@ volumes: name: SiFive, Inc. url: https://www.sifive.com/ chapters: - - riscv-isa-manual/src/priv-preface.adoc - - riscv-isa-manual/src/priv-intro.adoc - - riscv-isa-manual/src/priv-csrs.adoc - - riscv-isa-manual/src/machine.adoc - - riscv-isa-manual/src/smstateen.adoc - - riscv-isa-manual/src/indirect-csr.adoc - - riscv-isa-manual/src/smepmp.adoc - - riscv-isa-manual/src/smcntrpmf.adoc - - riscv-isa-manual/src/rnmi.adoc - - riscv-isa-manual/src/smcdeleg.adoc - - riscv-isa-manual/src/supervisor.adoc - - riscv-isa-manual/src/sstc.adoc - - riscv-isa-manual/src/sscofpmf.adoc - - riscv-isa-manual/src/hypervisor.adoc - - riscv-isa-manual/src/priv-insns.adoc - - riscv-isa-manual/src/priv-history.adoc - - riscv-isa-manual/src/bibliography.adoc + - src/priv-preface.adoc + - src/priv-intro.adoc + - src/priv-csrs.adoc + - src/machine.adoc + - src/smstateen.adoc + - src/indirect-csr.adoc + - src/smepmp.adoc + - src/smcntrpmf.adoc + - src/rnmi.adoc + - src/smcdeleg.adoc + - src/supervisor.adoc + - src/sstc.adoc + - src/sscofpmf.adoc + - src/hypervisor.adoc + - src/priv-insns.adoc + - src/priv-history.adoc + - src/bibliography.adoc extensions: - [Smstateen, "1.0.0"] - [Smcsrind, "1.0.0"] @@ -215,6 +214,5 @@ volumes: - [Svinval, "1.0.0"] - [Svadu, "1.0.0"] - [Svvptc, "1.0.0"] - - [Sstc, "1.0.0"] - [Sscofpmf, "1.0.0"] - [H, "1.0.0"] diff --git a/arch/profile/MP-S-64.yaml b/arch/profile/MP-S-64.yaml new file mode 100644 index 000000000..cd34a233c --- /dev/null +++ b/arch/profile/MP-S-64.yaml @@ -0,0 +1,68 @@ +# yaml-language-server: $schema=../../schemas/profile_schema.json + +$schema: profile_schema.json# +kind: profile +name: MP-S-64 +marketing_name: MockProfile 64-bit S-mode +description: This is the Mock Profile Supervisor Mode description. +mode: S +base: 64 +release: { $ref: profile_release/MockProfileRelease.yaml# } +contributors: + - name: Micky Mouse + email: micky@disney.com + company: Disney +extensions: + $inherits: "profile/MP-U-64.yaml#/extensions" + A: + presence: mandatory + note: This should be listed as mandatory in MP-S-64 and optional in MP-U-64. + S: + presence: + optional: localized + version: "= 1.12" + Zifencei: + presence: + optional: development + version: "= 2.0" + note: + Zihpm: + presence: + optional: expansion + version: "= 2.0" + note: Made this a expansion option + Sv48: + presence: + optional: transitory + version: "= 1.11" + note: Made this a transitory option +extra_notes: + - presence: mandatory + text: | + Here's the first extra note for the mandatory extensions section. + This note is multiple lines. + - presence: optional + text: | + Here's the first extra note for the optional extensions section. + In this case, we don't differentiate between optional types. + This note is multiple lines. + - presence: + optional: localized + text: Here's the first extra note for the localized optional extensions section. + - presence: + optional: localized + text: Here's the second extra note for the localized optional extensions section. + - presence: + optional: development + text: Here's the first extra note for the development optional extensions section. + - presence: + optional: expansion + text: Here's the first extra note for the expansion optional extensions section. + - presence: + optional: transitory + text: Here's the first extra note for the transitory optional extensions section. +recommendations: + - text: | + Implementations are strongly recommended to raise illegal-instruction + exceptions on attempts to execute unimplemented opcodes. + - text: Micky should give Pluto an extra treat diff --git a/arch/profile/MP-U-64.yaml b/arch/profile/MP-U-64.yaml new file mode 100644 index 000000000..3273c323b --- /dev/null +++ b/arch/profile/MP-U-64.yaml @@ -0,0 +1,17 @@ +$schema: profile_schema.json# +kind: profile +name: MP-U-64 +marketing_name: MockProfile 64-bit Unpriv +mode: Unpriv +base: 64 +release: { $ref: profile_release/MockProfileRelease.yaml# } +extensions: + A: + presence: optional + version: "= 2.1" + I: + presence: mandatory + version: "~> 2.1" + Svade: + presence: mandatory + note: Adding this to get coverage when extension "conflicts" with another (Svadu in this case). diff --git a/arch/profile/RVA20S64.yaml b/arch/profile/RVA20S64.yaml new file mode 100644 index 000000000..7af25aa36 --- /dev/null +++ b/arch/profile/RVA20S64.yaml @@ -0,0 +1,63 @@ +$schema: profile_schema.json# +kind: profile +name: RVA20S64 +marketing_name: RVA20S64 +mode: S +base: 64 +release: { $ref: profile_release/RVA20.yaml# } +introduction: | + The RVA20S64 profile specifies the ISA features available to a + supervisor-mode execution environment in 64-bit applications + processors. RVA20S64 is based on privileged architecture version 1.11. +extensions: + $inherits: "profile/RVI20U64.yaml#/extensions" + S: + presence: mandatory + version: "= 1.11" + Zifencei: + presence: mandatory + version: "= 2.0" + note: | + Zifencei is mandated as it is the only standard way to support + instruction-cache coherence in RVA20 application processors. A new + instruction-cache coherence mechanism is under development which might + be added as an option in the future. + Svbare: + presence: mandatory + version: "= 1.0" + note: | + Svbare is a new extension name introduced with RVA20. + Sv39: + presence: mandatory + version: "= 1.11" + Svade: + presence: mandatory + version: "~> 1.0" + note: | + Svbare is a new extension name introduced with RVA20. + + It is subsequently defined in more detail with the ratification of + `Svadu`. + Ssccptr: + presence: mandatory + version: "= 1.0" + note: | + Ssccptr is a new extension name introduced with RVA20. + Sstvecd: + presence: mandatory + version: "= 1.0" + note: | + Sstvecd is a new extension name introduced with RVA20. + Sstvala: + presence: mandatory + version: "= 1.0" + note: | + Sstvala is a new extension name introduced with RVA20. + Sv48: + presence: optional + version: "= 1.11" + Ssu64xl: + presence: optional + version: "= 1.0" + note: | + Ssu64xl is a new extension name introduced with RVA20. diff --git a/arch/profile/RVA20U64.yaml b/arch/profile/RVA20U64.yaml new file mode 100644 index 000000000..47d1cfa4b --- /dev/null +++ b/arch/profile/RVA20U64.yaml @@ -0,0 +1,98 @@ +$schema: profile_schema.json# +kind: profile +name: RVA20U64 +marketing_name: RVA20U64 +mode: Unpriv +base: 64 +release: { $ref: profile_release/RVA20.yaml# } +introduction: | + The RVA20U64 profile specifies the ISA features available to user-mode + execution environments in 64-bit applications processors. This is the + most important profile within application processors in + terms of the amount of software that targets this profile. +extensions: + $inherits: "profile/RVI20U64.yaml#/extensions" + $remove: Zifencei # Not allowed as an option for Unpriv ISA (only available in Priv ISA). + A: + presence: mandatory + C: + presence: mandatory + D: + presence: mandatory + F: + presence: mandatory + M: + presence: mandatory + U: + presence: mandatory + version: "~> 2.0" + param_constraints: + U_MODE_ENDIANESS: + schema: + const: little + Zicntr: + presence: mandatory + Ziccif: + presence: mandatory + version: "= 1.0" + note: | + Ziccif is a profile-defined extension introduced with RVA20. + The fetch atomicity requirement facilitates runtime patching + of aligned instructions. + Ziccrse: + presence: mandatory + version: "= 1.0" + note: Ziccrse is a profile-defined extension introduced with RVA20. + Ziccamoa: + presence: mandatory + version: "= 1.0" + note: Ziccamo is a profile-defined extension introduced with RVA20. + Za128rs: + presence: mandatory + version: "= 1.0" + note: | + Za128rs is a profile-defined extension introduced with RVA20. + The minimum reservation set size is effectively determined by the + size of atomic accesses in the `A` extension. + Zicclsm: + presence: mandatory + version: "= 1.0" + note: | + Zicclsm is a profile-defined extension introduced with RVA20. + This requires misaligned support for all regular load and store + instructions (including scalar and vector) but not AMOs or other + specialized forms of memory access. Even though mandated, misaligned + loads and stores might execute extremely slowly. Standard software + distributions should assume their existence only for correctness, not + for performance. +extra_notes: + - presence: optional + text: | + The rationale to not make Q an optional extension is that + quad-precision floating-point is unlikely to be implemented in + hardware, and so we do not require or expect A-profile software to + expend effort optimizing use of Q instructions in case they are + present. + - presence: optional + text: | + Zifencei is not classed as a supported option in the user-mode + profile because it is not sufficient by itself to produce the desired + effect in a multiprogrammed multiprocessor environment without OS + support, and so the instruction cache flush should always be performed + using an OS call rather than using the `fence.i` instruction. + `fence.i` semantics can be expensive to implement for some hardware + memory hierarchy designs, and so alternative non-standard + instruction-cache coherence mechanisms can be used behind the OS + abstraction. A separate extension is being developed for more general + and efficient instruction cache coherence. + - presence: optional + text: | + The execution environment must provide a means to synchronize writes to + instruction memory with instruction fetches, the implementation of which + likely relies on the Zifencei extension. + For example, RISC-V Linux supplies the `__riscv_flush_icache` system call and + a corresponding vDSO call. +recommendations: + - text: | + Implementations are strongly recommended to raise illegal-instruction + exceptions on attempts to execute unimplemented opcodes. diff --git a/arch/profile/RVA22S64.yaml b/arch/profile/RVA22S64.yaml new file mode 100644 index 000000000..6b558f6b9 --- /dev/null +++ b/arch/profile/RVA22S64.yaml @@ -0,0 +1,104 @@ +$schema: profile_schema.json# +kind: profile +name: RVA22S64 +marketing_name: RVA22S64 +mode: S +base: 64 +release: { $ref: profile_release/RVA22.yaml# } +introduction: | + The RVA22S64 profile specifies the ISA features available to a + supervisor-mode execution environment in 64-bit applications + processors. RVA22S64 is based on privileged architecture version + 1.12. +extensions: + $inherits: "profile/RVA20S64.yaml#/extensions" + S: + presence: mandatory + version: "= 1.12" + Sscounterenw: + presence: mandatory + version: "= 1.0" + note: | + Sstvala is a new extension name introduced with RVA22. + Svpbmt: + presence: mandatory + version: "~> 1.0" + Svinval: + presence: mandatory + version: "~> 1.0" + Ssstateen: + presence: mandatory + version: "~> 1.0" + when: + implemented: H + note: | + Ssstateen is a new extension name introduced with RVA22. + Shvstvala: + presence: mandatory + version: "~> 1.0" + when: + implemented: H + note: | + Shvstvala is a new extension name introduced with RVA22. + Shtvala: + presence: mandatory + version: "~> 1.0" + when: + implemented: H + note: | + Shtvala is a new extension name introduced with RVA22. + Shvstvecd: + presence: mandatory + version: "~> 1.0" + when: + implemented: H + note: | + Shvstvecd is a new extension name introduced with RVA22. + Shgatpa: + presence: mandatory + version: "~> 1.0" + when: + implemented: H + note: | + Shgatpa is a new extension name introduced with RVA22. + Sv57: + presence: optional + version: "~> 1.12" + Svnapot: + presence: optional + version: "~> 1.0" + note: | + It is expected that Svnapot will be mandatory in the next + profile release. + Sstc: + presence: optional + version: "~> 1.0" + note: | + Sstc was not made mandatory in RVA22S64 as it is a more + disruptive change affecting system-level architecture, and will take + longer for implementations to adopt. It is expected to be made + mandatory in the next profile release. + Sscofpmf: + presence: optional + version: "~> 1.0" + note: | + Platforms may choose to mandate the presence of Sscofpmf. + Zkr: + presence: optional + version: "~> 1.0" + note: | + Technically, Zk is also a privileged-mode option capturing that + Zkr, Zkn, and Zkt are all implemented. However, the Zk rollup is less + descriptive than specifying the individual extensions explicitly. + H: + presence: optional + version: "~> 1.0" + note: | + The following extensions become mandatory when H is implemented: + + * Ssstateen + * Shcounterenw + * Shvstvala + * Shtvala + * Shvstvecd + * Shgatpa diff --git a/arch/profile/RVA22U64.yaml b/arch/profile/RVA22U64.yaml new file mode 100644 index 000000000..48e4abc94 --- /dev/null +++ b/arch/profile/RVA22U64.yaml @@ -0,0 +1,126 @@ +$schema: profile_schema.json# +kind: profile +name: RVA22U64 +marketing_name: RVA22U64 +mode: Unpriv +base: 64 +release: { $ref: profile_release/RVA22.yaml# } +introduction: | + The RVA22U64 profile specifies the ISA features available to user-mode + execution environments in 64-bit applications processors. This is the + most important profile within application processors in + terms of the amount of software that targets this profile. +extensions: + $inherits: "profile/RVA20U64.yaml#/extensions" + Zihpm: + presence: mandatory + version: "= 2.0" + Zihintpause: + presence: mandatory + version: "= 2.0" + note: | + While the `pause` instruction is a HINT can be implemented as a + NOP and hence trivially supported by hardware implementers, its + inclusion in the mandatory extension list signifies that software + should use the instruction whenever it would make sense and that + implementors are expected to exploit this information to optimize + hardware execution. + Zba: + presence: mandatory + version: "~> 1.0" + Zbb: + presence: mandatory + version: "~> 1.0" + Zbs: + presence: mandatory + version: "~> 1.0" + Zic64b: + presence: mandatory + version: "= 1.0" + note: | + This is a new extension name for this feature. While the general + RISC-V specifications are agnostic to cache block size, selecting a + common cache block size simplifies the specification and use of the + following cache-block extensions within the application processor + profile. Software does not have to query a discovery mechanism and/or + provide dynamic dispatch to the appropriate code. We choose 64 bytes + at it is effectively an industry standard. Implementations may use + longer cache blocks to reduce tag cost provided they use 64-byte + sub-blocks to remain compatible. Implementations may use shorter cache + blocks provided they sequence cache operations across the multiple + cache blocks comprising a 64-byte block to remain compatible. + Zicbom: + presence: mandatory + version: "~> 1.0" + Zicbop: + presence: mandatory + version: "~> 1.0" + note: | + As with other HINTS, the inclusion of prefetches in the + mandatory set of extensions indicates that software should generate + these instructions where they are expected to be useful, and hardware + is expected to exploit that information. + Zicboz: + presence: mandatory + version: "~> 1.0" + Zfhmin: + presence: mandatory + version: "~> 1.0" + note: | + Zfhmin is a small extension that adds support to load/store and convert + IEEE 754 half-precision numbers to and from the IEEE 754 single-precision + format. The hardware cost for this extension is low, and mandating the + extension avoids adding an option to the profile. + Zkt: + presence: mandatory + version: "~> 1.0" + note: | + Zkt requires a certain subset of integer instructions execute + with data-independent latency. Mandating this feature enables + portable libraries for safe basic cryptographic operations. It is + expected that application processors will naturally have this property + and so implementation cost is low, if not zero, in most systems that + would support RVA22. + Zfh: + presence: optional + version: "~> 1.0" + note: A future profile might mandate V. + V: + presence: optional + version: "~> 1.0" + note: | + The smaller vector extensions (Zve32f, Zve32x, Zve64d, Zve64f, + Zve64x) are not provided as separately supported profile options. The + full V extension is specified as the only supported profile option. + + A future profile might mandate V. + Zkn: + presence: optional + version: "~> 1.0" + Zks: + presence: optional + version: "~> 1.0" +extra_notes: + - presence: optional + text: | + The scalar crypto extensions are expected to be superseded by + vector crypto standards in future profiles, and the scalar extensions + may be removed as supported options once vector crypto is present. + - presence: optional + text: | + The smaller component scalar crypto extensions (Zbc, Zbkb, Zbkc, + Zbkx, Zknd, Zkne, Zknh, Zksed, Zksh) are not provided as separate + options in the profile. Profile implementers should provide all of + the instructions in a given algorithm suite as part of the Zkn or Zks + supported options. + - presence: optional + text: | + Access to the entropy source (Zkr) in a system is usually + carefully controlled. While the design supports unprivileged access + to the entropy source, this is unlikely to be commonly used in an + application processor, and so Zkr was not added as a profile option. + This also means the roll-up Zk was not added as a profile option. + - presence: optional + text: | + The Zfinx, Zdinx, Zhinx, Zhinxmin extensions are incompatible + with the profile mandates to support the F and D extensions. diff --git a/arch/profile/RVI20U32.yaml b/arch/profile/RVI20U32.yaml new file mode 100644 index 000000000..4a1e17343 --- /dev/null +++ b/arch/profile/RVI20U32.yaml @@ -0,0 +1,68 @@ +$schema: profile_schema.json# +kind: profile +name: RVI20U32 +marketing_name: RVI20U32 +mode: Unpriv +base: 32 +release: { $ref: profile_release/RVI20.yaml# } +introduction: | + This profile specifies the ISA features available to generic unprivileged + execution environments. +extensions: + I: + presence: mandatory + version: "~> 2.1" + note: | + RVI is the mandatory base ISA for RVA, and is little-endian. + + As per the unprivileged architecture specification, the `ecall` + instruction causes a requested trap to the execution environment. + + Misaligned loads and stores might not be supported. + + The `fence.tso` instruction is mandatory. + + NOTE: The `fence.tso` instruction was incorrectly described as + optional in the 2019 ratified specifications. However, `fence.tso` is + encoded within the standard `fence` encoding such that implementations + must treat it as a simple global fence if they do not natively support + TSO-ordering optimizations. As software can always assume without any + penalty that `fence.tso` is being exploited by a hardware + implementation, there is no advantage to making the instruction a + profile option. Later versions of the unprivileged ISA specifications + correctly indicate that `fence.tso` is mandatory. + A: + presence: optional + version: "= 2.1" + C: + presence: optional + version: "= 2.2" + D: + presence: optional + version: "= 2.2" + note: | + NOTE: The rationale to not include Q as a profile option is that + quad-precision floating-point is unlikely to be implemented in + hardware, and so we do not require or expect software to expend effort + optimizing use of Q instructions in case they are present. + F: + presence: optional + version: "= 2.2" + M: + presence: optional + version: "= 2.0" + Zicntr: + presence: optional + version: "= 2.0" + Zihpm: + presence: optional + version: "= 2.0" + note: | + The number of counters is platform-specific. + Zifencei: + presence: optional + version: "= 2.0" +recommendations: + - text: | + Implementations are strongly recommended to raise illegal-instruction + exceptions on attempts to execute unimplemented opcodes. diff --git a/arch/profile/RVI20U64.yaml b/arch/profile/RVI20U64.yaml new file mode 100644 index 000000000..aa84d2e41 --- /dev/null +++ b/arch/profile/RVI20U64.yaml @@ -0,0 +1,6 @@ +$schema: profile_schema.json# +kind: profile +name: RVI20U64 +$inherits: "profile/RVI20U32.yaml#" +base: 64 +marketing_name: RVI20U64 diff --git a/arch/profile_class/MockProfileClass.yaml b/arch/profile_class/MockProfileClass.yaml index 613e4460b..6c6355bf3 100644 --- a/arch/profile_class/MockProfileClass.yaml +++ b/arch/profile_class/MockProfileClass.yaml @@ -1,15 +1,17 @@ -MockProfileClass: - marketing_name: Mock Profile Class - introduction: Here's the Mock Profile Class introduction. - description: | - This is the Mock Profile Class description. - It can be longer than the introduction since it gets its own sub-heading. - naming_scheme: | - Here's the Mock Profile Class naming scheme. - company: - name: RISC-V International - url: https://riscv.org - doc_license: - name: Creative Commons Attribution 4.0 International License - url: https://creativecommons.org/licenses/by/4.0/ - text_url: https://creativecommons.org/licenses/by/4.0/legalcode.txt +$schema: profile_class_schema.json# +kind: profile class +name: MockProfileClass +marketing_name: Mock Profile Class +introduction: Here's the Mock Profile Class introduction. +description: | + This is the Mock Profile Class description. + It can be longer than the introduction since it gets its own sub-heading. +naming_scheme: | + Here's the Mock Profile Class naming scheme. +company: + name: RISC-V International + url: https://riscv.org +doc_license: + name: Creative Commons Attribution 4.0 International License + url: https://creativecommons.org/licenses/by/4.0/ + text_url: https://creativecommons.org/licenses/by/4.0/legalcode.txt diff --git a/arch/profile_class/RVA.yaml b/arch/profile_class/RVA.yaml index 9c0df8d1a..d18ce00b8 100644 --- a/arch/profile_class/RVA.yaml +++ b/arch/profile_class/RVA.yaml @@ -1,145 +1,149 @@ -RVA: - marketing_name: RVA - introduction: | - The RVA profile class targets application processors for markets - requiring a high-degree of binary compatibility between compliant implementations. - description: | - RISC-V was designed to provide a highly modular and extensible - instruction set and includes a large and growing set of standard - extensions, where each standard extension is a bundle of - instruction-set features. This is no different than other industry - ISAs that continue to add new ISA features. Unlike other ISAs, - however, RISC-V has a broad set of contributors and implementers, and - also allows users to add their own custom extensions. For some deep - embedded markets, highly customized processor configurations are - desirable for efficiency, and all software is compiled, ported, and/or - developed in-house by the same organization for that specific - processor configuration. However, for other markets that expect a - substantial fraction of software to be delivered to end-customers in - binary form, compatibility across multiple implementations from - different RISC-V vendors is required. +# yaml-language-server: $schema=../../schemas/profile_class_schema.json - The RVIA ISA extension ratification process ensures that all processor - vendors have agreed to the specification of a standard extension if - present. However, by themselves, the ISA extension specifications do - not guarantee that a certain set of standard extensions will be - present in all implementations. +$schema: profile_class_schema.json# +kind: profile class +name: RVA +marketing_name: RVA +introduction: | + The RVA profile class targets application processors for markets + requiring a high-degree of binary compatibility between compliant implementations. +description: | + RISC-V was designed to provide a highly modular and extensible + instruction set and includes a large and growing set of standard + extensions, where each standard extension is a bundle of + instruction-set features. This is no different than other industry + ISAs that continue to add new ISA features. Unlike other ISAs, + however, RISC-V has a broad set of contributors and implementers, and + also allows users to add their own custom extensions. For some deep + embedded markets, highly customized processor configurations are + desirable for efficiency, and all software is compiled, ported, and/or + developed in-house by the same organization for that specific + processor configuration. However, for other markets that expect a + substantial fraction of software to be delivered to end-customers in + binary form, compatibility across multiple implementations from + different RISC-V vendors is required. - *The primary goal of the RVA profiles is to align processor vendors - targeting binary software markets, so software can rely on the - existence of a certain set of ISA features in a particular generation - of RISC-V implementations.* + The RVIA ISA extension ratification process ensures that all processor + vendors have agreed to the specification of a standard extension if + present. However, by themselves, the ISA extension specifications do + not guarantee that a certain set of standard extensions will be + present in all implementations. - Alignment is not only for compatibility, but also to ensure RISC-V is - competitive in these markets. The binary app markets are also - generally those with the most competitive performance requirements - (e.g., mobile, client, server). RVIA cannot mandate the ISA features - that a RISC-V binary software ecosystem should use, as each ecosystem - will typically select the lowest-common denominator they empirically - observe in the deployed devices in their target markets. But RVIA can - align hardware vendors to support a common set of features in each - generation through the RVA profiles. Without proactive alignment - through RVA profiles, RISC-V will be uncompetitive, as even if a - particular vendor implements a certain feature, if other vendors do - not, then binary distributions will not generally use that feature and - all implementations will suffer. While certain features may be - discoverable, and alternate code provided in case of presence/absence - of a feature, the added cost to support such options is only justified - for certain limited cases, and binary app markets will not support a - wide range of optional features, particularly for the nascent RISC-V - binary app ecosystems. + *The primary goal of the RVA profiles is to align processor vendors + targeting binary software markets, so software can rely on the + existence of a certain set of ISA features in a particular generation + of RISC-V implementations.* - To maintain alignment and increase RISC-V competitiveness over time, - the mandatory set of extensions must increase over time in successive - generations of RVA profile. (RVA profiles may eventually have to - deprecate previously mandatory instructions, but that is unlikely in - the near future.) Note that the RISC-V ISA will continue to evolve, - regardless of whether a given software ecosystem settles on a certain - generation of profile as the baseline for their ecosystem for many - years or even decades. There are many existing binary software - ecosystems, which will migrate to RISC-V and evolve at different rates, - and more new ones will doubtless be created over the hopefully long - lifetime of RISC-V. High-performance application processors require - considerable investment, and no single binary app ecosystem can - justify the development costs of these processors, especially for - RISC-V in its early stage of adoption. + Alignment is not only for compatibility, but also to ensure RISC-V is + competitive in these markets. The binary app markets are also + generally those with the most competitive performance requirements + (e.g., mobile, client, server). RVIA cannot mandate the ISA features + that a RISC-V binary software ecosystem should use, as each ecosystem + will typically select the lowest-common denominator they empirically + observe in the deployed devices in their target markets. But RVIA can + align hardware vendors to support a common set of features in each + generation through the RVA profiles. Without proactive alignment + through RVA profiles, RISC-V will be uncompetitive, as even if a + particular vendor implements a certain feature, if other vendors do + not, then binary distributions will not generally use that feature and + all implementations will suffer. While certain features may be + discoverable, and alternate code provided in case of presence/absence + of a feature, the added cost to support such options is only justified + for certain limited cases, and binary app markets will not support a + wide range of optional features, particularly for the nascent RISC-V + binary app ecosystems. - While the heart of the profile is the set of mandatory extensions, - there are several kinds of optional extension that serve important - roles in the profile. + To maintain alignment and increase RISC-V competitiveness over time, + the mandatory set of extensions must increase over time in successive + generations of RVA profile. (RVA profiles may eventually have to + deprecate previously mandatory instructions, but that is unlikely in + the near future.) Note that the RISC-V ISA will continue to evolve, + regardless of whether a given software ecosystem settles on a certain + generation of profile as the baseline for their ecosystem for many + years or even decades. There are many existing binary software + ecosystems, which will migrate to RISC-V and evolve at different rates, + and more new ones will doubtless be created over the hopefully long + lifetime of RISC-V. High-performance application processors require + considerable investment, and no single binary app ecosystem can + justify the development costs of these processors, especially for + RISC-V in its early stage of adoption. - The first kind are _localized_ _options_, whose presence or use - necessarily differs along geo-political and/or jurisdictional - boundaries, with crypto being the obvious example. These will always - be optional. At least for crypto, discovery has been found to be - perfectly acceptable to handle this optionality on other - architectures, as the use of the extensions is well contained in - certain libraries. + While the heart of the profile is the set of mandatory extensions, + there are several kinds of optional extension that serve important + roles in the profile. - The second kind of optional extension is a _development_ _option_, - which represents a new ISA extension in an early part of its lifecycle - but which is intended to become mandatory in a later generation of the - RVA profile. Processor vendors and software toolchain providers will - have varying development schedules, and providing an optional phase in - a new extension's lifecycle provides some flexibility while - maintaining overall alignment, and is particularly appropriate when - hardware or software development for the extension is complex. - Denoting an extension as a _development_ _option_ signals to the - community that development should be prioritized for such extensions - as they will become mandatory. + The first kind are _localized_ _options_, whose presence or use + necessarily differs along geo-political and/or jurisdictional + boundaries, with crypto being the obvious example. These will always + be optional. At least for crypto, discovery has been found to be + perfectly acceptable to handle this optionality on other + architectures, as the use of the extensions is well contained in + certain libraries. - The third kind of optional extension are _expansion_ _options_, which - are those that may have a large implementation cost but are not always - needed in a particular platform, and which can be readily handled by - discovery. These are also intended to remain available as expansion - options in future versions of the profile. Several supervisor-mode - extensions fall into this category, e.g., Sv57, which has a notable - PPA impact over Sv48 and is not needed on smaller platforms. Some - unprivileged extensions that may fall into this category are possible - future matrix extensions. These have large implementation costs, and - use of matrix instructions can be readily supported with discovery and - alternate math libraries. + The second kind of optional extension is a _development_ _option_, + which represents a new ISA extension in an early part of its lifecycle + but which is intended to become mandatory in a later generation of the + RVA profile. Processor vendors and software toolchain providers will + have varying development schedules, and providing an optional phase in + a new extension's lifecycle provides some flexibility while + maintaining overall alignment, and is particularly appropriate when + hardware or software development for the extension is complex. + Denoting an extension as a _development_ _option_ signals to the + community that development should be prioritized for such extensions + as they will become mandatory. - The fourth kind of optional extensions are _transitory_ _options_, - where it is not clear if the extension will change to a mandatory, - localized, or expansion option, or be possibly dropped over time. - Cryptography provides some examples where earlier cyphers have been - broken and are now deprecated. RVIA used this mechanism to enable - scalar crypto until vector crypto was ready. Software security - features may also be in this category, with examples of deprecated - security features occuring in other architectures. As another - example, the recent avalanche of new numeric datatypes for AI/ML may - eventually subside with a few survivors actually being used longer - term. Denoting an option as transitory signals to the community that - this extension may be removed in a future profile, though the time - scale may span many years. + The third kind of optional extension are _expansion_ _options_, which + are those that may have a large implementation cost but are not always + needed in a particular platform, and which can be readily handled by + discovery. These are also intended to remain available as expansion + options in future versions of the profile. Several supervisor-mode + extensions fall into this category, e.g., Sv57, which has a notable + PPA impact over Sv48 and is not needed on smaller platforms. Some + unprivileged extensions that may fall into this category are possible + future matrix extensions. These have large implementation costs, and + use of matrix instructions can be readily supported with discovery and + alternate math libraries. - Except for the localized options, it could be argued that other three - kinds of option could be left out of profiles. Binary distributions - of applications willing to invest in discovery can use an optional - extension, and customers compiling their own applications can take - advantage of the feature on a particular implementation, even when - that system is mostly running binary distributions that ignore the new - extension. However, there is value in providing guidance to align - hardware vendors and software developers around what extensions are - worth implementing and worth discovering, by designating only a few - important features as profile options and limiting their granularity. - naming_scheme: | - The profile class name is RVA (RISC-V Apps processor). - A profile release name is an integer (currently 2 digits, could grow in the future). - A full profile name is comprised of, in order: + The fourth kind of optional extensions are _transitory_ _options_, + where it is not clear if the extension will change to a mandatory, + localized, or expansion option, or be possibly dropped over time. + Cryptography provides some examples where earlier cyphers have been + broken and are now deprecated. RVIA used this mechanism to enable + scalar crypto until vector crypto was ready. Software security + features may also be in this category, with examples of deprecated + security features occuring in other architectures. As another + example, the recent avalanche of new numeric datatypes for AI/ML may + eventually subside with a few survivors actually being used longer + term. Denoting an option as transitory signals to the community that + this extension may be removed in a future profile, though the time + scale may span many years. - * Prefix *RVA* for RISC-V Applications - * Profile release - * Privilege mode: - ** *U* Unprivileged (available to any privilege mode, *U* is *not* User-mode) - ** *S* Supervisor mode (note that Hypervisor support is treated as an option) - ** *M* Machine mode - * A base ISA XLEN specifier (*32*, *64*) - company: - name: RISC-V International - url: https://riscv.org - doc_license: - name: Creative Commons Attribution 4.0 International License - url: https://creativecommons.org/licenses/by/4.0/ - text_url: https://creativecommons.org/licenses/by/4.0/legalcode.txt + Except for the localized options, it could be argued that other three + kinds of option could be left out of profiles. Binary distributions + of applications willing to invest in discovery can use an optional + extension, and customers compiling their own applications can take + advantage of the feature on a particular implementation, even when + that system is mostly running binary distributions that ignore the new + extension. However, there is value in providing guidance to align + hardware vendors and software developers around what extensions are + worth implementing and worth discovering, by designating only a few + important features as profile options and limiting their granularity. +naming_scheme: | + The profile class name is RVA (RISC-V Apps processor). + A profile release name is an integer (currently 2 digits, could grow in the future). + A full profile name is comprised of, in order: + + * Prefix *RVA* for RISC-V Applications + * Profile release + * Privilege mode: + ** *U* Unprivileged (available to any privilege mode, *U* is *not* User-mode) + ** *S* Supervisor mode (note that Hypervisor support is treated as an option) + ** *M* Machine mode + * A base ISA XLEN specifier (*32*, *64*) +company: + name: RISC-V International + url: https://riscv.org +doc_license: + name: Creative Commons Attribution 4.0 International License + url: https://creativecommons.org/licenses/by/4.0/ + text_url: https://creativecommons.org/licenses/by/4.0/legalcode.txt diff --git a/arch/profile_class/RVB.yaml b/arch/profile_class/RVB.yaml index ea4239f69..9a9517b70 100644 --- a/arch/profile_class/RVB.yaml +++ b/arch/profile_class/RVB.yaml @@ -1,50 +1,52 @@ -RVB: - marketing_name: RVB - introduction: | - The RVB profile class targets application processors for markets - running Bespoke (AKA custom, AKA Yocto) Linux Operating Systems - in embedded applications. - description: | - The RVB profile class is intended to be used for 64-bit application - processors running rich OS stacks. Only user-mode and - supervisor-mode profiles are specified in this class. +$schema: profile_class_schema.json# +kind: profile class +name: RVB +marketing_name: RVB +introduction: | + The RVB profile class targets application processors for markets + running Bespoke (AKA custom, AKA Yocto) Linux Operating Systems + in embedded applications. +description: | + The RVB profile class is intended to be used for 64-bit application + processors running rich OS stacks. Only user-mode and + supervisor-mode profiles are specified in this class. - NOTE: There is no machine-mode profile currently defined for RVB. - A machine-mode profile for application processors would only be used in specifying platforms for - portable machine-mode software. Given the relatively low volume of - portable M-mode software in this domain, the wide variety of potential - M-mode code, and the very specific needs of each type of M-mode - software, we are not specifying individual M-mode ISA requirements in - the RVB profiles. + NOTE: There is no machine-mode profile currently defined for RVB. + A machine-mode profile for application processors would only be used in specifying platforms for + portable machine-mode software. Given the relatively low volume of + portable M-mode software in this domain, the wide variety of potential + M-mode code, and the very specific needs of each type of M-mode + software, we are not specifying individual M-mode ISA requirements in + the RVB profiles. - NOTE: Only XLEN=64 application processor profiles are currently - defined. It would be possible to also define very similar XLEN=32 - variants. - naming_scheme: | - The profile class name is RVB (RISC-V Bespoke processor). - A profile release name is a integer (currently 2 digits, could grow in the future). - A full profile name is comprised of, in order: + NOTE: Only XLEN=64 application processor profiles are currently + defined. It would be possible to also define very similar XLEN=32 + variants. +naming_scheme: | + The profile class name is RVB (RISC-V Bespoke processor). + A profile release name is a integer (currently 2 digits, could grow in the future). + A full profile name is comprised of, in order: - * Prefix *RVB* for RISC-V Bespoke - * Profile release - * Privilege mode: - ** *U* Unprivileged (available to any privilege mode, *U* is *not* User-mode) - ** *S* Supervisor mode (note that Hypervisor support is treated as an option) - ** *M* Machine mode - * A base ISA XLEN specifier (*32*, *64*) + * Prefix *RVB* for RISC-V Bespoke + * Profile release + * Privilege mode: + ** *U* Unprivileged (available to any privilege mode, *U* is *not* User-mode) + ** *S* Supervisor mode (note that Hypervisor support is treated as an option) + ** *M* Machine mode + * A base ISA XLEN specifier (*32*, *64*) - The initial profiles based on specifications ratified in 2024 are: + The initial profiles based on specifications ratified in 2024 are: - * RVB23U64, RVB23S64 64-bit application-processor profiles + * RVB23U64, RVB23S64 64-bit application-processor profiles - NOTE: Profile names are embeddable into RISC-V ISA naming strings. - This implies that there will be no standard ISA extension with a name - that matches the profile naming convention. This allows tools that - process the RISC-V ISA naming string to parse and/or process a combined string. - company: - name: RISC-V International - url: https://riscv.org - doc_license: - name: Creative Commons Attribution 4.0 International License - url: https://creativecommons.org/licenses/by/4.0/ - text_url: https://creativecommons.org/licenses/by/4.0/legalcode.txt + NOTE: Profile names are embeddable into RISC-V ISA naming strings. + This implies that there will be no standard ISA extension with a name + that matches the profile naming convention. This allows tools that + process the RISC-V ISA naming string to parse and/or process a combined string. +company: + name: RISC-V International + url: https://riscv.org +doc_license: + name: Creative Commons Attribution 4.0 International License + url: https://creativecommons.org/licenses/by/4.0/ + text_url: https://creativecommons.org/licenses/by/4.0/legalcode.txt diff --git a/arch/profile_class/RVI.yaml b/arch/profile_class/RVI.yaml index daad9e90c..7c77a558f 100644 --- a/arch/profile_class/RVI.yaml +++ b/arch/profile_class/RVI.yaml @@ -1,28 +1,30 @@ -RVI: - marketing_name: RVI - introduction: The RVI profile class documents the initial set of unprivileged instructions. - description: | - The RVI profile class provides a generic target for software toolchains - and represent the minimum level of compatibility with RISC-V ratified standards. +$schema: profile_class_schema.json# +kind: profile class +name: RVI +marketing_name: RVI +introduction: The RVI profile class documents the initial set of unprivileged instructions. +description: | + The RVI profile class provides a generic target for software toolchains + and represent the minimum level of compatibility with RISC-V ratified standards. - NOTE: Profiles in this class are designated as _unprivileged_ profiles as opposed to - _user_-_mode_ profiles. Code using this profile class can run in any - privilege mode, and so requested and fatal traps may be horizontal - traps into an execution environment running in the same privilege mode. - naming_scheme: | - The profile class name is RVI (RISC-V base Integer instructions). - A profile release name is an integer (currently 2 digits, could grow in the future). - A full profile name is comprised of, in order: + NOTE: Profiles in this class are designated as _unprivileged_ profiles as opposed to + _user_-_mode_ profiles. Code using this profile class can run in any + privilege mode, and so requested and fatal traps may be horizontal + traps into an execution environment running in the same privilege mode. +naming_scheme: | + The profile class name is RVI (RISC-V base Integer instructions). + A profile release name is an integer (currently 2 digits, could grow in the future). + A full profile name is comprised of, in order: - * Prefix *RVI* for RISC-V Integer - * Profile release - * Privilege mode: - ** *U* Unprivileged (available to any privilege mode, *U* is *not* User-mode) - * A base ISA XLEN specifier (*32*, *64*) - company: - name: RISC-V International - url: https://riscv.org - doc_license: - name: Creative Commons Attribution 4.0 International License - url: https://creativecommons.org/licenses/by/4.0/ - text_url: https://creativecommons.org/licenses/by/4.0/legalcode.txt + * Prefix *RVI* for RISC-V Integer + * Profile release + * Privilege mode: + ** *U* Unprivileged (available to any privilege mode, *U* is *not* User-mode) + * A base ISA XLEN specifier (*32*, *64*) +company: + name: RISC-V International + url: https://riscv.org +doc_license: + name: Creative Commons Attribution 4.0 International License + url: https://creativecommons.org/licenses/by/4.0/ + text_url: https://creativecommons.org/licenses/by/4.0/legalcode.txt diff --git a/arch/profile_release/MockProfileRelease.yaml b/arch/profile_release/MockProfileRelease.yaml index 8d75ea6e2..8c4c63543 100644 --- a/arch/profile_release/MockProfileRelease.yaml +++ b/arch/profile_release/MockProfileRelease.yaml @@ -1,100 +1,24 @@ -MockProfileRelease: - name: MockProfileRelease - marketing_name: MockProfileRelease Marketing Name - class: MockProfileClass - release: 20 - state: ratified # current status ["ratified", "development"] - versions: - - version: "1.0" - ratification_date: "2024-01-01" - introduction: Here's the Mock Profile Release introduction. - description: | - This is the Mock Profile Release description. - It can be longer than the introduction since it gets its own sub-heading. - contributors: - - name: Joe Blow - email: joe.blow@riscv.org - company: Acme Inc - - name: Jane Doe - email: jane.doe@gmail.com - company: Universal Imports - profiles: - MP-U-64: - marketing_name: MockProfile 64-bit Unpriv - mode: Unpriv - base: 64 - release: MockProfileRelease - extensions: - A: - presence: optional - version: "~> 2.1" - I: - presence: mandatory - version: "~> 2.1" - Svade: - presence: mandatory - note: Adding this to get coverage when extension "conflicts" with another (Svadu in this case). - MP-S-64: - marketing_name: MockProfile 64-bit S-mode - description: This is the Mock Profile Supervisor Mode description. - mode: S - base: 64 - release: MockProfileRelease - contributors: - - name: Micky Mouse - email: micky@disney.com - company: Disney - extensions: - $inherits: "#/MockProfileRelease/profiles/MP-U-64/extensions" - A: - presence: mandatory - note: This should be listed as mandatory in MP-S-64 and optional in MP-U-64. - S: - presence: - optional: localized - version: "~> 1.12" - Zifencei: - presence: - optional: development - version: "~> 2.0" - note: - Zihpm: - presence: - optional: expansion - version: "~> 2.0" - note: Made this a expansion option - Sv48: - presence: - optional: transitory - version: "~> 1.11" - note: Made this a transitory option - extra_notes: - - presence: mandatory - text: | - Here's the first extra note for the mandatory extensions section. - This note is multiple lines. - - presence: optional - text: | - Here's the first extra note for the optional extensions section. - In this case, we don't differentiate between optional types. - This note is multiple lines. - - presence: - optional: localized - text: Here's the first extra note for the localized optional extensions section. - - presence: - optional: localized - text: Here's the second extra note for the localized optional extensions section. - - presence: - optional: development - text: Here's the first extra note for the development optional extensions section. - - presence: - optional: expansion - text: Here's the first extra note for the expansion optional extensions section. - - presence: - optional: transitory - text: Here's the first extra note for the transitory optional extensions section. - recommendations: - - text: | - Implementations are strongly recommended to raise illegal-instruction - exceptions on attempts to execute unimplemented opcodes. - - text: Micky should give Pluto an extra treat +$schema: profile_release_schema.json# +kind: profile release +name: MockProfileRelease +marketing_name: MockProfileRelease Marketing Name +class: MockProfileClass +release: 20 +state: ratified # current status ["ratified", "development"] +versions: + - version: "1.0" +ratification_date: "2024-01-01" +introduction: Here's the Mock Profile Release introduction. +description: | + This is the Mock Profile Release description. + It can be longer than the introduction since it gets its own sub-heading. +contributors: + - name: Joe Blow + email: joe.blow@riscv.org + company: Acme Inc + - name: Jane Doe + email: jane.doe@gmail.com + company: Universal Imports +profiles: + - { $ref: profile/MP-U-64.yaml# } + - { $ref: profile/MP-S-64.yaml# } diff --git a/arch/profile_release/RVA20.yaml b/arch/profile_release/RVA20.yaml index 162aba901..c524c091c 100644 --- a/arch/profile_release/RVA20.yaml +++ b/arch/profile_release/RVA20.yaml @@ -1,190 +1,37 @@ -RVA20: - name: RVA20 - marketing_name: RVA20 - class: RVA - release: 20 - state: ratified # current status ["ratified", "development"] - ratification_date: "2023-04-03" +$schema: profile_release_schema.json# +kind: profile release +name: RVA20 +marketing_name: RVA20 +class: RVA +release: 20 +state: ratified # current status ["ratified", "development"] +ratification_date: "2023-04-03" - # Semantic versions within the release - versions: - - version: "1.0.0" +# Semantic versions within the release +versions: + - version: "1.0.0" - introduction: | - This profile release targets 64-bit application processors for markets - requiring a high-degree of binary compatibility between compliant implementations. - description: | - This profile release is intended to be used for 64-bit application - processors running rich OS stacks. Only user-mode and - supervisor-mode profiles are specified in this release. +introduction: | + This profile release targets 64-bit application processors for markets + requiring a high-degree of binary compatibility between compliant implementations. +description: | + This profile release is intended to be used for 64-bit application + processors running rich OS stacks. Only user-mode and + supervisor-mode profiles are specified in this release. - NOTE: There is no machine-mode profile currently defined for this release. - A machine-mode profile for application processors would only be used in specifying platforms for - portable machine-mode software. Given the relatively low volume of - portable M-mode software in this domain, the wide variety of potential - M-mode code, and the very specific needs of each type of M-mode - software, we are not specifying individual M-mode ISA requirements in this release. + NOTE: There is no machine-mode profile currently defined for this release. + A machine-mode profile for application processors would only be used in specifying platforms for + portable machine-mode software. Given the relatively low volume of + portable M-mode software in this domain, the wide variety of potential + M-mode code, and the very specific needs of each type of M-mode + software, we are not specifying individual M-mode ISA requirements in this release. - NOTE: Only XLEN=64 application processor profiles are currently defined. - It would be possible to also define very similar XLEN=32 variants. - contributors: - - name: Krste Asanovic - email: krste@sifive.com - company: SiFive - profiles: - RVA20U64: - marketing_name: RVA20U64 - mode: Unpriv - base: 64 - release: RVA20 - introduction: | - The RVA20U64 profile specifies the ISA features available to user-mode - execution environments in 64-bit applications processors. This is the - most important profile within application processors in - terms of the amount of software that targets this profile. - extensions: - $inherits: "profile_release/RVI20.yaml#/RVI20/profiles/RVI20U64/extensions" - $remove: Zifencei # Not allowed as an option for Unpriv ISA (only available in Priv ISA). - A: - presence: mandatory - C: - presence: mandatory - D: - presence: mandatory - F: - presence: mandatory - M: - presence: mandatory - U: - presence: mandatory - version: "~> 2.0" - param_constraints: - U_MODE_ENDIANESS: - schema: - const: little - Zicntr: - presence: mandatory - Ziccif: - presence: mandatory - version: "~> 1.0" - note: | - Ziccif is a profile-defined extension introduced with RVA20. - The fetch atomicity requirement facilitates runtime patching - of aligned instructions. - Ziccrse: - presence: mandatory - version: "~> 1.0" - note: Ziccrse is a profile-defined extension introduced with RVA20. - Ziccamoa: - presence: mandatory - version: "~> 1.0" - note: Ziccamo is a profile-defined extension introduced with RVA20. - Za128rs: - presence: mandatory - version: "~> 1.0" - note: | - Za128rs is a profile-defined extension introduced with RVA20. - The minimum reservation set size is effectively determined by the - size of atomic accesses in the `A` extension. - Zicclsm: - presence: mandatory - version: "~> 1.0" - note: | - Zicclsm is a profile-defined extension introduced with RVA20. - This requires misaligned support for all regular load and store - instructions (including scalar and vector) but not AMOs or other - specialized forms of memory access. Even though mandated, misaligned - loads and stores might execute extremely slowly. Standard software - distributions should assume their existence only for correctness, not - for performance. - extra_notes: - - presence: optional - text: | - The rationale to not make Q an optional extension is that - quad-precision floating-point is unlikely to be implemented in - hardware, and so we do not require or expect A-profile software to - expend effort optimizing use of Q instructions in case they are - present. - - presence: optional - text: | - Zifencei is not classed as a supported option in the user-mode - profile because it is not sufficient by itself to produce the desired - effect in a multiprogrammed multiprocessor environment without OS - support, and so the instruction cache flush should always be performed - using an OS call rather than using the `fence.i` instruction. - `fence.i` semantics can be expensive to implement for some hardware - memory hierarchy designs, and so alternative non-standard - instruction-cache coherence mechanisms can be used behind the OS - abstraction. A separate extension is being developed for more general - and efficient instruction cache coherence. - - presence: optional - text: | - The execution environment must provide a means to synchronize writes to - instruction memory with instruction fetches, the implementation of which - likely relies on the Zifencei extension. - For example, RISC-V Linux supplies the `__riscv_flush_icache` system call and - a corresponding vDSO call. - recommendations: - - text: | - Implementations are strongly recommended to raise illegal-instruction - exceptions on attempts to execute unimplemented opcodes. - RVA20S64: - marketing_name: RVA20S64 - mode: S - base: 64 - release: RVA20 - introduction: | - The RVA20S64 profile specifies the ISA features available to a - supervisor-mode execution environment in 64-bit applications - processors. RVA20S64 is based on privileged architecture version 1.11. - extensions: - S: - presence: mandatory - version: "~> 1.11" - Zifencei: - presence: mandatory - version: "~> 2.0" - note: | - Zifencei is mandated as it is the only standard way to support - instruction-cache coherence in RVA20 application processors. A new - instruction-cache coherence mechanism is under development which might - be added as an option in the future. - Svbare: - presence: mandatory - version: "~> 1.0" - note: | - Svbare is a new extension name introduced with RVA20. - Sv39: - presence: mandatory - version: "~> 1.11" - Svade: - presence: mandatory - version: "~> 1.0" - note: | - Svbare is a new extension name introduced with RVA20. - - It is subsequently defined in more detail with the ratification of - `Svadu`. - Ssccptr: - presence: mandatory - version: "~> 1.0" - note: | - Ssccptr is a new extension name introduced with RVA20. - Sstvecd: - presence: mandatory - version: "~> 1.0" - note: | - Sstvecd is a new extension name introduced with RVA20. - Sstvala: - presence: mandatory - version: "~> 1.0" - note: | - Sstvala is a new extension name introduced with RVA20. - Sv48: - presence: optional - version: "~> 1.11" - Ssu64xl: - presence: optional - version: "~> 1.0" - note: | - Ssu64xl is a new extension name introduced with RVA20. + NOTE: Only XLEN=64 application processor profiles are currently defined. + It would be possible to also define very similar XLEN=32 variants. +contributors: + - name: Krste Asanovic + email: krste@sifive.com + company: SiFive +profiles: + - { $ref: profile/RVA20U64.yaml# } + - { $ref: profile/RVA20S64.yaml# } diff --git a/arch/profile_release/RVA22.yaml b/arch/profile_release/RVA22.yaml index 045779aef..284ed4e78 100644 --- a/arch/profile_release/RVA22.yaml +++ b/arch/profile_release/RVA22.yaml @@ -1,268 +1,37 @@ -RVA22: - name: RVA22 - marketing_name: RVA22 - class: RVA - release: 22 - state: ratified # current status ["ratified", "development"] - ratification_date: "2023-04-03" +$schema: profile_release_schema.json# +kind: profile release +name: RVA22 +marketing_name: RVA22 +class: RVA +release: 22 +state: ratified # current status ["ratified", "development"] +ratification_date: "2023-04-03" - # Semantic versions within the release - versions: - - version: "1.0.0" +# Semantic versions within the release +versions: + - version: "1.0.0" - introduction: | - This profile release targets 64-bit application processors for markets - requiring a high-degree of binary compatibility between compliant implementations. - description: | - This profile release is intended to be used for 64-bit application - processors running rich OS stacks. Only user-mode and - supervisor-mode profiles are specified in this release. +introduction: | + This profile release targets 64-bit application processors for markets + requiring a high-degree of binary compatibility between compliant implementations. +description: | + This profile release is intended to be used for 64-bit application + processors running rich OS stacks. Only user-mode and + supervisor-mode profiles are specified in this release. - NOTE: There is no machine-mode profile currently defined for this release. - A machine-mode profile for application processors would only be used in specifying platforms for - portable machine-mode software. Given the relatively low volume of - portable M-mode software in this domain, the wide variety of potential - M-mode code, and the very specific needs of each type of M-mode - software, we are not specifying individual M-mode ISA requirements in this release. + NOTE: There is no machine-mode profile currently defined for this release. + A machine-mode profile for application processors would only be used in specifying platforms for + portable machine-mode software. Given the relatively low volume of + portable M-mode software in this domain, the wide variety of potential + M-mode code, and the very specific needs of each type of M-mode + software, we are not specifying individual M-mode ISA requirements in this release. - NOTE: Only XLEN=64 application processor profiles are currently defined. - It would be possible to also define very similar XLEN=32 variants. - contributors: - - name: Krste Asanovic - email: krste@sifive.com - company: SiFive - profiles: - RVA22U64: - marketing_name: RVA22U64 - mode: Unpriv - base: 64 - release: RVA22 - introduction: | - The RVA22U64 profile specifies the ISA features available to user-mode - execution environments in 64-bit applications processors. This is the - most important profile within application processors in - terms of the amount of software that targets this profile. - extensions: - $inherits: "profile_release/RVA20.yaml#/RVA20/profiles/RVA20U64/extensions" - Zihpm: - presence: mandatory - version: "~> 2.0" - Zihintpause: - presence: mandatory - version: "~> 2.0" - note: | - While the `pause` instruction is a HINT can be implemented as a - NOP and hence trivially supported by hardware implementers, its - inclusion in the mandatory extension list signifies that software - should use the instruction whenever it would make sense and that - implementors are expected to exploit this information to optimize - hardware execution. - Zba: - presence: mandatory - version: "~> 1.0" - Zbb: - presence: mandatory - version: "~> 1.0" - Zbs: - presence: mandatory - version: "~> 1.0" - Zic64b: - presence: mandatory - version: "~> 1.0" - note: | - This is a new extension name for this feature. While the general - RISC-V specifications are agnostic to cache block size, selecting a - common cache block size simplifies the specification and use of the - following cache-block extensions within the application processor - profile. Software does not have to query a discovery mechanism and/or - provide dynamic dispatch to the appropriate code. We choose 64 bytes - at it is effectively an industry standard. Implementations may use - longer cache blocks to reduce tag cost provided they use 64-byte - sub-blocks to remain compatible. Implementations may use shorter cache - blocks provided they sequence cache operations across the multiple - cache blocks comprising a 64-byte block to remain compatible. - Zicbom: - presence: mandatory - version: "~> 1.0" - Zicbop: - presence: mandatory - version: "~> 1.0" - note: | - As with other HINTS, the inclusion of prefetches in the - mandatory set of extensions indicates that software should generate - these instructions where they are expected to be useful, and hardware - is expected to exploit that information. - Zicboz: - presence: mandatory - version: "~> 1.0" - Zfhmin: - presence: mandatory - version: "~> 1.0" - note: | - Zfhmin is a small extension that adds support to load/store and convert - IEEE 754 half-precision numbers to and from the IEEE 754 single-precision - format. The hardware cost for this extension is low, and mandating the - extension avoids adding an option to the profile. - Zkt: - presence: mandatory - version: "~> 1.0" - note: | - Zkt requires a certain subset of integer instructions execute - with data-independent latency. Mandating this feature enables - portable libraries for safe basic cryptographic operations. It is - expected that application processors will naturally have this property - and so implementation cost is low, if not zero, in most systems that - would support RVA22. - Zfh: - presence: optional - version: "~> 1.0" - note: A future profile might mandate V. - V: - presence: optional - version: "~> 1.0" - note: | - The smaller vector extensions (Zve32f, Zve32x, Zve64d, Zve64f, - Zve64x) are not provided as separately supported profile options. The - full V extension is specified as the only supported profile option. - - A future profile might mandate V. - Zkn: - presence: optional - version: "~> 1.0" - Zks: - presence: optional - version: "~> 1.0" - extra_notes: - - presence: optional - text: | - The scalar crypto extensions are expected to be superseded by - vector crypto standards in future profiles, and the scalar extensions - may be removed as supported options once vector crypto is present. - - presence: optional - text: | - The smaller component scalar crypto extensions (Zbc, Zbkb, Zbkc, - Zbkx, Zknd, Zkne, Zknh, Zksed, Zksh) are not provided as separate - options in the profile. Profile implementers should provide all of - the instructions in a given algorithm suite as part of the Zkn or Zks - supported options. - - presence: optional - text: | - Access to the entropy source (Zkr) in a system is usually - carefully controlled. While the design supports unprivileged access - to the entropy source, this is unlikely to be commonly used in an - application processor, and so Zkr was not added as a profile option. - This also means the roll-up Zk was not added as a profile option. - - presence: optional - text: | - The Zfinx, Zdinx, Zhinx, Zhinxmin extensions are incompatible - with the profile mandates to support the F and D extensions. - recommendations: - - text: | - Implementations are strongly recommended to raise illegal-instruction - exceptions on attempts to execute unimplemented opcodes. - RVA22S64: - marketing_name: RVA22S64 - mode: S - base: 64 - release: RVA22 - introduction: | - The RVA22S64 profile specifies the ISA features available to a - supervisor-mode execution environment in 64-bit applications - processors. RVA22S64 is based on privileged architecture version - 1.12. - extensions: - $inherits: "profile_release/RVA20.yaml#/RVA20/profiles/RVA20S64/extensions" - S: - presence: mandatory - version: "~> 1.12" - Sscounterenw: - presence: mandatory - version: "~> 1.0" - note: | - Sstvala is a new extension name introduced with RVA22. - Svpbmt: - presence: mandatory - version: "~> 1.0" - Svinval: - presence: mandatory - version: "~> 1.0" - Ssstateen: - presence: mandatory - version: "~> 1.0" - when: - implemented: H - note: | - Ssstateen is a new extension name introduced with RVA22. - Shvstvala: - presence: mandatory - version: "~> 1.0" - when: - implemented: H - note: | - Shvstvala is a new extension name introduced with RVA22. - Shtvala: - presence: mandatory - version: "~> 1.0" - when: - implemented: H - note: | - Shtvala is a new extension name introduced with RVA22. - Shvstvecd: - presence: mandatory - version: "~> 1.0" - when: - implemented: H - note: | - Shvstvecd is a new extension name introduced with RVA22. - Shgatpa: - presence: mandatory - version: "~> 1.0" - when: - implemented: H - note: | - Shgatpa is a new extension name introduced with RVA22. - Sv57: - presence: optional - version: "~> 1.12" - Svnapot: - presence: optional - version: "~> 1.0" - note: | - It is expected that Svnapot will be mandatory in the next - profile release. - Sstc: - presence: optional - version: "~> 1.0" - note: | - Sstc was not made mandatory in RVA22S64 as it is a more - disruptive change affecting system-level architecture, and will take - longer for implementations to adopt. It is expected to be made - mandatory in the next profile release. - Sscofpmf: - presence: optional - version: "~> 1.0" - note: | - Platforms may choose to mandate the presence of Sscofpmf. - Zkr: - presence: optional - version: "~> 1.0" - note: | - Technically, Zk is also a privileged-mode option capturing that - Zkr, Zkn, and Zkt are all implemented. However, the Zk rollup is less - descriptive than specifying the individual extensions explicitly. - H: - presence: optional - version: "~> 1.0" - note: | - The following extensions become mandatory when H is implemented: - - * Ssstateen - * Shcounterenw - * Shvstvala - * Shtvala - * Shvstvecd - * Shgatpa - recommendations: - - text: | - Implementations are strongly recommended to raise illegal-instruction - exceptions on attempts to execute unimplemented opcodes. + NOTE: Only XLEN=64 application processor profiles are currently defined. + It would be possible to also define very similar XLEN=32 variants. +contributors: + - name: Krste Asanovic + email: krste@sifive.com + company: SiFive +profiles: + - { $ref: profile/RVA22U64.yaml# } + - { $ref: profile/RVA22S64.yaml# } diff --git a/arch/profile_release/RVI20.yaml b/arch/profile_release/RVI20.yaml index 3c3b1f63d..55a5a265d 100644 --- a/arch/profile_release/RVI20.yaml +++ b/arch/profile_release/RVI20.yaml @@ -1,89 +1,22 @@ -RVI20: - name: RVI20 - marketing_name: RVI20 - class: RVI - release: 20 - state: ratified # current status ["ratified", "development"] - ratification_date: "2023-04-03" +$schema: profile_release_schema.json# +kind: profile release +name: RVI20 +marketing_name: RVI20 +class: RVI +release: 20 +state: ratified # current status ["ratified", "development"] +ratification_date: "2023-04-03" - # Semantic versions within the release - versions: - - version: "1.0.0" +# Semantic versions within the release +versions: + - version: "1.0.0" - introduction: | - The two profiles RVI20U32 and RVI20U64 correspond to the RV32I and RV64I base ISAs respectively. - contributors: - - name: Krste Asanovic - email: krste@sifive.com - company: SiFive - profiles: - RVI20U32: - marketing_name: RVI20U32 - mode: Unpriv - base: 32 - release: RVI20 - introduction: | - This profile specifies the ISA features available to generic unprivileged - execution environments. - extensions: - I: - presence: mandatory - version: "~> 2.1" - note: | - RVI is the mandatory base ISA for RVA, and is little-endian. - - As per the unprivileged architecture specification, the `ecall` - instruction causes a requested trap to the execution environment. - - Misaligned loads and stores might not be supported. - - The `fence.tso` instruction is mandatory. - - NOTE: The `fence.tso` instruction was incorrectly described as - optional in the 2019 ratified specifications. However, `fence.tso` is - encoded within the standard `fence` encoding such that implementations - must treat it as a simple global fence if they do not natively support - TSO-ordering optimizations. As software can always assume without any - penalty that `fence.tso` is being exploited by a hardware - implementation, there is no advantage to making the instruction a - profile option. Later versions of the unprivileged ISA specifications - correctly indicate that `fence.tso` is mandatory. - A: - presence: optional - version: "~> 2.1" - C: - presence: optional - version: "~> 2.2" - D: - presence: optional - version: "~> 2.2" - note: | - NOTE: The rationale to not include Q as a profile option is that - quad-precision floating-point is unlikely to be implemented in - hardware, and so we do not require or expect software to expend effort - optimizing use of Q instructions in case they are present. - F: - presence: optional - version: "~> 2.2" - M: - presence: optional - version: "~> 2.0" - Zicntr: - presence: optional - version: " = 2.0" - Zihpm: - presence: optional - version: "~> 2.0" - note: | - The number of counters is platform-specific. - Zifencei: - presence: optional - version: "~> 2.0" - recommendations: - - text: | - Implementations are strongly recommended to raise illegal-instruction - exceptions on attempts to execute unimplemented opcodes. - RVI20U64: - $inherits: "#/RVI20/profiles/RVI20U32" - base: 64 - marketing_name: RVI20U64 +introduction: | + The two profiles RVI20U32 and RVI20U64 correspond to the RV32I and RV64I base ISAs respectively. +contributors: + - name: Krste Asanovic + email: krste@sifive.com + company: SiFive +profiles: + - { $ref: profile/RVI20U32.yaml# } + - { $ref: profile/RVI20U64.yaml# } diff --git a/backends/arch_gen/lib/arch_gen.rb b/backends/arch_gen/lib/arch_gen.rb index e88857923..cea28daf9 100644 --- a/backends/arch_gen/lib/arch_gen.rb +++ b/backends/arch_gen/lib/arch_gen.rb @@ -8,7 +8,7 @@ require "tilt" require_relative "#{$lib}/validate" -require_relative "#{$lib}/arch_def" +require_relative "#{$lib}/cfg_arch" $root = Pathname.new(__FILE__).dirname.dirname.realpath if $root.nil? @@ -201,7 +201,7 @@ def generate gen_inst_def - gen_arch_def + gen_cfg_arch @generate_done = true end @@ -272,7 +272,7 @@ def implemented_extensions # Generate the config-specific, unified architecture spec data structure # - def gen_arch_def + def gen_cfg_arch csr_ary = Dir.glob(@gen_dir / "arch" / "csr" / "**" / "*.yaml").map do |f| YamlLoader.load(f, permitted_classes:[Date]) end @@ -286,24 +286,24 @@ def gen_arch_def profile_class_obj = YamlLoader.load(f, permitted_classes:[Date]) profile_class_name = profile_class_obj.keys[0] profile_class_obj[profile_class_name]["name"] = profile_class_name - profile_class_obj[profile_class_name]["__source"] = f + profile_class_obj[profile_class_name]["$source"] = f [profile_class_name, profile_class_obj[profile_class_name]] end.to_h profile_release_hash = Dir.glob($root / "arch" / "profile_release" / "**" / "*.yaml").map do |f| profile_release_obj = YamlLoader.load(f, permitted_classes:[Date]) profile_release_name = profile_release_obj.keys[0] profile_release_obj[profile_release_name]["name"] = profile_release_name - profile_release_obj[profile_release_name]["__source"] = f + profile_release_obj[profile_release_name]["$source"] = f [profile_release_name, profile_release_obj[profile_release_name]] end.to_h cert_class_ary = Dir.glob($root / "arch" / "certificate_class" / "**" / "*.yaml").map do |f| cert_class_obj = YamlLoader.load(f, permitted_classes:[Date]) - cert_class_obj["__source"] = f + cert_class_obj["$source"] = f cert_class_obj end cert_model_ary = Dir.glob($root / "arch" / "certificate_model" / "**" / "*.yaml").map do |f| cert_model_obj = YamlLoader.load(f, permitted_classes:[Date]) - cert_model_obj["__source"] = f + cert_model_obj["$source"] = f cert_model_obj end manual_hash = {} @@ -317,17 +317,17 @@ def gen_arch_def manual_info_file = manual_info_files.first manual_hash[manual_id] = YamlLoader.load(manual_info_file, permitted_classes:[Date]) - manual_hash[manual_id]["__source"] = manual_info_file + manual_hash[manual_id]["$source"] = manual_info_file # TODO: schema validation end manual_hash[manual_id]["versions"] ||= [] manual_hash[manual_id]["versions"] << YamlLoader.load(f, permitted_classes:[Date]) # TODO: schema validation - manual_hash[manual_id]["versions"].last["__source"] = f + manual_hash[manual_id]["versions"].last["$source"] = f end - arch_def = { + cfg_arch = { "type" => @cfg_opts["type"], "params" => params, "instructions" => inst_ary, @@ -343,24 +343,24 @@ def gen_arch_def "manuals" => manual_hash } - yaml = YAML.dump(arch_def) - arch_def_yaml = "# yaml-language-server: $schema=../../../arch/arch_schema.json\n\n#{yaml}" - abs_arch_def_path = @gen_dir / "arch" / "arch_def.yaml" + yaml = YAML.dump(cfg_arch) + cfg_arch_yaml = "# yaml-language-server: $schema=../../../arch/arch_schema.json\n\n#{yaml}" + abs_cfg_arch_path = @gen_dir / "arch" / "cfg_arch.yaml" - # return early if this arch_def hasn't changed - # return if abs_arch_def_path.exist? && (abs_arch_def_path.read == arch_def_yaml) + # return early if this cfg_arch hasn't changed + # return if abs_cfg_arch_path.exist? && (abs_cfg_arch_path.read == cfg_arch_yaml) - File.write(abs_arch_def_path, arch_def_yaml) + File.write(abs_cfg_arch_path, cfg_arch_yaml) # make sure it passes validation # begin - # @validator.validate_str(YAML.dump(arch_def), type: :arch) + # @validator.validate_str(YAML.dump(cfg_arch), type: :arch) # rescue Validator::SchemaValidationError => e - # warn "While validating the unified architecture defintion at #{abs_arch_def_path}" + # warn "While validating the unified architecture defintion at #{abs_cfg_arch_path}" # raise e # end end - private :gen_arch_def + private :gen_cfg_arch # @return [Object] An object that has a method (lowercase) or constant (uppercase) for every config option # so that obj.param_name or obj.PARAM_NAME gets you the value @@ -622,7 +622,7 @@ def maybe_add_csr(csr_name, extra_env = {}) # get the csr data (not including the name key), which is redundant at this point csr_data = YAML.load_file(merged_path) csr_data["fields"].each { |n, f| f["name"] = n } - csr_data["__source"] = og_path.to_s + csr_data["$source"] = og_path.to_s csr_yaml = YAML.dump(csr_data) begin @@ -633,18 +633,18 @@ def maybe_add_csr(csr_name, extra_env = {}) end csr_obj = Csr.new(csr_data) - arch_def_mock = Object.new - arch_def_mock.define_singleton_method(:fully_configured?) { true } + cfg_arch_mock = Object.new + cfg_arch_mock.define_singleton_method(:fully_configured?) { true } pos_xlen_local = possible_xlens - arch_def_mock.define_singleton_method(:possible_xlens) do + cfg_arch_mock.define_singleton_method(:possible_xlens) do pos_xlen_local end impl_ext = @cfg_impl_ext.map { |e| ExtensionVersion.new(e[0], e[1], nil) } - arch_def_mock.define_singleton_method(:implemented_extensions) do + cfg_arch_mock.define_singleton_method(:implemented_extensions) do impl_ext end belongs = - csr_obj.exists_in_cfg?(arch_def_mock) + csr_obj.exists_in_cfg?(cfg_arch_mock) @implemented_csrs ||= [] @@ -864,7 +864,7 @@ def maybe_add_inst(inst_name, extra_env = {}) # get the inst data (not including the name key), which is redundant at this point inst_data = YAML.load_file(merged_path) - inst_data["__source"] = og_path.to_s + inst_data["$source"] = og_path.to_s inst_yaml = YAML.dump(inst_data) begin @@ -890,15 +890,15 @@ def maybe_add_inst(inst_name, extra_env = {}) possible_xlens << 64 if [64, 3264].include?(params["VSXLEN"]) possible_xlens << 64 if [64, 3264].include?(params["VUXLEN"]) end - arch_def_mock = Object.new - arch_def_mock.define_singleton_method(:fully_configured?) { true } - arch_def_mock.define_singleton_method(:possible_xlens) { possible_xlens } + cfg_arch_mock = Object.new + cfg_arch_mock.define_singleton_method(:fully_configured?) { true } + cfg_arch_mock.define_singleton_method(:possible_xlens) { possible_xlens } impl_ext = @cfg_impl_ext.map { |e| ExtensionVersion.new(e[0], e[1], nil) } - arch_def_mock.define_singleton_method(:implemented_extensions) do + cfg_arch_mock.define_singleton_method(:implemented_extensions) do impl_ext end belongs = - inst_obj.exists_in_cfg?(arch_def_mock) + inst_obj.exists_in_cfg?(cfg_arch_mock) @implemented_instructions ||= [] @implemented_instructions << inst_name if belongs diff --git a/backends/arch_gen/tasks.rake b/backends/arch_gen/tasks.rake deleted file mode 100644 index fca19e28e..000000000 --- a/backends/arch_gen/tasks.rake +++ /dev/null @@ -1,181 +0,0 @@ -# frozen_string_literal: true - -# This file contains tasks related to the generation of a configured architecture specification - -require_relative "../../lib/yaml_loader" -require_relative "lib/arch_gen" - -ARCH_GEN_DIR = Pathname.new(__FILE__).dirname - -def arch_def_for(config_name) - config_name = "_" if config_name.nil? - @arch_defs ||= {} - return @arch_defs[config_name] if @arch_defs.key?(config_name) - - @arch_defs[config_name] = - if config_name == "_" - ArchDef.new("_", $root / "gen" / "_" / "arch" / "arch_def.yaml") - else - ArchDef.new( - config_name, - $root / "gen" / config_name / "arch" / "arch_def.yaml", - overlay_path: $root / "cfgs" / config_name / "arch_overlay" - ) - end -end - -file "#{$root}/.stamps/arch-gen.stamp" => ( - [ - "#{$root}/.stamps", - "#{ARCH_GEN_DIR}/lib/arch_gen.rb", - "#{$root}/lib/idl/ast.rb", - "#{ARCH_GEN_DIR}/tasks.rake", - __FILE__ - ] + Dir.glob($root / "arch" / "**" / "*.yaml") -) do |t| - csr_ary = Dir.glob($root / "arch" / "csr" / "**" / "*.yaml").map do |f| - csr_obj = YamlLoader.load(f, permitted_classes:[Date]) - csr_obj["fields"].map do |k, v| - v["name"] = k - [k, v] - end - csr_obj["__source"] = f - csr_obj - end - inst_ary = Dir.glob($root / "arch" / "inst" / "**" / "*.yaml").map do |f| - inst_obj = YamlLoader.load(f, permitted_classes:[Date]) - inst_obj["__source"] = f - inst_obj - end - ext_ary = Dir.glob($root / "arch" / "ext" / "**" / "*.yaml").map do |f| - ext_obj = YamlLoader.load(f, permitted_classes:[Date]) - ext_obj["__source"] = f - ext_obj - end - profile_class_hash = Dir.glob($root / "arch" / "profile_class" / "**" / "*.yaml").map do |f| - profile_class_obj = YamlLoader.load(f, permitted_classes:[Date]) - profile_class_name = profile_class_obj.keys[0] - profile_class_obj[profile_class_name]["name"] = profile_class_name - profile_class_obj[profile_class_name]["__source"] = f - [profile_class_name, profile_class_obj[profile_class_name]] - end.to_h - profile_release_hash = Dir.glob($root / "arch" / "profile_release" / "**" / "*.yaml").map do |f| - profile_release_obj = YamlLoader.load(f, permitted_classes:[Date]) - profile_release_name = profile_release_obj.keys[0] - profile_release_obj[profile_release_name]["name"] = profile_release_name - profile_release_obj[profile_release_name]["__source"] = f - [profile_release_name, profile_release_obj[profile_release_name]] - end.to_h - cert_class_ary = Dir.glob($root / "arch" / "certificate_class" / "**" / "*.yaml").map do |f| - cert_class_obj = YamlLoader.load(f, permitted_classes:[Date]) - cert_class_obj["__source"] = f - cert_class_obj - end - cert_model_ary = Dir.glob($root / "arch" / "certificate_model" / "**" / "*.yaml").map do |f| - cert_model_obj = YamlLoader.load(f, permitted_classes:[Date]) - cert_model_obj["__source"] = f - cert_model_obj - end - manual_hash = {} - Dir.glob($root / "arch" / "manual" / "**" / "contents.yaml").map do |f| - manual_version = YamlLoader.load(f, permitted_classes:[Date]) - manual_id = manual_version["manual"] - unless manual_hash.key?(manual_id) - manual_info_files = Dir.glob($root / "arch" / "manual" / "**" / "#{manual_id}.yaml") - raise "Could not find manual info '#{manual_id}'.yaml, needed by #{f}" if manual_info_files.empty? - raise "Found multiple manual infos '#{manual_id}'.yaml, needed by #{f}" if manual_info_files.size > 1 - - manual_info_file = manual_info_files.first - manual_hash[manual_id] = YamlLoader.load(manual_info_file, permitted_classes:[Date]) - manual_hash[manual_id]["__source"] = manual_info_file - # TODO: schema validation - end - - manual_hash[manual_id]["versions"] ||= [] - manual_hash[manual_id]["versions"] << YamlLoader.load(f, permitted_classes:[Date]) - # TODO: schema validation - manual_hash[manual_id]["versions"].last["__source"] = f - end - - arch_def = { - "type" => "unconfigured", - "instructions" => inst_ary, - "extensions" => ext_ary, - "csrs" => csr_ary, - "profile_classes" => profile_class_hash, - "profile_releases" => profile_release_hash, - "certificate_classes" => cert_class_ary, - "certificate_models" => cert_model_ary, - "manuals" => manual_hash - } - - dest = "#{$root}/gen/_/arch/arch_def.yaml" - FileUtils.mkdir_p File.dirname(dest) - File.write(dest, YAML.dump(arch_def)) - - FileUtils.touch(t.name) -end - -obj_model_files = Dir.glob($root / "lib" / "arch_obj_models" / "*.rb") -obj_model_files << ($root / "lib" / "arch_def.rb") - -arch_files = Dir.glob($root / "arch" / "**" / "*.yaml") - -# stamp to indicate completion of Arch Gen for a given config -rule %r{#{$root}/\.stamps/arch-gen-.*\.stamp} => proc { |tname| - config_name = Pathname.new(tname).basename(".stamp").sub("arch-gen-", "") - config_files = - Dir.glob($root / "cfgs" / config_name / "arch_overlay" / "**" / "*.yaml") + - [($root / "cfgs" / config_name / "params.yaml").to_s] - [ - "#{$root}/.stamps", - "#{ARCH_GEN_DIR}/lib/arch_gen.rb", - "#{$root}/lib/idl/ast.rb", - "#{ARCH_GEN_DIR}/tasks.rake", - arch_files, - config_files, - - # the stamp file is not actually dependent on the Ruby object model, - # but in general we want to rebuild anything using this stamp when the object model changes - obj_model_files.map(&:to_s) - ].flatten -} do |t| - config_name = Pathname.new(t.name).basename(".stamp").sub("arch-gen-", "") - - arch_gen = ArchGen.new(config_name) - puts "Generating architecture definition in #{arch_gen.gen_dir.relative_path_from($root)}" - - arch_gen.generate - - puts " Found #{arch_gen.implemented_csrs.size} CSRs" - puts " Found #{arch_gen.implemented_extensions.size} Extensions" - puts " Found #{arch_gen.implemented_instructions.size} Instructions" - - FileUtils.touch t.name -end - -namespace :gen do - desc "Generate the cfg-specific architecture files for config_name" - task :cfg_arch, [:config_name] do |_t, args| - raise "No config '#{args[:config_name]}' found in cfgs/" unless ($root / "cfgs" / args[:config_name]).directory? - - Rake::Task["#{$root}/.stamps/arch-gen-#{args[:config_name]}.stamp"].invoke(args[:config_name]) - end - - desc "Generate a unified architecture file (configuration independent)" - task :arch do - Rake::Task["#{$root}/.stamps/arch-gen.stamp"].invoke - end -end - -# TODO: Add this back once we settle on the config file format -# namespace :validate do -# desc "Validate that a configuration folder valid for the list of extensions it claims to implement" -# task :cfg, [:config_name] do |_t, args| -# raise "No config '#{args[:config_name]}' found in cfgs/" unless ($root / "cfgs" / args[:config_name]).directory? - -# ArchGen.new(args[:config_name]).validate_params - -# puts "Success! The '#{args[:config_name]}' configuration passes validation checks" -# end -# end diff --git a/backends/certificate_doc/tasks.rake b/backends/certificate_doc/tasks.rake index 33c3d0d29..3972b31f3 100644 --- a/backends/certificate_doc/tasks.rake +++ b/backends/certificate_doc/tasks.rake @@ -22,18 +22,17 @@ Dir.glob("#{$root}/arch/certificate_model/*.yaml") do |f| "#{$root}/arch/certificate_model/#{cert_model_name}.yaml", "#{$root}/arch/certificate_class/#{cert_class_name}.yaml", "#{CERT_DOC_DIR}/templates/certificate.adoc.erb", - __FILE__, - "#{$root}/.stamps/arch-gen-_#{base}.stamp" + __FILE__ ] do |t| # TODO: schema validation - arch_def = arch_def_for("_#{base}") - cert_model = arch_def.cert_model(cert_model_name) + cfg_arch = cfg_arch_for("rv#{base}") + cert_model = cfg_arch.cert_model(cert_model_name) raise "No certificate model defined for #{cert_model_name}" if cert_model.nil? # Switch to the generated certificate arch def # XXX - Add this to profile releases - arch_def = cert_model.to_arch_def - cert_model = arch_def.cert_model(cert_model_name) + cfg_arch = cert_model.to_cfg_arch + cert_model = cfg_arch.cert_model(cert_model_name) cert_class = cert_model.cert_class version = File.basename(t.name, '.adoc').split('-')[1..].join('-') @@ -42,7 +41,7 @@ Dir.glob("#{$root}/arch/certificate_model/*.yaml") do |f| erb.filename = "#{CERT_DOC_DIR}/templates/certificate.adoc.erb" FileUtils.mkdir_p File.dirname(t.name) - File.write t.name, AsciidocUtils.resolve_links(arch_def.find_replace_links(erb.result(binding))) + File.write t.name, AsciidocUtils.resolve_links(cfg_arch.find_replace_links(erb.result(binding))) puts "Generated adoc source at #{t.name}" end diff --git a/backends/certificate_doc/templates/certificate.adoc.erb b/backends/certificate_doc/templates/certificate.adoc.erb index 740a98f76..41e9fb32f 100644 --- a/backends/certificate_doc/templates/certificate.adoc.erb +++ b/backends/certificate_doc/templates/certificate.adoc.erb @@ -117,10 +117,10 @@ None | Requirement ID | Extension | Version | Long Name | Note <% ext_reqs.sort.each do |ext_req| -%> -<% ext = arch_def.extension(ext_req.name) -%> +<% ext = cfg_arch.extension(ext_req.name) -%> | <%= ext_req.req_id %> | <-def,<%= ext_req.name %>>> -| <%= ext_req.version_requirement %> +| <%= ext_req.requirement_specs.map(&:to_s).join(", ") %> | <%= ext.nil? ? "" : ext.long_name %> | <%= ext_req.note.nil? ? "" : ext_req.note %> <% end # each ext_req -%> @@ -219,7 +219,7 @@ None == CSR Summary <% - csrs = cert_model.in_scope_ext_reqs.map { |ext_req| ext_req.csrs(arch_def) }.flatten.uniq + csrs = cert_model.in_scope_ext_reqs.map { |ext_req| ext_req.csrs }.flatten.uniq -%> === By Name @@ -289,16 +289,16 @@ Requirement <%= req.name %> only apply when <%= req.when_pretty %>. [appendix] == Extension Details <% cert_model.in_scope_ext_reqs.sort.each do |ext_req| -%> -<% ext = arch_def.extension(ext_req.name) -%> +<% ext = cfg_arch.extension(ext_req.name) -%> [[ext-<%= ext_req.name %>-def]] === Extension <%= ext_req.name %> + <%= ext.nil? ? "" : "*Long Name*: " + ext.long_name + " +" %> -*Version Requirement*: <%= ext_req.version_requirement %> + +*Version Requirement*: <%= ext_req.requirement_specs.map(&:to_s).join(", ") %> + <% ext.versions.each do |v| -%> -<%= v.version %>:: +<%= v.version_spec %>:: State::: <%= v.state %> <% if v.state == "ratified" -%> @@ -320,7 +320,7 @@ Requirement <%= req.name %> only apply when <%= req.when_pretty %>. <% if v.implications.size > 0 -%> Implies::: <% v.implications.each do |i| -%> - * `<%= i.name %>` version <%= i.version %> + * `<%= i.name %>` version <%= i.version_spec %> <% end -%> <% end -%> <% end -%> @@ -341,7 +341,7 @@ Requirement <%= req.name %> only apply when <%= req.when_pretty %>. <% end -%> // TODO: GitHub issue 92: Use version specified by each profile. -<% insts = arch_def.instructions.select { |i| i.defined_by?(ext.min_version) } -%> +<% insts = cfg_arch.instructions.select { |i| i.defined_by?(ext.min_version) } -%> <% unless insts.empty? -%> ==== Instructions @@ -401,7 +401,7 @@ The following instructions are added by this extension: This instruction is defined by: -<%= inst.defined_by.to_asciidoc %> +<%= inst.defined_by_condition.to_asciidoc %> ==== Encoding @@ -495,13 +495,13 @@ RV64:: <% if inst.key?("operation()") -%> [source,idl,subs="specialchars,macros"] ---- -<%= inst.operation_ast(arch_def.symtab).gen_adoc %> +<%= inst.operation_ast(cfg_arch.symtab).gen_adoc %> ---- <% end -%> ==== Exceptions -<%- exception_list = inst.reachable_exceptions_str(arch_def.symtab) -%> +<%- exception_list = inst.reachable_exceptions_str(cfg_arch.symtab) -%> <%- if exception_list.empty? -%> This instruction does not generate synchronous exceptions. <%- else -%> @@ -521,7 +521,7 @@ This instruction may result in the following synchronous exceptions: == CSR Details <% - csrs = cert_model.in_scope_ext_reqs.map { |ext_req| ext_req.csrs(arch_def) }.flatten.uniq + csrs = cert_model.in_scope_ext_reqs.map { |ext_req| ext_req.csrs }.flatten.uniq csrs.sort_by!(&:name) -%> @@ -548,23 +548,23 @@ h| CSR Address | <%= "0x#{csr.address.to_s(16)}" %> <% if csr.priv_mode == 'VS' -%> h| Virtual CSR Address | <%= "0x#{csr.virtual_address.to_s(16)}" %> <% end -%> -h| Defining extension a| <%= csr.defined_by.to_asciidoc %> -<% if csr.dynamic_length?(arch_def) -%> -h| Length | <%= csr.length_pretty(arch_def) %> +h| Defining extension a| <%= csr.defined_by_condition.to_asciidoc %> +<% if csr.dynamic_length?(cfg_arch) -%> +h| Length | <%= csr.length_pretty(cfg_arch) %> <% else -%> -h| Length | <%= csr.length_pretty(arch_def) %> +h| Length | <%= csr.length_pretty(cfg_arch) %> <% end -%> h| Privilege Mode | <%= csr.priv_mode %> |=== ==== Format -<% unless csr.dynamic_length?(arch_def) || csr.implemented_fields(arch_def).any? { |f| f.dynamic_location?(arch_def) } -%> +<% unless csr.dynamic_length?(cfg_arch) || csr.implemented_fields(cfg_arch).any? { |f| f.dynamic_location?(cfg_arch) } -%> <%# CSR has a known static length, so there is only one format to display -%> .<%= csr.name %> format [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, csr.base.nil? ? 32 : csr.base, optional_type: 2) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, csr.base.nil? ? 32 : csr.base, optional_type: 2) %> .... <% else -%> <%# CSR has a dynamic length, or a field has a dynamic location, @@ -574,13 +574,13 @@ This CSR format changes dynamically with XLEN. .<%= csr.name %> Format when <%= csr.length_cond32 %> [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, 32, optional_type: 2) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, 32, optional_type: 2) %> .... .<%= csr.name %> Format when <%= csr.length_cond64 %> [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, 64, optional_type: 2) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, 64, optional_type: 2) %> .... @@ -593,34 +593,34 @@ This CSR format changes dynamically with XLEN. |=== @ Name @ Location @ Type @ Reset Value -<%- csr.implemented_fields(arch_def).each do |field| -%> +<%- csr.implemented_fields(cfg_arch).each do |field| -%> @ xref:<%=csr.name%>-<%=field.name%>-def[`<%= field.name %>`] a@ -<%- if field.dynamic_location?(arch_def) -%> +<%- if field.dynamic_location?(cfg_arch) -%> [when,"<%= field.location_cond32 %>"] -- -<%= field.location_pretty(arch_def, 32) %> +<%= field.location_pretty(cfg_arch, 32) %> -- [when,"<%= field.location_cond64 %>"] -- -<%= field.location_pretty(arch_def, 64) %> +<%= field.location_pretty(cfg_arch, 64) %> -- <%- else -%> -<%= field.location_pretty(arch_def) %> +<%= field.location_pretty(cfg_arch) %> <%- end -%> a@ -- -<%= field.type_pretty(arch_def.symtab) %> +<%= field.type_pretty(cfg_arch.symtab) %> -- a@ -- -<%= field.reset_value_pretty(arch_def) %> +<%= field.reset_value_pretty(cfg_arch) %> -- <%- end -%> @@ -628,11 +628,11 @@ a@ ==== Fields -<%- if csr.implemented_fields(arch_def).empty? -%> +<%- if csr.implemented_fields(cfg_arch).empty? -%> This CSR has no fields. However, it must still exist (not cause an `Illegal Instruction` trap) and always return zero on a read. <%- else -%> -<%- csr.implemented_fields(arch_def).each do |field| -%> +<%- csr.implemented_fields(cfg_arch).each do |field| -%> [[<%=csr.name%>-<%=field.name%>-def]] ===== `<%= field.name %>` @@ -642,23 +642,23 @@ IMPORTANT: <%= field.name %> is only defined in <%= field.base32_only? ? "RV32" **** Location:: -<%= field.location_pretty(arch_def) %> +<%= field.location_pretty(cfg_arch) %> Description:: <%= field.description.gsub("\n", " +\n") %> Type:: -<%= field.type_pretty(arch_def.symtab) %> +<%= field.type_pretty(cfg_arch.symtab) %> Reset value:: -<%= field.reset_value_pretty(arch_def) %> +<%= field.reset_value_pretty(cfg_arch) %> **** <%- end -%> <%- end -%> -<%- if csr.implemented_fields(arch_def).map(&:has_custom_sw_write?).any? -%> +<%- if csr.implemented_fields(cfg_arch).map(&:has_custom_sw_write?).any? -%> ==== Software write This CSR may store a value that is different from what software attempts to write. @@ -668,7 +668,7 @@ written value: [idl] ---- -<%- csr.implemented_fields(arch_def).each do |field| -%> +<%- csr.implemented_fields(cfg_arch).each do |field| -%> <%- if field.has_custom_sw_write? -%> <%= field.name %> = <%= field["sw_write(csr_value)"] %> <%- else -%> @@ -685,7 +685,7 @@ This CSR may return a value that is different from what is stored in hardware. [source,idl,subs="specialchars,macros"] ---- -<%= csr.sw_read_ast(arch_def.symtab).gen_adoc %> +<%= csr.sw_read_ast(cfg_arch.symtab).gen_adoc %> ---- <%- end -%> diff --git a/backends/cfg_html_doc/adoc_gen.rake b/backends/cfg_html_doc/adoc_gen.rake index 89ccf01ea..1bf407888 100644 --- a/backends/cfg_html_doc/adoc_gen.rake +++ b/backends/cfg_html_doc/adoc_gen.rake @@ -5,11 +5,9 @@ require "ruby-prof" # fill out templates for every csr, inst, ext, and func ["csr", "inst", "ext", "func"].each do |type| rule %r{#{$root}/\.stamps/adoc-gen-#{type}s-.*\.stamp} => proc { |tname| - config_name = Pathname.new(tname).basename(".stamp").sub("adoc-gen-#{type}s-", "") [ - "#{$root}/.stamps/arch-gen-#{config_name}.stamp", "#{CFG_HTML_DOC_DIR}/templates/#{type}.adoc.erb", - "#{$root}/lib/arch_def.rb", + "#{$root}/lib/cfg_arch.rb", "#{$root}/lib/idl/passes/gen_adoc.rb", __FILE__, "#{$root}/.stamps" @@ -17,7 +15,7 @@ require "ruby-prof" } do |t| config_name = Pathname.new(t.name).basename(".stamp").sub("adoc-gen-#{type}s-", "") - arch_def = arch_def_for(config_name) + cfg_arch = cfg_arch_for(config_name) adoc_template_path = CFG_HTML_DOC_DIR / "templates" / "#{type}.adoc.erb" adoc_template = adoc_template_path.read erb = ERB.new(adoc_template, trim_mode: "-") @@ -28,32 +26,32 @@ require "ruby-prof" case type when "csr" - arch_def.implemented_csrs.each do |csr| + cfg_arch.transitive_implemented_csrs.each do |csr| path = dir_path / "#{csr.name}.adoc" puts " Generating #{path}" - File.write(path, arch_def.find_replace_links(erb.result(binding))) + File.write(path, cfg_arch.find_replace_links(erb.result(binding))) end when "inst" - arch_def.implemented_instructions.each do |inst| + cfg_arch.transitive_implemented_instructions.each do |inst| path = dir_path / "#{inst.name}.adoc" puts " Generating #{path}" # RubyProf.start - File.write(path, arch_def.find_replace_links(erb.result(binding))) + File.write(path, cfg_arch.find_replace_links(erb.result(binding))) # result = RubyProf.stop # RubyProf::FlatPrinter.new(result).print(STDOUT) end when "ext" - arch_def.implemented_extensions.each do |ext_version| - ext = arch_def.extension(ext_version.name) + cfg_arch.transitive_implemented_extensions.each do |ext_version| + ext = cfg_arch.extension(ext_version.name) path = dir_path / "#{ext.name}.adoc" puts " Generating #{path}" - File.write(path, arch_def.find_replace_links(erb.result(binding))) + File.write(path, cfg_arch.find_replace_links(erb.result(binding))) end when "func" - global_symtab = arch_def.symtab + global_symtab = cfg_arch.symtab path = dir_path / "funcs.adoc" puts " Generating #{path}" - File.write(path, arch_def.find_replace_links(erb.result(binding))) + File.write(path, cfg_arch.find_replace_links(erb.result(binding))) else raise "todo" end @@ -74,55 +72,51 @@ require "ruby-prof" config_name = Pathname.new(t.name).relative_path_from("#{$root}/gen/cfg_html_doc").to_s.split("/")[0] - arch_def = arch_def_for(config_name) + cfg_arch = cfg_arch_for(config_name) lines = [ "= Implemented #{to_long[type]}", "", - "The following are implemented by the #{arch_def.name} configuration:", + "The following are implemented by the #{cfg_arch.name} configuration:", "" ] case type when "csr" puts "Generting full CSR list" - arch_def.implemented_csrs.each do |csr| + cfg_arch.transitive_implemented_csrs.each do |csr| lines << " * `#{csr.name}` #{csr.long_name}" end when "ext" puts "Generting full extension list" - arch_def.implemented_extensions.each do |ext_version| + cfg_arch.transitive_implemented_extensions.each do |ext_version| lines << " * `#{ext_version.name}` #{ext_version.ext.long_name}" end when "inst" puts "Generting full instruction list" - arch_def.implemented_instructions.each do |inst| + cfg_arch.transitive_implemented_instructions.each do |inst| lines << " * `#{inst.name}` #{inst.long_name}" end when "func" puts "Generting function list" - arch_def.implemented_functions.each do |func| + cfg_arch.implemented_functions.each do |func| lines << " * `#{func.name}`" end else raise "Unsupported type" end - File.write t.name, arch_def.find_replace_links(lines.join("\n")) + File.write t.name, cfg_arch.find_replace_links(lines.join("\n")) end end -rule %r{#{$root}/gen/cfg_html_doc/.*/adoc/ROOT/landing.adoc} => proc { |tname| - config_name = Pathname.new(tname).relative_path_from("#{$root}/gen/cfg_html_doc").to_s.split("/")[0] - [ - "#{$root}/\.stamps/arch-gen-#{config_name}\.stamp", - "#{CFG_HTML_DOC_DIR}/templates/landing.adoc.erb", - __FILE__ - ] -} do |t| +rule %r{#{$root}/gen/cfg_html_doc/.*/adoc/ROOT/landing.adoc} => [ + "#{CFG_HTML_DOC_DIR}/templates/landing.adoc.erb", + __FILE__ +] do |t| config_name = Pathname.new(t.name).relative_path_from("#{$root}/gen/cfg_html_doc").to_s.split("/")[0] - arch_def = arch_def_for(config_name) + cfg_arch = cfg_arch_for(config_name) puts "Generating landing page for #{config_name}" erb = ERB.new(File.read("#{CFG_HTML_DOC_DIR}/templates/landing.adoc.erb"), trim_mode: "-") @@ -131,7 +125,6 @@ rule %r{#{$root}/gen/cfg_html_doc/.*/adoc/ROOT/landing.adoc} => proc { |tname| File.write t.name, erb.result(binding) end - namespace :gen do desc "Generate Asciidoc source for config into gen/CONFIG_NAME/adoc" task :adoc, [:config_name] do |_t, args| diff --git a/backends/cfg_html_doc/html_gen.rake b/backends/cfg_html_doc/html_gen.rake index 24b1f3b69..1a2534fa2 100644 --- a/backends/cfg_html_doc/html_gen.rake +++ b/backends/cfg_html_doc/html_gen.rake @@ -58,7 +58,6 @@ rule %r{#{$root}/gen/cfg_html_doc/.*/antora/modules/nav.adoc} => proc { |tname| Dir.glob("#{$root}/gen/cfg_html_doc/#{config_name}/antora/modules/exts/**/*.adoc") + [ "#{CFG_HTML_DOC_DIR}/templates/toc.adoc.erb", - "#{$root}/.stamps/arch-gen-#{config_name}.stamp", __FILE__ ] } do |t| @@ -68,7 +67,7 @@ rule %r{#{$root}/gen/cfg_html_doc/.*/antora/modules/nav.adoc} => proc { |tname| erb = ERB.new(toc_path.read, trim_mode: "-") erb.filename = toc_path.to_s - arch_def = arch_def_for(config_name) + cfg_arch = cfg_arch_for(config_name) File.write t.name, AntoraUtils.resolve_links(erb.result(binding)) end @@ -76,7 +75,6 @@ rule %r{#{$root}/gen/cfg_html_doc/.*/antora/modules/ROOT/pages/config.adoc} => p config_name = Pathname.new(tname).relative_path_from("#{$root}/gen/cfg_html_doc").to_s.split("/")[0] [ "#{CFG_HTML_DOC_DIR}/templates/config.adoc.erb", - "#{$root}/.stamps/arch-gen-#{config_name}.stamp", __FILE__ ] } do |t| @@ -86,9 +84,9 @@ rule %r{#{$root}/gen/cfg_html_doc/.*/antora/modules/ROOT/pages/config.adoc} => p erb = ERB.new(config_path.read, trim_mode: "-") erb.filename = config_path.to_s - arch_def = arch_def_for(config_name) + cfg_arch = cfg_arch_for(config_name) FileUtils.mkdir_p File.dirname(t.name) - File.write t.name, AntoraUtils.resolve_links(arch_def.find_replace_links(erb.result(binding))) + File.write t.name, AntoraUtils.resolve_links(cfg_arch.find_replace_links(erb.result(binding))) end rule %r{#{$root}/gen/cfg_html_doc/.*/antora/modules/ROOT/pages/landing.adoc} => proc { |tname| @@ -99,10 +97,10 @@ rule %r{#{$root}/gen/cfg_html_doc/.*/antora/modules/ROOT/pages/landing.adoc} => ] } do |t| config_name = Pathname.new(t.name).relative_path_from("#{$root}/gen/cfg_html_doc").to_s.split("/")[0] - arch_def = arch_def_for(config_name) + cfg_arch = cfg_arch_for(config_name) FileUtils.mkdir_p File.dirname(t.name) - File.write t.name, AntoraUtils.resolve_links(arch_def.find_replace_links(File.read(t.prerequisites[0]))) + File.write t.name, AntoraUtils.resolve_links(cfg_arch.find_replace_links(File.read(t.prerequisites[0]))) end rule %r{#{$root}/gen/cfg_html_doc/.*/antora/antora.yml} => proc { |tname| diff --git a/backends/cfg_html_doc/tasks.rake b/backends/cfg_html_doc/tasks.rake index 3f31ece26..31a8e5f07 100644 --- a/backends/cfg_html_doc/tasks.rake +++ b/backends/cfg_html_doc/tasks.rake @@ -1,6 +1,6 @@ # frozen_string_literal: true -require_relative "#{$lib}/arch_def" +require_relative "#{$lib}/cfg_arch" CFG_HTML_DOC_DIR = Pathname.new(__FILE__).dirname diff --git a/backends/cfg_html_doc/templates/config.adoc.erb b/backends/cfg_html_doc/templates/config.adoc.erb index f6f75655a..73898d049 100644 --- a/backends/cfg_html_doc/templates/config.adoc.erb +++ b/backends/cfg_html_doc/templates/config.adoc.erb @@ -1,17 +1,17 @@ -= Configuration of <%= arch_def.name %> += Configuration of <%= cfg_arch.name %> == Extensions |=== | Name | Version -<%- arch_def.implemented_extensions.sort{ |a,b| a.name <=> b.name }.each do |e| -%> -| `<%= e.name %>` | <%= e.version.to_s %> +<%- cfg_arch.transitive_implemented_extensions.sort{ |a,b| a.name <=> b.name }.each do |e| -%> +| `<%= e.name %>` | <%= e.version_spec %> <%- end -%> |=== == Parameters -<%- arch_def.params_with_value.sort_by { |param| param.name }.each do |param| -%> +<%- cfg_arch.params_with_value.sort_by { |param| param.name }.each do |param| -%> === *<%= param.name %>* [cols="1,4,1"] diff --git a/backends/cfg_html_doc/templates/csr.adoc.erb b/backends/cfg_html_doc/templates/csr.adoc.erb index 671ddb68a..a2ef0c9ea 100644 --- a/backends/cfg_html_doc/templates/csr.adoc.erb +++ b/backends/cfg_html_doc/templates/csr.adoc.erb @@ -5,7 +5,7 @@ *<%= csr.long_name %>* -<%= arch_def.render_erb(csr.description) %> +<%= cfg_arch.render_erb(csr.description) %> == Attributes [%autowidth] @@ -14,22 +14,22 @@ h| CSR Address | <%= "0x#{csr.address.to_s(16)}" %> <%- if csr.priv_mode == 'VS' -%> h| Virtual CSR Address | <%= "0x#{csr.virtual_address.to_s(16)}" %> <%- end -%> -h| Defining extension a| <%= csr.defined_by.to_asciidoc %> -<%- if csr.dynamic_length?(arch_def) -%> -h| Length | <%= csr.length_pretty(arch_def) %> +h| Defining extension a| <%= csr.defined_by_condition.to_asciidoc %> +<%- if csr.dynamic_length?(cfg_arch) -%> +h| Length | <%= csr.length_pretty(cfg_arch) %> <%- else -%> -h| Length | <%= csr.length_pretty(arch_def) %> +h| Length | <%= csr.length_pretty(cfg_arch) %> <%- end -%> h| Privilege Mode | <%= csr.priv_mode %> |=== == Format -<%- unless csr.dynamic_length?(arch_def) || csr.fields.any? { |f| f.dynamic_location?(arch_def) } -%> +<%- unless csr.dynamic_length?(cfg_arch) || csr.fields.any? { |f| f.dynamic_location?(cfg_arch) } -%> <%# CSR has a known static length, so there is only one format to display -%> .<%= csr.name %> format [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, arch_def.param_values["XLEN"], exclude_unimplemented: true) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, cfg_arch.param_values["XLEN"], exclude_unimplemented: true) %> .... <%- else -%> <%# CSR has a dynamic length, or a field has a dynamic location, @@ -39,13 +39,13 @@ This CSR format changes dynamically. .<%= csr.name %> Format when <%= csr.length_cond32 %> [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, 32, exclude_unimplemented: true) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, 32, exclude_unimplemented: true) %> .... .<%= csr.name %> Format when <%= csr.length_cond64 %> [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, 64, exclude_unimplemented: true) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, 64, exclude_unimplemented: true) %> .... @@ -57,42 +57,42 @@ This CSR format changes dynamically. |=== |Name | Location | Type | Reset Value -<%- csr.implemented_fields(arch_def).each do |field| -%> +<%- csr.implemented_fields(cfg_arch).each do |field| -%> | `xref:<%=csr.name%>-<%=field.name%>-def[<%= field.name %>]` -| <%= field.location_pretty(arch_def) %> -| <%= field.type(arch_def.symtab) %> -| <%= field.reset_value(arch_def) %> +| <%= field.location_pretty(cfg_arch) %> +| <%= field.type(cfg_arch.symtab) %> +| <%= field.reset_value(cfg_arch) %> <%- end -%> |=== == Fields -<%- if csr.implemented_fields(arch_def).empty? -%> +<%- if csr.implemented_fields(cfg_arch).empty? -%> This CSR has no fields. However, it must still exist (not cause an `Illegal Instruction` trap) and always return zero on a read. <%- else -%> -<%- csr.implemented_fields(arch_def).each do |field| -%> +<%- csr.implemented_fields(cfg_arch).each do |field| -%> [[<%=csr.name%>-<%=field.name%>-def]] === `<%= field.name %>` [.csr-field-info] -- Location:: -`<%=field.csr.name%>[<%= field.location_pretty(arch_def) %>]` +`<%=field.csr.name%>[<%= field.location_pretty(cfg_arch) %>]` Description:: -<%= arch_def.render_erb(field.description).gsub("\n\n", "\n+\n") %> +<%= cfg_arch.render_erb(field.description).gsub("\n\n", "\n+\n") %> Type:: [%autowidth] |=== -| <%= field.type(arch_def.symtab) %> | <%= field.type_desc(arch_def) %> +| <%= field.type(cfg_arch.symtab) %> | <%= field.type_desc(cfg_arch) %> |=== Reset value:: -<%= field.reset_value(arch_def) %> +<%= field.reset_value(cfg_arch) %> <%- if field.has_custom_sw_write? -%> Software write:: @@ -100,27 +100,27 @@ This field has special behavior when written by software (_e.g._, through `csrrw + When software tries to write `csr_value`, the field will be written with the return value of the function below. + -<%- if arch_def.multi_xlen? && csr.defined_in_all_bases? && field.defined_in_all_bases? -%> +<%- if cfg_arch.multi_xlen? && csr.defined_in_all_bases? && field.defined_in_all_bases? -%> [tabs] ==== RV32:: + [source,idl,subs="specialchars,macros"] ---- -<%= field.pruned_sw_write_ast(arch_def, 32).gen_adoc %> +<%= field.pruned_sw_write_ast(cfg_arch, 32).gen_adoc %> ---- RV64:: + [source,idl,subs="specialchars,macros"] ---- -<%= field.pruned_sw_write_ast(arch_def, 64).gen_adoc %> +<%= field.pruned_sw_write_ast(cfg_arch, 64).gen_adoc %> ---- <%- else -%> -<%- xlen = !arch_def.multi_xlen? ? arch_def.mxlen : (!csr.defined_in_all_bases? ? csr.base : field.base) -%> +<%- xlen = !cfg_arch.multi_xlen? ? cfg_arch.mxlen : (!csr.defined_in_all_bases? ? csr.base : field.base) -%> [source,idl,subs="specialchars,macros"] ---- -<%= field.pruned_sw_write_ast(arch_def, xlen).gen_adoc %> +<%= field.pruned_sw_write_ast(cfg_arch, xlen).gen_adoc %> ---- <%- end -%> <%- end -%> @@ -141,14 +141,14 @@ Pruned:: + [source,idl,subs="specialchars,macros"] ---- -<%= csr.pruned_sw_read_ast(arch_def).gen_adoc %> +<%= csr.pruned_sw_read_ast(cfg_arch).gen_adoc %> ---- Original:: + [source,idl,subs="specialchars,macros"] ---- -<%= csr.type_checked_sw_read_ast(arch_def.symtab).gen_adoc %> +<%= csr.type_checked_sw_read_ast(cfg_arch.symtab).gen_adoc %> ---- ==== <%- end -%> diff --git a/backends/cfg_html_doc/templates/ext.adoc.erb b/backends/cfg_html_doc/templates/ext.adoc.erb index eaa8d34ef..8111a2ae5 100644 --- a/backends/cfg_html_doc/templates/ext.adoc.erb +++ b/backends/cfg_html_doc/templates/ext.adoc.erb @@ -2,13 +2,13 @@ = <%= ext.name %> Extension <%= ext.long_name %> -Implemented Version:: <%= ext_version.version %> +Implemented Version:: <%= ext_version.version_str %> == Versions <%- ext.versions.each do |v| -%> -<%- implemented = arch_def.implemented_extensions.include?(v) -%> -<%= v.version %>:: +<%- implemented = cfg_arch.transitive_implemented_extensions.include?(v) -%> +<%= v.version_str %>:: Ratification date::: <%= v.ratification_date %> <%- unless v.changes.empty? -%> @@ -26,7 +26,7 @@ Implemented Version:: <%= ext_version.version %> <%- unless v.implications.empty? -%> Implies::: <%- v.implications.each do |i| -%> - * `<%= i.name %>` version <%= i.version %> + * `<%= i.name %>` version <%= i.version_str %> <%- end -%> <%- end -%> <%- end -%> @@ -35,11 +35,11 @@ Implemented Version:: <%= ext_version.version %> <%= ext.description %> -<%- insts = arch_def.implemented_instructions.select { |i| i.definedBy == ext.name || i.definedBy.include?(ext.name) } -%> +<%- insts = cfg_arch.transitive_implemented_instructions.select { |i| i.definedBy == ext.name || i.definedBy.include?(ext.name) } -%> <%- unless insts.empty? -%> == Instructions -The following instructions are added by this extension in the <%= ext.arch_def.name %> configuration: +The following instructions are added by this extension in the <%= ext.cfg_arch.name %> configuration: [cols="1,3"] |=== diff --git a/backends/cfg_html_doc/templates/func.adoc.erb b/backends/cfg_html_doc/templates/func.adoc.erb index 3c281bbff..8a4746540 100644 --- a/backends/cfg_html_doc/templates/func.adoc.erb +++ b/backends/cfg_html_doc/templates/func.adoc.erb @@ -4,7 +4,7 @@ = Functions -<%- arch_def.implemented_functions.each do |f| -%> +<%- cfg_arch.implemented_functions.each do |f| -%> [#<%= f.name %>-func-def] == <%= f.name %><%- if f.builtin? -%> (builtin)<%- end -%> diff --git a/backends/cfg_html_doc/templates/inst.adoc.erb b/backends/cfg_html_doc/templates/inst.adoc.erb index dd0a2c5a1..d34aa2124 100644 --- a/backends/cfg_html_doc/templates/inst.adoc.erb +++ b/backends/cfg_html_doc/templates/inst.adoc.erb @@ -7,11 +7,11 @@ This instruction is defined by: -<%- inst.defined_by.to_asciidoc -%> +<%- inst.defined_by_condition.to_asciidoc -%> == Encoding -<%- if arch_def.multi_xlen? && inst.multi_encoding? -%> +<%- if cfg_arch.multi_xlen? && inst.multi_encoding? -%> [NOTE] This instruction has different encodings in RV32 and RV64. @@ -34,7 +34,7 @@ RV64:: <%- else -%> [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump inst.wavedrom_desc(arch_def.param_values["XLEN"]) %> +<%= JSON.dump inst.wavedrom_desc(cfg_arch.param_values["XLEN"]) %> .... <%- end -%> @@ -43,18 +43,18 @@ RV64:: <%= inst.description %> == Access -<%- if arch_def.ext?(:H) -%> +<%- if cfg_arch.ext?(:H) -%> [cols="^,^,^,^,^"] <%- else -%> [cols="^,^,^"] <%- end -%> |=== -| M | <%- if arch_def.ext?(:H) -%>HS<%- else -%>S<%- end -%> | U <%- if arch_def.ext?(:H) -%> | VS | VU <%- end -%> +| M | <%- if cfg_arch.ext?(:H) -%>HS<%- else -%>S<%- end -%> | U <%- if cfg_arch.ext?(:H) -%> | VS | VU <%- end -%> | [.access-always]#Always# | [.access-<%=inst.access['s']%>]#<%= inst.access['s'].capitalize %># | [.access-<%=inst.access['u']%>]#<%= inst.access['u'].capitalize %># -<% if arch_def.ext?(:H) %> +<% if cfg_arch.ext?(:H) %> | [.access-<%=inst.access['vs']%>]#<%= inst.access['vs'].capitalize %># | [.access-<%=inst.access['vu']%>]#<%= inst.access['vu'].capitalize %># <% end %> @@ -66,7 +66,7 @@ RV64:: == Decode Variables -<%- if arch_def.multi_xlen? && inst.multi_encoding? -%> +<%- if cfg_arch.multi_xlen? && inst.multi_encoding? -%> [tabs] ==== RV32:: @@ -90,7 +90,7 @@ RV64:: <%- else -%> [source,idl] ---- -<%- inst.decode_variables(arch_def.param_values["XLEN"]).each do |d| -%> +<%- inst.decode_variables(cfg_arch.param_values["XLEN"]).each do |d| -%> <%= d.sext? ? 'signed ' : '' %>Bits<<%= d.size %>> <%= d.name %> = <%= d.extract %>; <%- end -%> ---- @@ -98,7 +98,7 @@ RV64:: == Execution -<%- xlens = inst.base.nil? ? (arch_def.multi_xlen? ? [32, 64] : [arch_def.mxlen]) : [inst.base] -%> +<%- xlens = inst.base.nil? ? (cfg_arch.multi_xlen? ? [32, 64] : [cfg_arch.mxlen]) : [inst.base] -%> <%- if inst.key?("operation()") -%> [tabs] @@ -108,7 +108,7 @@ Pruned, XLEN == <%= effective_xlen %>:: + [source,idl,subs="specialchars,macros"] ---- -<%= inst.pruned_operation_ast(arch_def.symtab, effective_xlen).gen_adoc %> +<%= inst.pruned_operation_ast(cfg_arch.symtab, effective_xlen).gen_adoc %> ---- <%- end -%> @@ -116,12 +116,12 @@ Original:: + [source,idl,subs="specialchars,macros"] ---- -<%= inst.operation_ast(arch_def.idl_compiler).gen_adoc %> +<%= inst.operation_ast(cfg_arch.idl_compiler).gen_adoc %> ---- ==== <%- end -%> -<%- exception_list = inst.reachable_exceptions_str(arch_def.symtab) -%> +<%- exception_list = inst.reachable_exceptions_str(cfg_arch.symtab) -%> <%- unless exception_list.empty? -%> == Exceptions diff --git a/backends/cfg_html_doc/templates/landing.adoc.erb b/backends/cfg_html_doc/templates/landing.adoc.erb index 6f66fa3ce..2f2455393 100644 --- a/backends/cfg_html_doc/templates/landing.adoc.erb +++ b/backends/cfg_html_doc/templates/landing.adoc.erb @@ -1,6 +1,6 @@ -= RISC-V Specification for <%= arch_def.name %> += RISC-V Specification for <%= cfg_arch.name %> -This site contains the full ISA specification for the <%= arch_def.name %> configuration. +This site contains the full ISA specification for the <%= cfg_arch.name %> configuration. You can find the following: * xref:ROOT:config.adoc[Configuration parameters] diff --git a/backends/cfg_html_doc/templates/toc.adoc.erb b/backends/cfg_html_doc/templates/toc.adoc.erb index 6c951bccc..4f6b5d269 100644 --- a/backends/cfg_html_doc/templates/toc.adoc.erb +++ b/backends/cfg_html_doc/templates/toc.adoc.erb @@ -1,17 +1,17 @@ * xref:ROOT:config.adoc[Configuration] .Extensions -<%- arch_def.implemented_extensions.sort { |a, b| a.name <=> b.name }.each do |ext| -%> +<%- cfg_arch.transitive_implemented_extensions.sort { |a, b| a.name <=> b.name }.each do |ext| -%> * %%LINK%ext;<%= ext.name %>;<%= ext.name %>%% <%- end -%> .Control and Status Registers -<%- arch_def.implemented_csrs.sort { |a, b| a.name <=> b.name }.each do |csr| -%> +<%- cfg_arch.transitive_implemented_csrs.sort { |a, b| a.name <=> b.name }.each do |csr| -%> * %%LINK%csr;<%= csr.name %>;<%= csr.name %>%% <%- end -%> .Instructions -<%- arch_def.implemented_instructions.sort { |a, b| a.name <=> b.name }.each do |inst| -%> +<%- cfg_arch.transitive_implemented_instructions.sort { |a, b| a.name <=> b.name }.each do |inst| -%> * %%LINK%inst;<%= inst.name %>;<%= inst.name %>%% <%- end -%> diff --git a/backends/cfg_html_doc/ui/highlight.js b/backends/cfg_html_doc/ui/highlight.js index 89625009a..cd1cbaa22 100644 --- a/backends/cfg_html_doc/ui/highlight.js +++ b/backends/cfg_html_doc/ui/highlight.js @@ -1,4 +1,4 @@ -!(function() { +!(function () { function e(e) { return { aliases: ["adoc"], @@ -12,65 +12,65 @@ relevance: 10, variants: [ { begin: "^(={1,5}) .+?( \\1)?$" }, - { begin: "^[^\\[\\]\\n]+?\\n[=\\-~\\^\\+]{2,}$" } - ] + { begin: "^[^\\[\\]\\n]+?\\n[=\\-~\\^\\+]{2,}$" }, + ], }, { className: "meta", begin: "^:.+?:", end: "\\s", excludeEnd: !0, - relevance: 10 + relevance: 10, }, { className: "meta", begin: "^\\[.+?\\]$", relevance: 0 }, { className: "quote", begin: "^_{4,}\\n", end: "\\n_{4,}$", - relevance: 10 + relevance: 10, }, { className: "code", begin: "^[\\-\\.]{4,}\\n", end: "\\n[\\-\\.]{4,}$", - relevance: 10 + relevance: 10, }, { begin: "^\\+{4,}\\n", end: "\\n\\+{4,}$", contains: [ - { begin: "<", end: ">", subLanguage: "xml", relevance: 0 } + { begin: "<", end: ">", subLanguage: "xml", relevance: 0 }, ], - relevance: 10 + relevance: 10, }, { className: "bullet", begin: "^(\\*+|\\-+|\\.+|[^\\n]+?::)\\s+" }, { className: "symbol", begin: "^(NOTE|TIP|IMPORTANT|WARNING|CAUTION):\\s+", - relevance: 10 + relevance: 10, }, { className: "strong", begin: "\\B\\*(?![\\*\\s])", end: "(\\n{2}|\\*)", - contains: [{ begin: "\\\\*\\w", relevance: 0 }] + contains: [{ begin: "\\\\*\\w", relevance: 0 }], }, { className: "emphasis", begin: "\\B'(?!['\\s])", end: "(\\n{2}|')", contains: [{ begin: "\\\\'\\w", relevance: 0 }], - relevance: 0 + relevance: 0, }, { className: "emphasis", begin: "_(?![_\\s])", end: "(\\n{2}|_)", - relevance: 0 + relevance: 0, }, { className: "string", - variants: [{ begin: "``.+?''" }, { begin: "`.+?'" }] + variants: [{ begin: "``.+?''" }, { begin: "`.+?'" }], }, { className: "code", begin: "(`.+?`|\\+.+?\\+)", relevance: 0 }, { className: "code", begin: "^[ \\t]", end: "$", relevance: 0 }, @@ -87,18 +87,18 @@ end: "\\]", excludeBegin: !0, excludeEnd: !0, - relevance: 0 - } + relevance: 0, + }, ], - relevance: 10 - } - ] + relevance: 10, + }, + ], }; } function n(e) { var n = { className: "variable", - variants: [{ begin: /\$[\w\d#@][\w\d_]*/ }, { begin: /\$\{(.*?)}/ }] + variants: [{ begin: /\$[\w\d#@][\w\d_]*/ }, { begin: /\$\{(.*?)}/ }], }, a = { className: "string", @@ -111,9 +111,9 @@ className: "variable", begin: /\$\(/, end: /\)/, - contains: [e.BACKSLASH_ESCAPE] - } - ] + contains: [e.BACKSLASH_ESCAPE], + }, + ], }; return { aliases: ["sh", "zsh"], @@ -123,7 +123,7 @@ literal: "true false", built_in: "break cd continue eval exec exit export getopts hash pwd readonly return shift test times trap umask unset alias bind builtin caller command declare echo enable help let local logout mapfile printf read readarray source type typeset ulimit unalias set shopt autoload bg bindkey bye cap chdir clone comparguments compcall compctl compdescribe compfiles compgroups compquote comptags comptry compvalues dirs disable disown echotc echoti emulate fc fg float functions getcap getln history integer jobs kill limit log noglob popd print pushd pushln rehash sched setcap setopt stat suspend ttyctl unfunction unhash unlimit unsetopt vared wait whence where which zcompile zformat zftp zle zmodload zparseopts zprof zpty zregexparse zsocket zstyle ztcp", - _: "-ne -eq -lt -gt -f -d -e -s -l -a" + _: "-ne -eq -lt -gt -f -d -e -s -l -a", }, contains: [ { className: "meta", begin: /^#![^\n]+sh\s*$/, relevance: 10 }, @@ -132,21 +132,21 @@ begin: /\w[\w\d_]*\s*\(\s*\)\s*\{/, returnBegin: !0, contains: [e.inherit(e.TITLE_MODE, { begin: /\w[\w\d_]*/ })], - relevance: 0 + relevance: 0, }, e.HASH_COMMENT_MODE, a, { className: "", begin: /\\"/ }, { className: "string", begin: /'/, end: /'/ }, - n - ] + n, + ], }; } function a(e) { var n = { begin: (u = "[" + (u = "a-zA-Z_\\-!.?+*=<>&#'") + "][" + u + "0-9/;:]*"), - relevance: 0 + relevance: 0, }, a = { className: "number", begin: "[-+]?\\d+(\\.\\d+)?", relevance: 0 }, t = e.inherit(e.QUOTE_STRING_MODE, { illegal: null }), @@ -161,12 +161,12 @@ u = { keywords: { "builtin-name": - "def defonce cond apply if-not if-let if not not= = < > <= >= == + / * - rem quot neg? pos? delay? symbol? keyword? true? false? integer? empty? coll? list? set? ifn? fn? associative? sequential? sorted? counted? reversible? number? decimal? class? distinct? isa? float? rational? reduced? ratio? odd? even? char? seq? vector? string? map? nil? contains? zero? instance? not-every? not-any? libspec? -> ->> .. . inc compare do dotimes mapcat take remove take-while drop letfn drop-last take-last drop-while while intern condp case reduced cycle split-at split-with repeat replicate iterate range merge zipmap declare line-seq sort comparator sort-by dorun doall nthnext nthrest partition eval doseq await await-for let agent atom send send-off release-pending-sends add-watch mapv filterv remove-watch agent-error restart-agent set-error-handler error-handler set-error-mode! error-mode shutdown-agents quote var fn loop recur throw try monitor-enter monitor-exit defmacro defn defn- macroexpand macroexpand-1 for dosync and or when when-not when-let comp juxt partial sequence memoize constantly complement identity assert peek pop doto proxy defstruct first rest cons defprotocol cast coll deftype defrecord last butlast sigs reify second ffirst fnext nfirst nnext defmulti defmethod meta with-meta ns in-ns create-ns import refer keys select-keys vals key val rseq name namespace promise into transient persistent! conj! assoc! dissoc! pop! disj! use class type num float double short byte boolean bigint biginteger bigdec print-method print-dup throw-if printf format load compile get-in update-in pr pr-on newline flush read slurp read-line subvec with-open memfn time re-find re-groups rand-int rand mod locking assert-valid-fdecl alias resolve ref deref refset swap! reset! set-validator! compare-and-set! alter-meta! reset-meta! commute get-validator alter ref-set ref-history-count ref-min-history ref-max-history ensure sync io! new next conj set! to-array future future-call into-array aset gen-class reduce map filter find empty hash-map hash-set sorted-map sorted-map-by sorted-set sorted-set-by vec vector seq flatten reverse assoc dissoc list disj get union difference intersection extend extend-type extend-protocol int nth delay count concat chunk chunk-buffer chunk-append chunk-first chunk-rest max min dec unchecked-inc-int unchecked-inc unchecked-dec-inc unchecked-dec unchecked-negate unchecked-add-int unchecked-add unchecked-subtract-int unchecked-subtract chunk-next chunk-cons chunked-seq? prn vary-meta lazy-seq spread list* str find-keyword keyword symbol gensym force rationalize" + "def defonce cond apply if-not if-let if not not= = < > <= >= == + / * - rem quot neg? pos? delay? symbol? keyword? true? false? integer? empty? coll? list? set? ifn? fn? associative? sequential? sorted? counted? reversible? number? decimal? class? distinct? isa? float? rational? reduced? ratio? odd? even? char? seq? vector? string? map? nil? contains? zero? instance? not-every? not-any? libspec? -> ->> .. . inc compare do dotimes mapcat take remove take-while drop letfn drop-last take-last drop-while while intern condp case reduced cycle split-at split-with repeat replicate iterate range merge zipmap declare line-seq sort comparator sort-by dorun doall nthnext nthrest partition eval doseq await await-for let agent atom send send-off release-pending-sends add-watch mapv filterv remove-watch agent-error restart-agent set-error-handler error-handler set-error-mode! error-mode shutdown-agents quote var fn loop recur throw try monitor-enter monitor-exit defmacro defn defn- macroexpand macroexpand-1 for dosync and or when when-not when-let comp juxt partial sequence memoize constantly complement identity assert peek pop doto proxy defstruct first rest cons defprotocol cast coll deftype defrecord last butlast sigs reify second ffirst fnext nfirst nnext defmulti defmethod meta with-meta ns in-ns create-ns import refer keys select-keys vals key val rseq name namespace promise into transient persistent! conj! assoc! dissoc! pop! disj! use class type num float double short byte boolean bigint biginteger bigdec print-method print-dup throw-if printf format load compile get-in update-in pr pr-on newline flush read slurp read-line subvec with-open memfn time re-find re-groups rand-int rand mod locking assert-valid-fdecl alias resolve ref deref refset swap! reset! set-validator! compare-and-set! alter-meta! reset-meta! commute get-validator alter ref-set ref-history-count ref-min-history ref-max-history ensure sync io! new next conj set! to-array future future-call into-array aset gen-class reduce map filter find empty hash-map hash-set sorted-map sorted-map-by sorted-set sorted-set-by vec vector seq flatten reverse assoc dissoc list disj get union difference intersection extend extend-type extend-protocol int nth delay count concat chunk chunk-buffer chunk-append chunk-first chunk-rest max min dec unchecked-inc-int unchecked-inc unchecked-dec-inc unchecked-dec unchecked-negate unchecked-add-int unchecked-add unchecked-subtract-int unchecked-subtract chunk-next chunk-cons chunked-seq? prn vary-meta lazy-seq spread list* str find-keyword keyword symbol gensym force rationalize", }, lexemes: u, className: "name", begin: u, - starts: g + starts: g, }, n = [d, t, l, o, i, c, r, a, s, n]; return ( @@ -191,16 +191,16 @@ begin: '(u8?|U|L)?"', end: '"', illegal: "\\n", - contains: [e.BACKSLASH_ESCAPE] + contains: [e.BACKSLASH_ESCAPE], }, { begin: "(u8?|U|L)?'(\\\\(x[0-9A-Fa-f]{2}|u[0-9A-Fa-f]{4,8}|[0-7]{3}|\\S)|.)", end: "'", - illegal: "." + illegal: ".", }, - { begin: /(?:u8?|U|L)?R"([^()\\ ]{0,16})\((?:.|\n)*?\)\1"/ } - ] + { begin: /(?:u8?|U|L)?R"([^()\\ ]{0,16})\((?:.|\n)*?\)\1"/ }, + ], }, r = { className: "number", @@ -208,14 +208,14 @@ { begin: "\\b(0b[01']+)" }, { begin: - "(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)" + "(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)", }, { begin: - "(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)" - } + "(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)", + }, ], - relevance: 0 + relevance: 0, }, l = { className: "meta", @@ -223,7 +223,7 @@ end: /$/, keywords: { "meta-keyword": - "if else elif endif define undef warning error line pragma _Pragma ifdef ifndef include" + "if else elif endif define undef warning error line pragma _Pragma ifdef ifndef include", }, contains: [ { begin: /\\\n/, relevance: 0 }, @@ -232,11 +232,11 @@ className: "meta-string", begin: /<.*?>/, end: /$/, - illegal: "\\n" + illegal: "\\n", }, e.C_LINE_COMMENT_MODE, - e.C_BLOCK_COMMENT_MODE - ] + e.C_BLOCK_COMMENT_MODE, + ], }, o = { className: "title", begin: n(t) + e.IDENT_RE, relevance: 0 }, t = n(t) + e.IDENT_RE + "\\s*\\(", @@ -245,14 +245,14 @@ "int float while private char char8_t char16_t char32_t catch import module export virtual operator sizeof dynamic_cast|10 typedef const_cast|10 const for static_cast|10 union namespace unsigned long volatile static protected bool template mutable if public friend do goto auto void enum else break extern using asm case typeid wchar_tshort reinterpret_cast|10 default double register explicit signed typename try this switch continue inline delete alignas alignof constexpr consteval constinit decltype concept co_await co_return co_yield requires noexcept static_assert thread_local restrict final override atomic_bool atomic_char atomic_schar atomic_uchar atomic_short atomic_ushort atomic_int atomic_uint atomic_long atomic_ulong atomic_llong atomic_ullong new throw return and and_eq bitand bitor compl not not_eq or or_eq xor xor_eq", built_in: "std string wstring cin cout cerr clog stdin stdout stderr stringstream istringstream ostringstream auto_ptr deque list queue stack vector map set bitset multiset multimap unordered_set unordered_map unordered_multiset unordered_multimap array shared_ptr abort terminate abs acos asin atan2 atan calloc ceil cosh cos exit exp fabs floor fmod fprintf fputs free frexp fscanf future isalnum isalpha iscntrl isdigit isgraph islower isprint ispunct isspace isupper isxdigit tolower toupper labs ldexp log10 log malloc realloc memchr memcmp memcpy memset modf pow printf putchar puts scanf sinh sin snprintf sprintf sqrt sscanf strcat strchr strcmp strcpy strcspn strlen strncat strncmp strncpy strpbrk strrchr strspn strstr tanh tan vfprintf vprintf vsprintf endl initializer_list unique_ptr _Bool complex _Complex imaginary _Imaginary", - literal: "true false nullptr NULL" + literal: "true false nullptr NULL", }, d = [i, e.C_LINE_COMMENT_MODE, e.C_BLOCK_COMMENT_MODE, r, s], g = { variants: [ { begin: /=/, end: /;/ }, { begin: /\(/, end: /\)/ }, - { beginKeywords: "new throw return else", end: /;/ } + { beginKeywords: "new throw return else", end: /;/ }, ], keywords: c, contains: d.concat([ @@ -261,10 +261,10 @@ end: /\)/, keywords: c, contains: d.concat(["self"]), - relevance: 0 - } + relevance: 0, + }, ]), - relevance: 0 + relevance: 0, }, a = { className: "function", @@ -302,16 +302,16 @@ e.C_BLOCK_COMMENT_MODE, s, r, - i - ] - } - ] + i, + ], + }, + ], }, i, e.C_LINE_COMMENT_MODE, e.C_BLOCK_COMMENT_MODE, - l - ] + l, + ], }; return { aliases: ["c", "cc", "h", "c++", "h++", "hpp", "hh", "hxx", "cxx"], @@ -324,24 +324,27 @@ "\\b(deque|list|queue|stack|vector|map|set|bitset|multiset|multimap|unordered_map|unordered_set|unordered_multiset|unordered_multimap|array)\\s*<", end: ">", keywords: c, - contains: ["self", i] + contains: ["self", i], }, { begin: e.IDENT_RE + "::", keywords: c }, { className: "class", beginKeywords: "class struct", end: /[{;:]/, - contains: [{ begin: //, contains: ["self"] }, e.TITLE_MODE] - } + contains: [ + { begin: //, contains: ["self"] }, + e.TITLE_MODE, + ], + }, ]), - exports: { preprocessor: l, strings: s, keywords: c } + exports: { preprocessor: l, strings: s, keywords: c }, }; } function i(e) { var n = { keyword: "abstract as base bool break byte case catch char checked const continue decimal default delegate do double enum event explicit extern finally fixed float for foreach goto if implicit in int interface internal is lock long object operator out override params private protected public readonly ref sbyte sealed short sizeof stackalloc static string struct switch this try typeof uint ulong unchecked unsafe ushort using virtual void volatile while add alias ascending async await by descending dynamic equals from get global group into join let nameof on orderby partial remove select set value var when where yield", - literal: "null false true" + literal: "null false true", }, a = { className: "number", @@ -349,20 +352,20 @@ { begin: "\\b(0b[01']+)" }, { begin: - "(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)" + "(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)", }, { begin: - "(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)" - } + "(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)", + }, ], - relevance: 0 + relevance: 0, }, t = { className: "string", begin: '@"', end: '"', - contains: [{ begin: '""' }] + contains: [{ begin: '""' }], }, i = e.inherit(t, { illegal: /\n/ }), s = { className: "subst", begin: "{", end: "}", keywords: n }, @@ -372,17 +375,17 @@ begin: /\$"/, end: '"', illegal: /\n/, - contains: [{ begin: "{{" }, { begin: "}}" }, e.BACKSLASH_ESCAPE, r] + contains: [{ begin: "{{" }, { begin: "}}" }, e.BACKSLASH_ESCAPE, r], }, o = { className: "string", begin: /\$@"/, end: '"', - contains: [{ begin: "{{" }, { begin: "}}" }, { begin: '""' }, s] + contains: [{ begin: "{{" }, { begin: "}}" }, { begin: '""' }, s], }, c = e.inherit(o, { illegal: /\n/, - contains: [{ begin: "{{" }, { begin: "}}" }, { begin: '""' }, r] + contains: [{ begin: "{{" }, { begin: "}}" }, { begin: '""' }, r], }), s = ((s.contains = [ @@ -392,7 +395,7 @@ e.APOS_STRING_MODE, e.QUOTE_STRING_MODE, a, - e.C_BLOCK_COMMENT_MODE + e.C_BLOCK_COMMENT_MODE, ]), (r.contains = [ c, @@ -401,7 +404,7 @@ e.APOS_STRING_MODE, e.QUOTE_STRING_MODE, a, - e.inherit(e.C_BLOCK_COMMENT_MODE, { illegal: /\n/ }) + e.inherit(e.C_BLOCK_COMMENT_MODE, { illegal: /\n/ }), ]), { variants: [o, l, t, e.APOS_STRING_MODE, e.QUOTE_STRING_MODE] }), r = @@ -424,10 +427,10 @@ variants: [ { begin: "///", relevance: 0 }, { begin: "\x3c!--|--\x3e" }, - { begin: "" } - ] - } - ] + { begin: "" }, + ], + }, + ], }), e.C_LINE_COMMENT_MODE, e.C_BLOCK_COMMENT_MODE, @@ -437,8 +440,8 @@ end: "$", keywords: { "meta-keyword": - "if else elif endif define undef warning error line region endregion pragma checksum" - } + "if else elif endif define undef warning error line region endregion pragma checksum", + }, }, s, a, @@ -449,8 +452,8 @@ contains: [ e.TITLE_MODE, e.C_LINE_COMMENT_MODE, - e.C_BLOCK_COMMENT_MODE - ] + e.C_BLOCK_COMMENT_MODE, + ], }, { beginKeywords: "namespace", @@ -459,8 +462,8 @@ contains: [ e.inherit(e.TITLE_MODE, { begin: "[a-zA-Z](\\.?\\w)*" }), e.C_LINE_COMMENT_MODE, - e.C_BLOCK_COMMENT_MODE - ] + e.C_BLOCK_COMMENT_MODE, + ], }, { className: "meta", @@ -468,7 +471,7 @@ excludeBegin: !0, end: "\\]", excludeEnd: !0, - contains: [{ className: "meta-string", begin: /"/, end: /"/ }] + contains: [{ className: "meta-string", begin: /"/, end: /"/ }], }, { beginKeywords: "new return throw await else", relevance: 0 }, { @@ -483,7 +486,7 @@ begin: e.IDENT_RE + "\\s*\\(", returnBegin: !0, contains: [e.TITLE_MODE], - relevance: 0 + relevance: 0, }, { className: "params", @@ -493,13 +496,13 @@ excludeEnd: !0, keywords: n, relevance: 0, - contains: [s, a, e.C_BLOCK_COMMENT_MODE] + contains: [s, a, e.C_BLOCK_COMMENT_MODE], }, e.C_LINE_COMMENT_MODE, - e.C_BLOCK_COMMENT_MODE - ] - } - ] + e.C_BLOCK_COMMENT_MODE, + ], + }, + ], }; } function s(e) { @@ -523,19 +526,19 @@ contains: [ e.APOS_STRING_MODE, e.QUOTE_STRING_MODE, - e.CSS_NUMBER_MODE - ] - } - ] + e.CSS_NUMBER_MODE, + ], + }, + ], }, e.CSS_NUMBER_MODE, e.QUOTE_STRING_MODE, e.APOS_STRING_MODE, e.C_BLOCK_COMMENT_MODE, { className: "number", begin: "#[0-9A-Fa-f]+" }, - { className: "meta", begin: "!important" } - ] - } + { className: "meta", begin: "!important" }, + ], + }, }; return { case_insensitive: !0, @@ -549,16 +552,16 @@ begin: /\[/, end: /\]/, illegal: "$", - contains: [e.APOS_STRING_MODE, e.QUOTE_STRING_MODE] + contains: [e.APOS_STRING_MODE, e.QUOTE_STRING_MODE], }, { className: "selector-pseudo", - begin: /:(:)?[a-zA-Z0-9\_\-\+\(\)"'.]+/ + begin: /:(:)?[a-zA-Z0-9\_\-\+\(\)"'.]+/, }, { begin: "@(page|font-face)", lexemes: "@[a-z-]+", - keywords: "@page @font-face" + keywords: "@page @font-face", }, { begin: "@", @@ -577,15 +580,15 @@ { begin: /[a-z-]+:/, className: "attribute" }, e.APOS_STRING_MODE, e.QUOTE_STRING_MODE, - e.CSS_NUMBER_MODE - ] - } - ] + e.CSS_NUMBER_MODE, + ], + }, + ], }, { className: "selector-tag", begin: "[a-zA-Z-][a-zA-Z0-9_-]*", - relevance: 0 + relevance: 0, }, { begin: "{", @@ -598,11 +601,11 @@ returnBegin: !0, end: ";", endsWithParent: !0, - contains: [n] - } - ] - } - ] + contains: [n], + }, + ], + }, + ], }; } function r(e) { @@ -615,8 +618,8 @@ variants: [ { begin: /^@@ +\-\d+,\d+ +\+\d+,\d+ +@@$/ }, { begin: /^\*\*\* +\d+,\d+ +\*\*\*\*$/ }, - { begin: /^\-\-\- +\d+,\d+ +\-\-\-\-$/ } - ] + { begin: /^\-\-\- +\d+,\d+ +\-\-\-\-$/ }, + ], }, { className: "comment", @@ -626,13 +629,13 @@ { begin: /^\-{3}/, end: /$/ }, { begin: /^\*{3} /, end: /$/ }, { begin: /^\+{3}/, end: /$/ }, - { begin: /^\*{15}$/ } - ] + { begin: /^\*{15}$/ }, + ], }, { className: "addition", begin: "^\\+", end: "$" }, { className: "deletion", begin: "^\\-", end: "$" }, - { className: "addition", begin: "^\\!", end: "$" } - ] + { className: "addition", begin: "^\\!", end: "$" }, + ], }; } function l(e) { @@ -648,10 +651,10 @@ { beginKeywords: "run cmd entrypoint volume add copy workdir label healthcheck shell", - starts: { end: /[^\\]$/, subLanguage: "bash" } - } + starts: { end: /[^\\]$/, subLanguage: "bash" }, + }, ], - illegal: "/ } - ] - } - ] - } - ] + { begin: // }, + ], + }, + ], + }, + ], }, s = { className: "string", @@ -700,8 +703,8 @@ { begin: /\(/, end: /\)/ }, { begin: /\[/, end: /\]/ }, { begin: /\{/, end: /\}/ }, - { begin: /\/ } - ] + { begin: /\/ }, + ], }, r = { className: "string", @@ -714,19 +717,19 @@ { begin: /~S'''/, end: /'''/, contains: [] }, { begin: /~S'/, end: /'/, contains: [] }, { begin: /'/, end: /'/ }, - { begin: /"/, end: /"/ } - ] + { begin: /"/, end: /"/ }, + ], }, l = { className: "function", beginKeywords: "def defp defmacro", end: /\B\b/, - contains: [e.inherit(e.TITLE_MODE, { begin: n, endsParent: !0 })] + contains: [e.inherit(e.TITLE_MODE, { begin: n, endsParent: !0 })], }, o = e.inherit(l, { className: "class", beginKeywords: "defimpl defmodule defprotocol defrecord", - end: /\bdo\b|$|;/ + end: /\bdo\b|$|;/, }), s = [ r, @@ -743,17 +746,17 @@ r, { begin: - "[a-zA-Z_]\\w*[!?=]?|[-+~]\\@|<<|>>|=~|===?|<=>|[<>]=?|\\*\\*|[-/+%^&*~`|]|\\[\\]=?" - } + "[a-zA-Z_]\\w*[!?=]?|[-+~]\\@|<<|>>|=~|===?|<=>|[<>]=?|\\*\\*|[-/+%^&*~`|]|\\[\\]=?", + }, ], - relevance: 0 + relevance: 0, }, { className: "symbol", begin: n + ":(?!:)", relevance: 0 }, { className: "number", begin: "(\\b0o[0-7_]+)|(\\b0b[01_]+)|(\\b0x[0-9a-fA-F_]+)|(-?\\b[1-9][0-9_]*(.[0-9_]+([eE][-+]?[0-9]+)?)?)", - relevance: 0 + relevance: 0, }, { className: "variable", begin: "(\\$\\W)|((\\$|\\@\\@?)(\\w+))" }, { begin: "->" }, @@ -767,12 +770,12 @@ contains: [e.BACKSLASH_ESCAPE, t], variants: [ { begin: "/", end: "/[a-z]*" }, - { begin: "%r\\[", end: "\\][a-z]*" } - ] - } + { begin: "%r\\[", end: "\\][a-z]*" }, + ], + }, ], - relevance: 0 - } + relevance: 0, + }, ]; return { lexemes: n, keywords: a, contains: (t.contains = s) }; } @@ -782,7 +785,7 @@ "break default func interface select case map struct chan else goto package switch const fallthrough if range type continue for import return var go defer bool byte complex64 complex128 float32 float64 int8 int16 int32 int64 string uint8 uint16 uint32 uint64 int uint uintptr rune", literal: "true false iota nil", built_in: - "append cap close complex copy imag len make new panic print println real recover delete" + "append cap close complex copy imag len make new panic print println real recover delete", }; return { aliases: ["golang"], @@ -796,15 +799,15 @@ variants: [ e.QUOTE_STRING_MODE, e.APOS_STRING_MODE, - { begin: "`", end: "`" } - ] + { begin: "`", end: "`" }, + ], }, { className: "number", variants: [ { begin: e.C_NUMBER_RE + "[i]", relevance: 1 }, - e.C_NUMBER_MODE - ] + e.C_NUMBER_MODE, + ], }, { begin: /:=/ }, { @@ -819,11 +822,11 @@ begin: /\(/, end: /\)/, keywords: n, - illegal: /["']/ - } - ] - } - ] + illegal: /["']/, + }, + ], + }, + ], }; } function d(e) { @@ -831,15 +834,15 @@ keywords: { literal: "true false null", keyword: - "byte short char int long boolean float double void def as in assert trait super this abstract static volatile transient public private protected synchronized final class interface enum if else for while switch case break default continue throw throws try catch finally implements extends new import package return instanceof" + "byte short char int long boolean float double void def as in assert trait super this abstract static volatile transient public private protected synchronized final class interface enum if else for while switch case break default continue throw throws try catch finally implements extends new import package return instanceof", }, contains: [ e.COMMENT("/\\*\\*", "\\*/", { relevance: 0, contains: [ { begin: /\w+@/, relevance: 0 }, - { className: "doctag", begin: "@[A-Za-z]+" } - ] + { className: "doctag", begin: "@[A-Za-z]+" }, + ], }), e.C_LINE_COMMENT_MODE, e.C_BLOCK_COMMENT_MODE, @@ -850,14 +853,14 @@ { className: "regexp", begin: /~?\/[^\/\n]+\//, - contains: [e.BACKSLASH_ESCAPE] + contains: [e.BACKSLASH_ESCAPE], }, e.QUOTE_STRING_MODE, { className: "meta", begin: "^#!/usr/bin/env", end: "$", - illegal: "\n" + illegal: "\n", }, e.BINARY_NUMBER_MODE, { @@ -867,24 +870,24 @@ illegal: ":", contains: [ { beginKeywords: "extends implements" }, - e.UNDERSCORE_TITLE_MODE - ] + e.UNDERSCORE_TITLE_MODE, + ], }, e.C_NUMBER_MODE, { className: "meta", begin: "@[A-Za-z]+" }, { className: "string", begin: /[^\?]{0}[A-Za-z0-9_$]+ *:/ }, { begin: /\?/, end: /\:/ }, - { className: "symbol", begin: "^\\s*[A-Za-z0-9_$]+:", relevance: 0 } + { className: "symbol", begin: "^\\s*[A-Za-z0-9_$]+:", relevance: 0 }, ], - illegal: /#|<\// + illegal: /#|<\//, }; } function g(e) { var n = { variants: [ e.COMMENT("--", "$"), - e.COMMENT("{-", "-}", { contains: ["self"] }) - ] + e.COMMENT("{-", "-}", { contains: ["self"] }), + ], }, a = { className: "meta", begin: "{-#", end: "#-}" }, t = { className: "meta", begin: "^#", end: "$" }, @@ -898,11 +901,11 @@ t, { className: "type", - begin: "\\b[A-Z][\\w]*(\\((\\.\\.|,|\\w+)\\))?" + begin: "\\b[A-Z][\\w]*(\\((\\.\\.|,|\\w+)\\))?", }, e.inherit(e.TITLE_MODE, { begin: "[_a-z][\\w']*" }), - n - ] + n, + ], }; return { aliases: ["hs"], @@ -914,46 +917,52 @@ end: "where", keywords: "module where", contains: [s, n], - illegal: "\\W\\.|;" + illegal: "\\W\\.|;", }, { begin: "\\bimport\\b", end: "$", keywords: "import qualified as hiding", contains: [s, n], - illegal: "\\W\\.|;" + illegal: "\\W\\.|;", }, { className: "class", begin: "^(\\s*)?(class|instance)\\b", end: "where", keywords: "class family instance where", - contains: [i, s, n] + contains: [i, s, n], }, { className: "class", begin: "\\b(data|(new)?type)\\b", end: "$", keywords: "data family type newtype deriving", - contains: [a, i, s, { begin: "{", end: "}", contains: s.contains }, n] + contains: [ + a, + i, + s, + { begin: "{", end: "}", contains: s.contains }, + n, + ], }, { beginKeywords: "default", end: "$", contains: [i, s, n] }, { beginKeywords: "infix infixl infixr", end: "$", - contains: [e.C_NUMBER_MODE, n] + contains: [e.C_NUMBER_MODE, n], }, { begin: "\\bforeign\\b", end: "$", keywords: "foreign import export ccall stdcall cplusplus jvm dotnet safe unsafe", - contains: [i, e.QUOTE_STRING_MODE, n] + contains: [i, e.QUOTE_STRING_MODE, n], }, { className: "meta", begin: "#!\\/usr\\/bin\\/env runhaskell", - end: "$" + end: "$", }, a, t, @@ -962,8 +971,8 @@ i, e.inherit(e.TITLE_MODE, { begin: "^[_a-z][\\w']*" }), n, - { begin: "->|<-" } - ] + { begin: "->|<-" }, + ], }; } function u(e) { @@ -973,7 +982,7 @@ className: "number", begin: "\\b(0[bB]([01]+[01_]+[01]+|[01]+)|0[xX]([a-fA-F0-9]+[a-fA-F0-9_]+[a-fA-F0-9]+|[a-fA-F0-9]+)|(([\\d]+[\\d_]+[\\d]+|[\\d]+)(\\.([\\d]+[\\d_]+[\\d]+|[\\d]+))?|\\.([\\d]+[\\d_]+[\\d]+|[\\d]+))([eE][-+]?\\d+)?)[lLfF]?", - relevance: 0 + relevance: 0, }; return { aliases: ["jsp"], @@ -984,8 +993,8 @@ relevance: 0, contains: [ { begin: /\w+@/, relevance: 0 }, - { className: "doctag", begin: "@[A-Za-z]+" } - ] + { className: "doctag", begin: "@[A-Za-z]+" }, + ], }), e.C_LINE_COMMENT_MODE, e.C_BLOCK_COMMENT_MODE, @@ -1000,8 +1009,8 @@ illegal: /[:"\[\]]/, contains: [ { beginKeywords: "extends implements" }, - e.UNDERSCORE_TITLE_MODE - ] + e.UNDERSCORE_TITLE_MODE, + ], }, { beginKeywords: "new throw return else", relevance: 0 }, { @@ -1019,7 +1028,7 @@ begin: e.UNDERSCORE_IDENT_RE + "\\s*\\(", returnBegin: !0, relevance: 0, - contains: [e.UNDERSCORE_TITLE_MODE] + contains: [e.UNDERSCORE_TITLE_MODE], }, { className: "params", @@ -1031,16 +1040,16 @@ e.APOS_STRING_MODE, e.QUOTE_STRING_MODE, e.C_NUMBER_MODE, - e.C_BLOCK_COMMENT_MODE - ] + e.C_BLOCK_COMMENT_MODE, + ], }, e.C_LINE_COMMENT_MODE, - e.C_BLOCK_COMMENT_MODE - ] + e.C_BLOCK_COMMENT_MODE, + ], }, a, - { className: "meta", begin: "@[A-Za-z]+" } - ] + { className: "meta", begin: "@[A-Za-z]+" }, + ], }; } function _(e) { @@ -1054,23 +1063,23 @@ "in of if for while finally var new function do return void else break catch instanceof with throw case default try this switch continue typeof delete let yield const export super debugger as async await static import from as", literal: "true false null undefined NaN Infinity", built_in: - "eval isFinite isNaN parseFloat parseInt decodeURI decodeURIComponent encodeURI encodeURIComponent escape unescape Object Function Boolean Error EvalError InternalError RangeError ReferenceError StopIteration SyntaxError TypeError URIError Number Math Date String RegExp Array Float32Array Float64Array Int16Array Int32Array Int8Array Uint16Array Uint32Array Uint8Array Uint8ClampedArray ArrayBuffer DataView JSON Intl arguments require module console window document Symbol Set Map WeakSet WeakMap Proxy Reflect Promise" + "eval isFinite isNaN parseFloat parseInt decodeURI decodeURIComponent encodeURI encodeURIComponent escape unescape Object Function Boolean Error EvalError InternalError RangeError ReferenceError StopIteration SyntaxError TypeError URIError Number Math Date String RegExp Array Float32Array Float64Array Int16Array Int32Array Int8Array Uint16Array Uint32Array Uint8Array Uint8ClampedArray ArrayBuffer DataView JSON Intl arguments require module console window document Symbol Set Map WeakSet WeakMap Proxy Reflect Promise", }, l = { className: "number", variants: [ { begin: "\\b(0[bB][01]+)n?" }, { begin: "\\b(0[oO][0-7]+)n?" }, - { begin: e.C_NUMBER_RE + "n?" } + { begin: e.C_NUMBER_RE + "n?" }, ], - relevance: 0 + relevance: 0, }, o = { className: "subst", begin: "\\$\\{", end: "\\}", keywords: r, - contains: [] + contains: [], }, c = { begin: "html`", @@ -1079,8 +1088,8 @@ end: "`", returnEnd: !1, contains: [e.BACKSLASH_ESCAPE, o], - subLanguage: "xml" - } + subLanguage: "xml", + }, }, d = { begin: "css`", @@ -1089,14 +1098,14 @@ end: "`", returnEnd: !1, contains: [e.BACKSLASH_ESCAPE, o], - subLanguage: "css" - } + subLanguage: "css", + }, }, g = { className: "string", begin: "`", end: "`", - contains: [e.BACKSLASH_ESCAPE, o] + contains: [e.BACKSLASH_ESCAPE, o], }, o = ((o.contains = [ @@ -1106,7 +1115,7 @@ d, g, l, - e.REGEXP_MODE + e.REGEXP_MODE, ]), o.contains.concat([e.C_BLOCK_COMMENT_MODE, e.C_LINE_COMMENT_MODE])); return { @@ -1116,7 +1125,7 @@ { className: "meta", relevance: 10, - begin: /^\s*['"]use (strict|asm)['"]/ + begin: /^\s*['"]use (strict|asm)['"]/, }, { className: "meta", begin: /^#!/, end: /$/ }, e.APOS_STRING_MODE, @@ -1137,12 +1146,12 @@ className: "variable", begin: s + "(?=\\s*(-)|$)", endsParent: !0, - relevance: 0 + relevance: 0, }, - { begin: /(?=[^\n])\s/, relevance: 0 } - ] - } - ] + { begin: /(?=[^\n])\s/, relevance: 0 }, + ], + }, + ], }), e.C_BLOCK_COMMENT_MODE, l, @@ -1154,9 +1163,9 @@ begin: s + "\\s*:", returnBegin: !0, relevance: 0, - contains: [{ className: "attr", begin: s, relevance: 0 }] - } - ] + contains: [{ className: "attr", begin: s, relevance: 0 }], + }, + ], }, { begin: "(" + e.RE_STARTERS_RE + "|\\b(case|return|throw)\\b)\\s*", @@ -1182,23 +1191,23 @@ excludeBegin: !0, excludeEnd: !0, keywords: r, - contains: o - } - ] - } - ] + contains: o, + }, + ], + }, + ], }, { className: "", begin: /\s/, end: /\s*/, skip: !0 }, { variants: [ { begin: n, end: a }, - { begin: t, end: i } + { begin: t, end: i }, ], subLanguage: "xml", - contains: [{ begin: t, end: i, skip: !0, contains: ["self"] }] - } + contains: [{ begin: t, end: i, skip: !0, contains: ["self"] }], + }, ], - relevance: 0 + relevance: 0, }, { className: "function", @@ -1213,10 +1222,10 @@ end: /\)/, excludeBegin: !0, excludeEnd: !0, - contains: o - } + contains: o, + }, ], - illegal: /\[|%/ + illegal: /\[|%/, }, { begin: /\$[(.]/ }, e.METHOD_GUARD, @@ -1226,11 +1235,11 @@ end: /[{;=]/, excludeEnd: !0, illegal: /[:"\[\]]/, - contains: [{ beginKeywords: "extends" }, e.UNDERSCORE_TITLE_MODE] + contains: [{ beginKeywords: "extends" }, e.UNDERSCORE_TITLE_MODE], }, - { beginKeywords: "constructor get set", end: /\{/, excludeEnd: !0 } + { beginKeywords: "constructor get set", end: /\{/, excludeEnd: !0 }, ], - illegal: /#(?!!)/ + illegal: /#(?!!)/, }; } function m(e) { @@ -1242,7 +1251,7 @@ endsWithParent: !0, excludeEnd: !0, contains: t, - keywords: n + keywords: n, }, s = { begin: "{", @@ -1253,21 +1262,21 @@ begin: /"/, end: /"/, contains: [e.BACKSLASH_ESCAPE], - illegal: "\\n" + illegal: "\\n", }, - e.inherit(i, { begin: /:/ }) + e.inherit(i, { begin: /:/ }), ].concat(a), - illegal: "\\S" + illegal: "\\S", }, e = { begin: "\\[", end: "\\]", contains: [e.inherit(i)], - illegal: "\\S" + illegal: "\\S", }; return ( t.push(s, e), - a.forEach(function(e) { + a.forEach(function (e) { t.push(e); }), { contains: t, keywords: n, illegal: "\\S" } @@ -1279,14 +1288,14 @@ "abstract as val var vararg get set class object open private protected public noinline crossinline dynamic final enum if else do while for when throw try catch finally import package is in fun override companion reified inline lateinit init interface annotation data sealed internal infix operator out by constructor super tailrec where const inner suspend typealias external expect actual trait volatile transient native default", built_in: "Byte Short Char Int Long Boolean Float Double Void Unit Nothing", - literal: "true false null" + literal: "true false null", }, a = { className: "symbol", begin: e.UNDERSCORE_IDENT_RE + "@" }, t = { className: "subst", begin: "\\${", end: "}", - contains: [e.C_NUMBER_MODE] + contains: [e.C_NUMBER_MODE], }, i = { className: "string", @@ -1297,24 +1306,24 @@ contains: [ (i = { className: "variable", - begin: "\\$" + e.UNDERSCORE_IDENT_RE + begin: "\\$" + e.UNDERSCORE_IDENT_RE, }), - t - ] + t, + ], }, { begin: "'", end: "'", illegal: /\n/, - contains: [e.BACKSLASH_ESCAPE] + contains: [e.BACKSLASH_ESCAPE], }, { begin: '"', end: '"', illegal: /\n/, - contains: [e.BACKSLASH_ESCAPE, i, t] - } - ] + contains: [e.BACKSLASH_ESCAPE, i, t], + }, + ], }, t = (t.contains.push(i), @@ -1323,7 +1332,7 @@ begin: "@(?:file|property|field|get|set|receiver|param|setparam|delegate)\\s*:(?:\\s*" + e.UNDERSCORE_IDENT_RE + - ")?" + ")?", }), s = { className: "meta", @@ -1332,22 +1341,22 @@ { begin: /\(/, end: /\)/, - contains: [e.inherit(i, { className: "meta-string" })] - } - ] + contains: [e.inherit(i, { className: "meta-string" })], + }, + ], }, r = { className: "number", begin: "\\b(0[bB]([01]+[01_]+[01]+|[01]+)|0[xX]([a-fA-F0-9]+[a-fA-F0-9_]+[a-fA-F0-9]+|[a-fA-F0-9]+)|(([\\d]+[\\d_]+[\\d]+|[\\d]+)(\\.([\\d]+[\\d_]+[\\d]+|[\\d]+))?|\\.([\\d]+[\\d_]+[\\d]+|[\\d]+))([eE][-+]?\\d+)?)[lLfF]?", - relevance: 0 + relevance: 0, }, l = e.COMMENT("/\\*", "\\*/", { contains: [e.C_BLOCK_COMMENT_MODE] }), o = { variants: [ { className: "type", begin: e.UNDERSCORE_IDENT_RE }, - { begin: /\(/, end: /\)/, contains: [] } - ] + { begin: /\(/, end: /\)/, contains: [] }, + ], }, c = o; return ( @@ -1359,14 +1368,14 @@ contains: [ e.COMMENT("/\\*\\*", "\\*/", { relevance: 0, - contains: [{ className: "doctag", begin: "@[A-Za-z]+" }] + contains: [{ className: "doctag", begin: "@[A-Za-z]+" }], }), e.C_LINE_COMMENT_MODE, l, { className: "keyword", begin: /\b(break|continue|return|this)\b/, - starts: { contains: [{ className: "symbol", begin: /@\w+/ }] } + starts: { contains: [{ className: "symbol", begin: /@\w+/ }] }, }, a, t, @@ -1385,14 +1394,14 @@ begin: e.UNDERSCORE_IDENT_RE + "\\s*\\(", returnBegin: !0, relevance: 0, - contains: [e.UNDERSCORE_TITLE_MODE] + contains: [e.UNDERSCORE_TITLE_MODE], }, { className: "type", begin: //, keywords: "reified", - relevance: 0 + relevance: 0, }, { className: "params", @@ -1407,18 +1416,18 @@ end: /[=,\/]/, endsWithParent: !0, contains: [o, e.C_LINE_COMMENT_MODE, l], - relevance: 0 + relevance: 0, }, e.C_LINE_COMMENT_MODE, l, t, s, i, - e.C_NUMBER_MODE - ] + e.C_NUMBER_MODE, + ], }, - l - ] + l, + ], }, { className: "class", @@ -1428,7 +1437,7 @@ illegal: "extends implements", contains: [ { - beginKeywords: "public protected internal private constructor" + beginKeywords: "public protected internal private constructor", }, e.UNDERSCORE_TITLE_MODE, { @@ -1437,28 +1446,28 @@ end: />/, excludeBegin: !0, excludeEnd: !0, - relevance: 0 + relevance: 0, }, { className: "type", begin: /[,:]\s*/, end: /[<\(,]|$/, excludeBegin: !0, - returnEnd: !0 + returnEnd: !0, }, t, - s - ] + s, + ], }, i, { className: "meta", begin: "^#!/usr/bin/env", end: "$", - illegal: "\n" + illegal: "\n", }, - r - ] + r, + ], } ); } @@ -1468,7 +1477,7 @@ t = { begin: n, end: a, contains: ["self"] }, i = [ e.COMMENT("--(?!" + n + ")", "$"), - e.COMMENT("--" + n, a, { contains: [t], relevance: 10 }) + e.COMMENT("--" + n, a, { contains: [t], relevance: 10 }), ]; return { lexemes: e.UNDERSCORE_IDENT_RE, @@ -1477,7 +1486,7 @@ keyword: "and break do else elseif end for goto if in local not or repeat return then until while", built_in: - "_G _ENV _VERSION __index __newindex __mode __call __metatable __tostring __len __gc __add __sub __mul __div __mod __pow __concat __unm __eq __lt __le assert collectgarbage dofile error getfenv getmetatable ipairs load loadfile loadstringmodule next pairs pcall print rawequal rawget rawset require select setfenvsetmetatable tonumber tostring type unpack xpcall arg selfcoroutine resume yield status wrap create running debug getupvalue debug sethook getmetatable gethook setmetatable setlocal traceback setfenv getinfo setupvalue getlocal getregistry getfenv io lines write close flush open output type read stderr stdin input stdout popen tmpfile math log max acos huge ldexp pi cos tanh pow deg tan cosh sinh random randomseed frexp ceil floor rad abs sqrt modf asin min mod fmod log10 atan2 exp sin atan os exit setlocale date getenv difftime remove time clock tmpname rename execute package preload loadlib loaded loaders cpath config path seeall string sub upper len gfind rep find match char dump gmatch reverse byte format gsub lower table setn insert getn foreachi maxn foreach concat sort remove" + "_G _ENV _VERSION __index __newindex __mode __call __metatable __tostring __len __gc __add __sub __mul __div __mod __pow __concat __unm __eq __lt __le assert collectgarbage dofile error getfenv getmetatable ipairs load loadfile loadstringmodule next pairs pcall print rawequal rawget rawset require select setfenvsetmetatable tonumber tostring type unpack xpcall arg selfcoroutine resume yield status wrap create running debug getupvalue debug sethook getmetatable gethook setmetatable setlocal traceback setfenv getinfo setupvalue getlocal getregistry getfenv io lines write close flush open output type read stderr stdin input stdout popen tmpfile math log max acos huge ldexp pi cos tanh pow deg tan cosh sinh random randomseed frexp ceil floor rad abs sqrt modf asin min mod fmod log10 atan2 exp sin atan os exit setlocale date getenv difftime remove time clock tmpname rename execute package preload loadlib loaded loaders cpath config path seeall string sub upper len gfind rep find match char dump gmatch reverse byte format gsub lower table setn insert getn foreachi maxn foreach concat sort remove", }, contains: i.concat([ { @@ -1486,21 +1495,21 @@ end: "\\)", contains: [ e.inherit(e.TITLE_MODE, { - begin: "([_a-zA-Z]\\w*\\.)*([_a-zA-Z]\\w*:)?[_a-zA-Z]\\w*" + begin: "([_a-zA-Z]\\w*\\.)*([_a-zA-Z]\\w*:)?[_a-zA-Z]\\w*", }), { className: "params", begin: "\\(", endsWithParent: !0, - contains: i - } - ].concat(i) + contains: i, + }, + ].concat(i), }, e.C_NUMBER_MODE, e.APOS_STRING_MODE, e.QUOTE_STRING_MODE, - { className: "string", begin: n, end: a, contains: [t], relevance: 5 } - ]) + { className: "string", begin: n, end: a, contains: [t], relevance: 5 }, + ]), }; } function f(e) { @@ -1511,15 +1520,15 @@ className: "section", variants: [ { begin: "^#{1,6}", end: "$" }, - { begin: "^.+?\\n[=-]{2,}$" } - ] + { begin: "^.+?\\n[=-]{2,}$" }, + ], }, { begin: "<", end: ">", subLanguage: "xml", relevance: 0 }, { className: "bullet", begin: "^\\s*([*+-]|(\\d+\\.))\\s+" }, { className: "strong", begin: "[*_]{2}.+?[*_]{2}" }, { className: "emphasis", - variants: [{ begin: "\\*.+?\\*" }, { begin: "_.+?_", relevance: 0 }] + variants: [{ begin: "\\*.+?\\*" }, { begin: "_.+?_", relevance: 0 }], }, { className: "quote", begin: "^>\\s+", end: "$" }, { @@ -1527,8 +1536,8 @@ variants: [ { begin: "^```\\w*\\s*$", end: "^```[ ]*$" }, { begin: "`.+?`" }, - { begin: "^( {4}|\\t)", end: "$", relevance: 0 } - ] + { begin: "^( {4}|\\t)", end: "$", relevance: 0 }, + ], }, { begin: "^[-\\*]{3,}", end: "$" }, { @@ -1541,24 +1550,24 @@ end: "\\]", excludeBegin: !0, returnEnd: !0, - relevance: 0 + relevance: 0, }, { className: "link", begin: "\\]\\(", end: "\\)", excludeBegin: !0, - excludeEnd: !0 + excludeEnd: !0, }, { className: "symbol", begin: "\\]\\[", end: "\\]", excludeBegin: !0, - excludeEnd: !0 - } + excludeEnd: !0, + }, ], - relevance: 10 + relevance: 10, }, { begin: /^\[[^\n]+\]:/, @@ -1569,12 +1578,12 @@ begin: /\[/, end: /\]/, excludeBegin: !0, - excludeEnd: !0 + excludeEnd: !0, }, - { className: "link", begin: /:\s*/, end: /$/, excludeBegin: !0 } - ] - } - ] + { className: "link", begin: /:\s*/, end: /$/, excludeBegin: !0 }, + ], + }, + ], }; } function E(e) { @@ -1582,7 +1591,7 @@ keyword: "rec with let in inherit assert if else then", literal: "true false or and null", built_in: - "import abort baseNameOf dirOf isNull builtins map removeAttrs throw toString derivation" + "import abort baseNameOf dirOf isNull builtins map removeAttrs throw toString derivation", }, a = { className: "subst", begin: /\$\{/, end: /}/, keywords: n }, e = [ @@ -1594,15 +1603,15 @@ contains: [a], variants: [ { begin: "''", end: "''" }, - { begin: '"', end: '"' } - ] + { begin: '"', end: '"' }, + ], }, { begin: /[a-zA-Z0-9-_]+(\s*=)/, returnBegin: !0, relevance: 0, - contains: [{ className: "attr", begin: /\S+/ }] - } + contains: [{ className: "attr", begin: /\S+/ }], + }, ]; return { aliases: ["nixos"], keywords: n, contains: (a.contains = e) }; } @@ -1619,7 +1628,7 @@ "int float while char export sizeof typedef const struct for union unsigned long volatile static bool mutable if do return goto void enum else break extern asm case short default double register explicit signed typename this switch continue wchar_t inline readonly assign readwrite self @synchronized id typeof nonatomic super unichar IBOutlet IBAction strong weak copy in out inout bycopy byref oneway __strong __weak __block __autoreleasing @private @protected @public @try @property @end @throw @catch @finally @autoreleasepool @synthesize @dynamic @selector @optional @required @encode @package @import @defs @compatibility_alias __bridge __bridge_transfer __bridge_retained __bridge_retain __covariant __contravariant __kindof _Nonnull _Nullable _Null_unspecified __FUNCTION__ __PRETTY_FUNCTION__ __attribute__ getter setter retain unsafe_unretained nonnull nullable null_unspecified null_resettable class instancetype NS_DESIGNATED_INITIALIZER NS_UNAVAILABLE NS_REQUIRES_SUPER NS_RETURNS_INNER_POINTER NS_INLINE NS_AVAILABLE NS_DEPRECATED NS_ENUM NS_OPTIONS NS_SWIFT_UNAVAILABLE NS_ASSUME_NONNULL_BEGIN NS_ASSUME_NONNULL_END NS_REFINED_FOR_SWIFT NS_SWIFT_NAME NS_SWIFT_NOTHROW NS_DURING NS_HANDLER NS_ENDHANDLER NS_VALUERETURN NS_VOIDRETURN", literal: "false true FALSE TRUE nil YES NO NULL", built_in: - "BOOL dispatch_once_t dispatch_queue_t dispatch_sync dispatch_async dispatch_once" + "BOOL dispatch_once_t dispatch_queue_t dispatch_sync dispatch_async dispatch_once", }, lexemes: n, illegal: "/, end: /$/, - illegal: "\\n" + illegal: "\\n", }, e.C_LINE_COMMENT_MODE, - e.C_BLOCK_COMMENT_MODE - ] + e.C_BLOCK_COMMENT_MODE, + ], }, { className: "class", @@ -1673,10 +1682,10 @@ excludeEnd: !0, keywords: a, lexemes: n, - contains: [e.UNDERSCORE_TITLE_MODE] + contains: [e.UNDERSCORE_TITLE_MODE], }, - { begin: "\\." + e.UNDERSCORE_IDENT_RE, relevance: 0 } - ] + { begin: "\\." + e.UNDERSCORE_IDENT_RE, relevance: 0 }, + ], }; } function v(e) { @@ -1688,8 +1697,8 @@ variants: [ { begin: /\$\d/ }, { begin: /[\$%@](\^\w\b|#\w+(::\w+)*|{\w+}|\w+(::\w*)*)/ }, - { begin: /[\$%@][^\s\w{]/, relevance: 0 } - ] + { begin: /[\$%@][^\s\w{]/, relevance: 0 }, + ], }, s = [e.BACKSLASH_ESCAPE, a, i], i = [ @@ -1711,14 +1720,14 @@ { begin: '"', end: '"' }, { begin: "`", end: "`", contains: [e.BACKSLASH_ESCAPE] }, { begin: "{\\w+}", contains: [], relevance: 0 }, - { begin: "-?\\w+\\s*\\=\\>", contains: [], relevance: 0 } - ] + { begin: "-?\\w+\\s*\\=\\>", contains: [], relevance: 0 }, + ], }, { className: "number", begin: "(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b", - relevance: 0 + relevance: 0, }, { begin: @@ -1732,16 +1741,16 @@ { className: "regexp", begin: "(s|tr|y)/(\\\\.|[^/])*/(\\\\.|[^/])*/[a-z]*", - relevance: 10 + relevance: 10, }, { className: "regexp", begin: "(m|qr)?/", end: "/[a-z]*", contains: [e.BACKSLASH_ESCAPE], - relevance: 0 - } - ] + relevance: 0, + }, + ], }, { className: "function", @@ -1749,15 +1758,15 @@ end: "(\\s*\\(.*?\\))?[;{]", excludeEnd: !0, relevance: 5, - contains: [e.TITLE_MODE] + contains: [e.TITLE_MODE], }, { begin: "-\\w\\b", relevance: 0 }, { begin: "^__DATA__$", end: "^__END__$", subLanguage: "mojolicious", - contains: [{ begin: "^@@.*", end: "$", className: "comment" }] - } + contains: [{ begin: "^@@.*", end: "$", className: "comment" }], + }, ]; return ( (a.contains = i), @@ -1765,7 +1774,7 @@ aliases: ["pl", "pm"], lexemes: /[\w\.]+/, keywords: n, - contains: (t.contains = i) + contains: (t.contains = i), } ); } @@ -1779,8 +1788,8 @@ { begin: 'b"', end: '"' }, { begin: "b'", end: "'" }, e.inherit(e.APOS_STRING_MODE, { illegal: null }), - e.inherit(e.QUOTE_STRING_MODE, { illegal: null }) - ] + e.inherit(e.QUOTE_STRING_MODE, { illegal: null }), + ], }, i = { variants: [e.BINARY_NUMBER_MODE, e.C_NUMBER_MODE] }; return { @@ -1792,12 +1801,12 @@ e.HASH_COMMENT_MODE, e.COMMENT("//", "$", { contains: [a] }), e.COMMENT("/\\*", "\\*/", { - contains: [{ className: "doctag", begin: "@[A-Za-z]+" }] + contains: [{ className: "doctag", begin: "@[A-Za-z]+" }], }), e.COMMENT("__halt_compiler.+?;", !1, { endsWithParent: !0, keywords: "__halt_compiler", - lexemes: e.UNDERSCORE_IDENT_RE + lexemes: e.UNDERSCORE_IDENT_RE, }), { className: "string", @@ -1807,9 +1816,9 @@ e.BACKSLASH_ESCAPE, { className: "subst", - variants: [{ begin: /\$\w+/ }, { begin: /\{\$/, end: /\}/ }] - } - ] + variants: [{ begin: /\$\w+/ }, { begin: /\{\$/, end: /\}/ }], + }, + ], }, a, { className: "keyword", begin: /\$this\b/ }, @@ -1827,9 +1836,9 @@ className: "params", begin: "\\(", end: "\\)", - contains: ["self", n, e.C_BLOCK_COMMENT_MODE, t, i] - } - ] + contains: ["self", n, e.C_BLOCK_COMMENT_MODE, t, i], + }, + ], }, { className: "class", @@ -1839,20 +1848,20 @@ illegal: /[:\(\$"]/, contains: [ { beginKeywords: "extends implements" }, - e.UNDERSCORE_TITLE_MODE - ] + e.UNDERSCORE_TITLE_MODE, + ], }, { beginKeywords: "namespace", end: ";", illegal: /[\.']/, - contains: [e.UNDERSCORE_TITLE_MODE] + contains: [e.UNDERSCORE_TITLE_MODE], }, { beginKeywords: "use", end: ";", contains: [e.UNDERSCORE_TITLE_MODE] }, { begin: "=>" }, t, - i - ] + i, + ], }; } function w(e) { @@ -1867,8 +1876,8 @@ className: "string", end: /$/, relevance: 0, - contains: [{ begin: "\\\\\\n" }] - } + contains: [{ begin: "\\\\\\n" }], + }, }; return { case_insensitive: !0, @@ -1879,21 +1888,21 @@ begin: t + a, returnBegin: !0, contains: [ - { className: "attr", begin: t, endsParent: !0, relevance: 0 } + { className: "attr", begin: t, endsParent: !0, relevance: 0 }, ], - starts: s + starts: s, }, { begin: i + a, returnBegin: !0, relevance: 0, contains: [ - { className: "meta", begin: i, endsParent: !0, relevance: 0 } + { className: "meta", begin: i, endsParent: !0, relevance: 0 }, ], - starts: s + starts: s, }, - { className: "attr", relevance: 0, begin: i + n + "$" } - ] + { className: "attr", relevance: 0, begin: i + n + "$" }, + ], }; } function O(e) { @@ -1906,8 +1915,8 @@ contains: [e.BACKSLASH_ESCAPE, a], variants: [ { begin: /'/, end: /'/ }, - { begin: /"/, end: /"/ } - ] + { begin: /"/, end: /"/ }, + ], }; return { aliases: ["pp"], @@ -1919,14 +1928,14 @@ beginKeywords: "class", end: "\\{|;", illegal: /=/, - contains: [t, n] + contains: [t, n], }, { beginKeywords: "define", end: /\{/, contains: [ - { className: "section", begin: e.IDENT_RE, endsParent: !0 } - ] + { className: "section", begin: e.IDENT_RE, endsParent: !0 }, + ], }, { begin: e.IDENT_RE + "\\s+\\{", @@ -1943,7 +1952,7 @@ literal: "alias audit before loglevel noop require subscribe tag owner ensure group mode name|0 changes context force incl lens load_path onlyif provider returns root show_diff type_check en_address ip_address realname command environment hour monute month monthday special target weekday creates cwd ogoutput refresh refreshonly tries try_sleep umask backup checksum content ctime force ignore links mtime purge recurse recurselimit replace selinux_ignore_defaults selrange selrole seltype seluser source souirce_permissions sourceselect validate_cmd validate_replacement allowdupe attribute_membership auth_membership forcelocal gid ia_load_module members system host_aliases ip allowed_trunk_vlans description device_url duplex encapsulation etherchannel native_vlan speed principals allow_root auth_class auth_type authenticate_user k_of_n mechanisms rule session_owner shared options device fstype enable hasrestart directory present absent link atboot blockdevice device dump pass remounts poller_tag use message withpath adminfile allow_virtual allowcdrom category configfiles flavor install_options instance package_settings platform responsefile status uninstall_options vendor unless_system_user unless_uid binary control flags hasstatus manifest pattern restart running start stop allowdupe auths expiry gid groups home iterations key_membership keys managehome membership password password_max_age password_min_age profile_membership profiles project purge_ssh_keys role_membership roles salt shell uid baseurl cost descr enabled enablegroups exclude failovermethod gpgcheck gpgkey http_caching include includepkgs keepalive metadata_expire metalink mirrorlist priority protect proxy proxy_password proxy_username repo_gpgcheck s3_enabled skip_if_unavailable sslcacert sslclientcert sslclientkey sslverify mounted", built_in: - "architecture augeasversion blockdevices boardmanufacturer boardproductname boardserialnumber cfkey dhcp_servers domain ec2_ ec2_userdata facterversion filesystems ldom fqdn gid hardwareisa hardwaremodel hostname id|0 interfaces ipaddress ipaddress_ ipaddress6 ipaddress6_ iphostnumber is_virtual kernel kernelmajversion kernelrelease kernelversion kernelrelease kernelversion lsbdistcodename lsbdistdescription lsbdistid lsbdistrelease lsbmajdistrelease lsbminordistrelease lsbrelease macaddress macaddress_ macosx_buildversion macosx_productname macosx_productversion macosx_productverson_major macosx_productversion_minor manufacturer memoryfree memorysize netmask metmask_ network_ operatingsystem operatingsystemmajrelease operatingsystemrelease osfamily partitions path physicalprocessorcount processor processorcount productname ps puppetversion rubysitedir rubyversion selinux selinux_config_mode selinux_config_policy selinux_current_mode selinux_current_mode selinux_enforced selinux_policyversion serialnumber sp_ sshdsakey sshecdsakey sshrsakey swapencrypted swapfree swapsize timezone type uniqueid uptime uptime_days uptime_hours uptime_seconds uuid virtual vlans xendomains zfs_version zonenae zones zpool_version" + "architecture augeasversion blockdevices boardmanufacturer boardproductname boardserialnumber cfkey dhcp_servers domain ec2_ ec2_userdata facterversion filesystems ldom fqdn gid hardwareisa hardwaremodel hostname id|0 interfaces ipaddress ipaddress_ ipaddress6 ipaddress6_ iphostnumber is_virtual kernel kernelmajversion kernelrelease kernelversion kernelrelease kernelversion lsbdistcodename lsbdistdescription lsbdistid lsbdistrelease lsbmajdistrelease lsbminordistrelease lsbrelease macaddress macaddress_ macosx_buildversion macosx_productname macosx_productversion macosx_productverson_major macosx_productversion_minor manufacturer memoryfree memorysize netmask metmask_ network_ operatingsystem operatingsystemmajrelease operatingsystemrelease osfamily partitions path physicalprocessorcount processor processorcount productname ps puppetversion rubysitedir rubyversion selinux selinux_config_mode selinux_config_policy selinux_current_mode selinux_current_mode selinux_enforced selinux_policyversion serialnumber sp_ sshdsakey sshecdsakey sshrsakey swapencrypted swapfree swapsize timezone type uniqueid uptime uptime_days uptime_hours uptime_seconds uuid virtual vlans xendomains zfs_version zonenae zones zpool_version", }, relevance: 0, contains: [ @@ -1953,21 +1962,21 @@ begin: "[a-zA-Z_]+\\s*=>", returnBegin: !0, end: "=>", - contains: [{ className: "attr", begin: e.IDENT_RE }] + contains: [{ className: "attr", begin: e.IDENT_RE }], }, { className: "number", begin: "(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b", - relevance: 0 + relevance: 0, }, - a - ] - } + a, + ], + }, ], - relevance: 0 - } - ] + relevance: 0, + }, + ], }; } function M(e) { @@ -1975,7 +1984,7 @@ keyword: "and elif is global as in if from raise for except finally print import pass return exec else break not with class assert yield try while continue del or def lambda async await nonlocal|10", built_in: "Ellipsis NotImplemented", - literal: "False None True" + literal: "False None True", }, a = { className: "meta", begin: /^(>>>|\.\.\.) / }, t = { @@ -1983,7 +1992,7 @@ begin: /\{/, end: /\}/, keywords: n, - illegal: /#/ + illegal: /#/, }, i = { begin: /\{\{/, relevance: 0 }, i = { @@ -1994,23 +2003,23 @@ begin: /(u|b)?r?'''/, end: /'''/, contains: [e.BACKSLASH_ESCAPE, a], - relevance: 10 + relevance: 10, }, { begin: /(u|b)?r?"""/, end: /"""/, contains: [e.BACKSLASH_ESCAPE, a], - relevance: 10 + relevance: 10, }, { begin: /(fr|rf|f)'''/, end: /'''/, - contains: [e.BACKSLASH_ESCAPE, a, i, t] + contains: [e.BACKSLASH_ESCAPE, a, i, t], }, { begin: /(fr|rf|f)"""/, end: /"""/, - contains: [e.BACKSLASH_ESCAPE, a, i, t] + contains: [e.BACKSLASH_ESCAPE, a, i, t], }, { begin: /(u|r|ur)'/, end: /'/, relevance: 10 }, { begin: /(u|r|ur)"/, end: /"/, relevance: 10 }, @@ -2019,16 +2028,16 @@ { begin: /(fr|rf|f)'/, end: /'/, - contains: [e.BACKSLASH_ESCAPE, i, t] + contains: [e.BACKSLASH_ESCAPE, i, t], }, { begin: /(fr|rf|f)"/, end: /"/, - contains: [e.BACKSLASH_ESCAPE, i, t] + contains: [e.BACKSLASH_ESCAPE, i, t], }, e.APOS_STRING_MODE, - e.QUOTE_STRING_MODE - ] + e.QUOTE_STRING_MODE, + ], }, s = { className: "number", @@ -2036,14 +2045,14 @@ variants: [ { begin: e.BINARY_NUMBER_RE + "[lLjJ]?" }, { begin: "\\b(0o[0-7]+)[lLjJ]?" }, - { begin: e.C_NUMBER_RE + "[lLjJ]?" } - ] + { begin: e.C_NUMBER_RE + "[lLjJ]?" }, + ], }, r = { className: "params", begin: /\(/, end: /\)/, - contains: ["self", a, s, i, e.HASH_COMMENT_MODE] + contains: ["self", a, s, i, e.HASH_COMMENT_MODE], }; return ( (t.contains = [i, s, a]), @@ -2060,19 +2069,19 @@ { variants: [ { className: "function", beginKeywords: "def" }, - { className: "class", beginKeywords: "class" } + { className: "class", beginKeywords: "class" }, ], end: /:/, illegal: /[${=;\n,]/, contains: [ e.UNDERSCORE_TITLE_MODE, r, - { begin: /->/, endsWithParent: !0, keywords: "None" } - ] + { begin: /->/, endsWithParent: !0, keywords: "None" }, + ], }, { className: "meta", begin: /^[\t ]*@/, end: /$/ }, - { begin: /\b(print|exec)\(/ } - ] + { begin: /\b(print|exec)\(/ }, + ], } ); } @@ -2082,14 +2091,14 @@ a = { keyword: "and then defined module in return redo if BEGIN retry end for self when next until do begin unless END rescue else break undef not super class case require yield alias while ensure elsif or include attr_reader attr_writer attr_accessor", - literal: "true false nil" + literal: "true false nil", }, t = { className: "doctag", begin: "@[A-Za-z]+" }, i = { begin: "#<", end: ">" }, t = [ e.COMMENT("#", "$", { contains: [t] }), e.COMMENT("^\\=begin", "^\\=end", { contains: [t], relevance: 10 }), - e.COMMENT("^__END__", "\\n$") + e.COMMENT("^__END__", "\\n$"), ], s = { className: "subst", begin: "#\\{", end: "}", keywords: a }, r = { @@ -2108,7 +2117,8 @@ { begin: "%[qQwWx]?-", end: "-" }, { begin: "%[qQwWx]?\\|", end: "\\|" }, { - begin: /\B\?(\\\d{1,3}|\\x[A-Fa-f0-9]{1,2}|\\u[A-Fa-f0-9]{4}|\\?\S)\b/ + begin: + /\B\?(\\\d{1,3}|\\x[A-Fa-f0-9]{1,2}|\\u[A-Fa-f0-9]{4}|\\?\S)\b/, }, { begin: /<<[-~]?'?(\w+)(?:.|\n)*?\n\s*\1\b/, @@ -2118,18 +2128,18 @@ { begin: /\w+/, endSameAsBegin: !0, - contains: [e.BACKSLASH_ESCAPE, s] - } - ] - } - ] + contains: [e.BACKSLASH_ESCAPE, s], + }, + ], + }, + ], }, l = { className: "params", begin: "\\(", end: "\\)", endsParent: !0, - keywords: a + keywords: a, }, r = [ r, @@ -2141,37 +2151,37 @@ illegal: /=/, contains: [ e.inherit(e.TITLE_MODE, { - begin: "[A-Za-z_]\\w*(::\\w+)*(\\?|\\!)?" + begin: "[A-Za-z_]\\w*(::\\w+)*(\\?|\\!)?", }), { begin: "<\\s*", - contains: [{ begin: "(" + e.IDENT_RE + "::)?" + e.IDENT_RE }] - } - ].concat(t) + contains: [{ begin: "(" + e.IDENT_RE + "::)?" + e.IDENT_RE }], + }, + ].concat(t), }, { className: "function", beginKeywords: "def", end: "$|;", - contains: [e.inherit(e.TITLE_MODE, { begin: n }), l].concat(t) + contains: [e.inherit(e.TITLE_MODE, { begin: n }), l].concat(t), }, { begin: e.IDENT_RE + "::" }, { className: "symbol", begin: e.UNDERSCORE_IDENT_RE + "(\\!|\\?)?:", - relevance: 0 + relevance: 0, }, { className: "symbol", begin: ":(?!\\s)", contains: [r, { begin: n }], - relevance: 0 + relevance: 0, }, { className: "number", begin: "(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b", - relevance: 0 + relevance: 0, }, { begin: "(\\$\\W)|((\\$|\\@\\@?)(\\w+))" }, { className: "params", begin: /\|/, end: /\|/, keywords: a }, @@ -2189,12 +2199,12 @@ { begin: "%r{", end: "}[a-z]*" }, { begin: "%r\\(", end: "\\)[a-z]*" }, { begin: "%r!", end: "![a-z]*" }, - { begin: "%r\\[", end: "\\][a-z]*" } - ] - } + { begin: "%r\\[", end: "\\][a-z]*" }, + ], + }, ].concat(t), - relevance: 0 - } + relevance: 0, + }, ].concat(t); return ( (s.contains = r), @@ -2206,16 +2216,16 @@ .concat([ { begin: /^\s*=>/, - starts: { end: "$", contains: (l.contains = r) } + starts: { end: "$", contains: (l.contains = r) }, }, { className: "meta", begin: "^([>?]>|[\\w#]+\\(\\w+\\):\\d+:\\d+>|(\\w+-)?\\d+\\.\\d+\\.\\d(p\\d+)?[^>]+>)", - starts: { end: "$", contains: r } - } + starts: { end: "$", contains: r }, + }, ]) - .concat(r) + .concat(r), } ); } @@ -2229,7 +2239,7 @@ keyword: "abstract as async await become box break const continue crate do dyn else enum extern false final fn for if impl in let loop macro match mod move mut override priv pub ref return self Self static struct super trait true try type typeof unsafe unsized use virtual where while yield", literal: "true false Some None Ok Err", - built_in: a + built_in: a, }, lexemes: e.IDENT_RE + "!?", illegal: "" } - ] + { begin: "->" }, + ], }; } function S(e) { var n = { className: "subst", - variants: [{ begin: "\\$[A-Za-z0-9_]+" }, { begin: "\\${", end: "}" }] + variants: [{ begin: "\\$[A-Za-z0-9_]+" }, { begin: "\\${", end: "}" }], }, n = { className: "string", @@ -2299,35 +2309,36 @@ begin: '"', end: '"', illegal: "\\n", - contains: [e.BACKSLASH_ESCAPE] + contains: [e.BACKSLASH_ESCAPE], }, { begin: '"""', end: '"""', relevance: 10 }, { begin: '[a-z]+"', end: '"', illegal: "\\n", - contains: [e.BACKSLASH_ESCAPE, n] + contains: [e.BACKSLASH_ESCAPE, n], }, { className: "string", begin: '[a-z]+"""', end: '"""', contains: [n], - relevance: 10 - } - ] + relevance: 10, + }, + ], }, a = { className: "type", begin: "\\b[A-Z][A-Za-z0-9_]*", relevance: 0 }, t = { className: "title", - begin: /[^0-9\n\t "'(),.`{}\[\]:;][^\n\t "'(),.`{}\[\]:;]+|[^0-9\n\t "'(),.`{}\[\]:;=]/, - relevance: 0 + begin: + /[^0-9\n\t "'(),.`{}\[\]:;][^\n\t "'(),.`{}\[\]:;]+|[^0-9\n\t "'(),.`{}\[\]:;=]/, + relevance: 0, }; return { keywords: { literal: "true false null", keyword: - "type yield lazy override def with val var sealed abstract private trait object if forSome for while throw finally protected extends import final return else break new catch super class case package default try this match continue throws implicit" + "type yield lazy override def with val var sealed abstract private trait object if forSome for while throw finally protected extends import final return else break new catch super class case package default try this match continue throws implicit", }, contains: [ e.C_LINE_COMMENT_MODE, @@ -2340,7 +2351,7 @@ beginKeywords: "def", end: /[:={\[(\n;]/, excludeEnd: !0, - contains: [t] + contains: [t], }, { className: "class", @@ -2355,7 +2366,7 @@ excludeBegin: !0, excludeEnd: !0, relevance: 0, - contains: [a] + contains: [a], }, { className: "params", @@ -2364,14 +2375,14 @@ excludeBegin: !0, excludeEnd: !0, relevance: 0, - contains: [a] + contains: [a], }, - t - ] + t, + ], }, e.C_NUMBER_MODE, - { className: "meta", begin: "@[A-Za-z]+" } - ] + { className: "meta", begin: "@[A-Za-z]+" }, + ], }; } function T(e) { @@ -2381,9 +2392,9 @@ { className: "meta", begin: "^\\s{0,3}[/\\w\\d\\[\\]()@-]*[>%$#]", - starts: { end: "$", subLanguage: "bash" } - } - ] + starts: { end: "$", subLanguage: "bash" }, + }, + ], }; } function k(e) { @@ -2403,32 +2414,32 @@ "as abort abs absolute acc acce accep accept access accessed accessible account acos action activate add addtime admin administer advanced advise aes_decrypt aes_encrypt after agent aggregate ali alia alias all allocate allow alter always analyze ancillary and anti any anydata anydataset anyschema anytype apply archive archived archivelog are as asc ascii asin assembly assertion associate asynchronous at atan atn2 attr attri attrib attribu attribut attribute attributes audit authenticated authentication authid authors auto autoallocate autodblink autoextend automatic availability avg backup badfile basicfile before begin beginning benchmark between bfile bfile_base big bigfile bin binary_double binary_float binlog bit_and bit_count bit_length bit_or bit_xor bitmap blob_base block blocksize body both bound bucket buffer_cache buffer_pool build bulk by byte byteordermark bytes cache caching call calling cancel capacity cascade cascaded case cast catalog category ceil ceiling chain change changed char_base char_length character_length characters characterset charindex charset charsetform charsetid check checksum checksum_agg child choose chr chunk class cleanup clear client clob clob_base clone close cluster_id cluster_probability cluster_set clustering coalesce coercibility col collate collation collect colu colum column column_value columns columns_updated comment commit compact compatibility compiled complete composite_limit compound compress compute concat concat_ws concurrent confirm conn connec connect connect_by_iscycle connect_by_isleaf connect_by_root connect_time connection consider consistent constant constraint constraints constructor container content contents context contributors controlfile conv convert convert_tz corr corr_k corr_s corresponding corruption cos cost count count_big counted covar_pop covar_samp cpu_per_call cpu_per_session crc32 create creation critical cross cube cume_dist curdate current current_date current_time current_timestamp current_user cursor curtime customdatum cycle data database databases datafile datafiles datalength date_add date_cache date_format date_sub dateadd datediff datefromparts datename datepart datetime2fromparts day day_to_second dayname dayofmonth dayofweek dayofyear days db_role_change dbtimezone ddl deallocate declare decode decompose decrement decrypt deduplicate def defa defau defaul default defaults deferred defi defin define degrees delayed delegate delete delete_all delimited demand dense_rank depth dequeue des_decrypt des_encrypt des_key_file desc descr descri describ describe descriptor deterministic diagnostics difference dimension direct_load directory disable disable_all disallow disassociate discardfile disconnect diskgroup distinct distinctrow distribute distributed div do document domain dotnet double downgrade drop dumpfile duplicate duration each edition editionable editions element ellipsis else elsif elt empty enable enable_all enclosed encode encoding encrypt end end-exec endian enforced engine engines enqueue enterprise entityescaping eomonth error errors escaped evalname evaluate event eventdata events except exception exceptions exchange exclude excluding execu execut execute exempt exists exit exp expire explain explode export export_set extended extent external external_1 external_2 externally extract failed failed_login_attempts failover failure far fast feature_set feature_value fetch field fields file file_name_convert filesystem_like_logging final finish first first_value fixed flash_cache flashback floor flush following follows for forall force foreign form forma format found found_rows freelist freelists freepools fresh from from_base64 from_days ftp full function general generated get get_format get_lock getdate getutcdate global global_name globally go goto grant grants greatest group group_concat group_id grouping grouping_id groups gtid_subtract guarantee guard handler hash hashkeys having hea head headi headin heading heap help hex hierarchy high high_priority hosts hour hours http id ident_current ident_incr ident_seed identified identity idle_time if ifnull ignore iif ilike ilm immediate import in include including increment index indexes indexing indextype indicator indices inet6_aton inet6_ntoa inet_aton inet_ntoa infile initial initialized initially initrans inmemory inner innodb input insert install instance instantiable instr interface interleaved intersect into invalidate invisible is is_free_lock is_ipv4 is_ipv4_compat is_not is_not_null is_used_lock isdate isnull isolation iterate java join json json_exists keep keep_duplicates key keys kill language large last last_day last_insert_id last_value lateral lax lcase lead leading least leaves left len lenght length less level levels library like like2 like4 likec limit lines link list listagg little ln load load_file lob lobs local localtime localtimestamp locate locator lock locked log log10 log2 logfile logfiles logging logical logical_reads_per_call logoff logon logs long loop low low_priority lower lpad lrtrim ltrim main make_set makedate maketime managed management manual map mapping mask master master_pos_wait match matched materialized max maxextents maximize maxinstances maxlen maxlogfiles maxloghistory maxlogmembers maxsize maxtrans md5 measures median medium member memcompress memory merge microsecond mid migration min minextents minimum mining minus minute minutes minvalue missing mod mode model modification modify module monitoring month months mount move movement multiset mutex name name_const names nan national native natural nav nchar nclob nested never new newline next nextval no no_write_to_binlog noarchivelog noaudit nobadfile nocheck nocompress nocopy nocycle nodelay nodiscardfile noentityescaping noguarantee nokeep nologfile nomapping nomaxvalue nominimize nominvalue nomonitoring none noneditionable nonschema noorder nopr nopro noprom nopromp noprompt norely noresetlogs noreverse normal norowdependencies noschemacheck noswitch not nothing notice notnull notrim novalidate now nowait nth_value nullif nulls num numb numbe nvarchar nvarchar2 object ocicoll ocidate ocidatetime ociduration ociinterval ociloblocator ocinumber ociref ocirefcursor ocirowid ocistring ocitype oct octet_length of off offline offset oid oidindex old on online only opaque open operations operator optimal optimize option optionally or oracle oracle_date oradata ord ordaudio orddicom orddoc order ordimage ordinality ordvideo organization orlany orlvary out outer outfile outline output over overflow overriding package pad parallel parallel_enable parameters parent parse partial partition partitions pascal passing password password_grace_time password_lock_time password_reuse_max password_reuse_time password_verify_function patch path patindex pctincrease pctthreshold pctused pctversion percent percent_rank percentile_cont percentile_disc performance period period_add period_diff permanent physical pi pipe pipelined pivot pluggable plugin policy position post_transaction pow power pragma prebuilt precedes preceding precision prediction prediction_cost prediction_details prediction_probability prediction_set prepare present preserve prior priority private private_sga privileges procedural procedure procedure_analyze processlist profiles project prompt protection public publishingservername purge quarter query quick quiesce quota quotename radians raise rand range rank raw read reads readsize rebuild record records recover recovery recursive recycle redo reduced ref reference referenced references referencing refresh regexp_like register regr_avgx regr_avgy regr_count regr_intercept regr_r2 regr_slope regr_sxx regr_sxy reject rekey relational relative relaylog release release_lock relies_on relocate rely rem remainder rename repair repeat replace replicate replication required reset resetlogs resize resource respect restore restricted result result_cache resumable resume retention return returning returns reuse reverse revoke right rlike role roles rollback rolling rollup round row row_count rowdependencies rowid rownum rows rtrim rules safe salt sample save savepoint sb1 sb2 sb4 scan schema schemacheck scn scope scroll sdo_georaster sdo_topo_geometry search sec_to_time second seconds section securefile security seed segment select self semi sequence sequential serializable server servererror session session_user sessions_per_user set sets settings sha sha1 sha2 share shared shared_pool short show shrink shutdown si_averagecolor si_colorhistogram si_featurelist si_positionalcolor si_stillimage si_texture siblings sid sign sin size size_t sizes skip slave sleep smalldatetimefromparts smallfile snapshot some soname sort soundex source space sparse spfile split sql sql_big_result sql_buffer_result sql_cache sql_calc_found_rows sql_small_result sql_variant_property sqlcode sqldata sqlerror sqlname sqlstate sqrt square standalone standby start starting startup statement static statistics stats_binomial_test stats_crosstab stats_ks_test stats_mode stats_mw_test stats_one_way_anova stats_t_test_ stats_t_test_indep stats_t_test_one stats_t_test_paired stats_wsr_test status std stddev stddev_pop stddev_samp stdev stop storage store stored str str_to_date straight_join strcmp strict string struct stuff style subdate subpartition subpartitions substitutable substr substring subtime subtring_index subtype success sum suspend switch switchoffset switchover sync synchronous synonym sys sys_xmlagg sysasm sysaux sysdate sysdatetimeoffset sysdba sysoper system system_user sysutcdatetime table tables tablespace tablesample tan tdo template temporary terminated tertiary_weights test than then thread through tier ties time time_format time_zone timediff timefromparts timeout timestamp timestampadd timestampdiff timezone_abbr timezone_minute timezone_region to to_base64 to_date to_days to_seconds todatetimeoffset trace tracking transaction transactional translate translation treat trigger trigger_nestlevel triggers trim truncate try_cast try_convert try_parse type ub1 ub2 ub4 ucase unarchived unbounded uncompress under undo unhex unicode uniform uninstall union unique unix_timestamp unknown unlimited unlock unnest unpivot unrecoverable unsafe unsigned until untrusted unusable unused update updated upgrade upped upper upsert url urowid usable usage use use_stored_outlines user user_data user_resources users using utc_date utc_timestamp uuid uuid_short validate validate_password_strength validation valist value values var var_samp varcharc vari varia variab variabl variable variables variance varp varraw varrawc varray verify version versions view virtual visible void wait wallet warning warnings week weekday weekofyear wellformed when whene whenev wheneve whenever where while whitespace window with within without work wrapped xdb xml xmlagg xmlattributes xmlcast xmlcolattval xmlelement xmlexists xmlforest xmlindex xmlnamespaces xmlpi xmlquery xmlroot xmlschema xmlserialize xmltable xmltype xor year year_to_month years yearweek", literal: "true false null unknown", built_in: - "array bigint binary bit blob bool boolean char character date dec decimal float int int8 integer interval number numeric real record serial serial8 smallint text time timestamp tinyint varchar varchar2 varying void" + "array bigint binary bit blob bool boolean char character date dec decimal float int int8 integer interval number numeric real record serial serial8 smallint text time timestamp tinyint varchar varchar2 varying void", }, contains: [ { className: "string", begin: "'", end: "'", - contains: [{ begin: "''" }] + contains: [{ begin: "''" }], }, { className: "string", begin: '"', end: '"', - contains: [{ begin: '""' }] + contains: [{ begin: '""' }], }, { className: "string", begin: "`", end: "`" }, e.C_NUMBER_MODE, e.C_BLOCK_COMMENT_MODE, n, - e.HASH_COMMENT_MODE - ] + e.HASH_COMMENT_MODE, + ], }, e.C_BLOCK_COMMENT_MODE, n, - e.HASH_COMMENT_MODE - ] + e.HASH_COMMENT_MODE, + ], }; } function A(e) { @@ -2437,7 +2448,7 @@ "#available #colorLiteral #column #else #elseif #endif #file #fileLiteral #function #if #imageLiteral #line #selector #sourceLocation _ __COLUMN__ __FILE__ __FUNCTION__ __LINE__ Any as as! as? associatedtype associativity break case catch class continue convenience default defer deinit didSet do dynamic dynamicType else enum extension fallthrough false fileprivate final for func get guard if import in indirect infix init inout internal is lazy left let mutating nil none nonmutating open operator optional override postfix precedence prefix private protocol Protocol public repeat required rethrows return right self Self set static struct subscript super switch throw throws true try try! try? Type typealias unowned var weak where while willSet", literal: "true false nil", built_in: - "abs advance alignof alignofValue anyGenerator assert assertionFailure bridgeFromObjectiveC bridgeFromObjectiveCUnconditional bridgeToObjectiveC bridgeToObjectiveCUnconditional c contains count countElements countLeadingZeros debugPrint debugPrintln distance dropFirst dropLast dump encodeBitsAsWords enumerate equal fatalError filter find getBridgedObjectiveCType getVaList indices insertionSort isBridgedToObjectiveC isBridgedVerbatimToObjectiveC isUniquelyReferenced isUniquelyReferencedNonObjC join lazy lexicographicalCompare map max maxElement min minElement numericCast overlaps partition posix precondition preconditionFailure print println quickSort readLine reduce reflect reinterpretCast reverse roundUpToAlignment sizeof sizeofValue sort split startsWith stride strideof strideofValue swap toString transcode underestimateCount unsafeAddressOf unsafeBitCast unsafeDowncast unsafeUnwrap unsafeReflect withExtendedLifetime withObjectAtPlusZero withUnsafePointer withUnsafePointerToObject withUnsafeMutablePointer withUnsafeMutablePointers withUnsafePointer withUnsafePointers withVaList zip" + "abs advance alignof alignofValue anyGenerator assert assertionFailure bridgeFromObjectiveC bridgeFromObjectiveCUnconditional bridgeToObjectiveC bridgeToObjectiveCUnconditional c contains count countElements countLeadingZeros debugPrint debugPrintln distance dropFirst dropLast dump encodeBitsAsWords enumerate equal fatalError filter find getBridgedObjectiveCType getVaList indices insertionSort isBridgedToObjectiveC isBridgedVerbatimToObjectiveC isUniquelyReferenced isUniquelyReferencedNonObjC join lazy lexicographicalCompare map max maxElement min minElement numericCast overlaps partition posix precondition preconditionFailure print println quickSort readLine reduce reflect reinterpretCast reverse roundUpToAlignment sizeof sizeofValue sort split startsWith stride strideof strideofValue swap toString transcode underestimateCount unsafeAddressOf unsafeBitCast unsafeDowncast unsafeUnwrap unsafeReflect withExtendedLifetime withObjectAtPlusZero withUnsafePointer withUnsafePointerToObject withUnsafeMutablePointer withUnsafeMutablePointers withUnsafePointer withUnsafePointers withVaList zip", }, a = e.COMMENT("/\\*", "\\*/", { contains: ["self"] }), t = { @@ -2445,21 +2456,21 @@ begin: /\\\(/, end: "\\)", keywords: n, - contains: [] + contains: [], }, i = { className: "string", contains: [e.BACKSLASH_ESCAPE, t], variants: [ { begin: /"""/, end: /"""/ }, - { begin: /"/, end: /"/ } - ] + { begin: /"/, end: /"/ }, + ], }, s = { className: "number", begin: "\\b([\\d_]+(\\.[\\deE_]+)?|0x[a-fA-F0-9_]+(\\.[a-fA-F0-9p_]+)?|0b[01_]+|0o[0-7_]+)\\b", - relevance: 0 + relevance: 0, }; return ( (t.contains = [s]), @@ -2491,12 +2502,12 @@ s, i, e.C_BLOCK_COMMENT_MODE, - { begin: ":" } + { begin: ":" }, ], - illegal: /["']/ - } + illegal: /["']/, + }, ], - illegal: /\[|%/ + illegal: /\[|%/, }, { className: "class", @@ -2506,21 +2517,21 @@ excludeEnd: !0, contains: [ e.inherit(e.TITLE_MODE, { - begin: /[A-Za-z$_][\u00C0-\u02B80-9A-Za-z$_]*/ - }) - ] + begin: /[A-Za-z$_][\u00C0-\u02B80-9A-Za-z$_]*/, + }), + ], }, { className: "meta", begin: - "(@discardableResult|@warn_unused_result|@exported|@lazy|@noescape|@NSCopying|@NSManaged|@objc|@objcMembers|@convention|@required|@noreturn|@IBAction|@IBDesignable|@IBInspectable|@IBOutlet|@infix|@prefix|@postfix|@autoclosure|@testable|@available|@nonobjc|@NSApplicationMain|@UIApplicationMain|@dynamicMemberLookup|@propertyWrapper)" + "(@discardableResult|@warn_unused_result|@exported|@lazy|@noescape|@NSCopying|@NSManaged|@objc|@objcMembers|@convention|@required|@noreturn|@IBAction|@IBDesignable|@IBInspectable|@IBOutlet|@infix|@prefix|@postfix|@autoclosure|@testable|@available|@nonobjc|@NSApplicationMain|@UIApplicationMain|@dynamicMemberLookup|@propertyWrapper)", }, { beginKeywords: "import", end: /$/, - contains: [e.C_LINE_COMMENT_MODE, a] - } - ] + contains: [e.C_LINE_COMMENT_MODE, a], + }, + ], } ); } @@ -2532,9 +2543,9 @@ { className: "meta-keyword", begin: "#?[a-z_][a-z1-9_-]+", - illegal: "\\n" - } - ] + illegal: "\\n", + }, + ], }, t = e.inherit(a, { begin: "\\(", end: "\\)" }), i = e.inherit(e.APOS_STRING_MODE, { className: "meta-string" }), @@ -2555,12 +2566,12 @@ variants: [ { begin: /"/, end: /"/, contains: [n] }, { begin: /'/, end: /'/, contains: [n] }, - { begin: /[^\s"'=<>`]+/ } - ] - } - ] - } - ] + { begin: /[^\s"'=<>`]+/ }, + ], + }, + ], + }, + ], }; return { aliases: [ @@ -2573,7 +2584,7 @@ "xsl", "plist", "wsf", - "svg" + "svg", ], case_insensitive: !0, contains: [ @@ -2595,11 +2606,11 @@ className: "meta", begin: "", - contains: [a, t, s, i] - } - ] - } - ] + contains: [a, t, s, i], + }, + ], + }, + ], }, e.COMMENT("\x3c!--", "--\x3e", { relevance: 10 }), { begin: "<\\!\\[CDATA\\[", end: "\\]\\]>", relevance: 10 }, @@ -2617,15 +2628,15 @@ illegal: null, className: null, contains: null, - skip: !0 + skip: !0, }), e.inherit(e.QUOTE_STRING_MODE, { illegal: null, className: null, contains: null, - skip: !0 - }) - ] + skip: !0, + }), + ], }, { className: "tag", @@ -2636,8 +2647,8 @@ starts: { end: "", returnEnd: !0, - subLanguage: ["css", "xml"] - } + subLanguage: ["css", "xml"], + }, }, { className: "tag", @@ -2648,8 +2659,8 @@ starts: { end: "", returnEnd: !0, - subLanguage: ["actionscript", "javascript", "handlebars", "xml"] - } + subLanguage: ["actionscript", "javascript", "handlebars", "xml"], + }, }, { className: "tag", @@ -2657,10 +2668,10 @@ end: "/?>", contains: [ { className: "name", begin: /[^\/><\s]+/, relevance: 0 }, - r - ] - } - ] + r, + ], + }, + ], }; } function B(e) { @@ -2671,7 +2682,7 @@ variants: [ { begin: /'/, end: /'/ }, { begin: /"/, end: /"/ }, - { begin: /\S+/ } + { begin: /\S+/ }, ], contains: [ e.BACKSLASH_ESCAPE, @@ -2679,10 +2690,10 @@ className: "template-variable", variants: [ { begin: "{{", end: "}}" }, - { begin: "%{", end: "}" } - ] - } - ] + { begin: "%{", end: "}" }, + ], + }, + ], }; return { case_insensitive: !0, @@ -2693,13 +2704,13 @@ variants: [ { begin: "\\w[\\w :\\/.-]*:(?=[ \t]|$)" }, { begin: '"\\w[\\w :\\/.-]*":(?=[ \t]|$)' }, - { begin: "'\\w[\\w :\\/.-]*':(?=[ \t]|$)" } - ] + { begin: "'\\w[\\w :\\/.-]*':(?=[ \t]|$)" }, + ], }, { className: "meta", begin: "^---s*$", relevance: 10 }, { className: "string", - begin: "[\\|>]([0-9]?[+-])?[ ]*\\n( *)[\\S ]+\\n(\\2[\\S ]+\\n?)*" + begin: "[\\|>]([0-9]?[+-])?[ ]*\\n( *)[\\S ]+\\n(\\2[\\S ]+\\n?)*", }, { begin: "<%[%=-]?", @@ -2707,7 +2718,7 @@ subLanguage: "ruby", excludeBegin: !0, excludeEnd: !0, - relevance: 0 + relevance: 0, }, { className: "type", begin: "!" + e.UNDERSCORE_IDENT_RE }, { className: "type", begin: "!!" + e.UNDERSCORE_IDENT_RE }, @@ -2717,14 +2728,14 @@ e.HASH_COMMENT_MODE, { beginKeywords: n, keywords: { literal: n } }, { className: "number", begin: e.C_NUMBER_RE + "\\b" }, - a - ] + a, + ], }; } var D, L, I = {}; - (D = function(t) { + (D = function (t) { var a, g = [], s = Object.keys, @@ -2741,7 +2752,7 @@ classPrefix: "hljs-", tabReplace: null, useBR: !1, - languages: void 0 + languages: void 0, }, o = "of and for in not or if then".split(" "); function S(e) { @@ -2762,7 +2773,7 @@ t = Array.prototype.slice.call(arguments, 1); for (n in e) a[n] = e[n]; return ( - t.forEach(function(e) { + t.forEach(function (e) { for (n in e) a[n] = e[n]; }), a @@ -2796,18 +2807,18 @@ ? e : n : "start" === n[0].event - ? e - : n + ? e + : n : e.length - ? e - : n; + ? e + : n; } function l(e) { i += "<" + u(e) + g.map - .call(e.attributes, function(e) { + .call(e.attributes, function (e) { return ( " " + e.nodeName + @@ -2847,7 +2858,7 @@ return ( n.variants && !n.cached_variants && - (n.cached_variants = n.variants.map(function(e) { + (n.cached_variants = n.variants.map(function (e) { return d(n, { variants: null }, e); })), n.cached_variants || @@ -2856,8 +2867,8 @@ })(n) ? [d(n, { starts: n.starts ? d(n.starts) : null })] : Object.isFrozen(n) - ? [d(n)] - : [n]) + ? [d(n)] + : [n]) ); } function p(e) { @@ -2871,13 +2882,13 @@ return ( "string" == typeof n ? a("keyword", n) - : s(n).forEach(function(e) { + : s(n).forEach(function (e) { a(e, n[e]); }), i ); function a(a, e) { - (e = t ? e.toLowerCase() : e).split(" ").forEach(function(e) { + (e = t ? e.toLowerCase() : e).split(" ").forEach(function (e) { var n, e = e.split("|"); i[e[0]] = [ @@ -2885,11 +2896,11 @@ ((n = e[0]), (e = e[1]) ? Number(e) - : (function(e) { - return -1 != o.indexOf(e.toLowerCase()); - })(n) - ? 0 - : 1) + : (function (e) { + return -1 != o.indexOf(e.toLowerCase()); + })(n) + ? 0 + : 1), ]; }); } @@ -2901,7 +2912,7 @@ function g(e, n) { return new RegExp( d(e), - "m" + (t.case_insensitive ? "i" : "") + (n ? "g" : "") + "m" + (t.case_insensitive ? "i" : "") + (n ? "g" : ""), ); } function i(i) { @@ -2924,7 +2935,7 @@ i.terminator_end && e("end", i.terminator_end), i.illegal && e("illegal", i.illegal); var c = g( - (function(e, n) { + (function (e, n) { for ( var a = /\[(?:[^\\\]]|\\.)*\]|\(\??|\\([1-9][0-9]*)|\\./, t = 0, @@ -2951,16 +2962,16 @@ } return i; })( - r.map(function(e) { + r.map(function (e) { return e[1]; }), - "|" + "|", ), - !0 + !0, ); return ( (l.lastIndex = 0), - (l.exec = function(e) { + (l.exec = function (e) { var n; if (0 === r.length) return null; c.lastIndex = l.lastIndex; @@ -2984,9 +2995,9 @@ if (t.contains && -1 != t.contains.indexOf("self")) { if (!O) throw new Error( - "ERR: contains `self` is not supported at the top-level of a language. See documentation." + "ERR: contains `self` is not supported at the top-level of a language. See documentation.", ); - t.contains = t.contains.filter(function(e) { + t.contains = t.contains.filter(function (e) { return "self" != e; }); } @@ -3014,11 +3025,11 @@ a.contains || (a.contains = []), (a.contains = Array.prototype.concat.apply( [], - a.contains.map(function(e) { + a.contains.map(function (e) { return b("self" === e ? a : e); - }) + }), )), - a.contains.forEach(function(e) { + a.contains.forEach(function (e) { n(e, a); }), a.starts && n(a.starts, e), @@ -3081,7 +3092,7 @@ e.endSameAsBegin && (e.endRe = new RegExp( n.replace(/[-\/\\^$*+?.()|[\]{}]/g, "\\$&"), - "m" + "m", )), e.skip ? (E += n) @@ -3145,7 +3156,7 @@ a + '" for mode "' + (m.className || "") + - '"' + '"', ); if ("end" === n.type) { e = d(n); @@ -3155,8 +3166,10 @@ } var _ = R(n); if (!_) - throw (console.error(x.replace("{}", n)), - new Error('Unknown language: "' + n + '"')); + throw ( + (console.error(x.replace("{}", n)), + new Error('Unknown language: "' + n + '"')) + ); T(_); for (var m = a || _, b = {}, p = "", f = m; f !== _; f = f.parent) f.className && (p = s(f.className, "", !0) + p); @@ -3180,7 +3193,7 @@ value: S(i), language: n, top: m, - errorRaised: e + errorRaised: e, }; throw e; } @@ -3193,7 +3206,7 @@ e .filter(R) .filter(y) - .forEach(function(e) { + .forEach(function (e) { var n = k(e, a, !1); (n.language = e), n.relevance > i.relevance && (i = n), @@ -3205,12 +3218,12 @@ } function E(e) { return C.tabReplace || C.useBR - ? e.replace(i, function(e, n) { + ? e.replace(i, function (e, n) { return C.useBR && "\n" === e ? "
" : C.tabReplace - ? n.replace(/\t/g, C.tabReplace) - : ""; + ? n.replace(/\t/g, C.tabReplace) + : ""; }) : e; } @@ -3219,7 +3232,7 @@ a, t, i, - s = (function(e) { + s = (function (e) { var n, a, t, @@ -3234,7 +3247,7 @@ (console.warn(x.replace("{}", a[1])), console.warn( "Falling back to no-highlight mode for this block.", - e + e, )), s ? a[1] : "no-highlight" ); @@ -3243,9 +3256,7 @@ })(e); c(s) || (C.useBR - ? ((a = document.createElement( - "div" - )).innerHTML = e.innerHTML + ? ((a = document.createElement("div")).innerHTML = e.innerHTML .replace(/\n/g, "") .replace(//g, "\n")) : (a = e), @@ -3269,7 +3280,7 @@ n.second_best && (e.second_best = { language: n.second_best.language, - re: n.second_best.relevance + re: n.second_best.relevance, })); } function h() { @@ -3292,15 +3303,15 @@ (t.highlightAuto = A), (t.fixMarkup = E), (t.highlightBlock = N), - (t.configure = function(e) { + (t.configure = function (e) { C = d(C, e); }), (t.initHighlighting = h), - (t.initHighlightingOnLoad = function() { + (t.initHighlightingOnLoad = function () { window.addEventListener("DOMContentLoaded", h, !1), window.addEventListener("load", h, !1); }), - (t.registerLanguage = function(n, e) { + (t.registerLanguage = function (n, e) { var a; try { a = e(t); @@ -3309,8 +3320,8 @@ (console.error( "Language definition for '{}' could not be registered.".replace( "{}", - n - ) + n, + ), ), !O) ) @@ -3320,24 +3331,24 @@ p((w[n] = a)), (a.rawDefinition = e.bind(null, t)), a.aliases && - a.aliases.forEach(function(e) { + a.aliases.forEach(function (e) { r[e] = n; }); }), - (t.listLanguages = function() { + (t.listLanguages = function () { return s(w); }), (t.getLanguage = R), - (t.requireLanguage = function(e) { + (t.requireLanguage = function (e) { var n = R(e); if (n) return n; throw new Error( - "The '{}' language is required, but not loaded.".replace("{}", e) + "The '{}' language is required, but not loaded.".replace("{}", e), ); }), (t.autoDetection = y), (t.inherit = d), - (t.debugMode = function() { + (t.debugMode = function () { O = !1; }), (t.IDENT_RE = "[a-zA-Z]\\w*"), @@ -3354,29 +3365,30 @@ begin: "'", end: "'", illegal: "\\n", - contains: [t.BACKSLASH_ESCAPE] + contains: [t.BACKSLASH_ESCAPE], }), (t.QUOTE_STRING_MODE = { className: "string", begin: '"', end: '"', illegal: "\\n", - contains: [t.BACKSLASH_ESCAPE] + contains: [t.BACKSLASH_ESCAPE], }), (t.PHRASAL_WORDS_MODE = { - begin: /\b(a|an|the|are|I'm|isn't|don't|doesn't|won't|but|just|should|pretty|simply|enough|gonna|going|wtf|so|such|will|you|your|they|like|more)\b/ + begin: + /\b(a|an|the|are|I'm|isn't|don't|doesn't|won't|but|just|should|pretty|simply|enough|gonna|going|wtf|so|such|will|you|your|they|like|more)\b/, }), - (t.COMMENT = function(e, n, a) { + (t.COMMENT = function (e, n, a) { e = t.inherit( { className: "comment", begin: e, end: n, contains: [] }, - a || {} + a || {}, ); return ( e.contains.push(t.PHRASAL_WORDS_MODE), e.contains.push({ className: "doctag", begin: "(?:TODO|FIXME|NOTE|BUG|XXX):", - relevance: 0 + relevance: 0, }), e ); @@ -3387,24 +3399,24 @@ (t.NUMBER_MODE = { className: "number", begin: t.NUMBER_RE, - relevance: 0 + relevance: 0, }), (t.C_NUMBER_MODE = { className: "number", begin: t.C_NUMBER_RE, - relevance: 0 + relevance: 0, }), (t.BINARY_NUMBER_MODE = { className: "number", begin: t.BINARY_NUMBER_RE, - relevance: 0 + relevance: 0, }), (t.CSS_NUMBER_MODE = { className: "number", begin: t.NUMBER_RE + "(%|em|ex|ch|rem|vw|vh|vmin|vmax|cm|mm|in|pt|pc|px|deg|grad|rad|turn|s|ms|Hz|kHz|dpi|dpcm|dppx)?", - relevance: 0 + relevance: 0, }), (t.REGEXP_MODE = { className: "regexp", @@ -3417,19 +3429,19 @@ begin: /\[/, end: /\]/, relevance: 0, - contains: [t.BACKSLASH_ESCAPE] - } - ] + contains: [t.BACKSLASH_ESCAPE], + }, + ], }), (t.TITLE_MODE = { className: "title", begin: t.IDENT_RE, relevance: 0 }), (t.UNDERSCORE_TITLE_MODE = { className: "title", begin: t.UNDERSCORE_IDENT_RE, - relevance: 0 + relevance: 0, }), (t.METHOD_GUARD = { begin: "\\.\\s*" + t.UNDERSCORE_IDENT_RE, - relevance: 0 + relevance: 0, }), [ t.BACKSLASH_ESCAPE, @@ -3447,12 +3459,12 @@ t.REGEXP_MODE, t.TITLE_MODE, t.UNDERSCORE_TITLE_MODE, - t.METHOD_GUARD - ].forEach(function(e) { + t.METHOD_GUARD, + ].forEach(function (e) { !(function n(a) { Object.freeze(a); var t = "function" == typeof a; - Object.getOwnPropertyNames(a).forEach(function(e) { + Object.getOwnPropertyNames(a).forEach(function (e) { !a.hasOwnProperty(e) || null === a[e] || ("object" != typeof a[e] && "function" != typeof a[e]) || @@ -3473,11 +3485,11 @@ ? L && ((L.hljs = D({})), "function" == typeof define) && define.amd && - define([], function() { + define([], function () { return L.hljs; }) : D(I); - !(function() { + !(function () { "use strict"; I.registerLanguage("asciidoc", e), I.registerLanguage("bash", n), @@ -3513,7 +3525,7 @@ I.registerLanguage("swift", A), I.registerLanguage("xml", R), I.registerLanguage("yaml", B), - I.registerLanguage("idl", function(e) { + I.registerLanguage("idl", function (e) { return { name: "IDL", case_insensitive: !1, @@ -3524,7 +3536,7 @@ e.HASH_COMMENT_MODE, { className: "comment", - variants: [{ begin: "(?<=description\\s*){", end: "}" }] + variants: [{ begin: "(?<=description\\s*){", end: "}" }], }, { className: "type", @@ -3532,12 +3544,13 @@ { begin: "\\b(Bits)\\s*<(?!<)", end: ">" }, { begin: "XReg" }, { begin: "U32" }, - { begin: "U64" } - ] + { begin: "U64" }, + ], }, { - scope: 'string', - begin: '"', end: '"' + scope: "string", + begin: '"', + end: '"', }, { className: "number", @@ -3546,17 +3559,17 @@ { begin: /(('(s?([bhodBHOD])?))[0-9xzXZa-fA-F_]+)/ }, { begin: - "[+-]?\\b(?:0[Bb][01](?:'?[01])*|0[Xx][0-9A-Fa-f](?:'?[0-9A-Fa-f])*|0(?:'?[0-7])*|[1-9](?:'?[0-9])*)s?" + "[+-]?\\b(?:0[Bb][01](?:'?[01])*|0[Xx][0-9A-Fa-f](?:'?[0-9A-Fa-f])*|0(?:'?[0-7])*|[1-9](?:'?[0-9])*)s?", }, - { begin: /\b[0-9][0-9_]*/, relevance: 0 } - ] - } - ] + { begin: /\b[0-9][0-9_]*/, relevance: 0 }, + ], + }, + ], }; }), [].slice .call(document.querySelectorAll("pre code.hljs[data-lang]")) - .forEach(function(e) { + .forEach(function (e) { I.highlightBlock(e); }); })(); diff --git a/backends/common_templates/adoc/csr.adoc.erb b/backends/common_templates/adoc/csr.adoc.erb index 01f208f84..5a082ac1e 100644 --- a/backends/common_templates/adoc/csr.adoc.erb +++ b/backends/common_templates/adoc/csr.adoc.erb @@ -8,25 +8,22 @@ == Attributes [%autowidth] |=== +h| Defining Extension a| <%= csr.defined_by_condition.to_asciidoc %> h| CSR Address | <%= "0x#{csr.address.to_s(16)}" %> <%- if csr.priv_mode == 'VS' -%> h| Virtual CSR Address | <%= "0x#{csr.virtual_address.to_s(16)}" %> <%- end -%> -<%- if csr.dynamic_length?(arch_def) -%> -h| Length | <%= csr.length_pretty(arch_def) %> -<%- else -%> -h| Length | <%= csr.length_pretty(arch_def) %>-bit -<%- end -%> +h| Length | <%= csr.length_pretty(cfg_arch) %> h| Privilege Mode | <%= csr.priv_mode %> |=== == Format -<%- unless csr.dynamic_length?(arch_def) || csr.fields.any? { |f| f.dynamic_location?(arch_def) } -%> +<%- unless csr.dynamic_length?(cfg_arch) || csr.fields.any? { |f| f.dynamic_location?(cfg_arch) } -%> <%# CSR has a known static length, so there is only one format to display -%> .<%= csr.name %> format [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, 64) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, 64) %> .... <%- else -%> <%# CSR has a dynamic length, or a field has a dynamic location, @@ -36,27 +33,27 @@ This CSR format changes dynamically. .<%= csr.name %> Format when <%= csr.length_cond32 %> [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, 32) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, 32) %> .... .<%= csr.name %> Format when <%= csr.length_cond64 %> [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, 64) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, 64) %> .... <%- end -%> == Field Summary -[%autowidth,float="center",align="center",cols="^,<,<,<",options="header",role="stretch"] +[%autowidth,separator=@,float="center",align="center",cols="^,<,<,<",options="header",role="stretch"] |=== -|Name | Location | Type | Reset Value +@Name @ Location @ Type @ Reset Value <%- csr.fields.each do |field| -%> -| xref:<%=csr.name%>-<%=field.name%>-def[`<%= field.name %>`] -| <%= field.location_pretty(arch_def) %> -| <%= field.type_pretty(arch_def.symtab) %> -| <%= field.reset_value_pretty(arch_def) %> +@ <%= link_to_csr_field(csr.name, field.name) %> +@ <%= field.location_pretty(cfg_arch) %> +@ <%= field.type_pretty(cfg_arch.symtab) %> +@ <%= field.reset_value_pretty(cfg_arch) %> <%- end -%> |=== @@ -75,16 +72,16 @@ This CSR has no fields. However, it must still exist (not cause an `Illegal Inst [example] **** Location:: -<%= field.location_pretty(arch_def) %> +<%= field.location_pretty(cfg_arch) %> Description:: <%= field.description %> Type:: -<%= field.type_pretty(arch_def.symtab) %> +<%= field.type_pretty(cfg_arch.symtab) %> Reset value:: -<%= field.reset_value_pretty(arch_def) %> +<%= field.reset_value_pretty(cfg_arch) %> **** @@ -118,6 +115,6 @@ This CSR may return a value that is different from what is stored in hardware. [source,idl,subs="specialchars,macros"] ---- -<%= csr.sw_read_ast(arch_def.symtab).gen_adoc %> +<%= csr.sw_read_ast(cfg_arch.symtab).gen_adoc %> ---- <%- end -%> diff --git a/backends/common_templates/adoc/inst.adoc.erb b/backends/common_templates/adoc/inst.adoc.erb index 8d9422e47..85a704477 100644 --- a/backends/common_templates/adoc/inst.adoc.erb +++ b/backends/common_templates/adoc/inst.adoc.erb @@ -75,10 +75,10 @@ Operation:: <%- unless inst.data["operation()"].nil? -%> [source,idl,subs="specialchars,macros"] ---- -<%= inst.operation_ast(arch_def.symtab).gen_adoc %> +<%= inst.operation_ast(cfg_arch.symtab).gen_adoc %> ---- <%- end -%> Included in:: -<%= inst.defined_by.to_asciidoc %> +<%= inst.defined_by_condition.to_asciidoc %> diff --git a/backends/ext_pdf_doc/tasks.rake b/backends/ext_pdf_doc/tasks.rake index bbcb92c3a..fad64d6fa 100644 --- a/backends/ext_pdf_doc/tasks.rake +++ b/backends/ext_pdf_doc/tasks.rake @@ -120,10 +120,7 @@ rule %r{#{$root}/gen/ext_pdf_doc/.*/adoc/.*_extension\.adoc} => proc { |tname| end raise "Can't find extension '#{ext_name}'" if arch_yaml_paths.empty? - stamp = config_name == "_" ? "#{$root}/.stamps/arch-gen-_64.stamp" : "#{$root}/.stamps/arch-gen-#{config_name}.stamp" - [ - stamp, (EXT_PDF_DOC_DIR / "templates" / "ext_pdf.adoc.erb").to_s, arch_yaml_paths, __FILE__ @@ -131,12 +128,7 @@ rule %r{#{$root}/gen/ext_pdf_doc/.*/adoc/.*_extension\.adoc} => proc { |tname| } do |t| config_name = Pathname.new(t.name).relative_path_from("#{$root}/gen/ext_pdf_doc").to_s.split("/")[0] - arch_def = - if config_name == "_" - arch_def_for("_64") - else - arch_def_for(config_name) - end + cfg_arch = cfg_arch_for("_") ext_name = Pathname.new(t.name).basename(".adoc").to_s.split("_")[0..-2].join("_") @@ -144,21 +136,21 @@ rule %r{#{$root}/gen/ext_pdf_doc/.*/adoc/.*_extension\.adoc} => proc { |tname| erb = ERB.new(template_path.read, trim_mode: "-") erb.filename = template_path.to_s - ext = arch_def.extension(ext_name) + ext = cfg_arch.extension(ext_name) version_strs = ENV["VERSION"].split(",") versions = if version_strs.include?("all") ext.versions else vs = ext.versions.select do |ext_ver| - version_strs.include?(ext_ver.version.to_s) + version_strs.include?(ext_ver.version_spec.to_s) end vs << ext.max_version if version_strs.include?("latest") vs.uniq end max_version = versions.max { |a, b| a.version <=> b.version } FileUtils.mkdir_p File.dirname(t.name) - File.write t.name, AsciidocUtils.resolve_links(arch_def.find_replace_links(erb.result(binding))) + File.write t.name, AsciidocUtils.resolve_links(cfg_arch.find_replace_links(erb.result(binding))) end namespace :gen do diff --git a/backends/ext_pdf_doc/templates/ext_pdf.adoc.erb b/backends/ext_pdf_doc/templates/ext_pdf.adoc.erb index faf81fae8..2d214931f 100644 --- a/backends/ext_pdf_doc/templates/ext_pdf.adoc.erb +++ b/backends/ext_pdf_doc/templates/ext_pdf.adoc.erb @@ -1,7 +1,7 @@ [[header]] :description: <%= ext.long_name %> (<%= ext.name %>) :revdate: <%= max_version.ratification_date.nil? ? Date.today : max_version.ratification_date %> -:revnumber: <%= max_version.version %> +:revnumber: <%= max_version.version_spec %> :revmark: <%= case max_version.state when "ratified" @@ -129,7 +129,7 @@ Copyright <%= max_version.ratification_date.nil? ? Date.today.year : max_version == Acknowledgements <%- versions.each do |version| -%> -Contributors to version <%= version.version %> of the specification (in alphabetical order) include: + +Contributors to version <%= version.version_spec %> of the specification (in alphabetical order) include: + <%- unless version.contributors.empty? -%> <%- version.contributors.sort { |a, b| a.name.split(" ").last <=> b.name.split(" ").last }.each do |c| -%> @@ -146,16 +146,16 @@ improved this specification through their comments and questions. == Versions <%- if versions.size > 1 -%> -This specification documents versions <%= versions.map { |v| v.version }.join(', ') %> of <%= ext.name %>: +This specification documents versions <%= versions.map { |v| v.version_spec.to_s }.join(', ') %> of <%= ext.name %>: <%- else -%> -This specification documents version <%= max_version.version %> of <%= ext.name %>. +This specification documents version <%= max_version.version_spec %> of <%= ext.name %>. <%- end -%> === Version History <%- ext.versions.each do |version| -%> -- -Version:: <%= version.version %> +Version:: <%= version.version_spec %> State:: <%= version.state %> <%- unless version.ratification_date.nil? -%> Ratification Date:: <%= version.ratification_date %> @@ -173,10 +173,10 @@ Changes:: <%- end -%> <%- unless version.implications.empty? -%> Implies:: -* <%= version.implications.map { |i| "#{i.name} (#{i.version})" }.join("\n* ") %> -<%- unless version.requirements.empty? -%> +* <%= version.implications.map { |i| "#{i.name} (#{i.version_spec})" }.join("\n* ") %> +<%- unless version.requirement_condition.empty? -%> Requires:: -<%= version.requirements.to_asciidoc %> +<%= version.requirement_condition.to_asciidoc %> <%- end -%> <%- end -%> -- @@ -200,7 +200,7 @@ Requires:: <%- end -%> <%- implications.each do |sub_ext| -%> -==== <%= sub_ext.name %> (<%= sub_ext.version %>) +==== <%= sub_ext.name %> (<%= sub_ext.version_spec %>) <%- if versions.size > 1 -%> <%= sub_ext.name %> (<%= sub_ext.version %>) is implied by @@ -208,12 +208,12 @@ version <%= versions.select { |v| v.implications.include?(sub_ext)}.map(&:versio of <%= ext.name %>. <%- end -%> -<%= arch_def.extension(sub_ext.name).description %> +<%= cfg_arch.extension(sub_ext.name).description %> -<%- unless sub_ext.requirements.empty? -%> +<%- unless sub_ext.requirement_condition.empty? -%> <%= sub_ext.name %> requires: -<%= sub_ext.requirements.to_asciidoc %> +<%= sub_ext.requirement_condition.to_asciidoc %> <%- end -%> @@ -287,7 +287,7 @@ The following <%= ext.csrs.size %> are added by this extension. <%- ext.csrs.each do |csr| -%> <<< :leveloffset: +2 -<%= partial "adoc/csr.adoc.erb", { csr: csr, arch_def: arch_def } %> +<%= partial "adoc/csr.adoc.erb", { csr: csr, cfg_arch: cfg_arch } %> :leveloffset: -2 <%- end -%> @@ -300,7 +300,7 @@ The following <%= ext.csrs.size %> are added by this extension. <%- ext.instructions.each do |i| -%> :leveloffset: +2 -<%= partial "adoc/inst.adoc.erb", { inst: i, arch_def: arch_def } %> +<%= partial "adoc/inst.adoc.erb", { inst: i, cfg_arch: cfg_arch } %> :leveloffset: -2 <<< @@ -310,7 +310,7 @@ The following <%= ext.csrs.size %> are added by this extension. <<< == IDL Functions -<%- ext.reachable_functions(arch_def.symtab).sort { |a,b| a.name <=> b.name }.each do |f| -%> +<%- ext.reachable_functions(cfg_arch.symtab).sort { |a,b| a.name <=> b.name }.each do |f| -%> [#<%= f.name %>-func-def] === <%= f.name %><%- if f.builtin? -%> (builtin)<%- end -%> diff --git a/backends/indexer/index-unifieddb.js b/backends/indexer/index-unifieddb.js index cdc1404d6..ec3f0be40 100755 --- a/backends/indexer/index-unifieddb.js +++ b/backends/indexer/index-unifieddb.js @@ -1,9 +1,9 @@ #!/usr/bin/env node -'use strict'; +"use strict"; -const process = require('process'); -const path = require('path'); -const fs = require('fs'); +const process = require("process"); +const path = require("path"); +const fs = require("fs"); const { readdir, stat } = fs.promises; @@ -16,8 +16,8 @@ const rec = async (branch, root) => { const fileExt = path.extname(el); const baseName = path.basename(el, fileExt); if (isFile) { - if (['.yaml', '.json'].includes(fileExt)) { - node[baseName] = {$ref: path.join(...branch, el)}; + if ([".yaml", ".json"].includes(fileExt)) { + node[baseName] = { $ref: path.join(...branch, el) }; } } else { node[el] = await rec([...branch, el], root); @@ -29,10 +29,12 @@ const rec = async (branch, root) => { const main = async () => { const [, , root] = process.argv; if (root === undefined) { - console.error('usage: ./index-unifieddb.js '); + console.error( + "usage: ./index-unifieddb.js ", + ); return; } - const rootPath = path.resolve('.', root); + const rootPath = path.resolve(".", root); const tree = await rec([], rootPath); console.log(JSON.stringify(tree, null, 2)); }; diff --git a/backends/manual/tasks.rake b/backends/manual/tasks.rake index 99bd2101b..d7573071e 100644 --- a/backends/manual/tasks.rake +++ b/backends/manual/tasks.rake @@ -1,26 +1,33 @@ # frozen_string_literal: true -require_relative "#{$lib}/arch_def" +require "digest" + +require_relative "#{$lib}/cfg_arch" $root = Pathname.new(__FILE__).dirname.dirname.realpath if $root.nil? MANUAL_GEN_DIR = $root / "gen" / "manual" -def versions_from_env(manual) +def versions_from_env(manual_name) versions = ENV["VERSIONS"].split(",") output_hash = nil if versions.include?("all") raise ArgumentError, "'all' was given as a version, so nothing else should be" unless versions.length == 1 - versions = manual.versions + versions = [] + version_fns = Dir.glob("#{$root}/arch/manual_version/**/*.yaml") + raise "Cannot find version files" if version_fns.empty? + + version_fns.each do |manual_version_fn| + manual_version_obj = YAML.load_file(manual_version_fn, permitted_classes: [Date]) + versions << manual_version_obj["name"] if manual_version_obj["manual"]["$ref"] == "manual/#{manual_name}.yaml#" + end output_hash = "all" else - versions = versions.map { |vname| manual.versions.find { |v| v.name = vname } } - if versions.any?(&:nil?) - idx = versions.index(&:nil?) - raise "No manual version '#{ENV['VERSIONS'].split(',')[idx]}' for '#{args[:manual_name]}'" + versions.each do |version| + raise "No manual version #{version}" if Dir.glob("#{$root}/arch/manual_version/**/#{version}.yaml").empty? end - output_hash = versions.size == 1 ? versions[0] : versions.hash + output_hash = versions.size == 1 ? versions[0] : Digest::SHA2.hexdigest(versions.join("")) end [versions, output_hash] @@ -33,7 +40,7 @@ directory MANUAL_GEN_DIR / "html" file MANUAL_GEN_DIR / "antora" / "antora.yml" => (MANUAL_GEN_DIR / "antora").to_s do |t| File.write t.name, <<~ANTORA name: riscv_manual - version: #{arch_def.manual_version?} + version: #{cfg_arch.manual_version?} nav: - modules/nav.adoc title: RISC-V ISA Manual @@ -43,7 +50,10 @@ end # Rule to create a chapter page in antora hierarchy rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/modules/chapters/pages/.*\.adoc} do |t| parts = t.name.sub("#{MANUAL_GEN_DIR}/", "").split("/") - manual_version = arch_def_for("_64").manual(parts[0]).version(parts[1]) + manual_name = parts[0] + version_name = parts[1] + manual = cfg_arch_for("_").manual(parts[0]) + manual_version = manual.version(parts[1]) chapter_name = File.basename(t.name, ".adoc") volume = manual_version.volumes.find { |v| !v.chapter(chapter_name).nil? } @@ -52,7 +62,7 @@ rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/modules/chapters/pages/.*\.adoc} do |t| chapter = volume.chapter(chapter_name) FileUtils.mkdir_p File.dirname(t.name) - FileUtils.ln_s chapter.path, t.name + FileUtils.ln_s chapter.fullpath, t.name end # Rule to create antora.yml for a manual version @@ -61,20 +71,22 @@ rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/antora.yml} => proc { |tname| manual_name = parts[0] version_name = parts[1] - manual_yaml_path = $root / "arch" / "manual" / manual_name / "#{manual_name}.yaml" - contents_path = $root / "arch" / "manual" / manual_name / version_name / "contents.yaml" + manual_yaml_path = $root / "arch" / "manual" / "#{manual_name}.yaml" + version_paths = Dir.glob($root / "arch" / "manual_version" / "**" / "#{version_name}.yaml") + raise "Cannot find version" unless version_paths.size == 1 + + version_yaml_path = version_paths[0] raise "Cannot find #{manual_yaml_path}" unless manual_yaml_path.exist? - raise "Cannot find #{contents_path}" unless contents_path.exist? [ __FILE__, manual_yaml_path.to_s, - contents_path.to_s + version_yaml_path.to_s ] } do |t| parts = t.name.sub("#{MANUAL_GEN_DIR}/", "").split("/") - manual_version = arch_def_for("_64").manual(parts[0])&.version(parts[1]) + manual_version = cfg_arch_for("_").manual(parts[0])&.version(parts[1]) raise "Can't find any manual version for '#{parts[0]}' '#{parts[1]}'" if manual_version.nil? @@ -94,22 +106,24 @@ rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/nav.adoc} => proc { |tname| manual_name = parts[0] version_name = parts[1] - manual_yaml_path = $root / "arch" / "manual" / manual_name / "#{manual_name}.yaml" - contents_path = $root / "arch" / "manual" / manual_name / version_name / "contents.yaml" + manual_yaml_path = $root / "arch" / "manual" / "#{manual_name}.yaml" + version_paths = Dir.glob($root / "arch" / "manual_version" / "**" / "#{version_name}.yaml") + raise "Cannot find version" unless version_paths.size == 1 + + version_yaml_path = version_paths[0] nav_template_path = $root / "backends" / "manual" / "templates" / "#{manual_name}_nav.adoc.erb" raise "Cannot find #{manual_yaml_path}" unless manual_yaml_path.exist? - raise "Cannot find #{contents_path}" unless contents_path.exist? FileList[ __FILE__, manual_yaml_path.to_s, - contents_path.to_s, + version_yaml_path.to_s, nav_template_path.to_s ] } do |t| parts = t.name.sub("#{MANUAL_GEN_DIR}/", "").split("/") - manual_version = arch_def_for("_64").manual(parts[0])&.version(parts[1]) + manual_version = cfg_arch_for("_").manual(parts[0])&.version(parts[1]) raise "Can't find any manual version for '#{parts[0]}' '#{parts[1]}'" if manual_version.nil? @@ -118,9 +132,10 @@ rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/nav.adoc} => proc { |tname| raise "There is no navigation file for manual '#{parts[0]}' at '#{nav_template_path}'" end + raise "no cfg_arch" if manual_version.cfg_arch.nil? + erb = ERB.new(nav_template_path.read, trim_mode: "-") erb.filename = nav_template_path.to_s - puts erb.encoding FileUtils.mkdir_p File.dirname(t.name) File.write t.name, erb.result(binding) @@ -132,23 +147,26 @@ rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/modules/ROOT/pages/index.adoc} => proc { manual_name = parts[0] version_name = parts[1] - manual_yaml_path = $root / "arch" / "manual" / manual_name / "#{manual_name}.yaml" - contents_path = $root / "arch" / "manual" / manual_name / version_name / "contents.yaml" + manual_yaml_path = $root / "arch" / "manual" / "#{manual_name}.yaml" + version_paths = Dir.glob($root / "arch" / "manual_version" / "**" / "#{version_name}.yaml") + raise "Cannot find version" unless version_paths.size == 1 + + version_yaml_path = version_paths[0] + version_index_template_path = $root / "backends" / "manual" / "templates" / "#{manual_name}_version_index.adoc.erb" raise "Cannot find #{manual_yaml_path}" unless manual_yaml_path.exist? - raise "Cannot find #{contents_path}" unless contents_path.exist? raise "Cannot find #{version_index_template_path}" unless version_index_template_path.exist? FileList[ __FILE__, manual_yaml_path.to_s, - contents_path.to_s, + version_yaml_path.to_s, version_index_template_path.to_s ] } do |t| parts = t.name.sub("#{MANUAL_GEN_DIR}/", "").split("/") - manual_version = arch_def_for("_64").manual(parts[0])&.version(parts[1]) + manual_version = cfg_arch_for("_").manual(parts[0])&.version(parts[1]) raise "Can't find any manual version for '#{parts[0]}' '#{parts[1]}'" if manual_version.nil? @@ -168,13 +186,12 @@ end # rule to create instruction appendix page rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/modules/insts/pages/.*.adoc} => [ __FILE__, - "#{$root}/.stamps/arch-gen-_64.stamp", ($root / "backends" / "manual" / "templates" / "instruction.adoc.erb").to_s ] do |t| inst_name = File.basename(t.name, ".adoc") - arch_def = arch_def_for("_64") - inst = arch_def.instruction(inst_name) + cfg_arch = cfg_arch_for("_") + inst = cfg_arch.instruction(inst_name) raise "Can't find instruction '#{inst_name}'" if inst.nil? inst_template_path = $root / "backends" / "manual" / "templates" / "instruction.adoc.erb" @@ -182,44 +199,42 @@ rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/modules/insts/pages/.*.adoc} => [ erb.filename = inst_template_path.to_s FileUtils.mkdir_p File.dirname(t.name) - File.write t.name, AntoraUtils.resolve_links(arch_def.find_replace_links(erb.result(binding))) + File.write t.name, AntoraUtils.resolve_links(cfg_arch.find_replace_links(erb.result(binding))) end # rule to create csr appendix page -rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/modules/csrs/pages/.*.adoc} => [ +rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/modules/csrs/pages/.*\.adoc} => [ __FILE__, - "#{$root}/.stamps/arch-gen-_64.stamp", - "#{$root}/.stamps/arch-gen-_32.stamp", - ($root / "backends" / "manual" / "templates" / "csr.adoc.erb").to_s + "gen:arch", + ($root / "backends" / "common_templates" / "adoc" / "csr.adoc.erb").to_s ] do |t| csr_name = File.basename(t.name, ".adoc") - arch_def = arch_def_for("_64") - arch_def_32 = arch_def_for("_32") + cfg_arch = cfg_arch_for("_") + # cfg_arch_32 = cfg_arch_for("_32") - csr = arch_def.csr(csr_name) + csr = cfg_arch.csr(csr_name) raise "Can't find csr '#{csr_name}'" if csr.nil? - csr_32 = arch_def_32.csr(csr_name) + # csr_32 = cfg_arch_32.csr(csr_name) - csr_template_path = $root / "backends" / "manual" / "templates" / "csr.adoc.erb" + csr_template_path = $root / "backends" / "common_templates" / "adoc" / "csr.adoc.erb" erb = ERB.new(csr_template_path.read, trim_mode: "-") erb.filename = csr_template_path.to_s FileUtils.mkdir_p File.dirname(t.name) - File.write t.name, AntoraUtils.resolve_links(arch_def.find_replace_links(erb.result(binding))) + File.write t.name, AntoraUtils.resolve_links(cfg_arch.find_replace_links(erb.result(binding))) end # rule to create ext appendix page rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/modules/exts/pages/.*.adoc} => [ __FILE__, - "#{$root}/.stamps/arch-gen-_64.stamp", ($root / "backends" / "manual" / "templates" / "ext.adoc.erb").to_s ] do |t| ext_name = File.basename(t.name, ".adoc") - arch_def = arch_def_for("_64") - ext = arch_def.extension(ext_name) + cfg_arch = cfg_arch_for("_") + ext = cfg_arch.extension(ext_name) raise "Can't find extension '#{ext_name}'" if ext.nil? ext_template_path = $root / "backends" / "manual" / "templates" / "ext.adoc.erb" @@ -227,41 +242,39 @@ rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/modules/exts/pages/.*.adoc} => [ erb.filename = ext_template_path.to_s FileUtils.mkdir_p File.dirname(t.name) - File.write t.name, AntoraUtils.resolve_links(arch_def.find_replace_links(erb.result(binding))) + File.write t.name, AntoraUtils.resolve_links(cfg_arch.find_replace_links(erb.result(binding))) end # rule to create IDL function appendix page rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/modules/funcs/pages/funcs.adoc} => [ __FILE__, - "#{$root}/.stamps/arch-gen-_64.stamp", ($root / "backends" / "manual" / "templates" / "func.adoc.erb").to_s ] do |t| - arch_def = arch_def_for("_64") + cfg_arch = cfg_arch_for("_") funcs_template_path = $root / "backends" / "manual" / "templates" / "func.adoc.erb" erb = ERB.new(funcs_template_path.read, trim_mode: "-") erb.filename = funcs_template_path.to_s FileUtils.mkdir_p File.dirname(t.name) - File.write t.name, AntoraUtils.resolve_links(arch_def.find_replace_links(erb.result(binding))) + File.write t.name, AntoraUtils.resolve_links(cfg_arch.find_replace_links(erb.result(binding))) end # rule to create IDL function appendix page rule %r{#{MANUAL_GEN_DIR}/.*/.*/antora/modules/params/pages/param_list.adoc} => [ __FILE__, - "#{$root}/.stamps/arch-gen-_64.stamp", ($root / "backends" / "manual" / "templates" / "param_list.adoc.erb").to_s ] do |t| - arch_def = arch_def_for("_64") + cfg_arch = cfg_arch_for("_") parts = t.name.sub("#{MANUAL_GEN_DIR}/", "").split("/") - manual_version = arch_def.manual(parts[0])&.version(parts[1]) + manual_version = cfg_arch.manual(parts[0])&.version(parts[1]) param_list_template_path = $root / "backends" / "manual" / "templates" / "param_list.adoc.erb" erb = ERB.new(param_list_template_path.read, trim_mode: "-") erb.filename = param_list_template_path.to_s FileUtils.mkdir_p File.dirname(t.name) - File.write t.name, AntoraUtils.resolve_links(arch_def.find_replace_links(erb.result(binding))) + File.write t.name, AntoraUtils.resolve_links(cfg_arch.find_replace_links(erb.result(binding))) end rule %r{#{MANUAL_GEN_DIR}/.*/top/.*/antora/landing/antora.yml} => [ @@ -270,8 +283,8 @@ rule %r{#{MANUAL_GEN_DIR}/.*/top/.*/antora/landing/antora.yml} => [ parts = t.name.sub("#{MANUAL_GEN_DIR}/", "").split("/") manual_name = parts[0] - arch_def = arch_def_for("_64") - manual = arch_def.manual(manual_name) + cfg_arch = cfg_arch_for("_") + manual = cfg_arch.manual(manual_name) raise "Can't find any manual version for '#{manual_name}'" if manual.nil? FileUtils.mkdir_p File.basename(t.name) @@ -285,27 +298,26 @@ end rule %r{#{MANUAL_GEN_DIR}/.*/top/.*/antora/landing/modules/ROOT/pages/index.adoc} => proc { |tname| parts = tname.sub("#{MANUAL_GEN_DIR}/", "").split("/") manual_name = parts[0] + versions, _ = versions_from_env(ENV["MANUAL_NAME"]) + version_files = Dir.glob($root / "arch" / "manual_version" / "**" / "*.yaml").select { |f| versions.include?(File.basename(f, ".yaml"))} FileList[ __FILE__, - ($root / "arch" / "manual" / manual_name / "#{manual_name}.yaml").to_s, + ($root / "arch" / "manual" / "#{manual_name}.yaml").to_s, ($root / "backends" / "manual" / "templates" / "index.adoc.erb").to_s, - ($root / "arch" / "manual" / manual_name / "**" / "contents.yaml").to_s - ] + ] + version_files } do |t| parts = t.name.sub("#{MANUAL_GEN_DIR}/", "").split("/") manual_name = parts[0] - arch_def = arch_def_for("_64") - manual = arch_def.manual(manual_name) - raise "Can't find any manual version for '#{manual_name}'" if manual.nil? - - versions, output_hash = versions_from_env(manual) + versions, output_hash = versions_from_env(manual_name) raise "unexpected mismatch" unless output_hash == parts[2] landing_template_path = $root / "backends" / "manual" / "templates" / "index.adoc.erb" erb = ERB.new(landing_template_path.read, trim_mode: "-") erb.filename = landing_template_path.to_s + manual = cfg_arch_for("_").manual(manual_name) + FileUtils.mkdir_p File.dirname(t.name) File.write t.name, erb.result(binding) end @@ -313,27 +325,27 @@ end rule %r{#{MANUAL_GEN_DIR}/.*/top/.*/antora/playbook/playbook.yml} => proc { |tname| parts = tname.sub("#{MANUAL_GEN_DIR}/", "").split("/") manual_name = parts[0] + versions, _ = versions_from_env(ENV["MANUAL_NAME"]) + version_files = Dir.glob($root / "arch" / "manual_version" / "**" / "*.yaml").select { |f| versions.include?(File.basename(f, ".yaml"))} FileList[ __FILE__, - ($root / "arch" / "manual" / manual_name / "#{manual_name}.yaml").to_s, + ($root / "arch" / "manual" / "#{manual_name}.yaml").to_s, ($root / "backends" / "manual" / "templates" / "playbook.yml.erb").to_s, ($root / "arch" / "manual" / manual_name / "**" / "contents.yaml").to_s - ] + ] + version_files } do |t| parts = t.name.sub("#{MANUAL_GEN_DIR}/", "").split("/") manual_name = parts[0] - arch_def = arch_def_for("_64") - manual = arch_def.manual(manual_name) - raise "Can't find any manual version for '#{manual_name}'" if manual.nil? - - versions, output_hash = versions_from_env(manual) + versions, output_hash = versions_from_env(manual_name) raise "unexpected mismatch" unless output_hash == parts[2] playbook_template_path = $root / "backends" / "manual" / "templates" / "playbook.yml.erb" erb = ERB.new(playbook_template_path.read, trim_mode: "-") erb.filename = playbook_template_path.to_s + manual = cfg_arch_for("_").manual(manual_name) + FileUtils.mkdir_p File.dirname(t.name) File.write t.name, erb.result(binding) end @@ -342,19 +354,25 @@ file $root / "ext" / "riscv-isa-manual" / "README.md" do sh "git submodule update --init ext/riscv-isa-manual 2>&1" end -Dir.glob($root / "arch" / "manual" / "**" / "contents.yaml") do |content_fn| - file "#{File.dirname(content_fn)}/riscv-isa-manual/README.md" => ($root / "ext" / "riscv-isa-manual" / "README.md").to_s do |t| - content_obj = YAML.load_file(content_fn) - git_dir = `git rev-parse --git-dir`.strip - cmd = [ - "git", - "--git-dir=#{git_dir}/modules/ext/riscv-isa-manual", - "worktree add", - File.dirname(t.name), - content_obj["isa_manual_tree"], - "2>&1" - ].join(" ") - sh cmd +rule %r{#{MANUAL_GEN_DIR}/[^/]+/[^/]+/riscv-isa-manual/README.md} => ["#{$root}/ext/riscv-isa-manual/README.md"] do |t| + parts = t.name.sub("#{MANUAL_GEN_DIR}/","").split("/") + manual_version_name = parts[1] + + version_paths = Dir.glob("#{$root}/arch/manual_version/**/#{manual_version_name}.yaml") + raise "No manual version named '#{manual_version_name}' found" unless version_paths.size == 1 + + version_path = version_paths[0] + + version_obj = YAML.load_file(version_path, permitted_classes: [Date]) + raise "Not an isa manual version" unless version_obj["uses_isa_manual"] == true + + FileUtils.mkdir_p File.dirname(t.name) + tree = version_obj["isa_manual_tree"] + Dir.chdir($root / "ext" / "riscv-isa-manual") do + Tempfile.create("isa-manual") do |tmpfile| + sh "git archive --format=tar -o #{tmpfile.path} #{tree}" + sh "tar xf #{tmpfile.path} -C #{File.dirname(t.name)}" + end end end @@ -377,29 +395,32 @@ namespace :gen do A static HTML website will be written into gen/manual/MANUAL_NAME//html DESC desc html_manual_desc - task :html_manual => "#{$root}/.stamps/arch-gen-_64.stamp" do + task :html_manual do raise ArgumentError, "Missing required environment variable MANUAL_NAME\n\n#{html_manual_desc}" if ENV["MANUAL_NAME"].nil? raise ArgumentError, "Missing required environment variable VERSIONS\n\n#{html_manual_desc}" if ENV["VERSIONS"].nil? - arch_def = arch_def_for("_64") - manual = arch_def.manuals.find { |m| m.name == ENV["MANUAL_NAME"] } - raise "No manual '#{ENV['MANUAL_NAME']}'" if manual.nil? + versions, output_hash = versions_from_env(ENV["MANUAL_NAME"]) + cfg_arch = cfg_arch_for("_") + + manual = cfg_arch.manual(ENV["MANUAL_NAME"]) + raise "No manual named '#{ENV['MANUAL_NAME']}" if manual.nil? - versions, output_hash = versions_from_env(manual) # check out the correct version of riscv-isa-manual, if needed versions.each do |version| - next unless version.uses_isa_manual? + version_obj = cfg_arch.manual_version(version) - unless ($root / "arch" / "manual" / ENV["MANUAL_NAME"] / version.name / "riscv-isa-manual").exist? - Rake::Task[$root / "arch" / "manual" / ENV["MANUAL_NAME"] / version.name / "riscv-isa-manual" / "README.md"].invoke + manual.repo_path = MANUAL_GEN_DIR / ENV["MANUAL_NAME"] / version / "riscv-isa-manual" + + if version_obj.uses_isa_manual? == true \ + && !(MANUAL_GEN_DIR / ENV["MANUAL_NAME"] / version_obj.name / "riscv-isa-manual").exist? + Rake::Task[MANUAL_GEN_DIR / ENV["MANUAL_NAME"] / version_obj.name / "riscv-isa-manual" / "README.md"].invoke end - end - # create chapter pages in antora - versions.each do |version| - antora_path = MANUAL_GEN_DIR / ENV["MANUAL_NAME"] / version.name / "antora" - version.volumes.each do |volume| + # create chapter pages in antora + + antora_path = MANUAL_GEN_DIR / ENV["MANUAL_NAME"] / version_obj.name / "antora" + version_obj.volumes.each do |volume| volume.chapters.each do |chapter| Rake::Task[antora_path / "modules" / "chapters" / "pages" / "#{chapter.name}.adoc"].invoke end @@ -408,13 +429,14 @@ namespace :gen do Rake::Task[antora_path / "modules" / "ROOT" / "pages" / "index.adoc"].invoke Rake::Task[antora_path / "antora.yml"].invoke Rake::Task[antora_path / "nav.adoc"].invoke - version.instructions.each do |inst| - Rake::Task[antora_path / "modules" / "insts" / "pages" / "#{inst.name}.adoc"].invoke - end - version.csrs.each do |csr| + + version_obj.csrs.each do |csr| Rake::Task[antora_path / "modules" / "csrs" / "pages" / "#{csr.name}.adoc"].invoke end - version.extensions.each do |ext| + version_obj.instructions.each do |inst| + Rake::Task[antora_path / "modules" / "insts" / "pages" / "#{inst.name}.adoc"].invoke + end + version_obj.extensions.each do |ext| Rake::Task[antora_path / "modules" / "exts" / "pages" / "#{ext.name}.adoc"].invoke end Rake::Task[antora_path / "modules" / "params" / "pages" / "param_list.adoc"].invoke @@ -445,13 +467,14 @@ namespace :gen do end namespace :serve do - task :html_manual do |t| + desc "Serve an HTML site for one or more versions of the manual (gen:html_manual for options)" + task :html_manual do Rake::Task["gen:html_manual"].invoke port = ENV.key?("PORT") ? ENV["PORT"] : 8000 - arch_def = arch_def_for("_64") - manual = arch_def.manuals.find { |m| m.name == ENV["MANUAL_NAME"] } + cfg_arch = cfg_arch_for("_") + manual = cfg_arch.manuals.find { |m| m.name == ENV["MANUAL_NAME"] } raise "No manual '#{ENV['MANUAL_NAME']}'" if manual.nil? _, output_hash = versions_from_env(manual) diff --git a/backends/manual/templates/csr.adoc.erb b/backends/manual/templates/csr.adoc.erb index b5e59f47f..e4e241cf5 100644 --- a/backends/manual/templates/csr.adoc.erb +++ b/backends/manual/templates/csr.adoc.erb @@ -5,43 +5,43 @@ *<%= csr.long_name %>* -<%= arch_def.render_erb(csr.description, "#{csr.name}.description") %> +<%= cfg_arch.render_erb(csr.description, "#{csr.name}.description") %> == Attributes [%autowidth] |=== -h| Defining Extension a| <%= csr.defined_by.to_asciidoc %> +h| Defining Extension a| <%= csr.defined_by_condition.to_asciidoc %> h| CSR Address | <%= "0x#{csr.address.to_s(16)}" %> <%- if csr.priv_mode == 'VS' -%> h| Virtual CSR Address | <%= "0x#{csr.virtual_address.to_s(16)}" %> <%- end -%> -<%- if csr.dynamic_length?(arch_def) || csr.data["length"] == "MXLEN" -%> +<%- if csr.dynamic_length?(cfg_arch) || csr.data["length"] == "MXLEN" -%> h| Length a| [when,"<%= csr.length_cond32 %>"] -- -<%= csr_32.length_pretty(arch_def_32, 32) %> +<%= csr_32.length_pretty(cfg_arch_32, 32) %> -- [when,"<%= csr.length_cond64 %>"] -- -<%= csr.length_pretty(arch_def, 64) %> +<%= csr.length_pretty(cfg_arch, 64) %> -- <%- else -%> -h| Length | <%= csr.length_pretty(arch_def) %> +h| Length | <%= csr.length_pretty(cfg_arch) %> <%- end -%> h| Privilege Mode | <%= csr.priv_mode %> |=== == Format -<%- unless csr.dynamic_length?(arch_def) || csr.fields.any? { |f| f.dynamic_location?(arch_def) } || csr.data["length"] == "MXLEN" -%> +<%- unless csr.dynamic_length?(cfg_arch) || csr.fields.any? { |f| f.dynamic_location?(cfg_arch) } || csr.data["length"] == "MXLEN" -%> <%# CSR has a known static length, so there is only one format to display -%> .<%= csr.name %> format [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, 64) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, 64) %> .... <%- else -%> <%# CSR has a dynamic length, or a field has a dynamic location, @@ -52,7 +52,7 @@ This CSR format changes dynamically. -- [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr_32.wavedrom_desc(arch_def_32, 32) %> +<%= JSON.dump csr_32.wavedrom_desc(cfg_arch_32, 32) %> .... -- @@ -60,7 +60,7 @@ This CSR format changes dynamically. -- [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, 64) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, 64) %> .... -- <%- end -%> @@ -75,31 +75,31 @@ This CSR format changes dynamically. <%- csr.fields.each do |field| -%> @ xref:<%=csr.name%>-<%=field.name%>-def[`<%= field.name %>`] a@ -<%- if field.dynamic_location?(arch_def) -%> +<%- if field.dynamic_location?(cfg_arch) -%> [when,"<%= field.location_cond32 %>"] -- -<%= field.location_pretty(arch_def, 32) %> +<%= field.location_pretty(cfg_arch, 32) %> -- [when,"<%= field.location_cond64 %>"] -- -<%= field.location_pretty(arch_def, 64) %> +<%= field.location_pretty(cfg_arch, 64) %> -- <%- else -%> -<%= field.location_pretty(arch_def) %> +<%= field.location_pretty(cfg_arch) %> <%- end -%> a@ -- -<%= field.type_pretty(arch_def.symtab) %> +<%= field.type_pretty(cfg_arch.symtab) %> -- a@ -- -<%= field.reset_value_pretty(arch_def) %> +<%= field.reset_value_pretty(cfg_arch) %> -- <%- end -%> @@ -121,16 +121,16 @@ IMPORTANT: <%= field.name %> is only defined in <%= field.base32_only? ? "RV32" **** Location:: -<%= field.location_pretty(arch_def) %> +<%= field.location_pretty(cfg_arch) %> Description:: -<%= arch_def.render_erb(field.description, "#{csr.name}.#{field.name}.description").gsub("\n", " +\n") %> +<%= cfg_arch.render_erb(field.description, "#{csr.name}.#{field.name}.description").gsub("\n", " +\n") %> Type:: -<%= field.type_pretty(arch_def.symtab) %> +<%= field.type_pretty(cfg_arch.symtab) %> Reset value:: -<%= field.reset_value_pretty(arch_def) %> +<%= field.reset_value_pretty(cfg_arch) %> **** @@ -164,6 +164,6 @@ This CSR may return a value that is different from what is stored in hardware. [source,idl,subs="specialchars,macros"] ---- -<%= csr.sw_read_ast(arch_def.symtab).gen_adoc %> +<%= csr.sw_read_ast(cfg_arch.symtab).gen_adoc %> ---- <%- end -%> diff --git a/backends/manual/templates/ext.adoc.erb b/backends/manual/templates/ext.adoc.erb index a3e5fdfa1..71a46f857 100644 --- a/backends/manual/templates/ext.adoc.erb +++ b/backends/manual/templates/ext.adoc.erb @@ -5,7 +5,7 @@ == Versions <%- ext.versions.each do |v| -%> -<%= v.version %>:: +<%= v.canonical_version %>:: State::: <%= v.state %> <%- if v.state == "ratified" -%> @@ -36,7 +36,7 @@ <%= ext.description %> -<%- insts = arch_def.instructions.select { |i| ext.versions.any? { |v| i.defined_by?(ext.name, v.version) } } -%> +<%- insts = cfg_arch.instructions.select { |i| ext.versions.any? { |v| i.defined_by?(v) } } -%> <%- unless insts.empty? -%> == Instructions diff --git a/backends/manual/templates/func.adoc.erb b/backends/manual/templates/func.adoc.erb index a725f3014..01d693ae3 100644 --- a/backends/manual/templates/func.adoc.erb +++ b/backends/manual/templates/func.adoc.erb @@ -4,7 +4,7 @@ = Functions -<%- arch_def.functions.each do |f| -%> +<%- cfg_arch.functions.each do |f| -%> [#<%= f.name %>-func-def] == <%= f.name %><%- if f.builtin? -%> (builtin)<%- end -%> diff --git a/backends/manual/templates/index.adoc.erb b/backends/manual/templates/index.adoc.erb index b57680656..b555aae86 100644 --- a/backends/manual/templates/index.adoc.erb +++ b/backends/manual/templates/index.adoc.erb @@ -4,8 +4,8 @@ The following versions of the manual can be found here: -<%- versions.each do |version| -%> -* xref:<%= version.marketing_version %>@<%= manual.name %>:ROOT:index.adoc[Version <%= version.marketing_version %>] +<%- manual.versions.each do |version| -%> +* xref:<%= version.name %>@<%= manual.name %>:ROOT:index.adoc[Version <%= version.marketing_version %>] <%- end -%> This site was generated using https://github.com/riscv-software-src/riscv-unified-db[riscv-unified-db], diff --git a/backends/manual/templates/instruction.adoc.erb b/backends/manual/templates/instruction.adoc.erb index c1339a432..1c67a226c 100644 --- a/backends/manual/templates/instruction.adoc.erb +++ b/backends/manual/templates/instruction.adoc.erb @@ -7,18 +7,18 @@ This instruction is defined by: -<%= inst.defined_by.to_asciidoc %> +<%= inst.defined_by_condition.to_asciidoc %> This instruction is included in the following profiles: -<%- arch_def.profiles.each do |profile| -%> +<%- cfg_arch.profiles.each do |profile| -%> <%- in_profile_mandatory = profile.mandatory_ext_reqs.any? do |ext_req| - ext_versions = ext_req.satisfying_versions(arch_def) + ext_versions = ext_req.satisfying_versions ext_versions.any? { |ext_ver| inst.defined_by?(ext_ver) } end in_profile_optional = profile.optional_ext_reqs.any? do |ext_req| - ext_versions = ext_req.satisfying_versions(arch_def) + ext_versions = ext_req.satisfying_versions ext_versions.any? { |ext_ver| inst.defined_by?(ext_ver) } end if in_profile_mandatory @@ -128,7 +128,7 @@ IDL:: + [source,idl,subs="specialchars,macros"] ---- -<%= inst.operation_ast(inst.arch_def.symtab).gen_adoc %> +<%= inst.operation_ast(inst.cfg_arch.symtab).gen_adoc %> ---- <%- end -%> @@ -142,7 +142,7 @@ Sail:: <%- end -%> ==== -<% exception_list = inst.reachable_exceptions_str(inst.arch_def.symtab, 64) -%> +<% exception_list = inst.reachable_exceptions_str(inst.cfg_arch.symtab, 64) -%> <%- unless exception_list.empty? -%> == Exceptions diff --git a/backends/manual/templates/playbook.yml.erb b/backends/manual/templates/playbook.yml.erb index 3ffafdf7a..7c5018fe0 100644 --- a/backends/manual/templates/playbook.yml.erb +++ b/backends/manual/templates/playbook.yml.erb @@ -6,7 +6,7 @@ content: sources: - url: <%= $root %> start_path: gen/manual/<%= manual.name %>/top/<%= output_hash %>/antora/landing - <%- versions.each do |version| -%> + <%- manual.versions.each do |version| -%> - url: <%= $root %> start_path: gen/manual/<%= manual.name %>/<%= version.name %>/antora <%- end -%> diff --git a/backends/profile_doc/tasks.rake b/backends/profile_doc/tasks.rake index 167ba087e..8b68ba048 100644 --- a/backends/profile_doc/tasks.rake +++ b/backends/profile_doc/tasks.rake @@ -1,33 +1,30 @@ # frozen_string_literal: true -rule %r{#{$root}/gen/profile_doc/adoc/.*\.adoc} => proc { |tname| - profile_release_name = Pathname.new(tname).basename(".adoc") - - [ - "#{$root}/.stamps/arch-gen.stamp", - __FILE__, - "#{$root}/lib/arch_obj_models/profile.rb", - "#{$root}/backends/profile_doc/templates/profile.adoc.erb" - ] + Dir.glob("#{$root}/arch/profile_release/**/*.yaml") -} do |t| +rule %r{#{$root}/gen/profile_doc/adoc/.*\.adoc} => [ + __FILE__, + "#{$root}/lib/arch_obj_models/profile.rb", + "#{$root}/backends/profile_doc/templates/profile.adoc.erb", + Dir.glob("#{$root}/arch/profile_release/**/*.yaml") +].flatten do |t| profile_release_name = Pathname.new(t.name).basename(".adoc").to_s - profile_release = arch_def_for("_64").profile_release(profile_release_name) + profile_release = cfg_arch_for("_").profile_release(profile_release_name) raise ArgumentError, "No profile release named '#{profile_release_name}'" if profile_release.nil? + profile_class = profile_release.profile_class template_path = Pathname.new "#{$root}/backends/profile_doc/templates/profile.adoc.erb" erb = ERB.new(template_path.read, trim_mode: "-") erb.filename = template_path.to_s - arch_def = arch_def_for("_64") + cfg_arch = cfg_arch_for("_") - # XXX - Add call to to_arch_def() in portfolio instance class. + # XXX - Add call to to_cfg_arch() in portfolio instance class. # But somehow have to merge the multiple portofolios in one profile release to one since - # to_arch_def used to provide coloring of fields in CSRs in appendices that apply to all profiles in a release. + # to_cfg_arch used to provide coloring of fields in CSRs in appendices that apply to all profiles in a release. FileUtils.mkdir_p File.dirname(t.name) - File.write t.name, AsciidocUtils.resolve_links(arch_def.find_replace_links(erb.result(binding))) + File.write t.name, AsciidocUtils.resolve_links(cfg_arch.find_replace_links(erb.result(binding))) puts "Generated adoc source at #{t.name}" end @@ -86,22 +83,22 @@ end namespace :gen do desc "Create a specification PDF for +profile_release+" - task :profile, [:profile_release] => ["#{$root}/.stamps/arch-gen-_64.stamp"] do |_t, args| + task :profile, [:profile_release] do |_t, args| profile_release_name = args[:profile_release] raise ArgumentError, "Missing required option +profile_release+" if profile_release_name.nil? - profile_release = arch_def_for("_64").profile_release(profile_release_name) + profile_release = cfg_arch_for("_").profile_release(profile_release_name) raise ArgumentError, "No profile release named '#{profile_release_name}'" if profile_release.nil? Rake::Task["#{$root}/gen/profile_doc/pdf/#{profile_release_name}.pdf"].invoke end desc "Create a specification HTML for +profile_release+" - task :profile_html, [:profile_release] => ["#{$root}/.stamps/arch-gen-_64.stamp"] do |_t, args| + task :profile_html, [:profile_release] do |_t, args| profile_release_name = args[:profile_release] raise ArgumentError, "Missing required option +profile_release+" if profile_release_name.nil? - profile_release = arch_def_for("_64").profile_release(profile_release_name) + profile_release = cfg_arch_for("_").profile_release(profile_release_name) raise ArgumentError, "No profile release named '#{profile_release_name}" if profile_release.nil? Rake::Task["#{$root}/gen/profile_doc/html/#{profile_release_name}.html"].invoke diff --git a/backends/profile_doc/templates/profile.adoc.erb b/backends/profile_doc/templates/profile.adoc.erb index e310413d0..9eeef40c5 100644 --- a/backends/profile_doc/templates/profile.adoc.erb +++ b/backends/profile_doc/templates/profile.adoc.erb @@ -422,7 +422,7 @@ associated implementation-defined parameters. .Status |=== -| Profile | v<%= ext.versions.map { |v| v.version }.join(" | v") %> +| Profile | v<%= ext.versions.map { |ext_ver| ext_ver.canonical_version.to_s }.join(" | v") %> <% profile_release.profiles.each do |profile| -%> | <%= profile.marketing_name %> | <%= profile.version_strongest_presence(ext.name, ext.versions).join(" | ") -%> @@ -431,7 +431,7 @@ associated implementation-defined parameters. |=== <% ext.versions.each do |v| -%> -<%= v.version %>:: +<%= v.canonical_version %>:: Ratification date::: <%= v.ratification_date %> <% unless v.changes.empty? -%> @@ -463,7 +463,7 @@ associated implementation-defined parameters. :leveloffset: -3 // TODO: GitHub issue 92: Use version specified by each profile and add version info to inst table below. -<%- insts = arch_def.instructions.select { |i| i.defined_by?(ext.min_version) } -%> +<%- insts = cfg_arch.instructions.select { |i| i.defined_by?(ext.min_version) } -%> <%- unless insts.empty? -%> ==== Instructions @@ -511,7 +511,7 @@ This extension has the following implementation options: This instruction is defined by: -<%= inst.defined_by.to_asciidoc %> +<%= inst.defined_by_condition.to_asciidoc %> ==== Encoding @@ -608,13 +608,13 @@ RV64:: <% if inst.key?("operation()") -%> [source,idl,subs="specialchars,macros"] ---- -<%= inst.operation_ast(arch_def.symtab).gen_adoc %> +<%= inst.operation_ast(cfg_arch.symtab).gen_adoc %> ---- <% end -%> ==== Exceptions -<% exception_list = inst.reachable_exceptions_str(arch_def.symtab) -%> +<% exception_list = inst.reachable_exceptions_str(cfg_arch.symtab) -%> <% if exception_list.empty? -%> This instruction does not generate synchronous exceptions. <% else -%> @@ -660,23 +660,23 @@ h| CSR Address | <%= "0x#{csr.address.to_s(16)}" %> <% if csr.priv_mode == 'VS' -%> h| Virtual CSR Address | <%= "0x#{csr.virtual_address.to_s(16)}" %> <% end -%> -h| Defining extension a| <%= csr.defined_by.to_asciidoc %> -<% if csr.dynamic_length?(arch_def) -%> -h| Length | <%= csr.length_pretty(arch_def) %> +h| Defining extension a| <%= csr.defined_by_condition.to_asciidoc %> +<% if csr.dynamic_length?(cfg_arch) -%> +h| Length | <%= csr.length_pretty(cfg_arch) %> <% else -%> -h| Length | <%= csr.length_pretty(arch_def) %> +h| Length | <%= csr.length_pretty(cfg_arch) %> <% end -%> h| Privilege Mode | <%= csr.priv_mode %> |=== ==== Format -<% unless csr.dynamic_length?(arch_def) || csr.fields.any? { |f| f.dynamic_location?(arch_def) } -%> +<% unless csr.dynamic_length?(cfg_arch) || csr.fields.any? { |f| f.dynamic_location?(cfg_arch) } -%> <%# CSR has a known static length, so there is only one format to display -%> .<%= csr.name %> format [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, csr.base.nil? ? 32 : csr.base) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, csr.base.nil? ? 32 : csr.base) %> .... <% else -%> <%# CSR has a dynamic length, or a field has a dynamic location, @@ -686,13 +686,13 @@ This CSR format changes dynamically with XLEN. .<%= csr.name %> Format when <%= csr.length_cond32 %> [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, 32) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, 32) %> .... .<%= csr.name %> Format when <%= csr.length_cond64 %> [wavedrom, ,svg,subs='attributes',width="100%"] .... -<%= JSON.dump csr.wavedrom_desc(arch_def, 64) %> +<%= JSON.dump csr.wavedrom_desc(cfg_arch, 64) %> .... diff --git a/bin/clean b/bin/clean index bae76c356..22e665c5f 100755 --- a/bin/clean +++ b/bin/clean @@ -1,5 +1,5 @@ #!/bin/bash -ROOT=$(dirname $(realpath $BASH_SOURCE[0])) +ROOT=$(dirname $(dirname $(realpath $BASH_SOURCE[0]))) -rm -rf ${ROOT}/.stamps ${ROOT}/.home ${ROOT}/.bundle ${ROOT}/.singularity +rm -rf ${ROOT}/.stamps ${ROOT}/.home ${ROOT}/.bundle ${ROOT}/.singularity ${ROOT}/gen ${ROOT}/node_modules diff --git a/bin/setup b/bin/setup index 2f1c67342..9ea0591f5 100755 --- a/bin/setup +++ b/bin/setup @@ -15,7 +15,7 @@ if [ -v GITHUB_ACTIONS ]; then else CONTAINER_PATH=${ROOT}/.singularity/image-$CONTAINER_TAG.sif HOME_PATH=${HOME} - HOME_OPT="--bind ${ROOT}/.home:${HOME_PATH}" + HOME_OPT="--bind ${ROOT}/.home:${HOME_PATH} --bind /local/mnt/workspace/.vscode-server:/local/mnt/workspace/.vscode-server" SINGULARITY_CACHE= fi diff --git a/cfgs/_/cfg.yaml b/cfgs/_/cfg.yaml new file mode 100644 index 000000000..107a2c62e --- /dev/null +++ b/cfgs/_/cfg.yaml @@ -0,0 +1,8 @@ +# yaml-language-server: $schema=../../schemas/config_schema.json +--- +$schema: config_schema.json# +kind: architecture configuration +type: unconfigured +name: _ +description: | + A completely unconfigured RVI-standard architecture; not even MXLEN is known. diff --git a/cfgs/_32/cfg.yaml b/cfgs/_32/cfg.yaml deleted file mode 100644 index ca0f2d33d..000000000 --- a/cfgs/_32/cfg.yaml +++ /dev/null @@ -1,2 +0,0 @@ ---- -type: partially configured diff --git a/cfgs/_32/implemented_exts.yaml b/cfgs/_32/implemented_exts.yaml deleted file mode 100644 index a532cca51..000000000 --- a/cfgs/_32/implemented_exts.yaml +++ /dev/null @@ -1 +0,0 @@ -implemented_extensions: [] diff --git a/cfgs/_32/params.yaml b/cfgs/_32/params.yaml deleted file mode 100644 index b4f8fb3d4..000000000 --- a/cfgs/_32/params.yaml +++ /dev/null @@ -1,5 +0,0 @@ ---- -params: - NAME: _32 - - XLEN: 32 diff --git a/cfgs/_64/cfg.yaml b/cfgs/_64/cfg.yaml deleted file mode 100644 index ca0f2d33d..000000000 --- a/cfgs/_64/cfg.yaml +++ /dev/null @@ -1,2 +0,0 @@ ---- -type: partially configured diff --git a/cfgs/_64/implemented_exts.yaml b/cfgs/_64/implemented_exts.yaml deleted file mode 100644 index a532cca51..000000000 --- a/cfgs/_64/implemented_exts.yaml +++ /dev/null @@ -1 +0,0 @@ -implemented_extensions: [] diff --git a/cfgs/_64/params.yaml b/cfgs/_64/params.yaml deleted file mode 100644 index b4ff65b89..000000000 --- a/cfgs/_64/params.yaml +++ /dev/null @@ -1,4 +0,0 @@ -params: - NAME: _64 - - XLEN: 64 diff --git a/cfgs/generic_rv64/cfg.yaml b/cfgs/generic_rv64/cfg.yaml index c9fde9219..c15ff4800 100644 --- a/cfgs/generic_rv64/cfg.yaml +++ b/cfgs/generic_rv64/cfg.yaml @@ -1,2 +1,555 @@ ---- +$schema: config_schema.json# +kind: architecture configuration type: fully configured +name: generic_rv64 +description: An example fully-specified RV64 system +implemented_extensions: + - [A, "2.1.0"] + - [B, "1.0.0"] + - [C, "2.0.0"] + - [D, "2.2.0"] + - [F, "2.2.0"] + - [I, "2.1.0"] + - [H, "1.0.0"] + - [M, "2.0.0"] + - [S, "1.12.0"] + - [Sm, "1.12.0"] + - [Smhpm, "1.12.0"] + - [Smpmp, "1.12.0"] + - [U, "1.12.0"] + - [V, "1.0.0"] + - [Zicntr, "2.0.0"] + - [Zicsr, "2.0.0"] + - [Zihpm, "2.0.0"] + - [Smaia, "1.0.0"] + - [Smcdeleg, "1.0.0"] + - [Smcntrpmf, "1.0.0"] + - [Sscofpmf, "1.0.0"] + - [Ssaia, "1.0.0"] + - [Ssccfg, "1.0.0"] + - [Sstc, "0.9.0"] + - [Sv39, "1.12.0"] + - [Sv48, "1.12.0"] + - [Zicboz, "1.0.0"] + - [Zicbom, "1.0.0"] + +params: + XLEN: 64 + + # name of the configuration + NAME: generic_rv64 + + # vendor-specific architecture ID in marchid + ARCH_ID: 0x1000000000000000 + + # vendor-specific implementation ID in mimpid + IMP_ID: 0x0 + + # JEDEC Vendor ID bank + VENDOR_ID_BANK: 0x0 + + # JEDEC Vendor ID offset + VENDOR_ID_OFFSET: 0x0 + + # whether or not the implementation supports misaligned loads and stores in main memory (not including atomics) + # must be true when Zicclsm is supported + MISALIGNED_LDST: true + + MISALIGNED_LDST_EXCEPTION_PRIORITY: high + + MISALIGNED_MAX_ATOMICITY_GRANULE_SIZE: 0 + + MISALIGNED_SPLIT_STRATEGY: by_byte + + # whether or not the implementation supports misaligned atomics + MISALIGNED_AMO: false + + HPM_COUNTER_EN: + - false # CY + - false # empty + - false # IR + - true # HPM3 + - true # HPM4 + - true # HPM5 + - true # HPM6 + - true # HPM7 + - true # HPM8 + - true # HPM9 + - true # HPM10 + - false # HPM11 + - false # HPM12 + - false # HPM13 + - false # HPM14 + - false # HPM15 + - false # HPM16 + - false # HPM17 + - false # HPM18 + - false # HPM19 + - false # HPM20 + - false # HPM21 + - false # HPM22 + - false # HPM23 + - false # HPM24 + - false # HPM25 + - false # HPM26 + - false # HPM27 + - false # HPM28 + - false # HPM29 + - false # HPM30 + - false # HPM31 + + # list of defined HPM events + HPM_EVENTS: + - 0 + - 3 + + # Indicates which counters can be disabled from mcountinhibit + # + # An unimplemented counter cannot be specified, i.e., if + # NUM_HPM_COUNTERS == 8, it would be illegal to add index + # 11 in COUNTINHIBIT_EN since the highest implemented counter + # would be at bit 10 + COUNTINHIBIT_EN: + - true # CY + - false # empty + - true # IR + - true # HPM3 + - true # HPM4 + - true # HPM5 + - true # HPM6 + - true # HPM7 + - true # HPM8 + - true # HPM9 + - true # HPM10 + - false # HPM11 + - false # HPM12 + - false # HPM13 + - false # HPM14 + - false # HPM15 + - false # HPM16 + - false # HPM17 + - false # HPM18 + - false # HPM19 + - false # HPM20 + - false # HPM21 + - false # HPM22 + - false # HPM23 + - false # HPM24 + - false # HPM25 + - false # HPM26 + - false # HPM27 + - false # HPM28 + - false # HPM29 + - false # HPM30 + - false # HPM31 + + # Indicates which counters can delegated via mcounteren + # + # An unimplemented counter cannot be specified, i.e., if + # NUM_HPM_COUNTERS == 8, it would be illegal to add index + # 11 in COUNTEN_EN since the highest implemented counter + # would be at bit 10 + MCOUNTENABLE_EN: + - true # CY + - false # TM + - true # IR + - true # HPM3 + - true # HPM4 + - true # HPM5 + - true # HPM6 + - true # HPM7 + - true # HPM8 + - true # HPM9 + - true # HPM10 + - false # HPM11 + - false # HPM12 + - false # HPM13 + - false # HPM14 + - false # HPM15 + - false # HPM16 + - false # HPM17 + - false # HPM18 + - false # HPM19 + - false # HPM20 + - false # HPM21 + - false # HPM22 + - false # HPM23 + - false # HPM24 + - false # HPM25 + - false # HPM26 + - false # HPM27 + - false # HPM28 + - false # HPM29 + - false # HPM30 + - false # HPM31 + + # Indicates which counters can delegated via scounteren + # + # An unimplemented counter cannot be specified, i.e., if + # NUM_HPM_COUNTERS == 8, it would be illegal to add index + # 11 in COUNTEN_EN since the highest implemented counter + # would be at bit 10 + SCOUNTENABLE_EN: + - true # CY + - false # TM + - true # IR + - true # HPM3 + - true # HPM4 + - true # HPM5 + - true # HPM6 + - true # HPM7 + - true # HPM8 + - true # HPM9 + - true # HPM10 + - false # HPM11 + - false # HPM12 + - false # HPM13 + - false # HPM14 + - false # HPM15 + - false # HPM16 + - false # HPM17 + - false # HPM18 + - false # HPM19 + - false # HPM20 + - false # HPM21 + - false # HPM22 + - false # HPM23 + - false # HPM24 + - false # HPM25 + - false # HPM26 + - false # HPM27 + - false # HPM28 + - false # HPM29 + - false # HPM30 + - false # HPM31 + + # Indicates which counters can delegated via hcounteren + # + # An unimplemented counter cannot be specified, i.e., if + # NUM_HPM_COUNTERS == 8, it would be illegal to add index + # 11 in COUNTEN_EN since the highest implemented counter + # would be at bit 10 + HCOUNTENABLE_EN: + - true # CY + - false # TM + - true # IR + - true # HPM3 + - true # HPM4 + - true # HPM5 + - true # HPM6 + - true # HPM7 + - true # HPM8 + - true # HPM9 + - true # HPM10 + - false # HPM11 + - false # HPM12 + - false # HPM13 + - false # HPM14 + - false # HPM15 + - false # HPM16 + - false # HPM17 + - false # HPM18 + - false # HPM19 + - false # HPM20 + - false # HPM21 + - false # HPM22 + - false # HPM23 + - false # HPM24 + - false # HPM25 + - false # HPM26 + - false # HPM27 + - false # HPM28 + - false # HPM29 + - false # HPM30 + - false # HPM31 + + # when true, writing an illegal value to a WLRL CSR field raises an Illegal Instruction exception + # when false, writing an illegal value to a WLRL CSR field is ignored + TRAP_ON_ILLEGAL_WLRL: true + TRAP_ON_UNIMPLEMENTED_INSTRUCTION: true + TRAP_ON_RESERVED_INSTRUCTION: true + TRAP_ON_UNIMPLEMENTED_CSR: true + + # Whether or not a real hardware `time` CSR exists. Implementations can either provide a real + # CSR or trap and emulate access at M-mode. + TIME_CSR_IMPLEMENTED: true + + # Whether or not the `misa` CSR returns zero or a non-zero value. + MISA_CSR_IMPLEMENTED: true + + # when true, *tval is written with the virtual PC of the EBREAK instruction (same information as *epc) + # when false, *tval is written with 0 on an EBREAK instruction + # + # regardless, *tval is always written with a virtual PC when an external breakpoint is generated + REPORT_VA_IN_MTVAL_ON_BREAKPOINT: true + + REPORT_VA_IN_MTVAL_ON_LOAD_MISALIGNED: true + REPORT_VA_IN_MTVAL_ON_STORE_AMO_MISALIGNED: true + REPORT_VA_IN_MTVAL_ON_INSTRUCTION_MISALIGNED: true + REPORT_VA_IN_MTVAL_ON_LOAD_ACCESS_FAULT: true + REPORT_VA_IN_MTVAL_ON_STORE_AMO_ACCESS_FAULT: true + REPORT_VA_IN_MTVAL_ON_INSTRUCTION_ACCESS_FAULT: true + REPORT_VA_IN_MTVAL_ON_LOAD_PAGE_FAULT: true + REPORT_VA_IN_MTVAL_ON_STORE_AMO_PAGE_FAULT: true + REPORT_VA_IN_MTVAL_ON_INSTRUCTION_PAGE_FAULT: true + REPORT_ENCODING_IN_MTVAL_ON_ILLEGAL_INSTRUCTION: true + # REPORT_CAUSE_IN_MTVAL_ON_SOFTWARE_CHECK: true + MTVAL_WIDTH: 64 # must check that this can hold any valid VA if any REPORT_VA* or Sdext, and, if REPORT_ENCODING*, at least [MXLEN, ILEN].min bits + + REPORT_VA_IN_STVAL_ON_BREAKPOINT: true + REPORT_VA_IN_STVAL_ON_LOAD_MISALIGNED: true + REPORT_VA_IN_STVAL_ON_STORE_AMO_MISALIGNED: true + REPORT_VA_IN_STVAL_ON_INSTRUCTION_MISALIGNED: true + REPORT_VA_IN_STVAL_ON_LOAD_ACCESS_FAULT: true + REPORT_VA_IN_STVAL_ON_STORE_AMO_ACCESS_FAULT: true + REPORT_VA_IN_STVAL_ON_INSTRUCTION_ACCESS_FAULT: true + REPORT_VA_IN_STVAL_ON_LOAD_PAGE_FAULT: true + REPORT_VA_IN_STVAL_ON_STORE_AMO_PAGE_FAULT: true + REPORT_VA_IN_STVAL_ON_INSTRUCTION_PAGE_FAULT: true + REPORT_ENCODING_IN_STVAL_ON_ILLEGAL_INSTRUCTION: true + # REPORT_CAUSE_IN_STVAL_ON_SOFTWARE_CHECK: true + STVAL_WIDTH: 64 # must check that this can hold any valid VA, and, if REPORT_ENCODING*, at least [SXLEN, ILEN].min bits + + REPORT_VA_IN_VSTVAL_ON_BREAKPOINT: true + REPORT_VA_IN_VSTVAL_ON_LOAD_MISALIGNED: true + REPORT_VA_IN_VSTVAL_ON_STORE_AMO_MISALIGNED: true + REPORT_VA_IN_VSTVAL_ON_INSTRUCTION_MISALIGNED: true + REPORT_VA_IN_VSTVAL_ON_LOAD_ACCESS_FAULT: true + REPORT_VA_IN_VSTVAL_ON_STORE_AMO_ACCESS_FAULT: true + REPORT_VA_IN_VSTVAL_ON_INSTRUCTION_ACCESS_FAULT: true + REPORT_VA_IN_VSTVAL_ON_LOAD_PAGE_FAULT: true + REPORT_VA_IN_VSTVAL_ON_STORE_AMO_PAGE_FAULT: true + REPORT_VA_IN_VSTVAL_ON_INSTRUCTION_PAGE_FAULT: true + REPORT_ENCODING_IN_VSTVAL_ON_ILLEGAL_INSTRUCTION: true + # REPORT_CAUSE_IN_VSTVAL_ON_SOFTWARE_CHECK: true + # VSTVAL_WIDTH not needed; "vstval is a WARL register that must be able to hold the same set of values that stval can hold" + + # address of the unified discovery configuration data structure + # this address is reported in the mconfigptr CSR + CONFIG_PTR_ADDRESS: 0x1000 + + # number of implemented PMP entries. Can be any value between 0-64, inclusive. + # + # the number of implemented PMP registers must be 0, 16, or 64. + # + # Therefore, whether or not a pmpaddrN or pmpcfgN register exists depends on + # NUM_PMP_ENTRIES as follows: + # |=== + # | NUM_PMP_ENTRIES | pmpaddr<0-15> / pmpcfg<0-3> | pmpaddr<16-63> / pmpcfg<4-15> + # | 0 | N | N + # | 1-16 | Y | N + # | 17-64 | Y | Y + # |=== + # ** pmpcfgN for an odd N never exist when XLEN == 64 + # + # when NUM_PMP_ENTRIES is not exactly 0, 16, or 64, some extant pmp registers, + # and associated pmpNcfg, will be read-only zero (but will not cause an exception). + NUM_PMP_ENTRIES: 14 + + # log2 of the smallest supported PMP region + # generally, for systems with an MMU, should not be smaller than 12, + # as that would preclude caching PMP results in the TLB along with + # virtual memory translations + # + # Note that PMP_GRANULARITY is equal to G+2 (not G) as described in + # the privileged architecture + PMP_GRANULARITY: 12 + + # log2 of the smallest supported PMA region + # generally, for systems with an MMU, should not be smaller than 12, + # as that would preclude caching PMP results in the TLB along with + # virtual memory translations + PMA_GRANULARITY: 12 + + # number of bits in the physical address space + PHYS_ADDR_WIDTH: 56 + + # number of implemented ASID bits + # maximum value is 16 + ASID_WIDTH: 12 + + # when the A extensions is supported, indicates whether or not + # the extension can be disabled in the `misa.A` bit. + MUTABLE_MISA_A: false + + # when the B extensions is supported, indicates whether or not + # the extension can be disabled in the `misa.B` bit. + MUTABLE_MISA_B: false + + # when the C extensions is supported, indicates whether or not + # the extension can be disabled in the `misa.C` bit. + MUTABLE_MISA_C: false + + # when the D extensions is supported, indicates whether or not + # the extension can be disabled in the `misa.D` bit. + MUTABLE_MISA_D: false + + # when the F extensions is supported, indicates whether or not + # the extension can be disabled in the `misa.F` bit. + MUTABLE_MISA_F: false + + # when the H extensions is supported, indicates whether or not + # the extension can be disabled in the `misa.H` bit. + MUTABLE_MISA_H: false + + # when the M extensions is supported, indicates whether or not + # the extension can be disabled in the `misa.M` bit. + MUTABLE_MISA_M: false + + # when the S extensions is supported, indicates whether or not + # the extension can be disabled in the `misa.S` bit. + MUTABLE_MISA_S: false + + # when the U extensions is supported, indicates whether or not + # the extension can be disabled in the `misa.U` bit. + MUTABLE_MISA_U: false + + # when the V extensions is supported, indicates whether or not + # the extension can be disabled in the `misa.V` bit. + MUTABLE_MISA_V: false + + # size of a cache block, in bytes + CACHE_BLOCK_SIZE: 64 + + # number of supported virtualized guest interrupts + # corresponds to the `GEILEN` parameter in the RVI specs + NUM_EXTERNAL_GUEST_INTERRUPTS: 4 + + # Endianess of data in M-mode. Can be one of: + # + # * little: M-mode data is always little endian + # * big: M-mode data is always big endian + # * dynamic: M-mode data can be either little or big endian, depending on the RW CSR field mstatus.MBE + M_MODE_ENDIANESS: little + + # Endianess of data in M-mode. Can be one of: + # + # * little: S-mode data is always little endian + # * big: S-mode data is always big endian + # * dynamic: S-mode data can be either little or big endian, depending on the RW CSR field mstatus.SBE + S_MODE_ENDIANESS: little + + # Endianess of data in M-mode. Can be one of: + # + # * litte: U-mode data is always little endian + # * big: U-mode data is always big endian + # * dynamic: U-mode data can be either little or big endian, depending on the RW CSR field mstatus.UBE + U_MODE_ENDIANESS: little + + # Endianess of data in VU-mode. Can be one of: + # + # * little: VU-mode data is always little endian + # * big: VU-mode data is always big endian + # * dynamic: VU-mode data can be either little or big endian, depending on the RW CSR field vsstatus.UBE + VU_MODE_ENDIANESS: little + + # Endianess of data in VS-mode. Can be one of: + # + # * little: VS-mode data is always little endian + # * big: VS-mode data is always big endian + # * dynamic: VS-mode data can be either little or big endian, depending on the RW CSR field hstatus.VSBE + VS_MODE_ENDIANESS: little + + # XLENs supported in S-mode. Can be one of: + # + # * 32: SXLEN is always 32 + # * 64: SXLEN is always 64 + # * 3264: SXLEN can be changed (via mstatus.SXL) between 32 and 64 + SXLEN: 64 + + # XLENs supported in U-mode. Can be one of: + # + # * 32: SXLEN is always 32 + # * 64: SXLEN is always 64 + # * 3264: SXLEN can be changed (via mstatus.SXL) between 32 and 64 + UXLEN: 64 + + # XLENs supported in VS-mode. Can be one of: + # + # * 32: VSXLEN is always 32 + # * 64: VSXLEN is always 64 + # * 3264: VSXLEN can be changed (via hstatus.VSXL) between 32 and 64 + VSXLEN: 64 + + # XLENs supported in VS-mode. Can be one of: + # + # * 32: VSXLEN is always 32 + # * 64: VSXLEN is always 64 + # * 3264: VSXLEN can be changed (via hstatus.VSXL) between 32 and 64 + VUXLEN: 64 + + # Strategy used to handle reservation sets + # + # * "reserve naturally-aligned 64-byte region": Always reserve the 64-byte block containing the LR/SC address + # * "reserve naturally-aligned 128-byte region": Always reserve the 128-byte block containing the LR/SC address + # * "reserve exactly enough to cover the access": Always reserve exactly the LR/SC access, and no more + # * "custom": Custom behavior, leading to an 'unpredictable' call on any LR/SC + LRSC_RESERVATION_STRATEGY: reserve naturally-aligned 64-byte region + + # whether or not an SC will fail if its VA does not match the VA of the prior LR, + # even if the physical address of the SC and LR are the same + LRSC_FAIL_ON_VA_SYNONYM: false + + # what to do when an LR/SC address is misaligned: + # + # * 'always raise misaligned exception': self-explainitory + # * 'always raise access fault': self-explainitory + # * 'custom': Custom behavior; misaligned LR/SC may sometimes raise a misaligned exception and sometimes raise a access fault. Will lead to an 'unpredictable' call on any misaligned LR/SC access + LRSC_MISALIGNED_BEHAVIOR: always raise misaligned exception + + # whether or not a Store Conditional fails if its physical address and size do not + # exactly match the physical address and size of the last Load Reserved in program order + # (independent of whether or not the SC is in the current reservation set) + LRSC_FAIL_ON_NON_EXACT_LRSC: false + + # Whether writes from M-mode, U-mode, or S-mode to vsatp with an illegal mode setting are + # ignored (as they are with satp), or if they are treated as WARL, leading to undpredictable + # behavior. + IGNORE_INVALID_VSATP_MODE_WRITES_WHEN_V_EQ_ZERO: true + + GSTAGE_MODE_BARE: true + SV32_VSMODE_TRANSLATION: false + SV39_VSMODE_TRANSLATION: true + SV48_VSMODE_TRANSLATION: true + SV57_VSMODE_TRANSLATION: true + SV32X4_TRANSLATION: false + SV39X4_TRANSLATION: true + SV48X4_TRANSLATION: true + SV57X4_TRANSLATION: false + VMID_WIDTH: 8 + SV_MODE_BARE: true + STVEC_MODE_DIRECT: true + STVEC_MODE_VECTORED: true + SATP_MODE_BARE: true + REPORT_GPA_IN_TVAL_ON_LOAD_GUEST_PAGE_FAULT: true + REPORT_GPA_IN_TVAL_ON_STORE_AMO_GUEST_PAGE_FAULT: true + REPORT_GPA_IN_TVAL_ON_INSTRUCTION_GUEST_PAGE_FAULT: true + REPORT_GPA_IN_TVAL_ON_INTERMEDIATE_GUEST_PAGE_FAULT: true + TINST_VALUE_ON_FINAL_LOAD_GUEST_PAGE_FAULT: "always transformed standard instruction" + TINST_VALUE_ON_FINAL_STORE_AMO_GUEST_PAGE_FAULT: "always transformed standard instruction" + TINST_VALUE_ON_FINAL_INSTRUCTION_GUEST_PAGE_FAULT: "always zero" + TINST_VALUE_ON_INSTRUCTION_ADDRESS_MISALIGNED: "always zero" + TINST_VALUE_ON_BREAKPOINT: "always zero" + TINST_VALUE_ON_VIRTUAL_INSTRUCTION: "always zero" + TINST_VALUE_ON_LOAD_ADDRESS_MISALIGNED: "always zero" + TINST_VALUE_ON_LOAD_ACCESS_FAULT: "always zero" + TINST_VALUE_ON_STORE_AMO_ADDRESS_MISALIGNED: "always zero" + TINST_VALUE_ON_STORE_AMO_ACCESS_FAULT: "always zero" + TINST_VALUE_ON_UCALL: "always zero" + TINST_VALUE_ON_SCALL: "always zero" + TINST_VALUE_ON_MCALL: "always zero" + TINST_VALUE_ON_VSCALL: "always zero" + TINST_VALUE_ON_LOAD_PAGE_FAULT: "always zero" + TINST_VALUE_ON_STORE_AMO_PAGE_FAULT: "always zero" + MTVEC_MODES: [0, 1] + MSTATUS_FS_LEGAL_VALUES: [0, 1, 2, 3] + MSTATUS_FS_WRITEABLE: true + MSTATUS_TVM_IMPLEMENTED: true + HW_MSTATUS_FS_DIRTY_UPDATE: precise + MSTATUS_VS_WRITEABLE: true + MSTATUS_VS_LEGAL_VALUES: [0, 1, 2, 3] + HW_MSTATUS_VS_DIRTY_UPDATE: precise + FORCE_UPGRADE_CBO_INVAL_TO_FLUSH: true + REPORT_GPA_IN_HTVAL_ON_GUEST_PAGE_FAULT: true + VSTVEC_MODE_DIRECT: true + VSTVEC_MODE_VECTORED: true diff --git a/cfgs/generic_rv64/implemented_exts.yaml b/cfgs/generic_rv64/implemented_exts.yaml deleted file mode 100644 index 9a0b8a2cd..000000000 --- a/cfgs/generic_rv64/implemented_exts.yaml +++ /dev/null @@ -1,31 +0,0 @@ -# $schema=../../schemas/implemented_exts_schema.json - -implemented_extensions: - - [A, "2.1.0"] - - [B, "1.0.0"] - - [C, "2.2.0"] - - [D, "2.2.0"] - - [F, "2.2.0"] - - [I, "2.1.0"] - - [H, "1.0.0"] - - [M, "2.0.0"] - - [S, "1.12.0"] - - [Sm, "1.12.0"] - - [Smhpm, "1.12.0"] - - [Smpmp, "1.12.0"] - - [U, "1.12.0"] - - [V, "1.0.0"] - - [Zicntr, "2.0.0"] - - [Zicsr, "2.0.0"] - - [Zihpm, "2.0.0"] - - [Smaia, "1.0.0"] - - [Smcdeleg, "1.0.0"] - - [Smcntrpmf, "1.0.0"] - - [Sscofpmf, "1.0.0"] - - [Ssaia, "1.0.0"] - - [Ssccfg, "1.0.0"] - - [Sstc, "0.9.0"] - - [Sv39, "1.12.0"] - - [Sv48, "1.12.0"] - - [Zicboz, "1.0.1-b34ea8a"] - - [Zicbom, "1.0.1-b34ea8a"] diff --git a/cfgs/generic_rv64/params.yaml b/cfgs/generic_rv64/params.yaml deleted file mode 100644 index b7f97c19f..000000000 --- a/cfgs/generic_rv64/params.yaml +++ /dev/null @@ -1,518 +0,0 @@ ---- -params: - XLEN: 64 - - # name of the configuration - NAME: generic_rv64 - - # vendor-specific architecture ID in marchid - ARCH_ID: 0x1000000000000000 - - # vendor-specific implementation ID in mimpid - IMP_ID: 0x0 - - # JEDEC Vendor ID bank - VENDOR_ID_BANK: 0x0 - - # JEDEC Vendor ID offset - VENDOR_ID_OFFSET: 0x0 - - # whether or not the implementation supports misaligned loads and stores in main memory (not including atomics) - # must be true when Zicclsm is supported - MISALIGNED_LDST: true - - MISALIGNED_LDST_EXCEPTION_PRIORITY: high - - MISALIGNED_MAX_ATOMICITY_GRANULE_SIZE: 0 - - MISALIGNED_SPLIT_STRATEGY: by_byte - - # whether or not the implementation supports misaligned atomics - MISALIGNED_AMO: false - - HPM_COUNTER_EN: - - false # CY - - false # empty - - false # IR - - true # HPM3 - - true # HPM4 - - true # HPM5 - - true # HPM6 - - true # HPM7 - - true # HPM8 - - true # HPM9 - - true # HPM10 - - false # HPM11 - - false # HPM12 - - false # HPM13 - - false # HPM14 - - false # HPM15 - - false # HPM16 - - false # HPM17 - - false # HPM18 - - false # HPM19 - - false # HPM20 - - false # HPM21 - - false # HPM22 - - false # HPM23 - - false # HPM24 - - false # HPM25 - - false # HPM26 - - false # HPM27 - - false # HPM28 - - false # HPM29 - - false # HPM30 - - false # HPM31 - - # list of defined HPM events - HPM_EVENTS: - - 0 - - 3 - - # Indicates which counters can be disabled from mcountinhibit - # - # An unimplemented counter cannot be specified, i.e., if - # NUM_HPM_COUNTERS == 8, it would be illegal to add index - # 11 in COUNTINHIBIT_EN since the highest implemented counter - # would be at bit 10 - COUNTINHIBIT_EN: - - true # CY - - false # empty - - true # IR - - true # HPM3 - - true # HPM4 - - true # HPM5 - - true # HPM6 - - true # HPM7 - - true # HPM8 - - true # HPM9 - - true # HPM10 - - false # HPM11 - - false # HPM12 - - false # HPM13 - - false # HPM14 - - false # HPM15 - - false # HPM16 - - false # HPM17 - - false # HPM18 - - false # HPM19 - - false # HPM20 - - false # HPM21 - - false # HPM22 - - false # HPM23 - - false # HPM24 - - false # HPM25 - - false # HPM26 - - false # HPM27 - - false # HPM28 - - false # HPM29 - - false # HPM30 - - false # HPM31 - - # Indicates which counters can delegated via mcounteren - # - # An unimplemented counter cannot be specified, i.e., if - # NUM_HPM_COUNTERS == 8, it would be illegal to add index - # 11 in COUNTEN_EN since the highest implemented counter - # would be at bit 10 - MCOUNTENABLE_EN: - - true # CY - - false # TM - - true # IR - - true # HPM3 - - true # HPM4 - - true # HPM5 - - true # HPM6 - - true # HPM7 - - true # HPM8 - - true # HPM9 - - true # HPM10 - - false # HPM11 - - false # HPM12 - - false # HPM13 - - false # HPM14 - - false # HPM15 - - false # HPM16 - - false # HPM17 - - false # HPM18 - - false # HPM19 - - false # HPM20 - - false # HPM21 - - false # HPM22 - - false # HPM23 - - false # HPM24 - - false # HPM25 - - false # HPM26 - - false # HPM27 - - false # HPM28 - - false # HPM29 - - false # HPM30 - - false # HPM31 - - # Indicates which counters can delegated via scounteren - # - # An unimplemented counter cannot be specified, i.e., if - # NUM_HPM_COUNTERS == 8, it would be illegal to add index - # 11 in COUNTEN_EN since the highest implemented counter - # would be at bit 10 - SCOUNTENABLE_EN: - - true # CY - - false # TM - - true # IR - - true # HPM3 - - true # HPM4 - - true # HPM5 - - true # HPM6 - - true # HPM7 - - true # HPM8 - - true # HPM9 - - true # HPM10 - - false # HPM11 - - false # HPM12 - - false # HPM13 - - false # HPM14 - - false # HPM15 - - false # HPM16 - - false # HPM17 - - false # HPM18 - - false # HPM19 - - false # HPM20 - - false # HPM21 - - false # HPM22 - - false # HPM23 - - false # HPM24 - - false # HPM25 - - false # HPM26 - - false # HPM27 - - false # HPM28 - - false # HPM29 - - false # HPM30 - - false # HPM31 - - # Indicates which counters can delegated via hcounteren - # - # An unimplemented counter cannot be specified, i.e., if - # NUM_HPM_COUNTERS == 8, it would be illegal to add index - # 11 in COUNTEN_EN since the highest implemented counter - # would be at bit 10 - HCOUNTENABLE_EN: - - true # CY - - false # TM - - true # IR - - true # HPM3 - - true # HPM4 - - true # HPM5 - - true # HPM6 - - true # HPM7 - - true # HPM8 - - true # HPM9 - - true # HPM10 - - false # HPM11 - - false # HPM12 - - false # HPM13 - - false # HPM14 - - false # HPM15 - - false # HPM16 - - false # HPM17 - - false # HPM18 - - false # HPM19 - - false # HPM20 - - false # HPM21 - - false # HPM22 - - false # HPM23 - - false # HPM24 - - false # HPM25 - - false # HPM26 - - false # HPM27 - - false # HPM28 - - false # HPM29 - - false # HPM30 - - false # HPM31 - - # when true, writing an illegal value to a WLRL CSR field raises an Illegal Instruction exception - # when false, writing an illegal value to a WLRL CSR field is ignored - TRAP_ON_ILLEGAL_WLRL: true - TRAP_ON_UNIMPLEMENTED_INSTRUCTION: true - TRAP_ON_RESERVED_INSTRUCTION: true - TRAP_ON_UNIMPLEMENTED_CSR: true - - # Whether or not a real hardware `time` CSR exists. Implementations can either provide a real - # CSR or trap and emulate access at M-mode. - TIME_CSR_IMPLEMENTED: true - - # Whether or not the `misa` CSR returns zero or a non-zero value. - MISA_CSR_IMPLEMENTED: true - - # when true, *tval is written with the virtual PC of the EBREAK instruction (same information as *epc) - # when false, *tval is written with 0 on an EBREAK instruction - # - # regardless, *tval is always written with a virtual PC when an external breakpoint is generated - REPORT_VA_IN_MTVAL_ON_BREAKPOINT: true - - REPORT_VA_IN_MTVAL_ON_LOAD_MISALIGNED: true - REPORT_VA_IN_MTVAL_ON_STORE_AMO_MISALIGNED: true - REPORT_VA_IN_MTVAL_ON_INSTRUCTION_MISALIGNED: true - REPORT_VA_IN_MTVAL_ON_LOAD_ACCESS_FAULT: true - REPORT_VA_IN_MTVAL_ON_STORE_AMO_ACCESS_FAULT: true - REPORT_VA_IN_MTVAL_ON_INSTRUCTION_ACCESS_FAULT: true - REPORT_VA_IN_MTVAL_ON_LOAD_PAGE_FAULT: true - REPORT_VA_IN_MTVAL_ON_STORE_AMO_PAGE_FAULT: true - REPORT_VA_IN_MTVAL_ON_INSTRUCTION_PAGE_FAULT: true - REPORT_ENCODING_IN_MTVAL_ON_ILLEGAL_INSTRUCTION: true - # REPORT_CAUSE_IN_MTVAL_ON_SOFTWARE_CHECK: true - MTVAL_WIDTH: 64 # must check that this can hold any valid VA if any REPORT_VA* or Sdext, and, if REPORT_ENCODING*, at least [MXLEN, ILEN].min bits - - REPORT_VA_IN_STVAL_ON_BREAKPOINT: true - REPORT_VA_IN_STVAL_ON_LOAD_MISALIGNED: true - REPORT_VA_IN_STVAL_ON_STORE_AMO_MISALIGNED: true - REPORT_VA_IN_STVAL_ON_INSTRUCTION_MISALIGNED: true - REPORT_VA_IN_STVAL_ON_LOAD_ACCESS_FAULT: true - REPORT_VA_IN_STVAL_ON_STORE_AMO_ACCESS_FAULT: true - REPORT_VA_IN_STVAL_ON_INSTRUCTION_ACCESS_FAULT: true - REPORT_VA_IN_STVAL_ON_LOAD_PAGE_FAULT: true - REPORT_VA_IN_STVAL_ON_STORE_AMO_PAGE_FAULT: true - REPORT_VA_IN_STVAL_ON_INSTRUCTION_PAGE_FAULT: true - REPORT_ENCODING_IN_STVAL_ON_ILLEGAL_INSTRUCTION: true - # REPORT_CAUSE_IN_STVAL_ON_SOFTWARE_CHECK: true - STVAL_WIDTH: 64 # must check that this can hold any valid VA, and, if REPORT_ENCODING*, at least [SXLEN, ILEN].min bits - - REPORT_VA_IN_VSTVAL_ON_BREAKPOINT: true - REPORT_VA_IN_VSTVAL_ON_LOAD_MISALIGNED: true - REPORT_VA_IN_VSTVAL_ON_STORE_AMO_MISALIGNED: true - REPORT_VA_IN_VSTVAL_ON_INSTRUCTION_MISALIGNED: true - REPORT_VA_IN_VSTVAL_ON_LOAD_ACCESS_FAULT: true - REPORT_VA_IN_VSTVAL_ON_STORE_AMO_ACCESS_FAULT: true - REPORT_VA_IN_VSTVAL_ON_INSTRUCTION_ACCESS_FAULT: true - REPORT_VA_IN_VSTVAL_ON_LOAD_PAGE_FAULT: true - REPORT_VA_IN_VSTVAL_ON_STORE_AMO_PAGE_FAULT: true - REPORT_VA_IN_VSTVAL_ON_INSTRUCTION_PAGE_FAULT: true - REPORT_ENCODING_IN_VSTVAL_ON_ILLEGAL_INSTRUCTION: true - # REPORT_CAUSE_IN_VSTVAL_ON_SOFTWARE_CHECK: true - # VSTVAL_WIDTH not needed; "vstval is a WARL register that must be able to hold the same set of values that stval can hold" - - # address of the unified discovery configuration data structure - # this address is reported in the mconfigptr CSR - CONFIG_PTR_ADDRESS: 0x1000 - - # number of implemented PMP entries. Can be any value between 0-64, inclusive. - # - # the number of implemented PMP registers must be 0, 16, or 64. - # - # Therefore, whether or not a pmpaddrN or pmpcfgN register exists depends on - # NUM_PMP_ENTRIES as follows: - # |=== - # | NUM_PMP_ENTRIES | pmpaddr<0-15> / pmpcfg<0-3> | pmpaddr<16-63> / pmpcfg<4-15> - # | 0 | N | N - # | 1-16 | Y | N - # | 17-64 | Y | Y - # |=== - # ** pmpcfgN for an odd N never exist when XLEN == 64 - # - # when NUM_PMP_ENTRIES is not exactly 0, 16, or 64, some extant pmp registers, - # and associated pmpNcfg, will be read-only zero (but will not cause an exception). - NUM_PMP_ENTRIES: 14 - - # log2 of the smallest supported PMP region - # generally, for systems with an MMU, should not be smaller than 12, - # as that would preclude caching PMP results in the TLB along with - # virtual memory translations - # - # Note that PMP_GRANULARITY is equal to G+2 (not G) as described in - # the privileged architecture - PMP_GRANULARITY: 12 - - # log2 of the smallest supported PMA region - # generally, for systems with an MMU, should not be smaller than 12, - # as that would preclude caching PMP results in the TLB along with - # virtual memory translations - PMA_GRANULARITY: 12 - - # number of bits in the physical address space - PHYS_ADDR_WIDTH: 56 - - # number of implemented ASID bits - # maximum value is 16 - ASID_WIDTH: 12 - - # when the A extensions is supported, indicates whether or not - # the extension can be disabled in the `misa.A` bit. - MUTABLE_MISA_A: false - - # when the B extensions is supported, indicates whether or not - # the extension can be disabled in the `misa.B` bit. - MUTABLE_MISA_B: false - - # when the C extensions is supported, indicates whether or not - # the extension can be disabled in the `misa.C` bit. - MUTABLE_MISA_C: false - - # when the D extensions is supported, indicates whether or not - # the extension can be disabled in the `misa.D` bit. - MUTABLE_MISA_D: false - - # when the F extensions is supported, indicates whether or not - # the extension can be disabled in the `misa.F` bit. - MUTABLE_MISA_F: false - - # when the H extensions is supported, indicates whether or not - # the extension can be disabled in the `misa.H` bit. - MUTABLE_MISA_H: false - - # when the M extensions is supported, indicates whether or not - # the extension can be disabled in the `misa.M` bit. - MUTABLE_MISA_M: false - - # when the S extensions is supported, indicates whether or not - # the extension can be disabled in the `misa.S` bit. - MUTABLE_MISA_S: false - - # when the U extensions is supported, indicates whether or not - # the extension can be disabled in the `misa.U` bit. - MUTABLE_MISA_U: false - - # when the V extensions is supported, indicates whether or not - # the extension can be disabled in the `misa.V` bit. - MUTABLE_MISA_V: false - - # size of a cache block, in bytes - CACHE_BLOCK_SIZE: 64 - - # number of supported virtualized guest interrupts - # corresponds to the `GEILEN` parameter in the RVI specs - NUM_EXTERNAL_GUEST_INTERRUPTS: 4 - - # Endianess of data in M-mode. Can be one of: - # - # * little: M-mode data is always little endian - # * big: M-mode data is always big endian - # * dynamic: M-mode data can be either little or big endian, depending on the RW CSR field mstatus.MBE - M_MODE_ENDIANESS: little - - # Endianess of data in M-mode. Can be one of: - # - # * little: S-mode data is always little endian - # * big: S-mode data is always big endian - # * dynamic: S-mode data can be either little or big endian, depending on the RW CSR field mstatus.SBE - S_MODE_ENDIANESS: little - - # Endianess of data in M-mode. Can be one of: - # - # * litte: U-mode data is always little endian - # * big: U-mode data is always big endian - # * dynamic: U-mode data can be either little or big endian, depending on the RW CSR field mstatus.UBE - U_MODE_ENDIANESS: little - - # Endianess of data in VU-mode. Can be one of: - # - # * little: VU-mode data is always little endian - # * big: VU-mode data is always big endian - # * dynamic: VU-mode data can be either little or big endian, depending on the RW CSR field vsstatus.UBE - VU_MODE_ENDIANESS: little - - # Endianess of data in VS-mode. Can be one of: - # - # * little: VS-mode data is always little endian - # * big: VS-mode data is always big endian - # * dynamic: VS-mode data can be either little or big endian, depending on the RW CSR field hstatus.VSBE - VS_MODE_ENDIANESS: little - - # XLENs supported in S-mode. Can be one of: - # - # * 32: SXLEN is always 32 - # * 64: SXLEN is always 64 - # * 3264: SXLEN can be changed (via mstatus.SXL) between 32 and 64 - SXLEN: 64 - - # XLENs supported in U-mode. Can be one of: - # - # * 32: SXLEN is always 32 - # * 64: SXLEN is always 64 - # * 3264: SXLEN can be changed (via mstatus.SXL) between 32 and 64 - UXLEN: 64 - - # XLENs supported in VS-mode. Can be one of: - # - # * 32: VSXLEN is always 32 - # * 64: VSXLEN is always 64 - # * 3264: VSXLEN can be changed (via hstatus.VSXL) between 32 and 64 - VSXLEN: 64 - - # XLENs supported in VS-mode. Can be one of: - # - # * 32: VSXLEN is always 32 - # * 64: VSXLEN is always 64 - # * 3264: VSXLEN can be changed (via hstatus.VSXL) between 32 and 64 - VUXLEN: 64 - - # Strategy used to handle reservation sets - # - # * "reserve naturally-aligned 64-byte region": Always reserve the 64-byte block containing the LR/SC address - # * "reserve naturally-aligned 128-byte region": Always reserve the 128-byte block containing the LR/SC address - # * "reserve exactly enough to cover the access": Always reserve exactly the LR/SC access, and no more - # * "custom": Custom behavior, leading to an 'unpredictable' call on any LR/SC - LRSC_RESERVATION_STRATEGY: reserve naturally-aligned 64-byte region - - # whether or not an SC will fail if its VA does not match the VA of the prior LR, - # even if the physical address of the SC and LR are the same - LRSC_FAIL_ON_VA_SYNONYM: false - - # what to do when an LR/SC address is misaligned: - # - # * 'always raise misaligned exception': self-explainitory - # * 'always raise access fault': self-explainitory - # * 'custom': Custom behavior; misaligned LR/SC may sometimes raise a misaligned exception and sometimes raise a access fault. Will lead to an 'unpredictable' call on any misaligned LR/SC access - LRSC_MISALIGNED_BEHAVIOR: always raise misaligned exception - - # whether or not a Store Conditional fails if its physical address and size do not - # exactly match the physical address and size of the last Load Reserved in program order - # (independent of whether or not the SC is in the current reservation set) - LRSC_FAIL_ON_NON_EXACT_LRSC: false - - # Whether writes from M-mode, U-mode, or S-mode to vsatp with an illegal mode setting are - # ignored (as they are with satp), or if they are treated as WARL, leading to undpredictable - # behavior. - IGNORE_INVALID_VSATP_MODE_WRITES_WHEN_V_EQ_ZERO: true - - GSTAGE_MODE_BARE: true - SV32_VSMODE_TRANSLATION: false - SV39_VSMODE_TRANSLATION: true - SV48_VSMODE_TRANSLATION: true - SV57_VSMODE_TRANSLATION: true - SV32X4_TRANSLATION: false - SV39X4_TRANSLATION: true - SV48X4_TRANSLATION: true - SV57X4_TRANSLATION: false - VMID_WIDTH: 8 - SV_MODE_BARE: true - STVEC_MODE_DIRECT: true - STVEC_MODE_VECTORED: true - SATP_MODE_BARE: true - REPORT_GPA_IN_TVAL_ON_LOAD_GUEST_PAGE_FAULT: true - REPORT_GPA_IN_TVAL_ON_STORE_AMO_GUEST_PAGE_FAULT: true - REPORT_GPA_IN_TVAL_ON_INSTRUCTION_GUEST_PAGE_FAULT: true - REPORT_GPA_IN_TVAL_ON_INTERMEDIATE_GUEST_PAGE_FAULT: true - TINST_VALUE_ON_FINAL_LOAD_GUEST_PAGE_FAULT: "always transformed standard instruction" - TINST_VALUE_ON_FINAL_STORE_AMO_GUEST_PAGE_FAULT: "always transformed standard instruction" - TINST_VALUE_ON_FINAL_INSTRUCTION_GUEST_PAGE_FAULT: "always zero" - TINST_VALUE_ON_INSTRUCTION_ADDRESS_MISALIGNED: "always zero" - TINST_VALUE_ON_BREAKPOINT: "always zero" - TINST_VALUE_ON_VIRTUAL_INSTRUCTION: "always zero" - TINST_VALUE_ON_LOAD_ADDRESS_MISALIGNED: "always zero" - TINST_VALUE_ON_LOAD_ACCESS_FAULT: "always zero" - TINST_VALUE_ON_STORE_AMO_ADDRESS_MISALIGNED: "always zero" - TINST_VALUE_ON_STORE_AMO_ACCESS_FAULT: "always zero" - TINST_VALUE_ON_UCALL: "always zero" - TINST_VALUE_ON_SCALL: "always zero" - TINST_VALUE_ON_MCALL: "always zero" - TINST_VALUE_ON_VSCALL: "always zero" - TINST_VALUE_ON_LOAD_PAGE_FAULT: "always zero" - TINST_VALUE_ON_STORE_AMO_PAGE_FAULT: "always zero" - MTVEC_MODES: [0, 1] - MSTATUS_FS_LEGAL_VALUES: [0, 1, 2, 3] - MSTATUS_FS_WRITEABLE: true - MSTATUS_TVM_IMPLEMENTED: true - HW_MSTATUS_FS_DIRTY_UPDATE: precise - MSTATUS_VS_WRITEABLE: true - MSTATUS_VS_LEGAL_VALUES: [0, 1, 2, 3] - HW_MSTATUS_VS_DIRTY_UPDATE: precise - FORCE_UPGRADE_CBO_INVAL_TO_FLUSH: true diff --git a/cfgs/rv32/cfg.yaml b/cfgs/rv32/cfg.yaml new file mode 100644 index 000000000..ca61792af --- /dev/null +++ b/cfgs/rv32/cfg.yaml @@ -0,0 +1,14 @@ +# yaml-language-server: $schema=../../schemas/config_schema.json +--- +$schema: config_schema.json# +kind: architecture configuration +type: partially configured +name: rv32 +description: A generic RV32 system; only MXLEN is known +params: + XLEN: 32 +mandatory_extensions: + - name: "I" + version: ">= 0" + - name: "Sm" + version: ">= 0" diff --git a/cfgs/rv64/cfg.yaml b/cfgs/rv64/cfg.yaml new file mode 100644 index 000000000..f75b8da9e --- /dev/null +++ b/cfgs/rv64/cfg.yaml @@ -0,0 +1,14 @@ +# yaml-language-server: $schema=../../schemas/config_schema.json +--- +$schema: config_schema.json# +kind: architecture configuration +type: partially configured +name: rv64 +description: A generic RV32 system; only MXLEN is known +params: + XLEN: 64 +mandatory_extensions: + - name: "I" + version: ">= 0" + - name: "Sm" + version: ">= 0" diff --git a/container.def b/container.def index 544c9446c..3bb2c1104 100644 --- a/container.def +++ b/container.def @@ -28,6 +28,8 @@ From: ubuntu:24.04 apt-get install -y --no-install-recommends ditaa apt-get install -y --no-install-recommends libyaml-dev + apt-get install -y --no-install-recommends libyaml-dev + # cleanup apt-get clean autoclean apt-get autoremove -y diff --git a/do b/do index 2552807af..57ae6f102 100755 --- a/do +++ b/do @@ -7,5 +7,9 @@ else source $ROOT/bin/setup fi -# really long way of invoking rake, but renamed to 'do' -$BUNDLE exec --gemfile $ROOT/Gemfile ruby -r rake -e "Rake.application.init('do');Rake.application.load_rakefile;Rake.application.top_level" -- $@ +if [ "$1" == "clean" ]; then + ${ROOT}/bin/clean +else + # really long way of invoking rake, but renamed to 'do' + $BUNDLE exec --gemfile $ROOT/Gemfile ruby -r rake -e "Rake.application.init('do');Rake.application.load_rakefile;Rake.application.top_level" -- $@ +fi diff --git a/docs/index.html b/docs/index.html index 9786ceb70..fc169bf10 100644 --- a/docs/index.html +++ b/docs/index.html @@ -1,6 +1,5 @@ - + -

Arch specification schemas

    @@ -12,5 +11,4 @@

    Ruby documentation

  • lib
- diff --git a/ext/riscv-opcodes b/ext/riscv-opcodes index 9226b0d09..5ce8977a5 160000 --- a/ext/riscv-opcodes +++ b/ext/riscv-opcodes @@ -1 +1 @@ -Subproject commit 9226b0d091b0d2ea9ccad6f7f8ca1283a3b15e88 +Subproject commit 5ce8977a5961a6bbfc1638e6676e60489665d882 diff --git a/lib/arch_def.rb b/lib/arch_def.rb deleted file mode 100644 index 3c6821531..000000000 --- a/lib/arch_def.rb +++ /dev/null @@ -1,1155 +0,0 @@ -# frozen_string_literal: true - -# Many classes have an "arch_def" member which is an ArchDef (not ArchDefObject) class. -# The "arch_def" member contains the "database" of RISC-V standards including extensions, instructions, -# CSRs, Profiles, and Certificates. -# -# The arch_def member has methods such as: -# extensions() Array of all extensions known to the database (even if not implemented). -# extension(name) Extension object for "name" and nil if none. -# parameters() Array of all parameters defined in the architecture -# param(name) ExtensionParameter object for "name" and nil if none. -# csrs() Array of all CSRs defined by RISC-V, whether or not they are implemented -# csr(name) Csr object for "name" and nil if none. -# instructions() Array of all instructions, whether or not they are implemented -# inst(name) Instruction object for "name" and nil if none. -# profile_classes Array of all known profile classes. -# profile_class(class_name) ProfileClass object for "class_name" and nil if none. -# profile_releases Array of all profile releases for all profile classes -# profile_release(release_name) ProfileRelease object for "release_name" and nil if none. -# profiles Array of all profiles in all releases in all classes -# profile(name) Profile object for profile "name" and nil if none. -# cert_classes Array of all known certificate classes -# cert_class(name) CertClass object for "name" and nil if none. -# cert_models Array of all known certificate models across all classes. -# cert_model(name) CertModel object for "name" and nil if none. - -require "forwardable" -require "ruby-prof" - -require_relative "validate" -require_relative "idl" -require_relative "idl/passes/find_return_values" -require_relative "idl/passes/gen_adoc" -require_relative "idl/passes/prune" -require_relative "idl/passes/reachable_functions" -require_relative "idl/passes/reachable_functions_unevaluated" -require_relative "idl/passes/reachable_exceptions" -require_relative "arch_obj_models/manual" -require_relative "arch_obj_models/portfolio" -require_relative "arch_obj_models/profile" -require_relative "arch_obj_models/csr_field" -require_relative "arch_obj_models/csr" -require_relative "arch_obj_models/instruction" -require_relative "arch_obj_models/extension" -require_relative "arch_obj_models/certificate" -require_relative "template_helpers" - -include TemplateHelpers - -class ArchDef - # @return [Idl::Compiler] The IDL compiler - attr_reader :idl_compiler - - # @return [Idl::IsaAst] Abstract syntax tree of global scope - attr_reader :global_ast - - # @return [String] Name of this definition. Special names are: - # * '_' - The generic architecture, with no configuration settings. - # * '_32' - A generic RV32 architecture, with only one parameter set (XLEN == 32) - # * '_64' - A generic RV64 architecture, with only one parameter set (XLEN == 64) - attr_reader :name - - # @return [Hash] A hash mapping parameter name to value for any parameter that has been configured with a value. May be empty. - attr_reader :param_values - - # @return [Integer] 32 or 64, the XLEN in M-mode - # @return [nil] if the XLEN in M-mode is not configured - attr_reader :mxlen - - # hash for Hash lookup - def hash = @name_sym.hash - - # @return [Idl::SymbolTable] Symbol table with global scope - # @return [nil] if the architecture is not configured (use symtab_32 or symtab_64) - def symtab - raise NotImplementedError, "Un-configured ArchDefs have no symbol table" if @symtab.nil? - - @symtab - end - - def fully_configured? = @arch_def["type"] == "fully configured" - def partially_configured? = @arch_def["type"] == "partially configured" - def unconfigured? = @arch_def["type"] == "unconfigured" - def configured? = @arch_def["type"] != "unconfigured" - def type = @arch_def["type"] - - # Initialize a new configured architecture definition - # - # @param config_name [#to_s] The name of a configuration, which must correspond - # to a folder under $root/cfgs - def initialize(config_name, arch_def_path, overlay_path: nil) - @name = config_name.to_s.freeze - @name_sym = @name.to_sym.freeze - - @idl_compiler = Idl::Compiler.new(self) - - validator = Validator.instance - begin - validator.validate_str(arch_def_path.read, type: :arch) - rescue Validator::SchemaValidationError => e - warn "While parsing unified architecture definition at #{arch_def_path}" - raise e - end - - @arch_def = YAML.load_file(arch_def_path, permitted_classes: [Date]).freeze - @param_values = (@arch_def.key?("params") ? @arch_def["params"] : {}).freeze - @mxlen = @arch_def.dig("params", "XLEN") # might be nil - - unless @mxlen.nil? - # need at least XLEN specified to have a full architecture definition - # to populate the symbol table. - # - # if this is the fully generic config ("_"), then you need to use - # either symtab_32 or symtab_64 - @symtab = Idl::SymbolTable.new(self) - custom_globals_path = overlay_path.nil? ? Pathname.new("/does/not/exist") : overlay_path / "isa" / "globals.isa" - idl_path = File.exist?(custom_globals_path) ? custom_globals_path : $root / "arch" / "isa" / "globals.isa" - @global_ast = @idl_compiler.compile_file( - idl_path - ) - @global_ast.add_global_symbols(@symtab) - @symtab.deep_freeze - @global_ast.freeze_tree(@symtab) - @mxlen.freeze - else - # parse globals - @global_ast = @idl_compiler.compile_file( - $root / "arch" / "isa" / "globals.isa" - ) - @global_ast.add_global_symbols(symtab_32) - symtab_32.deep_freeze - @global_ast.freeze_tree(symtab_32) - - # do it again for rv64, but we don't need the ast this time - global_ast_64 = @idl_compiler.compile_file( - $root / "arch" / "isa" / "globals.isa" - ) - global_ast_64.add_global_symbols(symtab_64) - symtab_64.deep_freeze - global_ast_64.freeze_tree(symtab_64) - end - end - - # type check all IDL, including globals, instruction ops, and CSR functions - # - # @param show_progress [Boolean] whether to show progress bars - # @param io [IO] where to write progress bars - # @return [void] - def type_check(show_progress: true, io: $stdout) - io.puts "Type checking IDL code for #{name}..." - progressbar = - if show_progress - ProgressBar.create(title: "Instructions", total: instructions.size) - end - - instructions.each do |inst| - progressbar.increment if show_progress - if @mxlen == 32 - inst.type_checked_operation_ast(@idl_compiler, @symtab, 32) if inst.rv32? - elsif @mxlen == 64 - inst.type_checked_operation_ast(@idl_compiler, @symtab, 64) if inst.rv64? - inst.type_checked_operation_ast(@idl_compiler, @symtab, 32) if possible_xlens.include?(32) && inst.rv32? - end - end - - progressbar = - if show_progress - ProgressBar.create(title: "CSRs", total: csrs.size) - end - - csrs.each do |csr| - progressbar.increment if show_progress - if csr.has_custom_sw_read? - if (possible_xlens.include?(32) && csr.defined_in_base32?) || (possible_xlens.include?(64) && csr.defined_in_base64?) - csr.type_checked_sw_read_ast(@symtab) - end - end - csr.fields.each do |field| - unless field.type_ast(@symtab).nil? - if ((possible_xlens.include?(32) && csr.defined_in_base32? && field.defined_in_base32?) || - (possible_xlens.include?(64) && csr.defined_in_base64? && field.defined_in_base64?)) - field.type_checked_type_ast(@symtab) - end - end - unless field.reset_value_ast(@symtab).nil? - if ((possible_xlens.include?(32) && csr.defined_in_base32? && field.defined_in_base32?) || - (possible_xlens.include?(64) && csr.defined_in_base64? && field.defined_in_base64?)) - field.type_checked_reset_value_ast(@symtab) if csr.defined_in_base32? && field.defined_in_base32? - end - end - unless field.sw_write_ast(@symtab).nil? - field.type_checked_sw_write_ast(@symtab, 32) if possible_xlens.include?(32) && csr.defined_in_base32? && field.defined_in_base32? - field.type_checked_sw_write_ast(@symtab, 64) if possible_xlens.include?(64) && csr.defined_in_base64? && field.defined_in_base64? - end - end - end - - progressbar = - if show_progress - ProgressBar.create(title: "Functions", total: functions.size) - end - functions.each do |func| - progressbar.increment if show_progress - func.type_check(@symtab) - end - - puts "done" if show_progress - end - - # Returns whether or not it may be possible to switch XLEN given this definition. - # - # There are three cases when this will return true: - # 1. A mode (e.g., U) is known to be implemented, and the CSR bit that controls XLEN in that mode is known to be writeable. - # 2. A mode is known to be implemented, but the writability of the CSR bit that controls XLEN in that mode is not known. - # 3. It is not known if the mode is implemented. - # - # - # @return [Boolean] true if this configuration might execute in multiple xlen environments - # (e.g., that in some mode the effective xlen can be either 32 or 64, depending on CSR values) - def multi_xlen? - return true if @mxlen.nil? - - ["S", "U", "VS", "VU"].any? { |mode| multi_xlen_in_mode?(mode) } - end - - # Returns whether or not it may be possible to switch XLEN in +mode+ given this definition. - # - # There are three cases when this will return true: - # 1. +mode+ (e.g., U) is known to be implemented, and the CSR bit that controls XLEN in +mode+ is known to be writeable. - # 2. +mode+ is known to be implemented, but the writability of the CSR bit that controls XLEN in +mode+ is not known. - # 3. It is not known if +mode+ is implemented. - # - # Will return false if +mode+ is not possible (e.g., because U is a prohibited extension) - # - # @param mode [String] mode to check. One of "M", "S", "U", "VS", "VU" - # @return [Boolean] true if this configuration might execute in multiple xlen environments in +mode+ - # (e.g., that in some mode the effective xlen can be either 32 or 64, depending on CSR values) - def multi_xlen_in_mode?(mode) - return false if mxlen == 32 - - case mode - when "M" - mxlen.nil? - when "S" - return true if unconfigured? - - if fully_configured? - ext?(:S) && (@param_values["SXLEN"] == 3264) - elsif partially_configured? - return false if prohibited_ext?(:S) - - return true unless ext?(:S) # if S is not known to be implemented, we can't say anything about it - - return true unless @param_values.key?("SXLEN") - - @param_values["SXLEN"] == 3264 - else - raise "Unexpected configuration state" - end - when "U" - return false if prohibited_ext?(:U) - - return true if unconfigured? - - if fully_configured? - ext?(:U) && (@param_values["UXLEN"] == 3264) - elsif partially_configured? - return true unless ext?(:U) # if U is not known to be implemented, we can't say anything about it - - return true unless @param_values.key?("UXLEN") - - @param_values["UXLEN"] == 3264 - else - raise "Unexpected configuration state" - end - when "VS" - return false if prohibited_ext?(:H) - - return true if unconfigured? - - if fully_configured? - ext?(:H) && (@param_values["VSXLEN"] == 3264) - elsif partially_configured? - return true unless ext?(:H) # if H is not known to be implemented, we can't say anything about it - - return true unless @param_values.key?("VSXLEN") - - @param_values["VSXLEN"] == 3264 - else - raise "Unexpected configuration state" - end - when "VU" - return false if prohibited_ext?(:H) - - return true if unconfigured? - - if fully_configured? - ext?(:H) && (@param_values["VUXLEN"] == 3264) - elsif partially_configured? - return true unless ext?(:H) # if H is not known to be implemented, we can't say anything about it - - return true unless @param_values.key?("VUXLEN") - - @param_values["VUXLEN"] == 3264 - else - raise "Unexpected configuration state" - end - else - raise ArgumentError, "Bad mode" - end - end - - # @return [Array] List of possible XLENs in any mode for this config - def possible_xlens = multi_xlen? ? [32, 64] : [mxlen] - - # @return [Array] List of all available parameters with known values for the config - def params_with_value - return @params_with_value unless @params_with_value.nil? - - @params_with_value = [] - extensions.each do |ext_version| - ext = extension(ext_version.name) - ext.params.each do |ext_param| - if param_values.key?(ext_param.name) - @params_with_value << ExtensionParameterWithValue.new( - ext_param, - param_values[ext_param.name] - ) - end - end - end - @params_with_value - end - - # @return [Array] List of all available parameters without known values for the config - def params_without_value - return @params_without_value unless @params_without_value.nil? - - @params_without_value = [] - extensions.each do |ext_version| - ext = extension(ext_version.name) - ext.params.each do |ext_param| - unless param_values.key?(ext_param.name) - @params_without_value << ext_param - end - end - end - @params_without_value - end - - # Returns a string representation of the object, suitable for debugging. - # @return [String] A string representation of the object. - def inspect = "ArchDef##{name}" - - # @return [Array] List of all extensions, even those that are't implemented - def extensions - return @extensions unless @extensions.nil? - - @extensions = [] - @arch_def["extensions"].each do |ext_data| - @extensions << Extension.new(ext_data, self) - end - @extensions - end - - # may be overridden by subclass - # @return [Array] List of all extensions known to be implemented in this architecture - def implemented_extensions - raise "implemented_extensions is only valid for a fully configured defintion" unless fully_configured? - - return @implemented_extensions unless @implemented_extensions.nil? - - @implemented_extensions = [] - if @arch_def.key?("implemented_extensions") - @arch_def["implemented_extensions"].each do |e| - @implemented_extensions << ExtensionVersion.new(e["name"], e["version"], self) - end - end - @implemented_extensions - end - - # @return [Array] List of extensions that are explicitly required by an arch def - def mandatory_extensions - raise "mandatory_extensions is only valid for a partially configured defintion" unless partially_configured? - - return @mandatory_extensions unless @mandatory_extensions.nil? - - @mandatory_extensions = [] - if @arch_def.key?("mandatory_extensions") - @arch_def["mandatory_extensions"].each do |e| - @mandatory_extensions << ExtensionRequirement.new(e["name"], e["version"], presence: "mandatory") - end - end - @mandatory_extensions - end - - # @return [Array] List of extensions that are explicitly prohibited by an arch def - def prohibited_extensions - return @prohibited_extensions unless @prohibited_extensions.nil? - - @prohibited_extensions = [] - if @arch_def.key?("prohibited_extensions") - @arch_def["prohibited_extensions"].each do |e| - if e.is_a?(String) - @prohibited_extensions << ExtensionRequirement.new(e, nil) - else - @prohibited_extensions << ExtensionRequirement.new(e["name"], e["requirements"], presence: "prohibited") - end - end - end - @prohibited_extensions - end - - def prohibited_ext?(ext_name) - prohibited_extensions.any? { |ext_req| ext_req.name == ext_name.to_s } - end - - # @return [Hash] Hash of all extensions, even those that aren't implemented, indexed by extension name - def extension_hash - return @extension_hash unless @extension_hash.nil? - - @extension_hash = {} - extensions.each do |ext| - @extension_hash[ext.name] = ext - end - @extension_hash - end - - # @param name [#to_s] Extension name - # @return [Extension] Extension named `name` - # @return [nil] if no extension `name` exists - def extension(name) - extension_hash[name.to_s] - end - - # @overload ext?(ext_name) - # @param ext_name [#to_s] Extension name (case sensitive) - # @return [Boolean] True if the extension `name` is implemented - # @overload ext?(ext_name, ext_version_requirements) - # @param ext_name [#to_s] Extension name (case sensitive) - # @param ext_version_requirements [Number,String,Array] Extension version requirements, taking the same inputs as Gem::Requirement - # @see https://docs.ruby-lang.org/en/3.0/Gem/Requirement.html#method-c-new Gem::Requirement#new - # @return [Boolean] True if the extension `name` meeting `ext_version_requirements` is implemented - # @example Checking extension presence with a version requirement - # arch_def.ext?(:S, ">= 1.12") - # @example Checking extension presence with multiple version requirements - # arch_def.ext?(:S, ">= 1.12", "< 1.15") - # @example Checking extension precsence with a precise version requirement - # arch_def.ext?(:S, 1.12) - def ext?(ext_name, *ext_version_requirements) - @ext_cache ||= {} - cached_result = @ext_cache[[ext_name, ext_version_requirements]] - return cached_result unless cached_result.nil? - - result = - if fully_configured? - implemented_extensions.any? do |e| - if ext_version_requirements.empty? - e.name == ext_name.to_s - else - requirement = Gem::Requirement.new(ext_version_requirements) - (e.name == ext_name.to_s) && requirement.satisfied_by?(e.version) - end - end - else - raise "unexpected type" unless partially_configured? - - mandatory_extensions.any? do |e| - if ext_version_requirements.empty? - e.name == ext_name.to_s - else - requirement = Gem::Requirement.new(ext_version_requirements) - e.satisfying_versions.all? do |ext_ver| - (e.name == ext_name.to_s) && requirement.satisfied_by?(exrt_ver.version) - end - end - end - end - @ext_cache[[ext_name, ext_version_requirements]] = result - end - - # @return [Array] Array of all extensions that are prohibited because they are excluded by an implemented extension - def conflicting_extensions - extensions.map(&:conflicts).flatten - end - - # @return [Boolean] whether or not ext_name is prohibited because it is excluded by an implemented extension - def conflicting_ext?(ext_name) - prohibited_extensions.include? { |ext_req| ext_req.name == ext_name } - end - - # @return [Array] Alphabetical list of all parameters defined in the architecture - def params - return @params unless @params.nil? - - @params = extensions.map(&:params).flatten.uniq(&:name).sort_by!(&:name) - end - - # @return [Hash] Hash of all extension parameters defined in the architecture - def params_hash - return @params_hash unless @params_hash.nil? - - @params_hash = {} - params.each do |param| - @params_hash[param.name] = param - end - @param_hash - end - - # @return [ExtensionParameter] Parameter named +name+ - # @return [nil] if there is no parameter named +name+ - def param(name) - params_hash[name] - end - - # @return [Array] List of all CSRs defined by RISC-V, whether or not they are implemented - def csrs - return @csrs unless @csrs.nil? - - @csrs = @arch_def["csrs"].map do |csr_data| - Csr.new(csr_data) - end - end - - # @return [Array] List of all known CSRs, even those not implemented by - # this config - def all_known_csr_names - @arch_def["csrs"].map { |csr| csr[0] } - end - - # @return [Hash] All csrs, even unimplemented ones, indexed by CSR name - def csr_hash - return @csr_hash unless @csr_hash.nil? - - @csr_hash = {} - csrs.each do |csr| - @csr_hash[csr.name] = csr - end - @csr_hash - end - - # @param csr_name [#to_s] CSR name - # @return [Csr,nil] a specific csr, or nil if it doesn't exist - def csr(csr_name) - csr_hash[csr_name] - end - - # @return [Array] List of all instructions, whether or not they are implemented - def instructions - return @instructions unless @instructions.nil? - - @instructions = @arch_def["instructions"].map do |inst_data| - Instruction.new(inst_data, self) - end - - @instructions - end - - # @return [Hash] All instructions, indexed by name - def instruction_hash - return @instruction_hash unless @instruction_hash.nil? - - @instruction_hash = {} - instructions.each do |inst| - @instruction_hash[inst.name] = inst - end - @instruction_hash - end - - # @param inst_name [#to_s] Instruction name - # @return [Instruction,nil] An instruction named 'inst_name', or nil if it doesn't exist - def inst(inst_name) - instruction_hash[inst_name.to_s] - end - alias instruction inst - - # @return [Array] List of all functions defined by the architecture - def functions - return @functions unless @functions.nil? - - @functions = @global_ast.functions - end - - # @return [Hash] Function hash of name => FunctionBodyAst - def function_hash - return @function_hash unless @function_hash.nil? - - @function_hash = {} - functions.each do |func| - @function_hash[func.name] = func - end - - @function_hash - end - - # @param name [String] A function name - # @return [Idl::FunctionBodyAst] A function named +name+ - # @return [nil] if no function named +name+ is found - def function(name) - function_hash[name] - end - - # @return [Array] List of all manuals defined by the architecture - def manuals - return @manuals unless @manuals.nil? - - @manuals = [] - @arch_def["manuals"].each_value do |manual_data| - @manuals << Manual.new(manual_data, self) - end - @manuals - end - - # @return [Hash] All manuals, indexed by name - def manuals_hash - return @manuals_hash unless @manuals_hash.nil? - - @manuals_hash = {} - manuals.each do |manual| - @manuals_hash[manual.name] = manual - end - @manuals_hash - end - - # @return [Manual,nil] A manual named +name+, or nil if it doesn't exist - def manual(name) = manuals_hash[name] - - # @return [Array] All known profile classes (e.g. RVA) - def profile_classes - return @profile_classes unless @profile_classes.nil? - - @profile_classes = [] - @arch_def["profile_classes"].each_value do |pc_data| - @profile_classes << ProfileClass.new(pc_data, self) - end - @profile_classes - end - - # @return [Hash] Profile classes, indexed by profile class name - def profile_classes_hash - return @profile_classes_hash unless @profile_classes_hash.nil? - - @profile_classes_hash = {} - profile_classes.each do |pc| - @profile_classes_hash[pc.name] = pc - end - @profile_classes_hash - end - - # @return [ProfileClass] The profile class named +name+ - # @return [nil] if the profile class does not exist - def profile_class(profile_class_name) = profile_classes_hash[profile_class_name] - - # @return [ProfileRelease] List of all profile releases (e.g. RVA20, RVA22) for all profile classes. - def profile_releases - return @profile_releases unless @profile_releases.nil? - - @profile_releases = [] - @arch_def["profile_releases"].each_value do |pr_data| - raise ArgumentError, "Expecting pr_data to be a hash" unless pr_data.is_a?(Hash) - - profile_release = ProfileRelease.new(pr_data, self) - raise ArgumentError, "ProfileRelease constructor returned nil" if profile_release.nil? - - @profile_releases << profile_release - end - @profile_releases - end - - # @return [Hash], indexed by profile release name - def profile_releases_hash - return @profile_releases_hash unless @profile_releases_hash.nil? - - @profile_releases_hash = {} - profile_releases.each do |profile_release| - @profile_releases_hash[profile_release.name] = profile_release - end - @profile_releases_hash - end - - # @return [ProfileRelease] The profile release named +profile_release_name+ - # @return [nil] if the profile release does not exist - def profile_release(profile_release_name) = profile_releases_hash[profile_release_name] - - # @return [Profile] List of all defined profiles in all releases in all classes - def profiles - return @profiles unless @profiles.nil? - - @profiles = [] - @arch_def["profile_releases"].each_value do |pr_data| - raise ArgumentError, "Expecting pr_data to be a hash" unless pr_data.is_a?(Hash) - - pr_data["profiles"].each do |profile_name, profile_data| - profile_data["name"] = profile_name - profile = Profile.new(profile_data, self) - raise ArgumentError, "Profile constructor returned nil" if profile.nil? - - @profiles << profile - end - end - @profiles - end - - # @return [Hash] Profiles, indexed by profile name - def profiles_hash - return @profiles_hash unless @profiles_hash.nil? - - @profiles_hash = {} - profiles.each do |profile| - @profiles_hash[profile.name] = profile - end - @profiles_hash - end - - # @return [Profile] The profile named +name+ - # @return [nil] if the profile does not exist - def profile(name) = profiles_hash[name] - - def cert_classes - return @cert_classes unless @cert_classes.nil? - - @cert_classes = [] - @arch_def["certificate_classes"].each do |cc_data| - @cert_classes << CertClass.new(cc_data, self) - end - @cert_classes - end - - def cert_classes_hash - return @cert_classes_hash unless @cert_classes_hash.nil? - - @cert_classes_hash = {} - cert_classes.each do |cc| - @cert_classes_hash[cc.name] = cc - end - @cert_classes_hash - end - - # @return [CertClass] The certificate class named +name+ - # @return [nil] if the certificate class does not exist - def cert_class(name) = cert_classes_hash[name] - - # @return [CertModel] List of all defined certificate models across all certificate classes - def cert_models - return @cert_models unless @cert_models.nil? - - @cert_models = [] - @arch_def["certificate_models"].each do |cm_data| - @cert_models << CertModel.new(cm_data, self) - end - @cert_models - end - - def cert_models_hash - return @cert_models_hash unless @cert_models_hash.nil? - - @cert_models_hash = {} - cert_models.each do |cert_model| - @cert_models_hash[cert_model.name] = cert_model - end - @cert_models_hash - end - - # @return [CertModel] The CertModel named +name+ - # @return [nil] if the CertModel does not exist - def cert_model(name) = cert_models_hash[name] - - # @return [Array] All exception codes defined by RISC-V - def exception_codes - return @exception_codes unless @exception_codes.nil? - - @exception_codes = - extensions.reduce([]) do |list, ext_version| - ecodes = extension(ext_version.name)["exception_codes"] - next list if ecodes.nil? - - ecodes.each do |ecode| - # double check that all the codes are unique - raise "Duplicate exception code" if list.any? { |e| e.num == ecode["num"] || e.name == ecode["name"] || e.var == ecode["var"] } - - list << ExceptionCode.new(ecode["name"], ecode["var"], ecode["num"], self) - end - list - end - end - - # @return [Array] All exception codes known to be implemented - def implemented_exception_codes - return @implemented_exception_codes unless @implemented_exception_codes.nil? - - @implemented_exception_codes = - implemented_extensions.reduce([]) do |list, ext_version| - ecodes = extension(ext_version.name)["exception_codes"] - next list if ecodes.nil? - - ecodes.each do |ecode| - # double check that all the codes are unique - raise "Duplicate exception code" if list.any? { |e| e.num == ecode["num"] || e.name == ecode["name"] || e.var == ecode["var"] } - - unless ecode.dig("when", "version").nil? - # check version - next unless ext?(ext_version.name.to_sym, ecode["when"]["version"]) - end - list << ExceptionCode.new(ecode["name"], ecode["var"], ecode["num"], self) - end - list - end - end - - # @return [Array] All interrupt codes defined by extensions - def interrupt_codes - return @interrupt_codes unless @interrupt_codes.nil? - - @interupt_codes = - extensions.reduce([]) do |list, ext_version| - icodes = extension(ext_version.name)["interrupt_codes"] - next list if icodes.nil? - - icodes.each do |icode| - # double check that all the codes are unique - if list.any? { |i| i.num == icode["num"] || i.name == icode["name"] || i.var == icode["var"] } - raise "Duplicate interrupt code" - end - - list << InterruptCode.new(icode["name"], icode["var"], icode["num"], self) - end - list - end - end - - # @return [Array] All interrupt codes known to be implemented - def implemented_interrupt_codes - return @implemented_interrupt_codes unless @implemented_interrupt_codes.nil? - - @implemented_interupt_codes = - implemented_extensions.reduce([]) do |list, ext_version| - icodes = extension(ext_version.name)["interrupt_codes"] - next list if icodes.nil? - - icodes.each do |icode| - # double check that all the codes are unique - raise "Duplicate interrupt code" if list.any? { |i| i.num == icode["num"] || i.name == icode["name"] || i.var == icode["var"] } - - unless ecode.dig("when", "version").nil? - # check version - next unless ext?(ext_version.name.to_sym, ecode["when"]["version"]) - end - list << InterruptCode.new(icode["name"], icode["var"], icode["num"], self) - end - list - end - end - - # @return [Hash] The raw architecture defintion data structure - def data - @arch_def - end - - # given a `$ref` target, return the Ruby object - # - # @params uri [String] JSON Reference pointer - # @return [Object] The pointed-to object - def ref(uri) - raise ArgumentError, "JSON Reference must contain one '#'" unless uri.count("#") == 1 - - file_path, obj_path = uri.split("#") - obj = - case file_path - when /^certificate_class.*/ - cert_class_name = File.basename(file_path, ".yaml") - cert_class(cert_class_name) - when /^certificate_model.*/ - cert_mode_name = File.basename(file_path, ".yaml") - cert_model(cert_model_name) - when /^csr.*/ - csr_name = File.basename(file_path, ".yaml") - csr(csr_name) - when /^ext.*/ - ext_name = File.basename(file_path, ".yaml") - extension(ext_name) - when /^inst.*/ - inst_name = File.basename(file_path, ".yaml") - instruction(inst_name) - when /^manual.*/ - manual_name = File.basename(file_path, ".yaml") - manual(manual_name) - when /^profile_class.*/ - profile_class_name = File.basename(file_path, ".yaml") - profile_class(profile_class_name) - when /^profile_release.*/ - profile_release_name = File.basename(file_path, ".yaml") - profile_release(profile_release_name) - else - raise "Unhandled ref object: #{file_path}" - end - - if obj_path.nil? - obj - else - parts = obj_path.split("/") - parts.each do |part| - raise "Error in $ref. There is no method '#{part}' for a #{obj.class.name}" unless obj.respond_to?(part.to_sym) - - obj = obj.send(part) - end - obj - end - end - - # @return [Array] List of all implemented CSRs - def implemented_csrs - return @implemented_csrs unless @implemented_csrs.nil? - - @implemented_csrs = - if @arch_def.key?("implemented_csrs") - csrs.select { |c| @arch_def["implemented_csrs"].include?(c.name) } - else - [] - end - end - - # @return [Hash] Implemented csrs, indexed by CSR name - def implemented_csr_hash - return @implemented_csr_hash unless @implemented_csr_hash.nil? - - @implemented_csr_hash = {} - implemented_csrs.each do |csr| - @implemented_csr_hash[csr.name] = csr - end - @implemented_csr_hash - end - - # @param csr_name [#to_s] CSR name - # @return [Csr,nil] a specific csr, or nil if it doesn't exist or isn't implemented - def implemented_csr(csr_name) - implemented_csr_hash[csr_name] - end - - # @return [Array] List of all implemented instructions - def implemented_instructions - return @implemented_instructions unless @implemented_instructions.nil? - - @implemented_instructions = - if @arch_def.key?("implemented_instructions") - @arch_def["implemented_instructions"].map do |inst_name| - instruction_hash[inst_name] - end - else - [] - end - end - - - # @return [Array] List of all reachable IDL functions for the config - def implemented_functions - return @implemented_functions unless @implemented_functions.nil? - - @implemented_functions = [] - - puts " Finding all reachable functions from instruction operations" - - implemented_instructions.each do |inst| - @implemented_functions << - if inst.base.nil? - if multi_xlen? - (inst.reachable_functions(symtab, 32) + - inst.reachable_functions(symtab, 64)) - else - inst.reachable_functions(symtab, mxlen) - end - else - inst.reachable_functions(symtab, inst.base) - end - end - raise "?" unless @implemented_functions.is_a?(Array) - @implemented_functions = @implemented_functions.flatten - @implemented_functions.uniq!(&:name) - - puts " Finding all reachable functions from CSR operations" - - implemented_csrs.each do |csr| - csr_funcs = csr.reachable_functions(self) - csr_funcs.each do |f| - @implemented_functions << f unless @implemented_functions.any? { |i| i.name == f.name } - end - end - - @implemented_functions - end - - # given an adoc string, find names of CSR/Instruction/Extension enclosed in `monospace` - # and replace them with links to the relevant object page - # - # @param adoc [String] Asciidoc source - # @return [String] Asciidoc source, with link placeholders - def find_replace_links(adoc) - adoc.gsub(/`([\w.]+)`/) do |match| - name = Regexp.last_match(1) - csr_name, field_name = name.split(".") - csr = csr(csr_name) - if !field_name.nil? && !csr.nil? && csr.field?(field_name) - "%%LINK%csr_field;#{csr_name}.#{field_name};#{csr_name}.#{field_name}%%" - elsif !csr.nil? - "%%LINK%csr;#{csr_name};#{csr_name}%%" - elsif inst(name) - "%%LINK%inst;#{name};#{name}%%" - elsif extension(name) - "%%LINK%ext;#{name};#{name}%%" - else - match - end - end - end - - # Returns an environment hash suitable for use with ERb templates. - # - # This method returns a hash containing the architecture definition and other - # relevant data that can be used to generate ERb templates. - # - # @return [Hash] An environment hash suitable for use with ERb templates. - def erb_env - return @env unless @env.nil? - - @env = Class.new - @env.instance_variable_set(:@cfg, @cfg) - @env.instance_variable_set(:@params, @params) - @env.instance_variable_set(:@arch_gen, self) - - # add each parameter, either as a method (lowercase) or constant (uppercase) - params_with_value.each do |param| - @env.const_set(param.name, param.value) unless @env.const_defined?(param.name) - end - - params_without_value.each do |param| - @env.const_set(param.name, :unknown) unless @env.const_defined?(param.name) - end - - @env.instance_exec do - # method to check if a given extension (with an optional version number) is present - # - # @param ext_name [String,#to_s] Name of the extension - # @param ext_requirement [String, #to_s] Version string, as a Gem Requirement (https://guides.rubygems.org/patterns/#pessimistic-version-constraint) - # @return [Boolean] whether or not extension +ext_name+ meeting +ext_requirement+ is implemented in the config - def ext?(ext_name, ext_requirement = ">= 0") - @arch_gen.ext?(ext_name.to_s, ext_requirement) - end - - # @return [Array] List of possible XLENs for any implemented mode - def possible_xlens - @arch_gen.possible_xlens - end - - # insert a hyperlink to an object - # At this point, we insert a placeholder since it will be up - # to the backend to create a specific link - # - # @params type [Symbol] Type (:section, :csr, :inst, :ext) - # @params name [#to_s] Name of the object - def link_to(type, name) - "%%LINK%#{type};#{name}%%" - end - - # info on interrupt and exception codes - - # @returns [Hash] architecturally-defined exception codes and their names - def exception_codes - @arch_gen.exception_codes - end - - # returns [Hash] architecturally-defined interrupt codes and their names - def interrupt_codes - @arch_gen.interrupt_codes - end - - # @returns [Hash] architecturally-defined exception codes and their names - def implemented_exception_codes - @arch_gen.implemented_exception_codes - end - - # returns [Hash] architecturally-defined interrupt codes and their names - def implemented_interrupt_codes - @arch_gen.implemented_interrupt_codes - end - end - - @env - end - private :erb_env - - # create a new raw *unconfigured* architecture defintion data structure - # - # The data will not include anything configuration-dependent such as implemented_*/mandatory_*/etc. - # - # This function can be used to create a new arch_def for a different configuration - # - # @return [Hash] A unconfigured architecture definition - def unconfigured_data - { - "type" => "partially configured", - "instructions" => instructions.map(&:data), - "extensions" => extensions.map.map(&:data), - "csrs" => csrs.map(&:data), - "profile_classes" => profile_classes.map { |f| [f.name, f.data] }.to_h, - "profile_releases" => profile_releases.map { |p| [p.name, p.data] }.to_h, - "manuals" => manuals.map { |m| [m.name, m.data] }.to_h, - "certificate_classes" => cert_classes.map(&:data), - "certificate_models" => cert_models.map(&:data) - } - end - - # passes _erb_template_ through ERB within the content of this config - # - # @param erb_template [String] ERB source - # @return [String] The rendered text - def render_erb(erb_template, what='') - t = Tempfile.new("template") - t.write erb_template - t.flush - begin - Tilt["erb"].new(t.path, trim: "-").render(erb_env) - rescue - warn "While rendering ERB template: #{what}" - raise - ensure - t.close - t.unlink - end - end -end - -# a synchroncous exception code -class ExceptionCode - # @return [String] Long-form display name (can include special characters) - attr_reader :name - - # @return [String] Field name for an IDL enum - attr_reader :var - - # @return [Integer] Code, written into *mcause - attr_reader :num - - # @return [Extension] Extension that defines this code - attr_reader :ext - - def initialize(name, var, number, ext) - @name = name - @name.freeze - @var = var - @num = number - @ext = ext - end -end - -# all the same informatin as ExceptinCode, but for interrupts -InterruptCode = Class.new(ExceptionCode) diff --git a/lib/arch_obj_models/certificate.rb b/lib/arch_obj_models/certificate.rb index 833c2f172..d422ea6c5 100644 --- a/lib/arch_obj_models/certificate.rb +++ b/lib/arch_obj_models/certificate.rb @@ -10,12 +10,6 @@ # Holds information from certificate class YAML file. # The inherited "data" member is the database of extensions, instructions, CSRs, etc. class CertClass < PortfolioClass - # @param data [Hash] The data from YAML - # @param arch_def [ArchDef] Architecture spec - def initialize(data, arch_def) - super(data, arch_def) - end - def mandatory_priv_modes = @data["mandatory_priv_modes"] end @@ -26,12 +20,6 @@ def mandatory_priv_modes = @data["mandatory_priv_modes"] # Holds information about a certificate model YAML file. # The inherited "data" member is the database of extensions, instructions, CSRs, etc. class CertModel < PortfolioInstance - # @param data [Hash] The data from YAML - # @param arch_def [ArchDef] Architecture spec - def initialize(data, arch_def) - super(data, arch_def) - end - def unpriv_isa_manual_revision = @data["unpriv_isa_manual_revision"] def priv_isa_manual_revision = @data["priv_isa_manual_revision"] def debug_manual_revision = @data["debug_manual_revision"] @@ -39,7 +27,7 @@ def debug_manual_revision = @data["debug_manual_revision"] def tsc_profile return nil if @data["tsc_profile"].nil? - profile = arch_def.profile(@data["tsc_profile"]) + profile = cfg_arch.profile(@data["tsc_profile"]) raise "No profile '#{@data["tsc_profile"]}'" if profile.nil? @@ -48,7 +36,7 @@ def tsc_profile # @return [CertClass] The certification class that this model belongs to. def cert_class - cert_class = @arch_def.ref(@data["class"]['$ref']) + cert_class = @cfg_arch.ref(@data["class"]['$ref']) raise "No certificate class named '#{@data["class"]}'" if cert_class.nil? cert_class @@ -59,12 +47,14 @@ def cert_class ##################### # Holds extra requirements not associated with extensions or their parameters. - class Requirement < ArchDefObject - def initialize(data, arch_def) - super(data) - @arch_def = arch_def + class Requirement + def initialize(data, cfg_arch) + @data = data + @cfg_arch = cfg_arch end + def name = @data["name"] + def description = @data["description"] def when = @data["when"] @@ -91,12 +81,14 @@ def when_pretty # Holds a group of Requirement objects to provide a one-level group. # Can't nest RequirementGroup objects to make multi-level group. - class RequirementGroup < ArchDefObject - def initialize(data, arch_def) - super(data) - @arch_def = arch_def + class RequirementGroup + def initialize(data, cfg_arch) + @data = data + @cfg_arch = cfg_arch end + def name = @data["name"] + def description = @data["description"] def when = @data["when"] @@ -121,7 +113,7 @@ def requirements @requirements = [] @data["requirements"].each do |req| - @requirements << Requirement.new(req, @arch_def) + @requirements << Requirement.new(req, @cfg_arch) end @requirements end @@ -132,7 +124,7 @@ def requirement_groups @requirement_groups = [] @data["requirement_groups"]&.each do |req_group| - @requirement_groups << RequirementGroup.new(req_group, @arch_def) + @requirement_groups << RequirementGroup.new(req_group, @cfg_arch) end @requirement_groups end diff --git a/lib/arch_obj_models/csr.rb b/lib/arch_obj_models/csr.rb index 4747e05b3..20161ef58 100644 --- a/lib/arch_obj_models/csr.rb +++ b/lib/arch_obj_models/csr.rb @@ -2,9 +2,8 @@ require_relative "obj" - # CSR definition -class Csr < ArchDefObject +class Csr < DatabaseObjectect def ==(other) if other.is_a?(Csr) name == other.name @@ -47,68 +46,68 @@ def defined_in_base64? = @data["base"].nil? || @data["base"] == 64 # @return [Boolean] true if this CSR is defined regardless of the effective XLEN def defined_in_all_bases? = @data["base"].nil? - # @param arch_def [ArchDef] A configuration + # @param cfg_arch [ConfiguredArchitecture] A configuration # @return [Boolean] Whether or not the format of this CSR changes when the effective XLEN changes in some mode - def format_changes_with_xlen?(arch_def) - dynamic_length?(arch_def) || - implemented_fields(arch_def).any? do |f| - f.dynamic_location?(arch_def) + def format_changes_with_xlen?(cfg_arch) + dynamic_length?(cfg_arch) || + implemented_fields(cfg_arch).any? do |f| + f.dynamic_location?(cfg_arch) end end - # @param arch_def [ArchDef] A configuration + # @param cfg_arch [ConfiguredArchitecture] A configuration # @return [Array] List of functions reachable from this CSR's sw_read or a field's sw_wirte function - def reachable_functions(arch_def) + def reachable_functions(cfg_arch) return @reachable_functions unless @reachable_functions.nil? fns = [] if has_custom_sw_read? - ast = pruned_sw_read_ast(arch_def) - symtab = arch_def.symtab.deep_clone + ast = pruned_sw_read_ast(cfg_arch) + symtab = cfg_arch.symtab.deep_clone symtab.push(ast) fns.concat(ast.reachable_functions(symtab)) end - if arch_def.multi_xlen? - implemented_fields_for(arch_def, 32).each do |field| - fns.concat(field.reachable_functions(arch_def, 32)) + if cfg_arch.multi_xlen? + implemented_fields_for(cfg_arch, 32).each do |field| + fns.concat(field.reachable_functions(cfg_arch, 32)) end - implemented_fields_for(arch_def, 64).each do |field| - fns.concat(field.reachable_functions(arch_def, 64)) + implemented_fields_for(cfg_arch, 64).each do |field| + fns.concat(field.reachable_functions(cfg_arch, 64)) end else - implemented_fields_for(arch_def, arch_def.mxlen).each do |field| - fns.concat(field.reachable_functions(arch_def, arch_def.mxlen)) + implemented_fields_for(cfg_arch, cfg_arch.mxlen).each do |field| + fns.concat(field.reachable_functions(cfg_arch, cfg_arch.mxlen)) end end @reachable_functions = fns.uniq end - # @param arch_def [ArchDef] Architecture definition + # @param cfg_arch [ConfiguredArchitecture] Architecture definition # @return [Array] List of functions reachable from this CSR's sw_read or a field's sw_wirte function, irrespective of context - def reachable_functions_unevaluated(arch_def) + def reachable_functions_unevaluated(cfg_arch) return @reachable_functions_unevaluated unless @reachable_functions_unevaluated.nil? fns = [] if has_custom_sw_read? - ast = sw_read_ast(arch_def) - fns.concat(ast.reachable_functions_unevaluated(arch_def)) + ast = sw_read_ast(cfg_arch) + fns.concat(ast.reachable_functions_unevaluated(cfg_arch)) end fields.each do |field| - fns.concat(field.reachable_functions_unevaluated(arch_def)) + fns.concat(field.reachable_functions_unevaluated(cfg_arch)) end @reachable_functions_unevaluated = fns.uniq end - # @param arch_def [ArchDef] A configuration + # @param cfg_arch [ConfiguredArchitecture] A configuration # @return [Boolean] Whether or not the length of the CSR depends on a runtime value # (e.g., mstatus.SXL) - def dynamic_length?(arch_def) + def dynamic_length?(cfg_arch) return false if @data["length"].is_a?(Integer) # when a CSR is only defined in one base, its length can't change @@ -118,21 +117,21 @@ def dynamic_length?(arch_def) when "MXLEN" # mxlen can never change at runtime, so if we have it in the config, the length is not dynamic # if we don't have it in the config, we don't know what the length is - return arch_def.mxlen.nil? + cfg_arch.mxlen.nil? when "SXLEN" # dynamic if either we don't know SXLEN or SXLEN is explicitly mutable - [nil, 3264].include?(arch_def.param_values["SXLEN"]) + [nil, 3264].include?(cfg_arch.param_values["SXLEN"]) when "VSXLEN" # dynamic if either we don't know VSXLEN or VSXLEN is explicitly mutable - [nil, 3264].include?(arch_def.param_values["VSXLEN"]) + [nil, 3264].include?(cfg_arch.param_values["VSXLEN"]) else raise "Unexpected length" end end - # @param arch_def [ArchDef] Architecture definition + # @param cfg_arch [ConfiguredArchitecture] Architecture definition # @return [Integer] Smallest length of the CSR in any mode - def min_length(arch_def) + def min_length(cfg_arch) case @data["length"] when "MXLEN", "SXLEN", "VSXLEN" 32 @@ -143,27 +142,14 @@ def min_length(arch_def) end end - # @param arch_def [ArchDef] Architecture definition - # @return [Integer] Largest length of the CSR in any mode - def max_length(arch_def) - case @data["length"] - when "MXLEN", "SXLEN", "VSXLEN" - 64 - when Integer - @data["length"] - else - raise "Unexpected length" - end - end - - # @param arch_def [ArchDef] A configuration (can be nil if the lenth is not dependent on a config parameter) + # @param cfg_arch [ConfiguredArchitecture] A configuration (can be nil if the lenth is not dependent on a config parameter) # @param effective_xlen [Integer] The effective xlen, needed since some fields change location with XLEN. If the field location is not determined by XLEN, then this parameter can be nil # @return [Integer] Length, in bits, of the CSR, given effective_xlen - # @return [nil] if the length cannot be determined from the arch_def (e.g., because SXLEN is unknown and +effective_xlen+ was not provided) - def length(arch_def, effective_xlen = nil) + # @return [nil] if the length cannot be determined from the cfg_arch (e.g., because SXLEN is unknown and +effective_xlen+ was not provided) + def length(cfg_arch, effective_xlen = nil) case @data["length"] when "MXLEN" - return arch_def.mxlen unless arch_def.mxlen.nil? + return cfg_arch.mxlen unless cfg_arch.mxlen.nil? if !@data["base"].nil? @data["base"] @@ -172,11 +158,11 @@ def length(arch_def, effective_xlen = nil) effective_xlen end when "SXLEN" - if arch_def.param_values.key?("SXLEN") - if arch_def.param_values["SXLEN"] == 3264 + if cfg_arch.param_values.key?("SXLEN") + if cfg_arch.param_values["SXLEN"] == 3264 effective_xlen else - arch_def.param_values["SXLEN"] + cfg_arch.param_values["SXLEN"] end elsif !@data["base"].nil? # if this CSR is only available in one base, then we know its length @@ -186,11 +172,11 @@ def length(arch_def, effective_xlen = nil) effective_xlen end when "VSXLEN" - if arch_def.param_values.key?("VSXLEN") - if arch_def.param_values["VSXLEN"] == 3264 + if cfg_arch.param_values.key?("VSXLEN") + if cfg_arch.param_values["VSXLEN"] == 3264 effective_xlen else - arch_def.param_values["VSXLEN"] + cfg_arch.param_values["VSXLEN"] end elsif !@data["base"].nil? # if this CSR is only available in one base, then we know its length @@ -207,28 +193,28 @@ def length(arch_def, effective_xlen = nil) end # @return [Integer] The largest length of this CSR in any valid mode/xlen for the config - def max_length(arch_def) + def max_length(cfg_arch) return @data["base"] unless @data["base"].nil? case @data["length"] when "MXLEN" - arch_def.mxlen || 64 + cfg_arch.mxlen || 64 when "SXLEN" - if arch_def.param_values.key?("SXLEN") - if arch_def.param_values["SXLEN"] == 3264 + if cfg_arch.param_values.key?("SXLEN") + if cfg_arch.param_values["SXLEN"] == 3264 64 else - arch_def.param_values["SXLEN"] + cfg_arch.param_values["SXLEN"] end else 64 end when "VSXLEN" - if arch_def.param_values.key?("VSXLEN") - if arch_def.param_values["VSXLEN"] == 3264 + if cfg_arch.param_values.key?("VSXLEN") + if cfg_arch.param_values["VSXLEN"] == 3264 64 else - arch_def.param_values["VSXLEN"] + cfg_arch.param_values["VSXLEN"] end else 64 @@ -268,10 +254,10 @@ def length_cond64 end end - # @param arch_def [ArchDef] A configuration + # @param cfg_arch [ConfiguredArchitecture] A configuration # @return [String] Pretty-printed length string - def length_pretty(arch_def, effective_xlen=nil) - if dynamic_length?(arch_def) + def length_pretty(cfg_arch, effective_xlen=nil) + if dynamic_length?(cfg_arch) cond = case @data["length"] when "MXLEN" @@ -286,14 +272,14 @@ def length_pretty(arch_def, effective_xlen=nil) if effective_xlen.nil? <<~LENGTH - #{length(arch_def, 32)} when #{cond.sub('%%', '0')} - #{length(arch_def, 64)} when #{cond.sub('%%', '1')} + #{length(cfg_arch, 32)} when #{cond.sub('%%', '0')} + #{length(cfg_arch, 64)} when #{cond.sub('%%', '1')} LENGTH else - "#{length(arch_def, effective_xlen)}-bit" + "#{length(cfg_arch, effective_xlen)}-bit" end else - "#{length(arch_def)}-bit" + "#{length(cfg_arch)}-bit" end end @@ -320,39 +306,39 @@ def description_html Asciidoctor.convert description end - # @param arch_Def [ArchDef] A configuration + # @param cfg_arch [ConfiguredArchitecture] A configuration # @return [Array] All implemented fields for this CSR at the given effective XLEN, sorted by location (smallest location first) # Excluded any fields that are defined by unimplemented extensions or a base that is not effective_xlen - def implemented_fields_for(arch_def, effective_xlen) + def implemented_fields_for(cfg_arch, effective_xlen) @implemented_fields_for ||= {} - key = [arch_def.name, effective_xlen].hash + key = [cfg_arch.name, effective_xlen].hash return @implemented_fields_for[key] if @implemented_fields_for.key?(key) @implemented_fields_for[key] = - implemented_fields(arch_def).select do |f| + implemented_fields(cfg_arch).select do |f| !f.key?("base") || f.base == effective_xlen end end - # @param arch_def [ArchDef] A configuration + # @param cfg_arch [ConfiguredArchitecture] A configuration # @return [Array] All implemented fields for this CSR # Excluded any fields that are defined by unimplemented extensions - def implemented_fields(arch_def) + def implemented_fields(cfg_arch) return @implemented_fields unless @implemented_fields.nil? implemented_bases = - if arch_def.param_values["SXLEN"] == 3264 || - arch_def.param_values["UXLEN"] == 3264 || - arch_def.param_values["VSXLEN"] == 3264 || - arch_def.param_values["VUXLEN"] == 3264 + if cfg_arch.param_values["SXLEN"] == 3264 || + cfg_arch.param_values["UXLEN"] == 3264 || + cfg_arch.param_values["VSXLEN"] == 3264 || + cfg_arch.param_values["VUXLEN"] == 3264 [32, 64] else - [arch_def.param_values["XLEN"]] + [cfg_arch.param_values["XLEN"]] end @implemented_fields = fields.select do |f| - f.exists_in_cfg?(arch_def) + f.exists_in_cfg?(cfg_arch) end end @@ -360,7 +346,7 @@ def implemented_fields(arch_def) def fields return @fields unless @fields.nil? - @fields = @data["fields"].map { |_field_name, field_data| CsrField.new(self, field_data) } + @fields = @data["fields"].map { |field_name, field_data| CsrField.new(self, field_name, field_data) } end # @return [Array] All known fields of this CSR when XLEN == +effective_xlen+ @@ -391,15 +377,15 @@ def field(field_name) field_hash[field_name.to_s] end - # @param arch_def [ArchDef] A configuration + # @param cfg_arch [ConfiguredArchitecture] A configuration # @param effective_xlen [Integer] The effective XLEN to apply, needed when field locations change with XLEN in some mode # @return [Idl::BitfieldType] A bitfield type that can represent all fields of the CSR - def bitfield_type(arch_def, effective_xlen = nil) + def bitfield_type(cfg_arch, effective_xlen = nil) Idl::BitfieldType.new( "Csr#{name.capitalize}Bitfield", - length(arch_def, effective_xlen), + length(cfg_arch, effective_xlen), fields_for(effective_xlen).map(&:name), - fields_for(effective_xlen).map { |f| f.location(arch_def, effective_xlen) } + fields_for(effective_xlen).map { |f| f.location(cfg_arch, effective_xlen) } ) end @@ -428,7 +414,7 @@ def type_checked_sw_read_ast(symtab) ) ast = sw_read_ast(symtab) - symtab.archdef.idl_compiler.type_check( + symtab.cfg_arch.idl_compiler.type_check( ast, symtab, "CSR[#{name}].sw_read()" @@ -439,7 +425,7 @@ def type_checked_sw_read_ast(symtab) end # @return [FunctionBodyAst] The abstract syntax tree of the sw_read() function - # @param archdef [ArchDef] A configuration + # @param cfg_arch [ConfiguredArchitecture] A configuration def sw_read_ast(symtab) raise ArgumentError, "Argument should be a symtab" unless symtab.is_a?(Idl::SymbolTable) @@ -447,7 +433,7 @@ def sw_read_ast(symtab) return nil if @data["sw_read()"].nil? # now, parse the function - @sw_read_ast = symtab.archdef.idl_compiler.compile_func_body( + @sw_read_ast = symtab.cfg_arch.idl_compiler.compile_func_body( @data["sw_read()"], return_type: Idl::Type.new(:bits, width: 128), # big int to hold special return values name: "CSR[#{name}].sw_read()", @@ -464,14 +450,14 @@ def sw_read_ast(symtab) @sw_read_ast end - def pruned_sw_read_ast(arch_def) + def pruned_sw_read_ast(cfg_arch) @pruned_sw_read_asts ||= {} - ast = @pruned_sw_read_asts[arch_def.name] + ast = @pruned_sw_read_asts[cfg_arch.name] return ast unless ast.nil? - ast = type_checked_sw_read_ast(arch_def.symtab) + ast = type_checked_sw_read_ast(cfg_arch.symtab) - symtab = arch_def.symtab.global_clone + symtab = cfg_arch.symtab.global_clone symtab.push(ast) # all CSR instructions are 32-bit symtab.add( @@ -484,9 +470,9 @@ def pruned_sw_read_ast(arch_def) ) ast = ast.prune(symtab) - ast.freeze_tree(arch_def.symtab) + ast.freeze_tree(cfg_arch.symtab) - arch_def.idl_compiler.type_check( + cfg_arch.idl_compiler.type_check( ast, symtab, "CSR[#{name}].sw_read()" @@ -495,7 +481,7 @@ def pruned_sw_read_ast(arch_def) symtab.pop symtab.release - @pruned_sw_read_asts[arch_def.name] = ast + @pruned_sw_read_asts[cfg_arch.name] = ast end # @example Result for an I-type instruction @@ -507,12 +493,12 @@ def pruned_sw_read_ast(arch_def) # {bits: 12, name: 'imm12', attr: [''], type: 6} # ]} # - # @param arch_def [ArchDef] A configuration + # @param cfg_arch [ConfiguredArchitecture] A configuration # @param effective_xlen [Integer,nil] Effective XLEN to use when CSR length is dynamic # @param exclude_unimplemented [Boolean] If true, do not create include unimplemented fields in the figure - # @param optional_type [Integer] Wavedrom type (Fill color) for fields that are optional (not mandatory) in a partially-specified arch_def + # @param optional_type [Integer] Wavedrom type (Fill color) for fields that are optional (not mandatory) in a partially-specified cfg_arch # @return [Hash] A representation of the WaveDrom drawing for the CSR (should be turned into JSON for wavedrom) - def wavedrom_desc(arch_def, effective_xlen, exclude_unimplemented: false, optional_type: 2) + def wavedrom_desc(cfg_arch, effective_xlen, exclude_unimplemented: false, optional_type: 2) desc = { "reg" => [] } @@ -520,58 +506,58 @@ def wavedrom_desc(arch_def, effective_xlen, exclude_unimplemented: false, option field_list = if exclude_unimplemented - implemented_fields_for(arch_def, effective_xlen) + implemented_fields_for(cfg_arch, effective_xlen) else fields_for(effective_xlen) end - field_list.sort! { |a, b| a.location(arch_def, effective_xlen).min <=> b.location(arch_def, effective_xlen).min } + field_list.sort! { |a, b| a.location(cfg_arch, effective_xlen).min <=> b.location(cfg_arch, effective_xlen).min } field_list.each do |field| - if field.location(arch_def, effective_xlen).min != last_idx + 1 + if field.location(cfg_arch, effective_xlen).min != last_idx + 1 # have some reserved space - n = field.location(arch_def, effective_xlen).min - last_idx - 1 - raise "negative reserved space? #{n} #{name} #{field.location(arch_def, effective_xlen).min} #{last_idx + 1}" if n <= 0 + n = field.location(cfg_arch, effective_xlen).min - last_idx - 1 + raise "negative reserved space? #{n} #{name} #{field.location(cfg_arch, effective_xlen).min} #{last_idx + 1}" if n <= 0 desc["reg"] << { "bits" => n, type: 1 } end - if arch_def.partially_configured? && field.optional_in_cfg?(arch_def) - desc["reg"] << { "bits" => field.location(arch_def, effective_xlen).size, "name" => field.name, type: optional_type } + if cfg_arch.partially_configured? && field.optional_in_cfg?(cfg_arch) + desc["reg"] << { "bits" => field.location(cfg_arch, effective_xlen).size, "name" => field.name, type: optional_type } else - desc["reg"] << { "bits" => field.location(arch_def, effective_xlen).size, "name" => field.name, type: 3 } + desc["reg"] << { "bits" => field.location(cfg_arch, effective_xlen).size, "name" => field.name, type: 3 } end - last_idx = field.location(arch_def, effective_xlen).max + last_idx = field.location(cfg_arch, effective_xlen).max end - if !field_list.empty? && (field_list.last.location(arch_def, effective_xlen).max != (length(arch_def, effective_xlen) - 1)) + if !field_list.empty? && (field_list.last.location(cfg_arch, effective_xlen).max != (length(cfg_arch, effective_xlen) - 1)) # reserved space at the end - desc["reg"] << { "bits" => (length(arch_def, effective_xlen) - 1 - last_idx), type: 1 } + desc["reg"] << { "bits" => (length(cfg_arch, effective_xlen) - 1 - last_idx), type: 1 } # desc['reg'] << { 'bits' => 1, type: 1 } end - desc["config"] = { "bits" => length(arch_def, effective_xlen) } - desc["config"]["lanes"] = length(arch_def, effective_xlen) / 16 + desc["config"] = { "bits" => length(cfg_arch, effective_xlen) } + desc["config"]["lanes"] = length(cfg_arch, effective_xlen) / 16 desc end - # @param arch_def [ArchDef] Architecture def + # @param cfg_arch [ConfiguredArchitecture] Architecture def # @return [Boolean] whether or not the CSR is possibly implemented given the supplies config options - def exists_in_cfg?(arch_def) - if arch_def.fully_configured? - (@data["base"].nil? || (arch_def.possible_xlens.include? @data["base"])) && - arch_def.implemented_extensions.any? { |e| defined_by?(e) } + def exists_in_cfg?(cfg_arch) + if cfg_arch.fully_configured? + (@data["base"].nil? || (cfg_arch.possible_xlens.include? @data["base"])) && + cfg_arch.transitive_implemented_extensions.any? { |e| defined_by?(e) } else - (@data["base"].nil? || (arch_def.possible_xlens.include? @data["base"])) && - arch_def.prohibited_extensions.none? { |e| defined_by?(e) } + (@data["base"].nil? || (cfg_arch.possible_xlens.include? @data["base"])) && + cfg_arch.prohibited_extensions.none? { |ext_req| ext_req.satisfying_versions.any? { |e| defined_by?(e) } } end end - # @param arch_def [ArchDef] Architecture def + # @param cfg_arch [ConfiguredArchitecture] Architecture def # @return [Boolean] whether or not the CSR is optional in the config - def optional_in_cfg?(arch_def) - raise "optional_in_cfg? should only be used by a partially-specified arch def" unless arch_def.partially_configured? + def optional_in_cfg?(cfg_arch) + raise "optional_in_cfg? should only be used by a partially-specified arch def" unless cfg_arch.partially_configured? - exists_in_cfg?(arch_def) && - arch_def.mandatory_extensions.all? do |ext_req| - ext_req.satisfying_versions(arch_def).none? do |ext_ver| + exists_in_cfg?(cfg_arch) && + cfg_arch.mandatory_extensions.all? do |ext_req| + ext_req.satisfying_versions.none? do |ext_ver| defined_by?(ext_ver) end end diff --git a/lib/arch_obj_models/csr_field.rb b/lib/arch_obj_models/csr_field.rb index a793ba088..6eba5a21e 100644 --- a/lib/arch_obj_models/csr_field.rb +++ b/lib/arch_obj_models/csr_field.rb @@ -5,7 +5,7 @@ require_relative "../idl/passes/gen_option_adoc" # A CSR field object -class CsrField < ArchDefObject +class CsrField < DatabaseObjectect # @return [Csr] The Csr that defines this field attr_reader :parent @@ -23,39 +23,40 @@ def base # @param parent_csr [Csr] The Csr that defined this field # @param field_data [Hash] Field data from the arch spec - def initialize(parent_csr, field_data) - super(field_data) + def initialize(parent_csr, field_name, field_data) + super(field_data, parent_csr.data_path, arch: parent_csr.arch) + @name = field_name @parent = parent_csr end # @param possible_xlens [Array] List of xlens that be used in any implemented mode # @param extensions [Array] List of extensions implemented # @return [Boolean] whether or not the instruction is implemented given the supplies config options - def exists_in_cfg?(arch_def) - if arch_def.fully_configured? - parent.exists_in_cfg?(arch_def) && - (@data["base"].nil? || arch_def.possible_xlens.include?(@data["base"])) && - (@data["definedBy"].nil? || arch_def.implemented_extensions.any? { |ext_ver| defined_by?(ext_ver) }) + def exists_in_cfg?(cfg_arch) + if cfg_arch.fully_configured? + parent.exists_in_cfg?(cfg_arch) && + (@data["base"].nil? || cfg_arch.possible_xlens.include?(@data["base"])) && + (@data["definedBy"].nil? || cfg_arch.transitive_implemented_extensions.any? { |ext_ver| defined_by?(ext_ver) }) else - raise "unexpected type" unless arch_def.partially_configured? + raise "unexpected type" unless cfg_arch.partially_configured? - parent.exists_in_cfg?(arch_def) && - (@data["base"].nil? || arch_def.possible_xlens.include?(@data["base"])) && - (@data["definedBy"].nil? || arch_def.prohibited_extensions.none? { |ext_ver| defined_by?(ext_ver) }) + parent.exists_in_cfg?(cfg_arch) && + (@data["base"].nil? || cfg_arch.possible_xlens.include?(@data["base"])) && + (@data["definedBy"].nil? || cfg_arch.prohibited_extensions.none? { |ext_req| ext_req.satisfying_versions.any? { |ext_ver| defined_by?(ext_ver) } }) end end - # @return [Boolean] For a partially configured arch_def, whether or not the field is optional (not mandatory or prohibited) - def optional_in_cfg?(arch_def) - raise "optional_in_cfg? should only be called on a partially configured arch_def" unless arch_def.partially_configured? + # @return [Boolean] For a partially configured cfg_arch, whether or not the field is optional (not mandatory or prohibited) + def optional_in_cfg?(cfg_arch) + raise "optional_in_cfg? should only be called on a partially configured cfg_arch" unless cfg_arch.partially_configured? - exists_in_cfg?(arch_def) && + exists_in_cfg?(cfg_arch) && ( if data["definedBy"].nil? - parent.optional_in_cfg?(arch_def) + parent.optional_in_cfg?(cfg_arch) else - arch_def.mandatory_extensions.all? do |ext_req| - ext_req.satisfying_versions(arch_def).none? do |ext_ver| + cfg_arch.mandatory_extensions.all? do |ext_req| + ext_req.satisfying_versions.none? do |ext_ver| defined_by?(ext_ver) end end @@ -70,7 +71,7 @@ def type_ast(symtab) return @type_ast unless @type_ast.nil? return nil if @data["type()"].nil? - @type_ast = symtab.archdef.idl_compiler.compile_func_body( + @type_ast = symtab.cfg_arch.idl_compiler.compile_func_body( @data["type()"], name: "CSR[#{csr.name}].#{name}.type()", input_file: csr.__source, @@ -104,7 +105,7 @@ def type_checked_type_ast(symtab) ) ast = type_ast(symtab) - symtab.archdef.idl_compiler.type_check( + symtab.cfg_arch.idl_compiler.type_check( ast, symtab, "CSR[#{name}].type()" @@ -136,7 +137,7 @@ def pruned_type_ast(symtab) ast.freeze_tree(symtab) - symtab.archdef.idl_compiler.type_check( + symtab.cfg_arch.idl_compiler.type_check( ast, symtab, "CSR[#{name}].type()" @@ -161,9 +162,9 @@ def type(symtab) raise ArgumentError, "Argument 1 should be a symtab" unless symtab.is_a?(Idl::SymbolTable) unless @type_cache.nil? - raise "Different archdef for type #{@type_cache.keys}, #{symtab.archdef}" unless @type_cache.key?(symtab.archdef) + raise "Different cfg_arch for type #{@type_cache.keys}, #{symtab.cfg_arch}" unless @type_cache.key?(symtab.cfg_arch) - return @type_cache[symtab.archdef] + return @type_cache[symtab.cfg_arch] end type = @@ -174,8 +175,6 @@ def type(symtab) idl = @data["type()"] raise "type() is nil for #{csr.name}.#{name} #{@data}?" if idl.nil? - - # value_result = Idl::AstNode.value_try do ast = type_checked_type_ast(symtab) begin @@ -212,7 +211,7 @@ def type(symtab) end @type_cache ||= {} - @type_cache[symtab.archdef] = type + @type_cache[symtab.cfg_arch] = type end # @return [String] A pretty-printed type string @@ -243,7 +242,7 @@ def alias range_start = Regexp.last_match(4) range_end = Regexp.last_match(5) - csr_field = arch_def.csr(csr_name).field(csr_field) + csr_field = cfg_arch.csr(csr_name).field(csr_field) range = if range.nil? field.location @@ -258,31 +257,31 @@ def alias end # @return [Array] List of functions called thorugh this field - # @param archdef [ArchDef] a configuration + # @param cfg_arch [ConfiguredArchitecture] a configuration # @Param effective_xlen [Integer] 32 or 64; needed because fields can change in different XLENs - def reachable_functions(archdef, effective_xlen) + def reachable_functions(cfg_arch, effective_xlen) return @reachable_functions unless @reachable_functions.nil? symtab = - if (archdef.configured?) - archdef.symtab + if (cfg_arch.configured?) + cfg_arch.symtab else - raise ArgumentError, "Must supply effective_xlen for generic ArchDef" if effective_xlen.nil? + raise ArgumentError, "Must supply effective_xlen for generic ConfiguredArchitecture" if effective_xlen.nil? if effective_xlen == 32 - archdef.symtab_32 + cfg_arch.symtab_32 else - archdef.symtab_64 + cfg_arch.symtab_64 end end fns = [] if has_custom_sw_write? - ast = pruned_sw_write_ast(archdef, effective_xlen) + ast = pruned_sw_write_ast(cfg_arch, effective_xlen) unless ast.nil? sw_write_symtab = symtab.deep_clone sw_write_symtab.push(ast) - sw_write_symtab.add("csr_value", Idl::Var.new("csr_value", csr.bitfield_type(symtab.archdef, effective_xlen))) + sw_write_symtab.add("csr_value", Idl::Var.new("csr_value", csr.bitfield_type(symtab.cfg_arch, effective_xlen))) fns.concat ast.reachable_functions(sw_write_symtab) end end @@ -335,18 +334,18 @@ def reachable_functions_unevaluated(symtab) # @return [Csr] Parent CSR for this field alias csr parent - # @param arch_def [ArchDef] A configuration + # @param cfg_arch [ConfiguredArchitecture] A configuration # @return [Boolean] Whether or not the location of the field changes dynamically # (e.g., based on mstatus.SXL) in the configuration - def dynamic_location?(arch_def) + def dynamic_location?(cfg_arch) # if there is no location_rv32, the the field never changes return false unless @data["location"].nil? # the field changes *if* some mode with access can change XLEN - csr.modes_with_access.any? { |mode| arch_def.multi_xlen_in_mode?(mode) } + csr.modes_with_access.any? { |mode| cfg_arch.multi_xlen_in_mode?(mode) } end - # @param arch_def [IdL::Compiler] A compiler + # @param cfg_arch [IdL::Compiler] A compiler # @return [Idl::FunctionBodyAst] Abstract syntax tree of the reset_value function # @return [nil] If the reset_value is not a function def reset_value_ast(symtab) @@ -355,7 +354,7 @@ def reset_value_ast(symtab) return @reset_value_ast unless @reset_value_ast.nil? return nil unless @data.key?("reset_value()") - @reset_value_ast = symtab.archdef.idl_compiler.compile_func_body( + @reset_value_ast = symtab.cfg_arch.idl_compiler.compile_func_body( @data["reset_value()"], return_type: Idl::Type.new(:bits, width: 64), name: "CSR[#{parent.name}].#{name}.reset_value()", @@ -384,7 +383,7 @@ def type_checked_reset_value_ast(symtab) symtab = symtab.deep_clone symtab.push(ast) symtab.add("__expected_return_type", Idl::Type.new(:bits, width: 64)) - symtab.archdef.idl_compiler.type_check( + symtab.cfg_arch.idl_compiler.type_check( ast, symtab, "CSR[#{csr.name}].reset_value()" @@ -417,7 +416,7 @@ def pruned_reset_value_ast(symtab) symtab.push(ast) symtab.add("__expected_return_type", Idl::Type.new(:bits, width: 64)) - symtab.archdef.idl_compiler.type_check( + symtab.cfg_arch.idl_compiler.type_check( ast, symtab, "CSR[#{csr.name}].#{name}.reset_value()" @@ -426,27 +425,20 @@ def pruned_reset_value_ast(symtab) @type_checked_reset_value_asts[symtab_hash] = ast end - # @param arch_def [ArchDef] A config + # @param cfg_arch [ConfiguredArchitecture] A config # @return [Integer] The reset value of this field # @return [String] The string 'UNDEFINED_LEGAL' if, for this config, there is no defined reset value - def reset_value(arch_def, effective_xlen = nil) - cached_value = @reset_value_cache.nil? ? nil : @reset_value_cache[arch_def] + def reset_value(cfg_arch) + cached_value = @reset_value_cache.nil? ? nil : @reset_value_cache[cfg_arch] return cached_value if cached_value @reset_value_cache ||= {} - @reset_value_cache[arch_def] = + @reset_value_cache[cfg_arch] = if @data.key?("reset_value") @data["reset_value"] else - symtab = - if !arch_def.mxlen.nil? - arch_def.symtab - else - raise ArgumentError, "effective_xlen is required when using generic arch_def" if effective_xlen.nil? - - effective_xlen == 32 ? arch_def.symtab_32 : arch_def.symtab_64 - end + symtab = cfg_arch.symtab ast = pruned_reset_value_ast(symtab.deep_clone) val = ast.return_value(symtab.deep_clone.push(ast)) val = "UNDEFINED_LEGAL" if val == 0x1_0000_0000_0000_0000 @@ -454,39 +446,22 @@ def reset_value(arch_def, effective_xlen = nil) end end - def dynamic_reset_value?(arch_def) + def dynamic_reset_value?(cfg_arch) return false unless @data["reset_value"].nil? - value_result = value_try do - if arch_def.mxlen.nil? - # need to try with generic symtab_32/symtab_64 - reset_value_32 = reset_value(arch_def, 32) - reset_value_64 = reset_value(arch_def, 64) - reset_value_32 != reset_value_64 - else - # just call the function, see if we get a value error - reset_value(arch_def) - false - end + value_result = Idl::AstNode.value_try do + reset_value(cfg_arch) + false end || true end - def reset_value_pretty(arch_def) + def reset_value_pretty(cfg_arch) str = nil value_result = Idl::AstNode.value_try do - str = - if arch_def.mxlen.nil? - if dynamic_reset_value?(arch_def) - Idl::AstNode.value_error "" - else - reset_value(arch_def, 32) # 32 or 64, doesn't matter - end - else - reset_value(arch_def) - end + str = reset_value(cfg_arch) end Idl::AstNode.value_else(value_result) do - ast = reset_value_ast(arch_def.symtab) + ast = reset_value_ast(cfg_arch.symtab) str = ast.gen_option_adoc end str @@ -521,11 +496,11 @@ def type_checked_sw_write_ast(symtab, effective_xlen) ) symtab.add( "csr_value", - Idl::Var.new("csr_value", csr.bitfield_type(symtab.archdef, effective_xlen)) + Idl::Var.new("csr_value", csr.bitfield_type(symtab.cfg_arch, effective_xlen)) ) ast = sw_write_ast(symtab) - symtab.archdef.idl_compiler.type_check( + symtab.cfg_arch.idl_compiler.type_check( ast, symtab, "CSR[#{csr.name}].#{name}.sw_write()" @@ -537,7 +512,7 @@ def type_checked_sw_write_ast(symtab, effective_xlen) # @return [Idl::FunctionBodyAst] The abstract syntax tree of the sw_write() function # @return [nil] If there is no sw_write() function - # @param archdef [ArchDef] An architecture definition + # @param cfg_arch [ConfiguredArchitecture] An architecture definition def sw_write_ast(symtab) raise ArgumentError, "Argument should be a symtab" unless symtab.is_a?(Idl::SymbolTable) @@ -545,7 +520,7 @@ def sw_write_ast(symtab) return nil if @data["sw_write(csr_value)"].nil? # now, parse the function - @sw_write_ast = symtab.archdef.idl_compiler.compile_func_body( + @sw_write_ast = symtab.cfg_arch.idl_compiler.compile_func_body( @data["sw_write(csr_value)"], return_type: Idl::Type.new(:bits, width: 128), # big int to hold special return values name: "CSR[#{csr.name}].#{name}.sw_write(csr_value)", @@ -563,17 +538,17 @@ def sw_write_ast(symtab) # @return [Idl::FunctionBodyAst] The abstract syntax tree of the sw_write() function, type checked and pruned # @return [nil] if there is no sw_write() function # @param effective_xlen [Integer] effective xlen, needed because fields can change in different bases - # @param arch_def [ArchDef] A configuration - def pruned_sw_write_ast(arch_def, effective_xlen) + # @param cfg_arch [ConfiguredArchitecture] A configuration + def pruned_sw_write_ast(cfg_arch, effective_xlen) @pruned_sw_write_asts ||= {} - ast = @pruned_sw_write_asts[arch_def.name] + ast = @pruned_sw_write_asts[cfg_arch.name] return ast unless ast.nil? return nil unless @data.key?("sw_write(csr_value)") - raise ArgumentError, "arch_def must be configured to prune" if arch_def.unconfigured? + raise ArgumentError, "cfg_arch must be configured to prune" if cfg_arch.unconfigured? - symtab = arch_def.symtab.global_clone + symtab = cfg_arch.symtab.global_clone symtab.push(ast) # all CSR instructions are 32-bit symtab.add( @@ -586,17 +561,17 @@ def pruned_sw_write_ast(arch_def, effective_xlen) ) symtab.add( "csr_value", - Idl::Var.new("csr_value", csr.bitfield_type(arch_def, effective_xlen)) + Idl::Var.new("csr_value", csr.bitfield_type(cfg_arch, effective_xlen)) ) - ast = type_checked_sw_write_ast(arch_def.symtab, effective_xlen) + ast = type_checked_sw_write_ast(cfg_arch.symtab, effective_xlen) ast = ast.prune(symtab) raise "Symbol table didn't come back at global + 1" unless symtab.levels == 2 - ast.freeze_tree(arch_def.symtab) + ast.freeze_tree(cfg_arch.symtab) - arch_def.idl_compiler.type_check( + cfg_arch.idl_compiler.type_check( ast, symtab, "CSR[#{name}].sw_write(csr_value)" @@ -605,13 +580,13 @@ def pruned_sw_write_ast(arch_def, effective_xlen) symtab.pop symtab.release - @pruned_sw_write_asts[arch_def.name] = ast + @pruned_sw_write_asts[cfg_arch.name] = ast end - # @param arch_def [ArchDef] A config. May be nil if the location is not configturation-dependent + # @param cfg_arch [ConfiguredArchitecture] A config. May be nil if the location is not configturation-dependent # @param effective_xlen [Integer] The effective xlen, needed since some fields change location with XLEN. If the field location is not determined by XLEN, then this parameter can be nil # @return [Range] the location within the CSR as a range (single bit fields will be a range of size 1) - def location(arch_def, effective_xlen = nil) + def location(cfg_arch, effective_xlen = nil) key = if @data.key?("location") "location" @@ -624,14 +599,14 @@ def location(arch_def, effective_xlen = nil) raise "Missing location for #{csr.name}.#{name} (#{key})?" unless @data.key?(key) if @data[key].is_a?(Integer) - csr_length = csr.length(arch_def, effective_xlen || @data["base"]) + csr_length = csr.length(cfg_arch, effective_xlen || @data["base"]) if csr_length.nil? # we don't know the csr length for sure, so we can only check again max_length - if @data[key] > csr.max_length(arch_def) - raise "Location (#{key} = #{@data[key]}) is past the max csr length (#{csr.max_length(arch_def)}) in #{csr.name}.#{name}" + if @data[key] > csr.max_length(cfg_arch) + raise "Location (#{key} = #{@data[key]}) is past the max csr length (#{csr.max_length(cfg_arch)}) in #{csr.name}.#{name}" end elsif @data[key] > csr_length - raise "Location (#{key} = #{@data[key]}) is past the csr length (#{csr.length(arch_def, effective_xlen)}) in #{csr.name}.#{name}" + raise "Location (#{key} = #{@data[key]}) is past the csr length (#{csr.length(cfg_arch, effective_xlen)}) in #{csr.name}.#{name}" end @data[key]..@data[key] @@ -639,11 +614,11 @@ def location(arch_def, effective_xlen = nil) e, s = @data[key].split("-").map(&:to_i) raise "Invalid location" if s > e - csr_length = csr.length(arch_def, effective_xlen || @data["base"]) + csr_length = csr.length(cfg_arch, effective_xlen || @data["base"]) if csr_length.nil? # we don't know the csr length for sure, so we can only check again max_length - if e > csr.max_length(arch_def) - raise "Location (#{key} = #{@data[key]}) is past the max csr length (#{csr.max_length(arch_def)}) in #{csr.name}.#{name}" + if e > csr.max_length(cfg_arch) + raise "Location (#{key} = #{@data[key]}) is past the max csr length (#{csr.max_length(cfg_arch)}) in #{csr.name}.#{name}" end elsif e > csr_length raise "Location (#{key} = #{@data[key]}) is past the csr length (#{csr_length}) in #{csr.name}.#{name}" @@ -666,11 +641,11 @@ def defined_in_base64? = @data["base"].nil? || @data["base"] == 64 # @return [Boolean] Whether or not this field exists for any XLEN def defined_in_all_bases? = @data["base"].nil? - # @param arch_def [ArchDef] A config. May be nil if the width of the field is not configuration-dependent + # @param cfg_arch [ConfiguredArchitecture] A config. May be nil if the width of the field is not configuration-dependent # @param effective_xlen [Integer] The effective xlen, needed since some fields change location with XLEN. If the field location is not determined by XLEN, then this parameter can be nil # @return [Integer] Number of bits in the field - def width(arch_def, effective_xlen) - location(arch_def, effective_xlen).size + def width(cfg_arch, effective_xlen) + location(cfg_arch, effective_xlen).size end def location_cond32 @@ -700,14 +675,14 @@ def location_cond64 end # @return [String] Pretty-printed location string - def location_pretty(arch_def, effective_xlen = nil) + def location_pretty(cfg_arch, effective_xlen = nil) derangeify = proc { |loc| next loc.min.to_s if loc.size == 1 "#{loc.max}:#{loc.min}" } - if dynamic_location?(arch_def) + if dynamic_location?(cfg_arch) condition = case csr.priv_mode when "M" @@ -722,14 +697,14 @@ def location_pretty(arch_def, effective_xlen = nil) if effective_xlen.nil? <<~LOC - * #{derangeify.call(location(arch_def, 32))} when #{condition.sub('%%', '0')} - * #{derangeify.call(location(arch_def, 64))} when #{condition.sub('%%', '1')} + * #{derangeify.call(location(cfg_arch, 32))} when #{condition.sub('%%', '0')} + * #{derangeify.call(location(cfg_arch, 64))} when #{condition.sub('%%', '1')} LOC else - derangeify.call(location(arch_def, effective_xlen)) + derangeify.call(location(cfg_arch, effective_xlen)) end else - derangeify.call(location(arch_def, arch_def.mxlen)) + derangeify.call(location(cfg_arch, cfg_arch.mxlen)) end end @@ -783,7 +758,7 @@ def location_pretty(arch_def, effective_xlen = nil) }.freeze # @return [String] Long description of the field type - def type_desc(arch_def) - TYPE_DESC_MAP[type(arch_def.symtab)] + def type_desc(cfg_arch) + TYPE_DESC_MAP[type(cfg_arch.symtab)] end end diff --git a/lib/arch_obj_models/exception_code.rb b/lib/arch_obj_models/exception_code.rb new file mode 100644 index 000000000..633230315 --- /dev/null +++ b/lib/arch_obj_models/exception_code.rb @@ -0,0 +1,27 @@ +# frozen_string_literal: true + +# a synchroncous exception code +class ExceptionCode + # @return [String] Long-form display name (can include special characters) + attr_reader :name + + # @return [String] Field name for an IDL enum + attr_reader :var + + # @return [Integer] Code, written into *mcause + attr_reader :num + + # @return [Extension] Extension that defines this code + attr_reader :ext + + def initialize(name, var, number, ext) + @name = name + @name.freeze + @var = var + @num = number + @ext = ext + end +end + +# all the same informatin as ExceptinCode, but for interrupts +InterruptCode = Class.new(ExceptionCode) diff --git a/lib/arch_obj_models/extension.rb b/lib/arch_obj_models/extension.rb index 1d978c39b..9d02b671a 100644 --- a/lib/arch_obj_models/extension.rb +++ b/lib/arch_obj_models/extension.rb @@ -2,11 +2,12 @@ require_relative "obj" require_relative "schema" +require_relative "../version" # A parameter (AKA option, AKA implementation-defined value) supported by an extension class ExtensionParameter - # @return [ArchDef] The defining Arch def - attr_reader :archdef + # @return [ConfiguredArchitecture] The defining Arch def + attr_reader :cfg_arch # @return [String] Parameter name attr_reader :name @@ -36,7 +37,7 @@ def schema_type end def initialize(ext, name, data) - @archdef = ext.arch_def + @cfg_arch = ext.cfg_arch @data = data @name = name @desc = data["description"] @@ -45,8 +46,9 @@ def initialize(ext, name, data) also_defined_in = [] unless data["also_defined_in"].nil? if data["also_defined_in"].is_a?(String) - other_ext = @archdef.extension(data["also_defined_in"]) + other_ext = @cfg_arch.extension(data["also_defined_in"]) raise "Definition error in #{ext.name}.#{name}: #{data['also_defined_in']} is not a known extension" if other_ext.nil? + also_defined_in << other_ext else unless data["also_defined_in"].is_a?(Array) && data["also_defined_in"].all? { |e| e.is_a?(String) } @@ -54,8 +56,9 @@ def initialize(ext, name, data) end data["also_defined_in"].each do |other_ext_name| - other_ext = @archdef.extension(other_ext_name) + other_ext = @cfg_arch.extension(other_ext_name) raise "Definition error in #{ext.name}.#{name}: #{data['also_defined_in']} is not a known extension" if other_ext.nil? + also_defined_in << other_ext end end @@ -64,10 +67,15 @@ def initialize(ext, name, data) @idl_type = @schema.to_idl_type.make_const.freeze end + # @param version [ExtensionVersion] + # @return [Boolean] if this parameter is defined in +version+ def defined_in_extension_version?(version) + return false if @exts.none? { |ext| ext.name == version.ext.name } return true if @data.dig("when", "version").nil? - Gem::Requirement.new(@data["when"]["version"]).satisfied_by?(Gem::Version.new(version)) + @exts.any? do |ext| + ExtensionRequirement.new(ext.name, @data["when"]["version"], cfg_arch: ext.cfg_arch).satisfied_by?(version) + end end # @return [String] @@ -78,7 +86,7 @@ def name_potentially_with_link(exts) if exts.size == 1 "<>" else - "#{name}" + name end end @@ -117,9 +125,9 @@ def initialize(param, value) end # Extension definition -class Extension < ArchDefObject - # @return [ArchDef] The architecture defintion - attr_reader :arch_def +class Extension < DatabaseObjectect + # @return [ConfiguredArchitecture] The architecture defintion + attr_reader :cfg_arch # @return [String] Long name of the extension def long_name = @data["long_name"] @@ -136,13 +144,15 @@ def doc_license @data["doc_license"] end - # @return [Array] versions hash from config + # @return [Array] versions hash from config, sorted by version number def versions return @versions unless @versions.nil? @versions = @data["versions"].map do |v| - ExtensionVersion.new(name, v["version"], arch_def) + ExtensionVersion.new(name, v["version"], cfg_arch) end + @versions.sort! + @versions end # @return [Array] Ratified versions hash from config @@ -152,12 +162,12 @@ def ratified_versions # @return [ExtensionVersion] Mimumum defined version of this extension def min_version - versions.min { |a, b| a.version <=> b.version } + versions.min { |a, b| a.version_spec <=> b.version_spec } end # @return [ExtensionVersion] Maximum defined version of this extension def max_version - versions.max { |a, b| a.version <=> b.version } + versions.max { |a, b| a.version_spec <=> b.version_spec } end # @return [ExtensionVersion] Mimumum defined ratified version of this extension @@ -165,7 +175,7 @@ def max_version def min_ratified_version return nil if ratified_versions.empty? - ratified_versions.min { |a, b| a.version <=> b.version } + ratified_versions.min { |a, b| a.version_spec <=> b.version_spec } end # @return [Array] List of parameters added by this extension @@ -181,61 +191,53 @@ def params @params end - # @param ext_data [Hash] The extension data from the architecture spec - # @param arch_def [ArchDef] The architecture definition - def initialize(ext_data, arch_def) - super(ext_data) - @arch_def = arch_def - end - # @param version_requirement [String] Version requirement # @return [Array] Array of extensions implied by any version of this extension meeting version_requirement - def implies(version_requirement = ">= 0") - return [] unless Gem::Requirement.new(version_requirement).satisfied_by?(max_version.version) + def implies(version_requirement = nil) + if version_requirement.nil? + return [] unless ExtensionRequirement.new(@new, cfg_arch: @cfg_arch).satisfied_by?(max_version.version) + else + return [] unless ExtensionRequirement.new(@new, version_requirement, cfg_arch: @cfg_arch).satisfied_by?(max_version.version) + end max_version.implications end + # @return [Array] List of conflicting extension requirements def conflicts return [] if @data["conflicts"].nil? - to_extension_requirement_list(@data["conflicts"]) + if @data["conflicts"].is_a?(String) + [ExtensionRequirement.new(@data["conflicts"], cfg_arch: @cfg_arch)] + elsif @data["conflicts"].is_a?(Hash) + [ExtensionRequirement.new(@data["conflicts"]["name"], @data["conflicts"]["version"], cfg_arch: @cfg_arch)] + elsif @data["conflicts"].is_a?(Array) + @data["conflicts"].map do |conflict| + if conflict.is_a?(String) + ExtensionRequirement.new(conflict, cfg_arch: @cfg_arch) + elsif conflict.is_a?(Array) + ExtensionRequirement.new(conflict["name"], conflict["version"], cfg_arch: @cfg_arch) + else + raise "Invalid conflicts data: #{conflict.inspect}" + end + end + else + raise "Invalid conflicts data: #{@data["conflicts"].inspect}" + end end # @return [Array] the list of instructions implemented by *any version* of this extension (may be empty) def instructions return @instructions unless @instructions.nil? - @instructions = arch_def.instructions.select { |i| versions.any? { |v| i.defined_by?(v) }} + @instructions = cfg_arch.instructions.select { |i| versions.any? { |v| i.defined_by?(v) }} end # @return [Array] the list of CSRs implemented by *any version* of this extension (may be empty) def csrs return @csrs unless @csrs.nil? - @csrs = arch_def.csrs.select { |csr| versions.any? { |v| csr.defined_by?(v) } } - end - - # @return [Array] the list of CSRs implemented by this extension (may be empty) - def implemented_csrs(archdef) - raise "should only be called with a fully configured arch def" unless archdef.fully_configured? - - return @implemented_csrs unless @implemented_csrs.nil? - - @implemented_csrs = archdef.implemented_csrs.select do |csr| - versions.any? { |ver| csr.defined_by?(ExtensionVersion.new(name, ver["version"], @arch_def)) } - end - end - - # @return [Array] the list of CSRs implemented by this extension (may be empty) - def implemented_instructions(archdef) - raise "should only be called with a fully configured arch def" unless archdef.fully_configured? - - return @implemented_instructions unless @implemented_instructions.nil? - - @implemented_instructions = archdef.implemented_instructions.select do |inst| - versions.any? { |ver| inst.defined_by?(ExtensionVersion.new(name, ver["version"], @arch_def)) } - end + @csrs = cfg_arch.csrs.select { |csr| versions.any? { |v| csr.defined_by?(v) } } end # return the set of reachable functions from any of this extensions's CSRs or instructions in the given evaluation context @@ -257,27 +259,11 @@ def reachable_functions(symtab) end csrs.each do |csr| - funcs += csr.reachable_functions(arch_def) + funcs += csr.reachable_functions(cfg_arch) end @reachable_functions[symtab] = funcs.uniq end - - # @return [Array] Array of IDL functions reachable from any instruction or CSR in the extension, irrespective of a specific evaluation context - def reachable_functions_unevaluated - return @reachable_functions_unevaluated unless @reachable_functions_unevaluated.nil? - - funcs = [] - instructions.each do |inst| - funcs += inst.operation_ast(arch_def.symtab).reachable_functions(arch_def.symtab) - end - - csrs.each do |csr| - funcs += csr.reachable_functions(arch_def) - end - - @reachable_functions_unevaluated = funcs.uniq(&:name) - end end # A specific version of an extension @@ -285,28 +271,79 @@ class ExtensionVersion # @return [String] Name of the extension attr_reader :name - # @return [Gem::Version] Version of the extension - attr_reader :version - # @return [Extension] Extension attr_reader :ext + # @return [VersionSpec] + attr_reader :version_spec + + # @return [String] + attr_reader :version_str + # @param name [#to_s] The extension name - # @param version [Integer,String] The version specifier - # @param arch_def [ArchDef] The architecture definition - def initialize(name, version, arch_def) + # @param version [String] The version specifier + # @param cfg_arch [ConfiguredArchitecture] The architecture definition + def initialize(name, version_str, cfg_arch, fail_if_version_does_not_exist: false) @name = name.to_s - @version = Gem::Version.new(version) - @arch_def = arch_def - unless arch_def.nil? - @ext = arch_def.extension(@name) - raise "Extension #{name} not found in arch def" if @ext.nil? - - @data = @ext.data["versions"].find { |v| v["version"] == version.to_s } - raise "Extension #{name} version #{version} not found in arch def" if @data.nil? + @version_str = version_str + @version_spec = VersionSpec.new(version_str) + + raise ArgumentError, "Must supply arch" if cfg_arch.nil? + + @cfg_arch = cfg_arch + + @ext = @cfg_arch.extension(@name) + raise "Extension #{name} not found in arch def" if @ext.nil? + + @data = @ext.data["versions"].find { |v| VersionSpec.new(v["version"]) == @version_spec } + + if fail_if_version_does_not_exist && @data.nil? + raise ArgumentError, "#{@name}, Version #{version_str} is not defined" + elsif @data.nil? + warn "#{@name}, Version #{version_str} is not defined" end end + # @return [Array] List of known ExtensionVersions that are compatible with this ExtensionVersion (i.e., have larger version number and are not breaking) + def compatible_versions + return @compatible_versions unless @compatible_versions.nil? + + @compatible_versions = [] + @ext.versions.each do |v| + @compatible_versions << v if v.version_spec >= @version_spec + break if @compatible_versions.size.positive? && v.breaking? + end + raise "Didn't even find self?" if compatible_versions.empty? + + @compatible_versions + end + + # @param other [ExtensionVersion] + # @return [Boolean] Whether or not +other+ is compatible with self + def compatible?(other) = compatible_versions.include?(other) + + # @return [Boolean] Whether or not this is a breaking version (i.e., incompatible with all prior versions) + def breaking? + !@data["breaking"].nil? + end + + # @return [String] Canonical version string + def canonical_version = @version_spec.canonical + + # @param other [ExtensionVersion] An extension name and version + # @return [Boolean] whether or not this ExtensionVersion has the exact same name and version as other + def eql?(other) + raise "ExtensionVersion is not comparable to #{other.class}" unless other.is_a?(ExtensionVersion) + + @ext.name == other.ext.name && @version_spec.eql?(other.version_spec) + end + + # @param other [ExtensionVersion] An extension name and version + # @return [Boolean] whether or not this ExtensionVersion has the exact same name and version as other + def ==(other) + eql?(other) + end + # @return [String] The state of the extension version ('ratified', 'developemnt', etc) def state = @data["state"] @@ -316,6 +353,7 @@ def changes = @data["changes"].nil? ? [] : @data["changes"] def url = @data["url"] + # @return [Array] List of contributors to this extension version def contributors return @contributors unless @contributors.nil? @@ -328,57 +366,70 @@ def contributors # @return [Array] The list of parameters for this extension version def params - @ext.params.select { |p| p.defined_in_extension_version?(@version) } + @ext.params.select { |p| p.defined_in_extension_version?(self) } end - def to_s - "#{name}@#{version}" + # @return [String] formatted like the RVI manual + # + # @example + # ExtensionVersion.new("A", "2.2").to_rvi_s #=> "A2p2" + def to_rvi_s + "#{name}#{@version_spec.to_rvi_s}" end - # @overload ==(other) - # @param other [String] An extension name - # @return [Boolean] whether or not this ExtensionVersion is named 'other' - # @overload ==(other) - # @param other [ExtensionVersion] An extension name and version - # @return [Boolean] whether or not this ExtensionVersion has the exact same name and version as other - def ==(other) - case other - when String - @name == other - when ExtensionVersion - @name == other.name && @version == other.version - else - raise "Unexpected comparison" - end + # @return [String] Ext@Version + def to_s + "#{name}@#{@version_spec.canonical}" + end + + # @return [SchemaCondition] Condition that must be met for this version to be allowed. + # Transitively includes any requirements from an implied extension. + def requirement_condition + @requirement_condition ||= + begin + r = case @data["requires"] + when nil + AlwaysTrueSchemaCondition.new + when Hash + SchemaCondition.new(@data["requires"], @cfg_arch) + else + SchemaCondition.new({ "oneOf" => [@data["requires"]] }, @cfg_arch) + end + if @data.key?("implies") + rs = [r] + implications.map(&:requirement_condition) + rs = rs.reject(&:empty?) + r = SchemaCondition.all_of(*rs.map(&:to_h)) unless rs.empty? + end + r + end end - # @param other [ExtensionVersion] Comparison - # @return [Boolean] Whether or not +other+ is an ExtensionVersion with the same name and version - def eql?(other) - return false unless other.is_a?(ExtensionVersion) - - @name == other.name && @version == other.version + # @return [Array] List of extensions that conflict with this ExtensionVersion + # The list is *not* transitive; if conflict C1 implies C2, + # only C1 shows up in the list + def conflicts + @conflicts ||= extension.conflicts.map(&:satisfying_versions).flatten.uniq.sort end - def requirements - r = case @data["requires"] - when nil - AlwaysTrueSchemaCondition.new - when Hash - SchemaCondition.new(@data["requires"]) - else - SchemaCondition.new({"oneOf" => [@data["requires"]]}) - end - if @data.key?("implies") - rs = [r] + implications.map { |e| e.requirements } - rs = rs.reject { |r| r.empty? } - unless rs.empty? - r = SchemaCondition.all_of(*rs.map { |r| r.to_h }) - end + # @return [Array] List of extensions that conflict with this ExtensionVersion + # The list *is* transitive; if conflict C1 implies C2, + # both C1 and C2 show up in the list + def transitive_conflicts + return @transitive_conflicts unless @transive_conflicts.nil? + + @transitive_conflicts = [] + conflicts.each do |c| + @transitive_conflicts << c + @transitive_conflicts.concat(c.transitive_implications) end - r + @transitive_conflicts.uniq! + @transitive_conflicts.sort! + @transitive_conflicts end + # @return [Array] List of extension versions that are implied by with this ExtensionVersion + # This list is *not* transitive; if an implication I1 implies another extension I2, + # only I1 shows up in the list def implications return @implications unless @implications.nil? @@ -388,52 +439,83 @@ def implications return @implications when Array if @data["implies"][0].is_a?(Array) - @implications += @data["implies"].map { |e| ExtensionVersion.new(e[0], e[1], @arch_def) } + @implications.concat(@data["implies"].map { |e| ExtensionVersion.new(e[0], e[1], @cfg_arch) }) else - @implications << ExtensionVersion.new(@data["implies"][0], @data["implies"][1], @arch_def) + @implications << ExtensionVersion.new(@data["implies"][0], @data["implies"][1], @cfg_arch) end end - @implications.uniq! + @implications.sort! @implications end + # @return [Array] List of extension versions that are implied by with this ExtensionVersion + # This list is transitive; if an implication I1 implies another extension I2, + # both I1 and I2 are in the returned list + def transitive_implications + return @transitive_implications unless @transitive_implications.nil? + + @transitive_implications = [] + case @data["implies"] + when nil + return @transitive_implications + when Array + if @data["implies"][0].is_a?(Array) + impls = @data["implies"].map { |e| ExtensionVersion.new(e[0], e[1], @cfg_arch) } + @transitive_implications.concat(impls) + impls.each do |i| + transitive_impls = i.implications + @transitive_implications.concat(transitive_impls) unless transitive_impls.empty? + end + else + impl = ExtensionVersion.new(@data["implies"][0], @data["implies"][1], @cfg_arch) + @transitive_implications << impl + transitive_impls = impl.implications + @transitive_implications.concat(transitive_impls) unless transitive_impls.empty? + end + end + @transitive_implications.uniq! + @transitive_implications.sort! + @transitive_implications + end + # @param ext_name [String] Extension name - # @param ext_version_requirements [Number,String,Array] Extension version requirements, taking the same inputs as Gem::Requirement - # @see https://docs.ruby-lang.org/en/3.0/Gem/Requirement.html#method-c-new Gem::Requirement#new + # @param ext_version_requirements [String,Array] Extension version requirements # @return [Boolean] whether or not this ExtensionVersion is named `ext_name` and satifies the version requirements def satisfies?(ext_name, *ext_version_requirements) - @name == ext_name && Gem::Requirement.new(ext_version_requirements).satisfied_by?(@version) + ExtensionRequirement.new(ext_name, ext_version_requirements).satisfied_by?(self) end # sorts extension by name, then by version def <=>(other) - raise ArgumentError, "ExtensionVersions are only comparable to other extension versions" unless other.is_a?(ExtensionVersion) + unless other.is_a?(ExtensionVersion) + raise ArgumentError, "ExtensionVersions are only comparable to other extension versions" + end if other.name != @name @name <=> other.name else - @version <=> other.version + @version_spec <=> other.version_spec end end # @return [Array] the list of CSRs implemented by this extension version (may be empty) - def implemented_csrs(archdef) - raise "should only be called with a fully configured arch def" unless archdef.fully_configured? - + def implemented_csrs return @implemented_csrs unless @implemented_csrs.nil? - @implemented_csrs = archdef.implemented_csrs.select do |csr| + raise "implemented_csrs needs an cfg_arch" if @cfg_arch.nil? + + @implemented_csrs = @cfg_arch.csrs.select do |csr| csr.defined_by?(self) end end - # @return [Array] the list of CSRs implemented by this extension version (may be empty) - def implemented_instructions(archdef) - raise "should only be called with a fully configured arch def" unless archdef.fully_configured? - + # @return [Array] the list of insts implemented by this extension version (may be empty) + def implemented_instructions return @implemented_instructions unless @implemented_instructions.nil? - @implemented_instructions = archdef.implemented_instructions.select do |inst| + raise "implemented_instructions needs an cfg_arch" if @cfg_arch.nil? + + @implemented_instructions = @cfg_arch.instructions.select do |inst| inst.defined_by?(self) end end @@ -555,41 +637,63 @@ def <=>(other) class ExtensionRequirement # @return [String] Extension name attr_reader :name - attr_reader :note # Optional note. Can be nil. - attr_reader :req_id # Optional Requirement ID. Can be nil. - attr_reader :presence # Optional presence (e.g., mandatory, optional, etc.). Can be nil. - # @return [Gem::Requirement] Version requirement - def version_requirement - @requirement - end + # @return [String,nil] Optional note + attr_reader :note + + # @return [String,nil] Optional Requirement ID. + attr_reader :req_id + + # @return [String,nil], Optional presence (e.g., mandatory, optional, etc.) + attr_reader :presence + + # @return [Array] Set of requirement specifications + def requirement_specs = @requirements def to_s - "#{name} #{@requirement}" + "#{name} #{@requirements.map(&:to_s).join(', ')}" + end + + # @return [Extension] The extension that this requirement is for + def extension + return @extension unless @extension.nil? + + raise "Cannot get extension; cfg_arch was not initialized" if @cfg_arch.nil? + + @extension = @cfg_arch.extension(@name) end # @param name [#to_s] Extension name - # @param requirements (see Gem::Requirement#new) - def initialize(name, *requirements, note: nil, req_id: nil, presence: nil) - @name = name.to_s + # @param requirements [String] Single requirement + # @param requirements [Array] List of requirements, all of which must hold + def initialize(name, *requirements, cfg_arch: nil, note: nil, req_id: nil, presence: nil) + raise ArgumentError, "Arch is required" if cfg_arch.nil? + + @name = name.to_s.freeze + @cfg_arch = cfg_arch + @ext = @cfg_arch.extension(@name) + + raise ArgumentError, "Could not find extension named '#{@name}'" if @ext.nil? + requirements = if requirements.empty? - [">= 0"] + ["~> #{@ext.min_version.version_str}"] else requirements end - @requirement = Gem::Requirement.new(requirements) - @note = note - @req_id = req_id - @presence = presence + @requirements = requirements.map { |r| RequirementSpec.new(r) } + + @note = note.freeze + @req_id = req_id.freeze + @presence = presence.freeze end # @return [Array] The list of extension versions that satisfy this requirement - def satisfying_versions(archdef) - ext = archdef.extension(@name) + def satisfying_versions + ext = @cfg_arch.extension(@name) return [] if ext.nil? - ext.versions.select { |v| @requirement.satisfied_by?(v.version) } + ext.versions.select { |v| @requirements.all? { |r| r.satisfied_by?(v.version_spec, ext) } } end # @overload @@ -605,11 +709,16 @@ def satisfying_versions(archdef) def satisfied_by?(*args) if args.size == 1 if args[0].is_a?(ExtensionVersion) - args[0].name == @name && - @requirement.satisfied_by?(Gem::Version.new(args[0].version)) + return false if args[0].name != @name + + @requirements.all? { |r| r.satisfied_by?(args[0].version_spec, @ext) } elsif args[0].is_a?(ExtensionRequirement) - satisfying_versions.all? do |ext_ver| - satified_by?(ext_ver) + return false if args[0].name != @name + + @requirements.all? do |r| + args[0].satisfying_versions.all? do |ext_ver| + r.satisfied_by?(ext_ver.version_spec, @ext) + end end else raise ArgumentError, "Single argument must be an ExtensionVersion or ExtensionRquirement" @@ -618,19 +727,18 @@ def satisfied_by?(*args) raise ArgumentError, "First parameter must be an extension name" unless args[0].respond_to?(:to_s) raise ArgumentError, "First parameter must be an extension version" unless args[1].respond_to?(:to_s) - args[0] == @name && - @requirement.satisfied_by?(Gem::Version.new(args[1])) + return false if args[0] != @name + + @requirements.all? { |r| r.satisfied_by?(args[1], @ext) } else raise ArgumentError, "Wrong number of args (expecting 1 or 2)" end end # @return [Array] List of CSRs defined by any extension satisfying this requirement - def csrs(arch_def) - return @csrs unless @csrs.nil? - - @csrs = arch_def.csrs.select do |csr| - satisfying_versions(arch_def).any? do |ext_ver| + def csrs + @csrs ||= @cfg_arch.csrs.select do |csr| + satisfying_versions.any? do |ext_ver| csr.defined_by?(ext_ver) end end diff --git a/lib/arch_obj_models/instruction.rb b/lib/arch_obj_models/instruction.rb index ee347bb4b..ea8a6470d 100644 --- a/lib/arch_obj_models/instruction.rb +++ b/lib/arch_obj_models/instruction.rb @@ -6,7 +6,59 @@ # model of a specific instruction in a specific base (RV32/RV64) -class Instruction < ArchDefObject +class Instruction < DatabaseObjectect + def self.ary_from_location(location_str_or_int) + return [location_str_or_int] if location_str_or_int.is_a?(Integer) + + bits = [] + parts = location_str_or_int.split("|") + parts.each do |part| + if part.include?("-") + msb, lsb = part.split("-").map(&:to_i) + (lsb..msb).each { |i| bits << i } + else + bits << part.to_i + end + end + bits + end + + def self.validate_encoding(encoding, inst_name) + match = encoding["match"] + raise "No match for instruction #{inst_name}?" if match.nil? + + variables = encoding.key?("variables") ? encoding["variables"] : [] + match.size.times do |i| + if match[match.size - 1 - i] == "-" + # make sure exactly one variable covers this bit + vars_match = variables.count { |variable| ary_from_location(variable["location"]).include?(i) } + if vars_match.zero? + raise ValidationError, "In instruction #{inst_name}, no variable or encoding bit covers bit #{i}" + elsif vars_match != 1 + raise ValidationError, "In instruction, #{inst_name}, bit #{i} is covered by more than one variable" + end + else + # make sure no variable covers this bit + unless variables.nil? + unless variables.none? { |variable| ary_from_location(variable["location"]).include?(i) } + raise ValidationError, "In instruction, #{inst_name}, bit #{i} is covered by both a variable and the match string" + end + end + end + end + end + + def validate + super + + if @data["encoding"]["RV32"].nil? + Instruction.validate_encoding(@data["encoding"], name) + else + Instruction.validate_encoding(@data["encoding"]["RV32"], name) + Instruction.validate_encoding(@data["encoding"]["RV64"], name) + end + end + def ==(other) if other.is_a?(Instruction) name == other.name @@ -85,25 +137,25 @@ def fill_symtab(global_symtab, effective_xlen, ast) def pruned_operation_ast(global_symtab, effective_xlen) @pruned_asts ||= {} - arch_def = global_symtab.archdef + cfg_arch = global_symtab.cfg_arch - pruned_ast = @pruned_asts[arch_def.name] + pruned_ast = @pruned_asts[cfg_arch.name] return pruned_ast unless pruned_ast.nil? return nil unless @data.key?("operation()") - type_checked_ast = type_checked_operation_ast(arch_def.idl_compiler, global_symtab, effective_xlen) + type_checked_ast = type_checked_operation_ast(cfg_arch.idl_compiler, global_symtab, effective_xlen) print "Pruning #{name} operation()..." pruned_ast = type_checked_ast.prune(fill_symtab(global_symtab, effective_xlen, type_checked_ast)) puts "done" pruned_ast.freeze_tree(global_symtab) - arch_def.idl_compiler.type_check( + cfg_arch.idl_compiler.type_check( pruned_ast, fill_symtab(global_symtab, effective_xlen, pruned_ast), "#{name}.operation() (pruned)" ) - @pruned_asts[arch_def.name] = pruned_ast + @pruned_asts[cfg_arch.name] = pruned_ast end # @param symtab [Idl::SymbolTable] Symbol table with global scope populated @@ -114,7 +166,7 @@ def reachable_functions(symtab, effective_xlen) [] else # RubyProf.start - ast = type_checked_operation_ast(symtab.archdef.idl_compiler, symtab, effective_xlen) + ast = type_checked_operation_ast(symtab.cfg_arch.idl_compiler, symtab, effective_xlen) print "Determining reachable funcs from #{name}..." fns = ast.reachable_functions(fill_symtab(symtab, effective_xlen, ast)) puts "done" @@ -134,7 +186,7 @@ def reachable_exceptions(symtab, effective_xlen) else # pruned_ast = pruned_operation_ast(symtab) # type_checked_operation_ast() - type_checked_ast = type_checked_operation_ast(symtab.arch_def.idl_compiler, symtab, effective_xlen) + type_checked_ast = type_checked_operation_ast(symtab.cfg_arch.idl_compiler, symtab, effective_xlen) symtab = fill_symtab(symtab, effective_xlen, pruned_ast) type_checked_ast.reachable_exceptions(symtab) end @@ -162,7 +214,7 @@ def reachable_exceptions_str(symtab, effective_xlen=nil) else etype = symtab.get("ExceptionCode") if effective_xlen.nil? - if symtab.archdef.multi_xlen? + if symtab.cfg_arch.multi_xlen? if base.nil? ( pruned_ast = pruned_operation_ast(symtab, 32) @@ -189,7 +241,7 @@ def reachable_exceptions_str(symtab, effective_xlen=nil) e end else - effective_xlen = symtab.archdef.mxlen + effective_xlen = symtab.cfg_arch.mxlen pruned_ast = pruned_operation_ast(symtab, effective_xlen) print "Determining reachable exceptions from #{name}..." e = mask_to_array(pruned_ast.reachable_exceptions(fill_symtab(symtab, effective_xlen, pruned_ast))).map { |code| @@ -211,14 +263,6 @@ def reachable_exceptions_str(symtab, effective_xlen=nil) end end - # @return [ArchDef] The architecture definition - attr_reader :arch_def - - def initialize(data, arch_def) - super(data) - @arch_def = arch_def - end - # represents a single contiguous instruction encoding field # Multiple EncodingFields may make up a single DecodeField, e.g., when an immediate # is split across multiple locations @@ -600,10 +644,10 @@ def operation_ast(symtab) return nil if @data["operation()"].nil? # now, parse the operation - @operation_ast = symtab.archdef.idl_compiler.compile_inst_operation( + @operation_ast = symtab.cfg_arch.idl_compiler.compile_inst_operation( self, symtab:, - input_file: @data["__source"], + input_file: @data["$source"], input_line: source_line("operation()") ) @@ -675,39 +719,38 @@ def rv64? def excluded_by?(*args) return false if @data["excludedBy"].nil? - excluded_by = SchemaCondition.new(@data["excludedBy"]) + excluded_by = SchemaCondition.new(@data["excludedBy"], @cfg_arch) - if args.size == 1 - raise ArgumentError, "Parameter must be an ExtensionVersion" unless args[0].is_a?(ExtensionVersion) + ext_ver = + if args.size == 1 + raise ArgumentError, "Parameter must be an ExtensionVersion" unless args[0].is_a?(ExtensionVersion) - excluded_by.satisfied_by? do |r| - r.name == args[0].name && r.version_requirement.satisfied_by?(args[0].version) - end - elsif args.size == 2 - raise ArgumentError, "First parameter must be an extension name" unless args[0].respond_to?(:to_s) - raise ArgumentError, "Second parameter must be an extension version" unless args[0].respond_to?(:to_s) - - version = args[1].is_a?(Gem::Version) ? args[1] : Gem::Version.new(args[1]) + args[0] + elsif args.size == 2 + raise ArgumentError, "First parameter must be an extension name" unless args[0].respond_to?(:to_s) + raise ArgumentError, "Second parameter must be an extension version" unless args[1].respond_to?(:to_s) - excluded_by.satisfied_by? do |r| - r.name == args[0] && r.version_requirement.satisfied_by?(version) + ExtensionVersion.new(args[0], args[1], @cfg_arch) end + + excluded_by.satisfied_by? do |r| + r.satisfied_by?(ext_ver) end end - # @param arch_def [ArchDef] The architecture definition + # @param cfg_arch [ConfiguredArchitecture] The architecture definition # @return [Boolean] whether or not the instruction is implemented given the supplies config options - def exists_in_cfg?(arch_def) - if arch_def.fully_configured? - (@data["base"].nil? || (arch_def.possible_xlens.include? @data["base"])) && - arch_def.implemented_extensions.any? { |e| defined_by?(e) } && - arch_def.implemented_extensions.none? { |e| excluded_by?(e) } + def exists_in_cfg?(cfg_arch) + if cfg_arch.fully_configured? + (@data["base"].nil? || (cfg_arch.possible_xlens.include? @data["base"])) && + cfg_arch.implemented_extensions.any? { |e| defined_by?(e) } && + cfg_arch.implemented_extensions.none? { |e| excluded_by?(e) } else - raise "unexpected arch_def type" unless arch_def.partially_configured? + raise "unexpected cfg_arch type" unless cfg_arch.partially_configured? - (@data["base"].nil? || (arch_def.possible_xlens.include? @data["base"])) && - arch_def.prohibited_extensions.none? { |e| defined_by?(e) } && - arch_def.mandatory_extensions.none? { |e| excluded_by?(e) } + (@data["base"].nil? || (cfg_arch.possible_xlens.include? @data["base"])) && + cfg_arch.prohibited_extensions.none? { |e| defined_by?(e) } && + cfg_arch.mandatory_extensions.none? { |e| excluded_by?(e) } end end end diff --git a/lib/arch_obj_models/manual.rb b/lib/arch_obj_models/manual.rb index 86a99576a..0a276c5f6 100644 --- a/lib/arch_obj_models/manual.rb +++ b/lib/arch_obj_models/manual.rb @@ -4,21 +4,16 @@ require_relative "obj" -class Manual < ArchDefObject - def initialize(data, arch_def) - super(data) - @arch_def = arch_def - end - +class Manual < DatabaseObjectect def versions return @versions unless @versions.nil? - @versions = [] - @data["versions"].each do |version| - @versions << ManualVersion.new(version, self, @arch_def) - end - - @versions + @versions = + if @cfg_arch.nil? + @specification.manual_versions.select { |mv| mv.manual == self } + else + @cfg_arch.manual_versions.select { |mv| mv.manual == self } + end end def version(name) @@ -27,6 +22,12 @@ def version(name) # @return [String] The title of the manual, as used by marketing def marketing_name = @data["marketing_name"] + + # for manuals that reference an external repo, set the url to that repo data (file path) + def repo_path=(path) + @repo_path = Pathname.new(path) + versions.each { |v| v.repo_path = @repo_path } + end end class ManualChapter @@ -34,23 +35,30 @@ def initialize(volume, path) @volume = volume @version = volume.version - fullpath = "#{@version.path}/#{path}" - raise "Path '#{fullpath}' does not exist" unless File.exist?(fullpath) - - @path = fullpath + @path = Pathname.new path end def name - File.basename(@path, ".adoc") + @path.basename(".adoc").to_s end def title return @title unless @title.nil? - @title = (Asciidoctor.load File.read(path).scrub).doctitle.encode("US-ASCII") + @title = (Asciidoctor.load File.read(fullpath).scrub).doctitle.encode("US-ASCII") end - # @return [String] The absolute path to the chapter + def fullpath + raise "Must call repo_path= first" if @repo_path.nil? + + @repo_path / @path + end + + def repo_path=(path) + @repo_path = path + end + + # @return [Pathname] The relative path to the chapter attr_reader :path end @@ -58,8 +66,7 @@ class ManualVolume # @return [ManualVersion] The version this volume belongs to attr_reader :version - # @return [ArchDef] The architecture definition - def arch_def = version.arch_def + def cfg_arch = version.cfg_arch def initialize(data, version) @data = data @@ -91,34 +98,43 @@ def extensions return @extensions if @data["extensions"].nil? @data["extensions"].each do |ext| - ext_obj = arch_def.extension(ext[0]) + ext_obj = cfg_arch.extension(ext[0]) if ext_obj.nil? warn "Extension '#{ext[0]}' is not in the database" next end - unless ext_obj.versions.any? { |v| v.version == ext[1] } + ext_ver = ExtensionVersion.new(ext[0], ext[1], cfg_arch) + unless ext_obj.versions.any? { |known_ver| known_ver == ext_ver } warn "Extension '#{ext[0]}', version '#{ext[1]}' is not defined in the database" next end - @extensions << ExtensionVersion.new(ext[0], ext[1], arch_def) + @extensions << ext_ver end @extensions end + + def repo_path=(path) + @repo_path = path + chapters.each { |c| c.repo_path = path } + end end -class ManualVersion < ArchDefObject +class ManualVersion < DatabaseObjectect # @return [Manual] The manual this version belongs to - attr_reader :manual - - # @return [ArchDef] The architecture definition - attr_reader :arch_def + def manual + return @manual unless @manual.nil? + + @manual = + if @cfg_arch.nil? + @specification.ref(@data["manual"]["$ref"]) + else + @cfg_arch.ref(@data["manual"]["$ref"]) + end + raise "Error: manual #{@data['manual']['$ref']} is not found" if @manual.nil? - def initialize(data, manual, arch_def) - super(data) - @manual = manual - @arch_def = arch_def + @manual end # @return [String] Semantic version number @@ -129,7 +145,7 @@ def marketing_version = @data["marketing_version"] # @return [String] Path to the directory containing contents.yaml file for this version def path - File.dirname(@data["__source"]) + File.dirname(@data["$source"]) end # @return [Boolean] Whether or not this version is using riscv-isa-manual as a source @@ -165,7 +181,7 @@ def instructions @instructions = [] extensions.each do |ext| - ext_obj = @arch_def.extension(ext.name) + ext_obj = @cfg_arch.extension(ext.name) ext_obj.instructions.each do |inst| @instructions << inst end @@ -179,11 +195,16 @@ def csrs @csrs = [] extensions.each do |ext| - ext_obj = @arch_def.extension(ext.name) + ext_obj = @cfg_arch.extension(ext.name) ext_obj.csrs.each do |csr| @csrs << csr end end @csrs = @csrs.uniq(&:name) end + + def repo_path=(path) + @repo_path = path + volumes.each { |v| v.repo_path = path } + end end diff --git a/lib/arch_obj_models/obj.rb b/lib/arch_obj_models/obj.rb index 2e6b62176..567893b5d 100644 --- a/lib/arch_obj_models/obj.rb +++ b/lib/arch_obj_models/obj.rb @@ -11,7 +11,7 @@ # ... # } # -# obj = ArchDefObject.new(data) +# obj = DatabaseObjectect.new(data) # obj['name'] # 'mstatus' # obj['address'] # 0x320 # @@ -24,8 +24,140 @@ # Subclasses may override the accessors when a more complex data structure # is warranted, e.g., the CSR Field 'alias' returns a CsrFieldAlias object # instead of a simple string -class ArchDefObject - attr_reader :data, :name, :long_name, :description +class DatabaseObjectect + # Exception raised when there is a problem with a schema file + class SchemaError < ::StandardError + # result from JsonSchemer.validate + attr_reader :result + + def initialize(result) + if result.is_a?(Enumerator) + super(result.to_a.map { |e| "At #{e['schema_pointer']}: #{e['type']}" }) + else + super(result["error"]) + end + @result = result + end + end + + # exception raised when an object does not validate against its schema + class SchemaValidationError < ::StandardError + + # result from JsonSchemer.validate + attr_reader :result + + # create a new SchemaValidationError + # + # @param result [JsonSchemer::Result] JsonSchemer result + def initialize(path, result) + msg = "While validating #{path}:\n\n" + nerrors = result.count + msg << "#{nerrors} error(s) during validations\n\n" + result.to_a.each do |r| + msg << + if r["type"] == "required" && !r.dig("details", "missing_keys").nil? + " At '#{r['data_pointer']}': Missing required parameter(s) '#{r['details']['missing_keys']}'\n" + elsif r["type"] == "schema" + if r["schema_pointer"] == "/additionalProperties" + " At #{r['data_pointer']}, there is an unallowed additional key\n" + else + " At #{r['data_pointer']}, endpoint is an invalid key\n" + end + elsif r["type"] == "enum" + " At #{r['data_pointer']}, '#{r['data']}' is not a valid enum value (#{r['schema']['enum']})\n" + elsif r["type"] == "maxProperties" + " Maximum number of properties exceeded\n" + elsif r["type"] == "object" + " At #{r['data_pointer']}, Expecting object, got #{r['data']}\n" + elsif r["type"] == "pattern" + " At #{r['data_pointer']}, RegEx validation failed; '#{r['data']}' does not match '#{r['schema']['pattern']}'\n" + elsif r["type"] == "integer" + " At #{r['data_pointer']}, '#{r['data']}' is not a integer\n" + elsif r["type"] == "array" + " At #{r['data_pointer']}, '#{r['data']}' is not a array\n" + elsif r["type"] == "oneOf" + " At #{r['data_pointer']}, '#{r['data']}' matches more than one of #{r['schema']['oneOf']}\n" + elsif r["type"] == "const" + " At #{r['data_pointer']}, '#{r['data']}' does not match required value '#{r['schema']['const']}'\n" + else + " #{r}\n\n" + end + end + msg << "\n" + # msg << result.to_a.to_s + super(msg) + @result = result + end + end + + attr_reader :data, :data_path, :specification, :cfg_arch, :name, :long_name, :description + + # @return [Architecture] If only a specification (no config) is known + # @return [ConfiguredArchitecture] If a specification and config is known + attr_reader :arch + + def kind = @data["kind"] + + @@schemas ||= {} + @@schema_ref_resolver ||= proc do |pattern| + if pattern.to_s =~ /^http/ + JSON.parse(Net::HTTP.get(pattern)) + else + JSON.load_file($root / "schemas" / pattern.to_s) + end + end + + # validate the data against it's schema + # @raise [SchemaError] if the data is invalid + def validate + schemas = @@schemas + ref_resolver = @@schema_ref_resolver + + if @data.key?("$schema") + schema_path = data["$schema"] + schema_file, obj_path = schema_path.split("#") + schema = + if schemas.key?(schema_file) + schemas[schema_file] + else + schemas[schema_file] = JSONSchemer.schema( + File.read("#{$root}/schemas/#{schema_file}"), + regexp_resolver: "ecma", + ref_resolver:, + insert_property_defaults: true + ) + raise SchemaError, schemas[schema_file].validate_schema unless schemas[schema_file].valid_schema? + + schemas[schema_file] + end + + unless obj_path.nil? + obj_path_parts = obj_path.split("/")[1..] + + obj_path_parts.each do |k| + schema = schema.fetch(k) + end + end + + # convert through JSON to handle anything supported in YAML but not JSON + # (e.g., integer object keys will be coverted to strings) + jsonified_obj = JSON.parse(JSON.generate(@data)) + + raise "Nothing there?" if jsonified_obj.nil? + + raise SchemaValidationError.new(@data_path, schema.validate(jsonified_obj)) unless schema.valid?(jsonified_obj) + else + warn "No $schema for #{@data_path}" + end + end + + # clone this, and set the cfg_arch at the same time + # @return [ExtensionRequirement] The new object + def clone(cfg_arch: nil) + obj = super() + obj.instance_variable_set(:@cfg_arch, cfg_arch) + obj + end def <=>(other) name <=> other.name @@ -34,7 +166,7 @@ def <=>(other) # @return [String] Source file that data for this object can be attributed to # @return [nil] if the source isn't known def __source - @data["__source"] + @data["$source"] end # The raw content of definedBy in the data. @@ -48,14 +180,22 @@ def definedBy end # @param data [Hash] Hash with fields to be added - def initialize(data) + # @param data_path [Pathname] Path to the data file + def initialize(data, data_path, arch: nil) raise "Bad data" unless data.is_a?(Hash) @data = data + @data_path = data_path + if arch.is_a?(ConfiguredArchitecture) + @cfg_arch = arch + @specification = arch + elsif arch.is_a?(Architecture) + @specification = arch + end + @arch = arch @name = data["name"] @long_name = data["long_name"] @description = data["description"] - end def inspect @@ -66,30 +206,13 @@ def inspect extend Forwardable def_delegator :@data, :[] - # @return [Array] List of keys added by this ArchDefObject + # @return [Array] List of keys added by this DatabaseObjectect def keys = @data.keys # @param k (see Hash#key?) # @return (see Hash#key?) def key?(k) = @data.key?(k) - # adds accessor functions for any properties in the data - # def method_missing(method_name, *args, &block) - # if @data.key?(method_name.to_s) - # raise "Unexpected argument to '#{method_name}" unless args.empty? - - # raise "Unexpected block given to '#{method_name}" if block_given? - - # @data[method_name.to_s] - # else - # super - # end - # end - - # def respond_to_missing?(method_name, include_private = false) - # @data.key?(method_name.to_s) || super - # end - # @overload defined_by?(ext_name, ext_version) # @param ext_name [#to_s] An extension name # @param ext_version [#to_s] A specific extension version @@ -98,67 +221,44 @@ def key?(k) = @data.key?(k) # @param ext_version [ExtensionVersion] An extension version # @return [Boolean] Whether or not the instruction is defined by ext_version def defined_by?(*args) - if args.size == 1 - raise ArgumentError, "Parameter must be an ExtensionVersion" unless args[0].is_a?(ExtensionVersion) + ext_ver = + if args.size == 1 + raise ArgumentError, "Parameter must be an ExtensionVersion" unless args[0].is_a?(ExtensionVersion) - defined_by.satisfied_by? do |r| - r.name == args[0].name && r.version_requirement.satisfied_by?(args[0].version) - end - elsif args.size == 2 - raise ArgumentError, "First parameter must be an extension name" unless args[0].respond_to?(:to_s) - version = args[1].is_a?(Gem::Version) ? args[1] : Gem::Version.new(args[1]) + args[0] + elsif args.size == 2 + raise ArgumentError, "First parameter must be an extension name" unless args[0].respond_to?(:to_s) + raise ArgumentError, "First parameter must be an extension version" unless args[1].respond_to?(:to_s) - defined_by.satisfied_by? do |r| - r.name == args[0] && r.version_requirement.satisfied_by?(version) + ExtensionVersion.new(args[0], args[1], cfg_arch) + else + raise ArgumentError, "Unsupported number of arguments (#{args.size})" end - else - raise ArgumentError, "Unsupported number of arguments of " + args.size - end + + defined_by_condition.satisfied_by? { |req| req.satisfied_by?(ext_ver) } end - # def to_extension_requirement(obj) - # if obj.is_a?(String) - # ExtensionRequirement.new(obj, ">= 0") - # else - # ExtensionRequirement.new(*obj) - # end - # end - # private :to_extension_requirement - - # def to_extension_requirement_list(obj) - # list = [] - # if obj.is_a?(Array) - # # could be either a single extension with exclusion, or a list of exclusions - # if extension_exclusion?(obj[0]) - # list << to_extension_requirement(obj[0]) - # else - # # this is a list - # obj.each do |r| - # list << to_extension_exclusion(r) - # end - # end - # else - # list << to_extension_requirement(obj) - # end - # list - # end + # because of multiple ("allOf") conditions, we generally can't return a list of extension versions here.... + # # @return [Array] Extension(s) that define the instruction. If *any* requirement is met, the instruction is defined. + # def defined_by + # raise "ERROR: definedBy is nul for #{name}" if @data["definedBy"].nil? - # def extension_requirement?(obj) - # obj.is_a?(String) && obj =~ /^([A-WY])|([SXZ][a-z]+)$/ || - # obj.is_a?(Array) && obj[0] =~ /^([A-WY])|([SXZ][a-z]+)$/ + # SchemaCondition.new(@data["definedBy"], @cfg_arch).satisfying_ext_versions # end - # private :extension_requirement? # @return [SchemaCondition] Extension(s) that define the instruction. If *any* requirement is met, the instruction is defined. - def defined_by - raise "ERROR: definedBy is nul for #{name}" if @data["definedBy"].nil? + def defined_by_condition + @defined_by_condition ||= + begin + raise "ERROR: definedBy is nul for #{name}" if @data["definedBy"].nil? - SchemaCondition.new(@data["definedBy"]) + SchemaCondition.new(@data["definedBy"], @cfg_arch) + end end # @return [String] Name of an extension that "primarily" defines the object (i.e., is the first in a list) def primary_defined_by - defined_by.first_requirement.name + defined_by_condition.first_requirement end # @return [Integer] THe source line number of +path+ in the YAML file @@ -176,8 +276,8 @@ def primary_defined_by def source_line(*path) # find the line number of this operation() in the *original* file - yaml_filename = @data["__source"] - raise "No __source for #{name}" if yaml_filename.nil? + yaml_filename = @data["$source"] + raise "No $source for #{name}" if yaml_filename.nil? line = nil path_idx = 0 Psych.parse_stream(File.read(yaml_filename), filename: yaml_filename) do |doc| @@ -208,12 +308,16 @@ def source_line(*path) end end end - raise "Didn't find key '#{path}' in #{@data['__source']}" + raise "Didn't find key '#{path}' in #{@data['$source']}" end end # A company description -class Company < ArchDefObject +class Company + def initialize(data) + @data = data + end + # @return [String] Company name def name = @data["name"] @@ -222,7 +326,11 @@ def url = @data["url"] end # License information -class License < ArchDefObject +class License + def initialize(data) + @data = data + end + # @return [String] License name def name = @data["name"] @@ -241,7 +349,9 @@ def text end # Personal information about a contributor -class Person < ArchDefObject +class Person + include Comparable + # @return [String] Person's name def name = @data["name"] @@ -252,6 +362,16 @@ def email = @data["email"] # @return [String] Company the person works for # @return [nil] if the company is not known, or if the person is an individual contributor def company = @data["company"] + + def initialize(data) + @data = data + end + + def <=>(other) + raise ArgumentError, "Person is only comparable to Person (not #{other.class.name})" unless other.is_a?(Person) + + name <=> other.name + end end # represents a JSON Schema compoisition, e.g.: @@ -264,7 +384,7 @@ def company = @data["company"] # class SchemaCondition # @param composition_hash [Hash] A possibly recursive hash of "allOf", "anyOf", "oneOf" - def initialize(composition_hash) + def initialize(composition_hash, cfg_arch) raise ArgumentError, "composition_hash is nil" if composition_hash.nil? unless is_a_condition?(composition_hash) @@ -272,6 +392,7 @@ def initialize(composition_hash) end @hsh = composition_hash + @cfg_arch = cfg_arch end def to_h = @hsh @@ -346,10 +467,14 @@ def is_a_condition?(hsh) def first_requirement(req = @hsh) case req when String - ExtensionRequirement.new(req, ">= 0") + ExtensionRequirement.new(req, cfg_arch: @cfg_arch) when Hash if req.key?("name") - ExtensionRequirement.new(req["name"], req["version"] || ">= 0") + if req["version"].nil? + ExtensionRequirement.new(req["name"], cfg_arch: @cfg_arch) + else + ExtensionRequirement.new(req["name"], req["version"], cfg_arch: @cfg_arch) + end else first_requirement(req[req.keys[0]]) end @@ -361,12 +486,12 @@ def first_requirement(req = @hsh) end # combine all conds into one using AND - def self.all_of(*conds) + def self.all_of(*conds, cfg_arch:) cond = SchemaCondition.new({ "allOf" => conds - }) + }, cfg_arch) - SchemaCondition.new(cond.minimize) + SchemaCondition.new(cond.minimize, cfg_arch) end # @return [Object] Schema for this condition, with basic logic minimization @@ -404,14 +529,14 @@ def to_rb_helper(hsh) if hsh.key?("name") if hsh.key?("version") if hsh["version"].is_a?(String) - "(yield ExtensionRequirement.new('#{hsh["name"]}', '#{hsh["version"]}'))" + "(yield ExtensionRequirement.new('#{hsh["name"]}', '#{hsh["version"]}', cfg_arch: @cfg_arch))" elsif hsh["version"].is_a?(Array) - "(yield ExtensionRequirement.new('#{hsh["name"]}', #{hsh["version"].map { |v| "'#{v}'" }.join(', ')}))" + "(yield ExtensionRequirement.new('#{hsh["name"]}', #{hsh["version"].map { |v| "'#{v}'" }.join(', ')}, cfg_arch: @cfg_arch))" else raise "unexpected" end else - "(yield ExtensionRequirement.new('#{hsh["name"]}'))" + "(yield ExtensionRequirement.new('#{hsh["name"]}', cfg_arch: @cfg_arch))" end else key = hsh.keys[0] @@ -432,7 +557,7 @@ def to_rb_helper(hsh) end end else - "(yield ExtensionRequirement.new('#{hsh}'))" + "(yield ExtensionRequirement.new('#{hsh}', cfg_arch: @cfg_arch))" end end @@ -470,6 +595,16 @@ def satisfied_by?(&block) eval to_rb end + + def satisfying_ext_versions + list = [] + cfg_arch.extensions.each do |ext| + ext.versions.each do |ext_ver| + list << ext_ver if satisfied_by? { |ext_req| ext_req.satisfied_by?(ext_ver) } + end + end + list + end end class AlwaysTrueSchemaCondition @@ -480,4 +615,5 @@ def satisfied_by? = true def empty? = true def to_h = {} + def minimize = {} end diff --git a/lib/arch_obj_models/portfolio.rb b/lib/arch_obj_models/portfolio.rb index 3ad3306d5..0afa26720 100644 --- a/lib/arch_obj_models/portfolio.rb +++ b/lib/arch_obj_models/portfolio.rb @@ -4,11 +4,13 @@ # RVA20U64 and MC100 are examples of portfolio instances # RVA and MC are examples of portfolio classes # -# Many classes inherit from the ArchDefObject class. This provides facilities for accessing the contents of a +# Many classes inherit from the DatabaseObjectect class. This provides facilities for accessing the contents of a # Portfolio Class YAML or Portfolio Model YAML file via the "data" member (hash holding releated YAML file contents). # # A variable name with a "_data" suffix indicates it is the raw hash data from the porfolio YAML file. +require "tmpdir" + require_relative "obj" require_relative "schema" @@ -18,16 +20,9 @@ # Holds information from Portfolio class YAML file (certificate class or profile class). # The inherited "data" member is the database of extensions, instructions, CSRs, etc. -class PortfolioClass < ArchDefObject - # @return [ArchDef] The defining ArchDef - attr_reader :arch_def - - # @param data [Hash] The data from YAML - # @param arch_def [ArchDef] Architecture spec - def initialize(data, arch_def) - super(data) - @arch_def = arch_def - end +class PortfolioClass < DatabaseObjectect + # @return [ConfiguredArchitecture] The defining ConfiguredArchitecture + attr_reader :cfg_arch def introduction = @data["introduction"] def naming_scheme = @data["naming_scheme"] @@ -45,16 +40,9 @@ def eql?(other) # Holds information about a PortfolioInstance YAML file (certificate or profile). # The inherited "data" member is the database of extensions, instructions, CSRs, etc. -class PortfolioInstance < ArchDefObject - # @return [ArchDef] The defining ArchDef - attr_reader :arch_def - - # @param data [Hash] The data from YAML - # @param arch_def [ArchDef] Architecture spec - def initialize(data, arch_def) - super(data) - @arch_def = arch_def - end +class PortfolioInstance < DatabaseObjectect + # @return [ConfiguredArchitecture] The defining ConfiguredArchitecture + attr_reader :cfg_arch def description = @data["description"] @@ -79,17 +67,18 @@ def version_strongest_presence(ext_name, ext_versions) # See if any extension requirement in this profile lists this version as either mandatory or optional. ext_versions.map do |v| - mandatory = mandatory_ext_reqs.any? { |ext_req| ext_req.satisfied_by?(ext_name, v.version) } - optional = optional_ext_reqs.any? { |ext_req| ext_req.satisfied_by?(ext_name, v.version) } + mandatory = mandatory_ext_reqs.any? { |ext_req| ext_req.satisfied_by?(v) } + optional = optional_ext_reqs.any? { |ext_req| ext_req.satisfied_by?(v) } # Just show strongest presence (mandatory stronger than optional). - if mandatory - presences << ExtensionPresence.mandatory - elsif optional - presences << ExtensionPresence.optional - else - presences << "-" - end + presences << + if mandatory + ExtensionPresence.mandatory + elsif optional + ExtensionPresence.optional + else + "-" + end end presences @@ -120,6 +109,8 @@ def in_scope_ext_reqs(desired_presence = nil) ExtensionPresence.new(desired_presence) @data["extensions"]&.each do |ext_name, ext_data| + next if ext_name[0] == "$" + actual_presence = ext_data["presence"] # Could be a String or Hash raise "Missing extension presence for extension #{ext_name}" if actual_presence.nil? @@ -135,8 +126,15 @@ def in_scope_ext_reqs(desired_presence = nil) if match in_scope_ext_reqs << - ExtensionRequirement.new(ext_name, ext_data["version"], presence: actual_presence_obj, - note: ext_data["note"], req_id: "REQ-EXT-" + ext_name) + if ext_data.key?("version") + ExtensionRequirement.new( + ext_name, ext_data["version"], cfg_arch: @cfg_arch, + presence: actual_presence_obj, note: ext_data["note"], req_id: "REQ-EXT-#{ext_name}") + else + ExtensionRequirement.new( + ext_name, cfg_arch: @cfg_arch, + presence: actual_presence_obj, note: ext_data["note"], req_id: "REQ-EXT-#{ext_name}") + end end end in_scope_ext_reqs @@ -151,7 +149,7 @@ def in_scope_extensions return @in_scope_extensions unless @in_scope_extensions.nil? @in_scope_extensions = in_scope_ext_reqs.map do |er| - obj = arch_def.extension(er.name) + obj = cfg_arch.extension(er.name) # @todo: change this to raise once all the profile extensions # are defined @@ -180,27 +178,34 @@ def uses_optional_types? @uses_optional_types end - # @return [ArchDef] A partially-configured architecture definition corresponding to this certificate. - def to_arch_def - return @generated_arch_def unless @generated_arch_def.nil? - - arch_def_data = arch_def.unconfigured_data - - arch_def_data["mandatory_extensions"] = mandatory_ext_reqs.map do |ext_req| - { - "name" => ext_req.name, - "version" => ext_req.version_requirement.requirements.map { |r| "#{r[0]} #{r[1]}" } - } - end - arch_def_data["params"] = all_in_scope_ext_params.select(&:single_value?).map { |p| [p.name, p.value] }.to_h + # @return [ConfiguredArchitecture] A partially-configured architecture definition corresponding to this portfolio. + def to_cfg_arch + return @generated_cfg_arch unless @generated_cfg_arch.nil? + + # build up a config for the portfolio + config_data = { + "$schema" => "config_schema.json", + "type" => "partially configured", + "kind" => "architecture configuration", + "name" => name, + "description" => "A partially configured architecture definition corresponding to the #{name} portfolio.", + "mandatory_extensions" => mandatory_ext_reqs.map do |ext_req| + { + "name" => ext_req.name, + "version" => ext_req.requirement_specs.map(&:to_s) + } + end, + "params" => all_in_scope_ext_params.select(&:single_value?).map { |p| [p.name, p.value] }.to_h + } # XXX Add list of prohibited_extensions - file = Tempfile.new("archdef") - file.write(YAML.safe_dump(arch_def_data, permitted_classes: [Date])) - file.flush - file.close - @generated_arch_def = ArchDef.new(name, Pathname.new(file.path)) + @generated_cfg_arch = + Dir.mktmpdir do |dir| + FileUtils.mkdir("#{dir}/#{name}") + File.write("#{dir}/#{name}/cfg.yaml", YAML.safe_dump(config_data, permitted_classes: [Date])) + @generated_cfg_arch = ConfiguredArchitecture.new(name, @cfg_arch.path, cfg_path: dir) + end end ################################### @@ -273,8 +278,10 @@ def all_in_scope_ext_params @all_in_scope_ext_params = [] @data["extensions"].each do |ext_name, ext_data| + next if ext_name[0] == "$" + # Find Extension object from database - ext = @arch_def.extension(ext_name) + ext = @cfg_arch.extension(ext_name) raise "Cannot find extension named #{ext_name}" if ext.nil? ext_data["parameters"]&.each do |param_name, param_data| @@ -282,8 +289,9 @@ def all_in_scope_ext_params raise "There is no param '#{param_name}' in extension '#{ext_name}" if param.nil? next unless ext.versions.any? do |ext_ver| - Gem::Requirement.new(ext_data["version"]).satisfied_by?(ext_ver.version) && - param.defined_in_extension_version?(ext_ver.version) + ver_req = ext_data["version"] || ">= #{ext.min_version.version_spec}" + ExtensionRequirement.new(ext_name, ver_req, cfg_arch: @cfg_arch).satisfied_by?(ext_ver) && + param.defined_in_extension_version?(ext_ver) end @all_in_scope_ext_params << @@ -305,23 +313,23 @@ def in_scope_ext_params(ext_req) raise "Cannot find extension named #{ext_req.name}" if ext_data.nil? # Find Extension object from database - ext = @arch_def.extension(ext_req.name) + ext = @cfg_arch.extension(ext_req.name) raise "Cannot find extension named #{ext_req.name}" if ext.nil? # Loop through an extension's parameter constraints (hash) from the portfolio. # Note that "&" is the Ruby safe navigation operator (i.e., skip do loop if nil). ext_data["parameters"]&.each do |param_name, param_data| - # Find ExtensionParameter object from database - ext_param = ext.params.find { |p| p.name == param_name } - raise "There is no param '#{param_name}' in extension '#{ext_req.name}" if ext_param.nil? + # Find ExtensionParameter object from database + ext_param = ext.params.find { |p| p.name == param_name } + raise "There is no param '#{param_name}' in extension '#{ext_req.name}" if ext_param.nil? - next unless ext.versions.any? do |ext_ver| - Gem::Requirement.new(ext_data["version"]).satisfied_by?(ext_ver.version) && - ext_param.defined_in_extension_version?(ext_ver.version) - end + next unless ext.versions.any? do |ext_ver| + ext_req.satisfied_by?(ext_ver) && + ext_param.defined_in_extension_version?(ext_ver) + end - ext_params << - InScopeExtensionParameter.new(ext_param, param_data["schema"], param_data["note"]) + ext_params << + InScopeExtensionParameter.new(ext_param, param_data["schema"], param_data["note"]) end ext_params @@ -333,13 +341,13 @@ def all_out_of_scope_params @all_out_of_scope_params = [] in_scope_ext_reqs.each do |ext_req| - ext = @arch_def.extension(ext_req.name) + ext = @cfg_arch.extension(ext_req.name) ext.params.each do |param| next if all_in_scope_ext_params.any? { |c| c.param.name == param.name } next unless ext.versions.any? do |ext_ver| - Gem::Requirement.new(ext_req.version_requirement).satisfied_by?(ext_ver.version) && - param.defined_in_extension_version?(ext_ver.version) + ext_req.satisfied_by?(ext_ver) && + param.defined_in_extension_version?(ext_ver) end @all_out_of_scope_params << param @@ -350,7 +358,7 @@ def all_out_of_scope_params # @return [Array] Parameters that are out of scope for named extension. def out_of_scope_params(ext_name) - all_out_of_scope_params.select{|param| param.exts.any? {|ext| ext.name == ext_name} } + all_out_of_scope_params.select{ |param| param.exts.any? { |ext| ext.name == ext_name } } end # @return [Array] @@ -373,8 +381,8 @@ def all_in_scope_exts_with_param(param) end if found - # Only add extensions that exist in this portfolio. - exts << ext + # Only add extensions that exist in this portfolio. + exts << ext end end @@ -418,9 +426,9 @@ def all_in_scope_exts_without_param(param) # Tracks history of portfolio document. This is separate from its version since # a document may be revised several times before a new version is released. - class RevisionHistory < ArchDefObject + class RevisionHistory def initialize(data) - super(data) + @data = data end def revision = @data["revision"] @@ -442,9 +450,9 @@ def revision_history # ExtraNote Subclass # ###################### - class ExtraNote < ArchDefObject + class ExtraNote def initialize(data) - super(data) + @data = data @presence_obj = ExtensionPresence.new(@data["presence"]) end @@ -476,9 +484,9 @@ def extra_notes_for_presence(desired_presence_obj) # Recommendation Subclass # ########################### - class Recommendation < ArchDefObject + class Recommendation def initialize(data) - super(data) + @data = data end def text = @data["text"] diff --git a/lib/arch_obj_models/profile.rb b/lib/arch_obj_models/profile.rb index c63be19a2..1149689c8 100644 --- a/lib/arch_obj_models/profile.rb +++ b/lib/arch_obj_models/profile.rb @@ -7,12 +7,6 @@ # that each include an unprivileged profile (e.g., RVA20U64) and one more # privileged profiles (e.g., RVA20S64). class ProfileClass < PortfolioClass - # @param data [Hash] The data from YAML - # @param arch_def [ArchDef] Architecture spec - def initialize(data, arch_def) - super(data, arch_def) - end - # @return [String] Name of the class def marketing_name = @data["marketing_name"] @@ -28,7 +22,7 @@ def doc_license def profile_releases return @profile_releases unless @profile_releases.nil? - @profile_releases = @arch_def.profile_releases.select { |pr| pr.profile_class.name == name } + @profile_releases = @cfg_arch.profile_releases.select { |pr| pr.profile_class.name == name } @profile_releases end @@ -38,7 +32,7 @@ def profiles return @profiles unless @profiles.nil? @profiles = [] - @arch_def.profiles.each do |profile| + @cfg_arch.profiles.each do |profile| if profile.profile_class.name == name @profiles << profile end @@ -67,14 +61,7 @@ def referenced_extensions # For example, the RVA20 profile release has profiles RVA20U64 and RVA20S64. # Note there is no Portfolio* base class for a ProfileRelease to inherit from since there is no # equivalent to a ProfileRelease in a Certificate so no potential for a shared base class. -class ProfileRelease < ArchDefObject - # @param data [Hash] The data from YAML - # @param arch_def [ArchDef] Architecture spec - def initialize(data, arch_def) - super(data) - @arch_def = arch_def - end - +class ProfileRelease < DatabaseObjectect def marketing_name = @data["marketing_name"] def introduction = @data["introduction"] def state = @data["state"] @@ -94,7 +81,7 @@ def contributors # @return [ProfileClass] Profile Class that this ProfileRelease belongs to def profile_class - profile_class = @arch_def.profile_class(@data["class"]) + profile_class = @cfg_arch.profile_class(@data["class"]) raise "No profile class named '#{@data["class"]}'" if profile_class.nil? profile_class @@ -105,10 +92,8 @@ def profiles return @profiles unless @profiles.nil? @profiles = [] - @arch_def.profiles.each do |profile| - if profile.profile_release.name == name - @profiles << profile - end + @data["profiles"].each do |profile_ref| + @profiles << @cfg_arch.ref(profile_ref["$ref"]) end @profiles end @@ -130,9 +115,6 @@ def referenced_extensions # Representation of a specific profile in a profile release. class Profile < PortfolioInstance - def initialize(data, arch_def) - super(data, arch_def) - end # @return [String] The marketing name of the Profile def introduction = @data["introduction"] @@ -140,8 +122,8 @@ def marketing_name = @data["marketing_name"] # @return [ProfileRelease] The profile release this profile belongs to def profile_release - profile_release = @arch_def.profile_release(@data["release"]) - raise "No profile release named '#{@data["release"]}'" if profile_release.nil? + profile_release = @cfg_arch.ref(@data["release"]["$ref"]) + raise "No profile release named '#{@data["release"]["$ref"]}'" if profile_release.nil? profile_release end @@ -219,10 +201,10 @@ def extensions_to_adoc(presence_type, heading_level) def ext_req_to_adoc(ext_req) ret = [] - ext = arch_def.extension(ext_req.name) + ext = cfg_arch.extension(ext_req.name) ret << "* *#{ext_req.name}* " + (ext.nil? ? "" : ext.long_name) ret << "+" - ret << "Version #{ext_req.version_requirement}" + ret << "Version #{ext_req.requirement_specs}" ret end diff --git a/lib/architecture.rb b/lib/architecture.rb new file mode 100644 index 000000000..ab92f5505 --- /dev/null +++ b/lib/architecture.rb @@ -0,0 +1,283 @@ +# frozen_string_literal: true + +require "active_support/inflector/methods" + +require "json" +require "json_schemer" +require "pathname" +require "yaml" + +require_relative "idl" + +require_relative "arch_obj_models/certificate" +require_relative "arch_obj_models/csr" +require_relative "arch_obj_models/csr_field" +require_relative "arch_obj_models/exception_code" +require_relative "arch_obj_models/extension" +require_relative "arch_obj_models/instruction" +require_relative "arch_obj_models/manual" +require_relative "arch_obj_models/portfolio" +require_relative "arch_obj_models/profile" + +# Represents the entire RISC-V Architecture. +# +# Could be either the standard spec (defined by RISC-V International) +# of a custom spec (defined as an arch_overlay in cfgs/) +class Architecture + # @return [Pathname] Path to the directory with the standard YAML files + attr_reader :path + + # @param arch_dir [Sting,Pathname] Path to a directory with a fully merged/resolved architecture defintion + def initialize(arch_dir) + @arch_dir = Pathname.new(arch_dir) + raise "Arch directory not found: #{arch_dir}" unless @arch_dir.exist? + + @arch_dir = @arch_dir.realpath + @path = @arch_dir # alias + @objects ||= {} + @object_hashes ||= {} + end + + # validate the architecture against JSON Schema and any object-specific verification + # @param show_progress [Boolean] Whether to show a progress bar + def validate(show_progress: true) + progressbar = ProgressBar.create(total: objs.size) if show_progress + + objs.each do |obj| + progressbar.increment if show_progress + obj.validate + end + end + + # @!macro [attach] generate_obj_methods + # @method $1s + # @return [Array<$3>] List of all $1s defined in the standard + # + # @method $1_hash + # @return [Hash] Hash of all $1s + # + # @method $1 + # @param name [String] The $1 name + # @return [$3] The $1 + # @return [nil] if there is no $1 named +name+ + def self.generate_obj_methods(fn_name, arch_dir, obj_class) + plural_fn = ActiveSupport::Inflector.pluralize(fn_name) + + define_method(plural_fn) do + return @objects[arch_dir] unless @objects[arch_dir].nil? + + @objects[arch_dir] = [] + @object_hashes[arch_dir] = {} + Dir.glob(@arch_dir / arch_dir / "**" / "*.yaml") do |obj_path| + obj_yaml = YAML.load_file(obj_path, permitted_classes: [Date]) + @objects[arch_dir] << obj_class.new(obj_yaml, Pathname.new(obj_path).realpath, arch: self) + @object_hashes[arch_dir][@objects[arch_dir].last.name] = @objects[arch_dir].last + end + @objects[arch_dir] + end + + define_method("#{fn_name}_hash") do + return @object_hashes[arch_dir] unless @object_hashes[arch_dir].nil? + + send(plural_fn) # create the hash + + @object_hashes[arch_dir] + end + + define_method(fn_name) do |name| + return @object_hashes[arch_dir][name] unless @object_hashes[arch_dir].nil? + + send(plural_fn) # create the hash + + @object_hashes[arch_dir][name] + end + end + + OBJS = [ + { + fn_name: "extension", + arch_dir: "ext", + klass: Extension + }, + { + fn_name: "instruction", + arch_dir: "inst", + klass: Instruction + }, + { + fn_name: "csr", + arch_dir: "csr", + klass: Csr + }, + { + fn_name: "cert_class", + arch_dir: "certificate_class", + klass: CertClass + }, + { + fn_name: "cert_model", + arch_dir: "certificate_model", + klass: CertModel + }, + { + fn_name: "manual", + arch_dir: "manual", + klass: Manual + }, + { + fn_name: "manual_version", + arch_dir: "manual_version", + klass: ManualVersion + }, + { + fn_name: "profile_release", + arch_dir: "profile_release", + klass: ProfileRelease + }, + { + fn_name: "profile_class", + arch_dir: "profile_class", + klass: ProfileClass + }, + { + fn_name: "profile", + arch_dir: "profile", + klass: Profile + } + ].freeze + + OBJS.each do |obj_info| + generate_obj_methods(obj_info[:fn_name], obj_info[:arch_dir], obj_info[:klass]) + end + + # @return [Array] All known objects + def objs + return @objs unless @objs.nil? + + @objs = [] + OBJS.each do |obj_info| + @objs.concat(send(ActiveSupport::Inflector.pluralize(obj_info[:fn_name]))) + end + @objs.freeze + end + + # @return [Array] Alphabetical list of all parameters defined in the architecture + def params + return @params unless @params.nil? + + @params = extensions.map(&:params).flatten.uniq(&:name).sort_by!(&:name) + end + + # @return [Hash] Hash of all extension parameters defined in the architecture + def params_hash + return @params_hash unless @params_hash.nil? + + @params_hash = {} + params.each do |param| + @params_hash[param.name] = param + end + @param_hash + end + + # @return [ExtensionParameter] Parameter named +name+ + # @return [nil] if there is no parameter named +name+ + def param(name) + params_hash[name] + end + + # @return [Array] All exception codes defined by the spec + def exception_codes + return @exception_codes unless @exception_codes.nil? + + @exception_codes = + extensions.reduce([]) do |list, ext_version| + ecodes = extension(ext_version.name)["exception_codes"] + next list if ecodes.nil? + + ecodes.each do |ecode| + # double check that all the codes are unique + raise "Duplicate exception code" if list.any? { |e| e.num == ecode["num"] || e.name == ecode["name"] || e.var == ecode["var"] } + + list << ExceptionCode.new(ecode["name"], ecode["var"], ecode["num"], self) + end + list + end + end + + # @return [Array] All interrupt codes defined by extensions + def interrupt_codes + return @interrupt_codes unless @interrupt_codes.nil? + + @interupt_codes = + extensions.reduce([]) do |list, ext_version| + icodes = extension(ext_version.name)["interrupt_codes"] + next list if icodes.nil? + + icodes.each do |icode| + # double check that all the codes are unique + if list.any? { |i| i.num == icode["num"] || i.name == icode["name"] || i.var == icode["var"] } + raise "Duplicate interrupt code" + end + + list << InterruptCode.new(icode["name"], icode["var"], icode["num"], self) + end + list + end + end + + # given a `$ref` target, return the Ruby object + # + # @params uri [String] JSON Reference pointer + # @return [Object] The pointed-to object + def ref(uri) + raise ArgumentError, "JSON Reference (#{uri}) must contain one '#'" unless uri.count("#") == 1 + + file_path, obj_path = uri.split("#") + obj = + case file_path + when /^certificate_class.*/ + cert_class_name = File.basename(file_path, ".yaml") + cert_class(cert_class_name) + when /^certificate_model.*/ + cert_model_name = File.basename(file_path, ".yaml") + cert_model(cert_model_name) + when /^csr.*/ + csr_name = File.basename(file_path, ".yaml") + csr(csr_name) + when /^ext.*/ + ext_name = File.basename(file_path, ".yaml") + extension(ext_name) + when /^inst.*/ + inst_name = File.basename(file_path, ".yaml") + instruction(inst_name) + when /^manual.*/ + manual_name = File.basename(file_path, ".yaml") + manual(manual_name) + when /^manual_version.*/ + manual_name = File.basename(file_path, ".yaml") + manual_version(manual_name) + when /^profile_class.*/ + profile_class_name = File.basename(file_path, ".yaml") + profile_class(profile_class_name) + when /^profile_release.*/ + profile_release_name = File.basename(file_path, ".yaml") + profile_release(profile_release_name) + when /^profile.*/ + profile_name = File.basename(file_path, ".yaml") + profile(profile_name) + else + raise "Unhandled ref object: #{file_path}" + end + + unless obj_path.nil? + parts = obj_path.split("/") + parts.each do |part| + raise "Error in $ref. There is no method '#{part}' for a #{obj.class.name}" unless obj.respond_to?(part.to_sym) + + obj = obj.send(part) + end + end + + obj + end +end diff --git a/lib/asciidoc_extensions.js b/lib/asciidoc_extensions.js index 91520a803..269f676de 100644 --- a/lib/asciidoc_extensions.js +++ b/lib/asciidoc_extensions.js @@ -1,3 +1,3 @@ -const asciidoctor = require('asciidoctor')() -const registry = asciidoctor.Extensions.create() -require('./asciidoc_when_extension.js')(registry) +const asciidoctor = require("asciidoctor")(); +const registry = asciidoctor.Extensions.create(); +require("./asciidoc_when_extension.js")(registry); diff --git a/lib/asciidoc_when_extension.js b/lib/asciidoc_when_extension.js index 1bd198f47..90fc51eb7 100644 --- a/lib/asciidoc_when_extension.js +++ b/lib/asciidoc_when_extension.js @@ -1,18 +1,25 @@ module.exports.register = function (registry, context = {}) { - registry.block('when', function () { - var self = this - self.named('when') - self.onContexts('paragraph', 'open') - self.positionalAttributes(['condition']) - self.process(function (parent, reader, attributes) { - var condition = attributes.condition - let new_block_attrs = {} - new_block_attrs.role = "when" - new_block_attrs.name = "when" - new_block_attrs["textlabel"] = `When ${condition}` - let content_model = attributes["cloaked-context"] == "paragraph" ? "simple" : "compound" - return self.createBlock(parent, 'admonition', reader.getLines(), new_block_attrs, {"content_model": content_model}) - }) - }) - return registry -} + registry.block("when", function () { + var self = this; + self.named("when"); + self.onContexts("paragraph", "open"); + self.positionalAttributes(["condition"]); + self.process(function (parent, reader, attributes) { + var condition = attributes.condition; + let new_block_attrs = {}; + new_block_attrs.role = "when"; + new_block_attrs.name = "when"; + new_block_attrs["textlabel"] = `When ${condition}`; + let content_model = + attributes["cloaked-context"] == "paragraph" ? "simple" : "compound"; + return self.createBlock( + parent, + "admonition", + reader.getLines(), + new_block_attrs, + { content_model: content_model }, + ); + }); + }); + return registry; +}; diff --git a/lib/cfg_arch.rb b/lib/cfg_arch.rb new file mode 100644 index 000000000..f86df3dad --- /dev/null +++ b/lib/cfg_arch.rb @@ -0,0 +1,723 @@ +# frozen_string_literal: true + +# Many classes have an "cfg_arch" member which is an ConfiguredArchitecture (not DatabaseObjectect) class. +# The "cfg_arch" member contains the "database" of RISC-V standards including extensions, instructions, +# CSRs, Profiles, and Certificates. +# +# The cfg_arch member has methods such as: +# extensions() Array of all extensions known to the database (even if not implemented). +# extension(name) Extension object for "name" and nil if none. +# parameters() Array of all parameters defined in the architecture +# param(name) ExtensionParameter object for "name" and nil if none. +# csrs() Array of all CSRs defined by RISC-V, whether or not they are implemented +# csr(name) Csr object for "name" and nil if none. +# instructions() Array of all instructions, whether or not they are implemented +# inst(name) Instruction object for "name" and nil if none. +# profile_classes Array of all known profile classes. +# profile_class(class_name) ProfileClass object for "class_name" and nil if none. +# profile_releases Array of all profile releases for all profile classes +# profile_release(release_name) ProfileRelease object for "release_name" and nil if none. +# profiles Array of all profiles in all releases in all classes +# profile(name) Profile object for profile "name" and nil if none. +# cert_classes Array of all known certificate classes +# cert_class(name) CertClass object for "name" and nil if none. +# cert_models Array of all known certificate models across all classes. +# cert_model(name) CertModel object for "name" and nil if none. + +require "forwardable" +require "ruby-prof" +require "tilt" + +require_relative "config" +require_relative "architecture" + +require_relative "idl" +require_relative "idl/passes/find_return_values" +require_relative "idl/passes/gen_adoc" +require_relative "idl/passes/prune" +require_relative "idl/passes/reachable_exceptions" +require_relative "idl/passes/reachable_functions" + +require_relative "template_helpers" + +include TemplateHelpers + +class ConfiguredArchitecture < Architecture + extend Forwardable + + # @return [Idl::Compiler] The IDL compiler + attr_reader :idl_compiler + + # @return [Idl::IsaAst] Abstract syntax tree of global scope + attr_reader :global_ast + + # @return [String] Name of this definition. Special names are: + # * '_' - The generic architecture, with no configuration settings. + # * 'rv32' - A generic RV32 architecture, with only one parameter set (XLEN == 32) + # * 'rv64' - A generic RV64 architecture, with only one parameter set (XLEN == 64) + attr_reader :name + + def_delegators \ + :@config, \ + :fully_configured?, :partially_configured?, :unconfigured?, :configured?, \ + :mxlen, :param_values + + # Returns whether or not it may be possible to switch XLEN given this definition. + # + # There are three cases when this will return true: + # 1. A mode (e.g., U) is known to be implemented, and the CSR bit that controls XLEN in that mode is known to be writeable. + # 2. A mode is known to be implemented, but the writability of the CSR bit that controls XLEN in that mode is not known. + # 3. It is not known if the mode is implemented. + # + # + # @return [Boolean] true if this configuration might execute in multiple xlen environments + # (e.g., that in some mode the effective xlen can be either 32 or 64, depending on CSR values) + def multi_xlen? + return true if @mxlen.nil? + + ["S", "U", "VS", "VU"].any? { |mode| multi_xlen_in_mode?(mode) } + end + + # Returns whether or not it may be possible to switch XLEN in +mode+ given this definition. + # + # There are three cases when this will return true: + # 1. +mode+ (e.g., U) is known to be implemented, and the CSR bit that controls XLEN in +mode+ is known to be writeable. + # 2. +mode+ is known to be implemented, but the writability of the CSR bit that controls XLEN in +mode+ is not known. + # 3. It is not known if +mode+ is implemented. + # + # Will return false if +mode+ is not possible (e.g., because U is a prohibited extension) + # + # @param mode [String] mode to check. One of "M", "S", "U", "VS", "VU" + # @return [Boolean] true if this configuration might execute in multiple xlen environments in +mode+ + # (e.g., that in some mode the effective xlen can be either 32 or 64, depending on CSR values) + def multi_xlen_in_mode?(mode) + return false if mxlen == 32 + + case mode + when "M" + mxlen.nil? + when "S" + return true if unconfigured? + + if fully_configured? + ext?(:S) && (param_values["SXLEN"] == 3264) + elsif partially_configured? + return false if prohibited_ext?(:S) + + return true unless ext?(:S) # if S is not known to be implemented, we can't say anything about it + + return true unless param_values.key?("SXLEN") + + param_values["SXLEN"] == 3264 + else + raise "Unexpected configuration state" + end + when "U" + return false if prohibited_ext?(:U) + + return true if unconfigured? + + if fully_configured? + ext?(:U) && (param_values["UXLEN"] == 3264) + elsif partially_configured? + return true unless ext?(:U) # if U is not known to be implemented, we can't say anything about it + + return true unless param_values.key?("UXLEN") + + param_values["UXLEN"] == 3264 + else + raise "Unexpected configuration state" + end + when "VS" + return false if prohibited_ext?(:H) + + return true if unconfigured? + + if fully_configured? + ext?(:H) && (param_values["VSXLEN"] == 3264) + elsif partially_configured? + return true unless ext?(:H) # if H is not known to be implemented, we can't say anything about it + + return true unless param_values.key?("VSXLEN") + + param_values["VSXLEN"] == 3264 + else + raise "Unexpected configuration state" + end + when "VU" + return false if prohibited_ext?(:H) + + return true if unconfigured? + + if fully_configured? + ext?(:H) && (param_values["VUXLEN"] == 3264) + elsif partially_configured? + return true unless ext?(:H) # if H is not known to be implemented, we can't say anything about it + + return true unless param_values.key?("VUXLEN") + + param_values["VUXLEN"] == 3264 + else + raise "Unexpected configuration state" + end + else + raise ArgumentError, "Bad mode" + end + end + + # @return [Array] List of possible XLENs in any mode for this config + def possible_xlens = multi_xlen? ? [32, 64] : [mxlen] + + # hash for Hash lookup + def hash = @name_sym.hash + + # @return [Idl::SymbolTable] Symbol table with global scope + # @return [nil] if the architecture is not configured (use symtab_32 or symtab_64) + def symtab + raise NotImplementedError, "Un-configured ConfiguredArchitectures have no symbol table" if @symtab.nil? + + @symtab + end + + def config_type = @config.type + + # Initialize a new configured architecture definition + # + # @param config_name [#to_s] The name of a configuration, which must correspond + # to a folder name under cfg_path + def initialize(config_name, arch_path, overlay_path: nil, cfg_path: "#{$root}/cfgs") + super(arch_path) + + @name = config_name.to_s.freeze + @name_sym = @name.to_sym.freeze + + @obj_cache = {} + + @config = Config.create("#{cfg_path}/#{config_name}/cfg.yaml") + @mxlen = @config.mxlen + @mxlen.freeze + + @idl_compiler = Idl::Compiler.new + + @symtab = Idl::SymbolTable.new(self) + custom_globals_path = overlay_path.nil? ? Pathname.new("/does/not/exist") : overlay_path / "isa" / "globals.isa" + idl_path = File.exist?(custom_globals_path) ? custom_globals_path : $root / "arch" / "isa" / "globals.isa" + @global_ast = @idl_compiler.compile_file( + idl_path + ) + @global_ast.add_global_symbols(@symtab) + @symtab.deep_freeze + @global_ast.freeze_tree(@symtab) + end + + # type check all IDL, including globals, instruction ops, and CSR functions + # + # @param config [Config] Configuration + # @param show_progress [Boolean] whether to show progress bars + # @param io [IO] where to write progress bars + # @return [void] + def type_check(show_progress: true, io: $stdout) + io.puts "Type checking IDL code for #{@config.name}..." + progressbar = + if show_progress + ProgressBar.create(title: "Instructions", total: instructions.size) + end + + instructions.each do |inst| + progressbar.increment if show_progress + if @mxlen == 32 + inst.type_checked_operation_ast(@idl_compiler, @symtab, 32) if inst.rv32? + elsif @mxlen == 64 + inst.type_checked_operation_ast(@idl_compiler, @symtab, 64) if inst.rv64? + inst.type_checked_operation_ast(@idl_compiler, @symtab, 32) if possible_xlens.include?(32) && inst.rv32? + end + end + + progressbar = + if show_progress + ProgressBar.create(title: "CSRs", total: csrs.size) + end + + csrs.each do |csr| + progressbar.increment if show_progress + if csr.has_custom_sw_read? + if (possible_xlens.include?(32) && csr.defined_in_base32?) || (possible_xlens.include?(64) && csr.defined_in_base64?) + csr.type_checked_sw_read_ast(@symtab) + end + end + csr.fields.each do |field| + unless field.type_ast(@symtab).nil? + if ((possible_xlens.include?(32) && csr.defined_in_base32? && field.defined_in_base32?) || + (possible_xlens.include?(64) && csr.defined_in_base64? && field.defined_in_base64?)) + field.type_checked_type_ast(@symtab) + end + end + unless field.reset_value_ast(@symtab).nil? + if ((possible_xlens.include?(32) && csr.defined_in_base32? && field.defined_in_base32?) || + (possible_xlens.include?(64) && csr.defined_in_base64? && field.defined_in_base64?)) + field.type_checked_reset_value_ast(@symtab) if csr.defined_in_base32? && field.defined_in_base32? + end + end + unless field.sw_write_ast(@symtab).nil? + field.type_checked_sw_write_ast(@symtab, 32) if possible_xlens.include?(32) && csr.defined_in_base32? && field.defined_in_base32? + field.type_checked_sw_write_ast(@symtab, 64) if possible_xlens.include?(64) && csr.defined_in_base64? && field.defined_in_base64? + end + end + end + + progressbar = + if show_progress + ProgressBar.create(title: "Functions", total: functions.size) + end + functions.each do |func| + progressbar.increment if show_progress + func.type_check(@symtab) + end + + puts "done" if show_progress + end + + # @return [Array] List of all available parameters with known values for the config + def params_with_value + return @params_with_value unless @params_with_value.nil? + + @params_with_value = [] + return @params_with_value if @config.unconfigured? + + if @config.fully_configured? + transitive_implemented_extensions.each do |ext_version| + ext = extension(ext_version.name) + ext.params.each do |ext_param| + next unless @config.param_values.key?(ext_param.name) + + @params_with_value << ExtensionParameterWithValue.new( + ext_param, + @config.param_values[ext_param.name] + ) + end + end + elsif @config.partially_configured? + mandatory_extensions.each do |ext_requirement| + ext = extension(ext_requirement.name) + ext.params.each do |ext_param| + next unless @config.param_values.key?(ext_param.name) + + @params_with_value << ExtensionParameterWithValue.new( + ext_param, + @config.param_values[ext_param.name] + ) + end + end + else + raise "ERROR: unexpected config type" + end + @params_with_value + end + + # @return [Array] List of all available parameters without known values for the config + def params_without_value + return @params_without_value unless @params_without_value.nil? + + @params_without_value = [] + extensions.each do |ext| + ext.params.each do |ext_param| + next if @config.param_values.key?(ext_param.name) + + @params_without_value << ext_param + end + end + @params_without_value + end + + # Returns a string representation of the object, suitable for debugging. + # @return [String] A string representation of the object. + def inspect = "ConfiguredArchitecture##{name}" + + def implemented_extensions + @implemented_extensions ||= + @config.implemented_extensions.map do |e| + ExtensionVersion.new(e["name"], e["version"], self, fail_if_version_does_not_exist: true) + end + end + + # @return [Array] List of all extensions known to be implemented in this config, including transitive implications + def transitive_implemented_extensions + return @transitive_implemented_extensions unless @transitive_implemented_extensions.nil? + + raise "implemented_extensions is only valid for a fully configured defintion" unless @config.fully_configured? + + list = implemented_extensions + list.each do |e| + implications = e.transitive_implications + list.concat(implications) unless implications.empty? + end + @transitive_implemented_extensions = list.uniq.sort + end + + # @return [Array] List of all mandatory extension requirements + def mandatory_extensions + @mandatory_extensions ||= + @config.mandatory_extensions.map do |e| + ext = extension(e["name"]) + raise "Cannot find extension #{e['name']} in the architecture definition" if ext.nil? + + ExtensionRequirement.new(e["name"], *e["version"], presence: "mandatory", cfg_arch: self) + end + end + + # @return [Array] List of all extensions that are prohibited. + # This includes extensions explicitly prohibited by the config file + # and extensions that conflict with a mandatory extension. + def prohibited_extensions + return @prohibited_extensions unless @prohibited_extensions.nil? + + if @config.partially_configured? + @prohibited_extensions = + @config.prohibited_extensions.map do |e| + ext = extension(e["name"]) + raise "Cannot find extension #{e['name']} in the architecture definition" if ext.nil? + + ExtensionRequirement.new(e["name"], *e["version"], presence: "mandatory", cfg_arch: self) + end + + # now add any extensions that are prohibited by a mandatory extension + mandatory_extensions.each do |ext_req| + ext_req.extension.conflicts.each do |conflict| + if @prohibited_extensions.none? { |prohibited_ext| prohibited_ext.name == conflict.name } + @prohibited_extensions << conflict + else + # pick whichever requirement is more expansive + p = @prohibited_extensions.find { |prohibited_ext| prohibited_ext.name == confict.name } + if p.version_requirement.subsumes?(conflict.version_requirement) + @prohibited_extensions.delete(p) + @prohibited_extensions << conflict + end + end + end + end + + @prohibited_extensions + elsif @config.fully_configured? + prohibited_ext_versions = [] + extensions.each do |ext| + ext.versions.each do |ext_ver| + prohibited_ext_versions << ext_ver unless transitive_implemented_extensions.include?(ext_ver) + end + end + @prohibited_extensions = [] + prohibited_ext_versions.group_by(&:name).each_value do |ext_ver_list| + if ext_ver_list.sort == ext_ver_list[0].ext.versions.sort + # excludes every version + @prohibited_extensions << + ExtensionRequirement.new( + ext_ver_list[0].ext.name, ">= #{ext_ver_list.min.version_spec.canonical}", + presence: "prohibited", cfg_arch: self + ) + elsif ext_ver_list.size == (ext_ver_list[0].ext.versions.size - 1) + # excludes all but one version + allowed_version_list = (ext_ver_list[0].ext.versions - ext_ver_list) + raise "Expected only a single element" unless allowed_version_list.size == 1 + + allowed_version = allowed_version_list[0] + @prohibited_extensions << + ExtensionRequirement.new( + ext_ver_list[0].ext.name, "!= #{allowed_version.version_spec.canonical}", + presence: "prohibited", cfg_arch: self + ) + else + # need to group + raise "TODO" + end + end + else + @prohibited_extensions = [] + end + @prohibited_extensions + end + + # @overload prohibited_ext?(ext) + # Returns true if the ExtensionVersion +ext+ is prohibited + # @param ext [ExtensionVersion] An extension version + # @return [Boolean] + # + # @overload prohibited_ext?(ext) + # Returns true if any version of the extension named +ext+ is prohibited + # @param ext [String] An extension name + # @return [Boolean] + def prohibited_ext?(ext) + if ext.is_a?(ExtensionVersion) + prohibited_extensions.any? { |ext_req| ext_req.satisfied_by?(ext) } + elsif ext.is_a?(String) || ext.is_a?(Symbol) + prohibited_extensions.any? { |ext_req| ext_req.name == ext.to_s } + else + raise ArgumentError, "Argument to prohibited_ext? should be an ExtensionVersion or a String" + end + end + + # @overload ext?(ext_name) + # @param ext_name [#to_s] Extension name (case sensitive) + # @return [Boolean] True if the extension `name` is implemented + # @overload ext?(ext_name, ext_version_requirements) + # @param ext_name [#to_s] Extension name (case sensitive) + # @param ext_version_requirements [Number,String,Array] Extension version requirements + # @return [Boolean] True if the extension `name` meeting `ext_version_requirements` is implemented + # @example Checking extension presence with a version requirement + # cfg_arch.ext?(:S, ">= 1.12") + # @example Checking extension presence with multiple version requirements + # cfg_arch.ext?(:S, ">= 1.12", "< 1.15") + # @example Checking extension precsence with a precise version requirement + # cfg_arch.ext?(:S, 1.12) + def ext?(ext_name, *ext_version_requirements) + @ext_cache ||= {} + cached_result = @ext_cache[[ext_name, ext_version_requirements]] + return cached_result unless cached_result.nil? + + result = + if @config.fully_configured? + transitive_implemented_extensions.any? do |e| + if ext_version_requirements.empty? + e.name == ext_name.to_s + else + requirement = ExtensionRequirement.new(ext_name, *ext_version_requirements, cfg_arch: self) + requirement.satisfied_by?(e) + end + end + elsif @config.partially_configured? + mandatory_extensions.any? do |e| + if ext_version_requirements.empty? + e.name == ext_name.to_s + else + requirement = ExtensionRequirement.new(ext_name, *ext_version_requirements, cfg_arch: self) + e.satisfying_versions.all? do |ext_ver| + requirement.satisfied_by?(ext_ver) + end + end + end + else + raise "unexpected type" unless unconfigured? + + false + end + @ext_cache[[ext_name, ext_version_requirements]] = result + end + + # @return [Array] All exception codes known to be implemented + def implemented_exception_codes + return @implemented_exception_codes unless @implemented_exception_codes.nil? + + @implemented_exception_codes = + implemented_extensions.reduce([]) do |list, ext_version| + ecodes = extension(ext_version.name)["exception_codes"] + next list if ecodes.nil? + + ecodes.each do |ecode| + # double check that all the codes are unique + raise "Duplicate exception code" if list.any? { |e| e.num == ecode["num"] || e.name == ecode["name"] || e.var == ecode["var"] } + + unless ecode.dig("when", "version").nil? + # check version + next unless ext?(ext_version.name.to_sym, ecode["when"]["version"]) + end + list << ExceptionCode.new(ecode["name"], ecode["var"], ecode["num"], self) + end + list + end + end + + # @return [Array] All interrupt codes known to be implemented + def implemented_interrupt_codes + return @implemented_interrupt_codes unless @implemented_interrupt_codes.nil? + + @implemented_interupt_codes = + implemented_extensions.reduce([]) do |list, ext_version| + icodes = extension(ext_version.name)["interrupt_codes"] + next list if icodes.nil? + + icodes.each do |icode| + # double check that all the codes are unique + raise "Duplicate interrupt code" if list.any? { |i| i.num == icode["num"] || i.name == icode["name"] || i.var == icode["var"] } + + unless ecode.dig("when", "version").nil? + # check version + next unless ext?(ext_version.name.to_sym, ecode["when"]["version"]) + end + list << InterruptCode.new(icode["name"], icode["var"], icode["num"], self) + end + list + end + end + + # @return [Array] List of all functions defined by the architecture + def functions + return @functions unless @functions.nil? + + @functions = @global_ast.functions + end + + # @return [Array] List of all implemented CSRs + def transitive_implemented_csrs + @transitive_implemented_csrs ||= + transitive_implemented_extensions.map(&:implemented_csrs).flatten.uniq.sort + end + + # @return [Array] List of all implemented instructions + def transitive_implemented_instructions + @transitive_implemented_instructions ||= + transitive_implemented_extensions.map(&:implemented_instructions).flatten.uniq.sort + end + + # @return [Array] List of all reachable IDL functions for the config + def implemented_functions + return @implemented_functions unless @implemented_functions.nil? + + @implemented_functions = [] + + puts " Finding all reachable functions from instruction operations" + + transitive_implemented_instructions.each do |inst| + @implemented_functions << + if inst.base.nil? + if multi_xlen? + (inst.reachable_functions(symtab, 32) + + inst.reachable_functions(symtab, 64)) + else + inst.reachable_functions(symtab, mxlen) + end + else + inst.reachable_functions(symtab, inst.base) + end + end + raise "?" unless @implemented_functions.is_a?(Array) + @implemented_functions = @implemented_functions.flatten + @implemented_functions.uniq!(&:name) + + puts " Finding all reachable functions from CSR operations" + + transitive_implemented_csrs.each do |csr| + csr_funcs = csr.reachable_functions(self) + csr_funcs.each do |f| + @implemented_functions << f unless @implemented_functions.any? { |i| i.name == f.name } + end + end + + @implemented_functions + end + + # given an adoc string, find names of CSR/Instruction/Extension enclosed in `monospace` + # and replace them with links to the relevant object page + # + # @param adoc [String] Asciidoc source + # @return [String] Asciidoc source, with link placeholders + def find_replace_links(adoc) + adoc.gsub(/`([\w.]+)`/) do |match| + name = Regexp.last_match(1) + csr_name, field_name = name.split(".") + csr = csr(csr_name) + if !field_name.nil? && !csr.nil? && csr.field?(field_name) + "%%LINK%csr_field;#{csr_name}.#{field_name};#{csr_name}.#{field_name}%%" + elsif !csr.nil? + "%%LINK%csr;#{csr_name};#{csr_name}%%" + elsif instruction(name) + "%%LINK%inst;#{name};#{name}%%" + elsif extension(name) + "%%LINK%ext;#{name};#{name}%%" + else + match + end + end + end + + # Returns an environment hash suitable for use with ERb templates. + # + # This method returns a hash containing the architecture definition and other + # relevant data that can be used to generate ERb templates. + # + # @return [Hash] An environment hash suitable for use with ERb templates. + def erb_env + return @env unless @env.nil? + + @env = Class.new + @env.instance_variable_set(:@cfg, @cfg) + @env.instance_variable_set(:@params, @params) + @env.instance_variable_set(:@arch_gen, self) + + # add each parameter, either as a method (lowercase) or constant (uppercase) + params_with_value.each do |param| + @env.const_set(param.name, param.value) unless @env.const_defined?(param.name) + end + + params_without_value.each do |param| + @env.const_set(param.name, :unknown) unless @env.const_defined?(param.name) + end + + @env.instance_exec do + # method to check if a given extension (with an optional version number) is present + # + # @param ext_name [String,#to_s] Name of the extension + # @param ext_requirement [String, #to_s] Version string, as a Gem Requirement (https://guides.rubygems.org/patterns/#pessimistic-version-constraint) + # @return [Boolean] whether or not extension +ext_name+ meeting +ext_requirement+ is implemented in the config + def ext?(ext_name, ext_requirement = ">= 0") + @arch_gen.ext?(ext_name.to_s, ext_requirement) + end + + # @return [Array] List of possible XLENs for any implemented mode + def possible_xlens + @arch_gen.possible_xlens + end + + # insert a hyperlink to an object + # At this point, we insert a placeholder since it will be up + # to the backend to create a specific link + # + # @params type [Symbol] Type (:section, :csr, :inst, :ext) + # @params name [#to_s] Name of the object + def link_to(type, name) + "%%LINK%#{type};#{name}%%" + end + + # info on interrupt and exception codes + + # @returns [Hash] architecturally-defined exception codes and their names + def exception_codes + @arch_gen.exception_codes + end + + # returns [Hash] architecturally-defined interrupt codes and their names + def interrupt_codes + @arch_gen.interrupt_codes + end + + # @returns [Hash] architecturally-defined exception codes and their names + def implemented_exception_codes + @arch_gen.implemented_exception_codes + end + + # returns [Hash] architecturally-defined interrupt codes and their names + def implemented_interrupt_codes + @arch_gen.implemented_interrupt_codes + end + end + + @env + end + private :erb_env + + # passes _erb_template_ through ERB within the content of this config + # + # @param erb_template [String] ERB source + # @return [String] The rendered text + def render_erb(erb_template, what = "") + t = Tempfile.new("template") + t.write erb_template + t.flush + begin + Tilt["erb"].new(t.path, trim: "-").render(erb_env) + rescue + warn "While rendering ERB template: #{what}" + raise + ensure + t.close + t.unlink + end + end +end diff --git a/lib/config.rb b/lib/config.rb new file mode 100644 index 000000000..92a9b9d04 --- /dev/null +++ b/lib/config.rb @@ -0,0 +1,183 @@ +# frozen_string_literal: true + +require "pathname" + +# this class represents a configuration file (e.g., cfgs/*/cfg.yaml), independent of the Architecture +class Config + # @return [Hash] A hash mapping parameter name to value for any parameter that has + # been configured with a value. May be empty. + attr_reader :param_values + + # use Config#create instead + private_class_method :new + + def self.freeze_data(obj) + if obj.is_a?(Hash) + obj.each do |k, v| + obj[k] = freeze_data(v) + end + elsif obj.is_a?(Array) + obj.each { |v| freeze_data(v) } + end + + obj.freeze + end + private_class_method :freeze_data + + # factory method to create a FullConfig, PartialConfig, or Unconfig based on the contents of cfg_filename + # + # @return [Config] A new Config + def self.create(cfg_filename) + cfg_file_path = Pathname.new(cfg_filename) + raise ArgumentError, "Cannot find #{cfg_filename}" unless cfg_file_path.exist? + + data = YAML.load(cfg_file_path.read, permitted_classes: [Date]) + + # now deep freeze the data + freeze_data(data) + + case data["type"] + when "fully configured" + FullConfig.send(:new, cfg_file_path, data) + when "partially configured" + PartialConfig.send(:new, cfg_file_path, data) + when "unconfigured" + Unconfig.send(:new, cfg_file_path, data) + else + raise "Unexpected type in config" + end + end + + def initialize(cfg_file_path, data) + @cfg_file_path = cfg_file_path + @data = data + end + + def name = @data["name"] + + def fully_configured? = @data["type"] == "fully configured" + def partially_configured? = @data["type"] == "partially configured" + def unconfigured? = @data["type"] == "unconfigured" + def configured? = @data["type"] != "unconfigured" + def type = @data["type"] +end + +# this class represents a configuration file (e.g., cfgs/*/cfg.yaml) that is "unconfigured" +# (i.e., we don't know any implemented/mandatory extensions or parameter values) +class Unconfig < Config + attr_reader :param_values + + def initialize(cfg_file_path, data) + super(cfg_file_path, data) + + @param_values = {}.freeze + end + + def mxlen = nil + + def implemented_extensions = raise "implemented_extensions is only availabe for a FullConfig" + def mandatory_extensions = raise "mandatory_extensions is only availabe for a PartialConfig" + def prohibited_extensions = raise "prohibited_extensions is only availabe for a PartialConfig" +end + +# this class represents a configuration file (e.g., cfgs/*/cfg.yaml) that is "partially configured" +# (i.e., we have a list of mandatory/prohibited extensions and a paritial list of parameter values) +# +# This would, for example, represent a Profile or configurable IP +class PartialConfig < Config + attr_reader :param_values, :mxlen + + def initialize(cfg_file_path, data) + super(cfg_file_path, data) + + @param_values = @data.key?("params") ? @data["params"] : [].freeze + + @mxlen = @data.dig("params", "XLEN") + raise "Must set XLEN for a configured config" if @mxlen.nil? + + @mxlen.freeze + end + + def implemented_extensions = raise "implemented_extensions is only availabe for a FullConfig" + + # @return [Array String,Array] + # List of all extensions that must be implemented, as specified in the config file + # The first entry in the nested array is an Extension name. + # The second entry in the nested array is an Extension version requirement + # + # @example + # partial_config.mandatory_extensions #=> [{ "name" => "A", "version" => ["~> 2.0"] }, { "name" => "B", "version" => ["~> 1.0"] }, ...] + def mandatory_extensions + @mandatory_extensions ||= + if @data["mandatory_extensions"].nil? + [] + else + @data["mandatory_extensions"].map do |e| + # convert the requirement to always be an array + { "name" => e["name"], "version" => e["version"].is_a?(String) ? [e["version"]] : e["version"]} + end + end + end + + # @return [Array String,Array] + # List of all extensions that are explicitly prohibited. + # The first entry in the nested array is an Extension name. + # The second entry in the nested array is an Extension version requirement. + # + # @example + # partial_config.prohibited_extensions #=> [{ "name" => "F", "version" => [">= 2.0"] }, { "name" => "Zfa", "version" => ["> = 1.0"] }, ...] + def prohibited_extensions + @prohibited_extensions ||= + if @data["prohibited_extensions"].nil? + [] + else + @data["prohibited_extensions"].map do |e| + # convert the requirement to always be an array + { "name" => e["name"], "version" => e["version"].is_a?(String) ? [e["version"]] : e["version"]} + end + end + end + + # def prohibited_ext?(ext_name, cfg_arch) = prohibited_extensions(cfg_arch).any? { |e| e.name == ext_name.to_s } + + # def ext?(ext_name, cfg_arch) = mandatory_extensions(cfg_arch).any? { |e| e.name == ext_name.to_s } +end + +# this class represents a configuration file (e.g., cfgs/*/cfg.yaml) that is "fully configured" +# (i.e., we have a complete list of implemented extensions and a complete list of parameter values) +# +# This would, for example, represent a specific silicon tapeout/SKU +class FullConfig < Config + attr_reader :param_values, :mxlen + + def initialize(cfg_file_path, data) + super(cfg_file_path, data) + + @param_values = @data["params"] + + @mxlen = @data.dig("params", "XLEN").freeze + raise "Must set XLEN for a configured config" if @mxlen.nil? + end + + # @return [Array>] List of all extensions known to be implemented in this architecture + def implemented_extensions + @implemented_extensions ||= + if @data["implemented_extensions"].nil? + [] + else + @data["implemented_extensions"].map do |e| + if e.is_a?(Array) + { "name" => e[0], "version" => e[1] } + else + e + end + end + end + end + + def mandatory_extensions = raise "mandatory_extensions is only availabe for a PartialConfig" + def prohibited_extensions = raise "prohibited_extensions is only availabe for a PartialConfig" + + # def prohibited_ext?(ext_name, cfg_arch) = !ext?(ext_name, cfg_arch) + # def ext?(ext_name, cfg_arch) = implemented_extensions(cfg_arch).any? { |e| e.name == ext_name.to_s } +end diff --git a/lib/idl.rb b/lib/idl.rb index 47250ddb2..287761e22 100644 --- a/lib/idl.rb +++ b/lib/idl.rb @@ -35,10 +35,8 @@ def instantiate_node(node_type, *args) module Idl # the Idl compiler class Compiler - # @param arch_def [ArchDef] Architecture defintion, the context of the compilation - def initialize(arch_def) + def initialize @parser = IdlParser.new - @arch_def = arch_def end def compile_file(path) @@ -248,7 +246,16 @@ def compile_expression(expression, symtab, pass_error: false) ast = m.to_ast ast.set_input_file("[EXPRESSION]", 0) - ast.freeze_tree(symtab) + value_result = ast.value_try do + ast.freeze_tree(symtab) + end + if value_result == :unknown_value + raise AstNode::TypeError, "Bad literal value" if pass_error + + warn "Compiling #{expression}" + warn "Bad literal value" + exit 1 + end begin ast.type_check(symtab) rescue AstNode::TypeError => e diff --git a/lib/idl/ast.rb b/lib/idl/ast.rb index b2e58f54c..001233991 100644 --- a/lib/idl/ast.rb +++ b/lib/idl/ast.rb @@ -47,6 +47,8 @@ class AstNode Bits1Type = Type.new(:bits, width: 1).freeze Bits32Type = Type.new(:bits, width: 32).freeze Bits64Type = Type.new(:bits, width: 64).freeze + BitsUnknownType = Type.new(:bits, width: :unknown).freeze + ConstBitsUnknownType = Type.new(:bits, width: :unknown, qualifiers: [:const]).freeze ConstBoolType = Type.new(:boolean, qualifiers: [:const]).freeze BoolType = Type.new(:boolean).freeze VoidType = Type.new(:void).freeze @@ -562,7 +564,7 @@ def type_check(symtab) type_error "no symbol named '#{name}' on line #{lineno}" if symtab.get(name).nil? end - # @!macro type_no_archdef + # @!macro type_no_cfg_arch def type(symtab) return @type unless @type.nil? @@ -586,11 +588,11 @@ def const? = @const def value(symtab) # can't do this.... a const might be in a template function, with different values at call time # if @const - # # consts never change, so we can look them up by arch_def - # var = @vars[symtab.archdef] + # # consts never change, so we can look them up by cfg_arch + # var = @vars[symtab.cfg_arch] # if var.nil? # var = symtab.get(name) - # @vars[symtab.archdef] = var + # @vars[symtab.cfg_arch] = var # end # type_error "Variable '#{name}' was not found" if var.nil? # value_error "Value of '#{name}' not known" if var.value.nil? @@ -791,7 +793,7 @@ def type_check(symtab) type_error "#{expression.text_value} is not an array" unless expression_type.kind == :array type_error "#{expression.text_value} must be a constant" unless expression_type.const? - if symtab.archdef.fully_configured? && (expression_type.width == :unknown) + if symtab.cfg_arch.fully_configured? && (expression_type.width == :unknown) type_error "#{expression.text_value} must have a known value at compile time" end end @@ -1061,7 +1063,7 @@ def type(symtab) end # @!macro value_no_args - def value(_symtab, _archdef) = raise InternalError, "Enum defintions have no value" + def value(_symtab, _cfg_arch) = raise InternalError, "Enum defintions have no value" # @return [String] enum name def name = @user_type.text_value @@ -1117,11 +1119,11 @@ def type_check(_symtab) def element_names(symtab) case name when "ExtensionName" - symtab.archdef.extensions.map(&:name) + symtab.cfg_arch.extensions.map(&:name) when "ExceptionCode" - symtab.archdef.exception_codes.map(&:var) + symtab.cfg_arch.exception_codes.map(&:var) when "InterruptCode" - symtab.archdef.interrupt_codes.map(&:var) + symtab.cfg_arch.interrupt_codes.map(&:var) else type_error "Unknown builtin enum type '#{name}'" end @@ -1130,17 +1132,17 @@ def element_names(symtab) def element_values(symtab) case name when "ExtensionName" - (0...symtab.archdef.extensions.size).to_a + (0...symtab.cfg_arch.extensions.size).to_a when "ExceptionCode" - symtab.archdef.exception_codes.map(&:num) + symtab.cfg_arch.exception_codes.map(&:num) when "InterruptCode" - symtab.archdef.interrupt_codes.map(&:num) + symtab.cfg_arch.interrupt_codes.map(&:num) else type_error "Unknown builtin enum type '#{name}'" end end - # @!macro type_no_archdef + # @!macro type_no_cfg_arch def type(symtab) return @type unless @type.nil? @@ -1326,7 +1328,7 @@ def type(symtab) def name = @name.text_value # @!macro value_no_args - def value(_symtab, _archdef) = raise AstNode::InternalError, "Bitfield defintions have no value" + def value(_symtab, _cfg_arch) = raise AstNode::InternalError, "Bitfield defintions have no value" # @!macro to_idl def to_idl @@ -1501,20 +1503,21 @@ def type_check(symtab) type_error "Array index must be integral" unless index.type(symtab).integral? - if var.type(symtab).kind == :array + var_type = var.type(symtab) + if var_type.kind == :array value_result = value_try do index_value = index.value(symtab) - if var.type(symtab).width != :unknown - type_error "Array index out of range" if index_value >= var.type(symtab).width + if var_type.width != :unknown + type_error "Array index out of range" if index_value >= var_type.width end end # Ok, doesn't need to be known - elsif var.type(symtab).integral? - if var.type(symtab).kind == :bits + elsif var_type.integral? + if var_type.kind == :bits value_result = value_try do index_value = index.value(symtab) - if index_value >= var.type(symtab).width - type_error "Bits element index (#{index_value}) out of range (max #{var.type(symtab).width - 1}) in access '#{text_value}'" + if (var_type.width != :unknown) && (index_value >= var_type.width) + type_error "Bits element index (#{index_value}) out of range (max #{var_type.width - 1}) in access '#{text_value}'" end end # OK, doesn need to be known end @@ -1525,9 +1528,10 @@ def type_check(symtab) end def type(symtab) - if var.type(symtab).kind == :array - var.type(symtab).sub_type - elsif var.type(symtab).integral? + var_type = var.type(symtab) + if var_type.kind == :array + var_type.sub_type + elsif var_type.integral? Bits1Type else internal_error "Bad ary element access" @@ -1582,8 +1586,9 @@ def type_check(symtab) msb_value = msb.value(symtab) lsb_value = lsb.value(symtab) - if var.type(symtab).kind == :bits && msb_value >= var.type(symtab).width - type_error "Range too large for bits (msb = #{msb_value}, range size = #{var.type(symtab).width})" + var_type = var.type(symtab) + if var_type.kind == :bits && var_type.width != :unknown && msb_value >= var_type.width + type_error "Range too large for bits (msb = #{msb_value}, range size = #{var_type.width})" end range_size = msb_value - lsb_value + 1 @@ -1684,10 +1689,10 @@ def type_check(symtab) end def var(symtab) - variable = @vars[symtab.archdef] + variable = @vars[symtab.cfg_arch] if variable.nil? variable = symtab.get(lhs.text_value) - @vars[symtab.archdef] = variable + @vars[symtab.cfg_arch] = variable end variable end @@ -2018,15 +2023,15 @@ def initialize(input, interval, csr_field, write_value) def type(symtab) if field(symtab).defined_in_all_bases? - if symtab.archdef.mxlen == 64 && symtab.archdef.multi_xlen? - Type.new(:bits, width: [field(symtab).location(symtab.archdef, 32).size, field(symtab).location(symtab.archdef, 64).size].max) + if symtab.cfg_arch.mxlen == 64 && symtab.cfg_arch.multi_xlen? + Type.new(:bits, width: [field(symtab).location(symtab.cfg_arch, 32).size, field(symtab).location(symtab.cfg_arch, 64).size].max) else - Type.new(:bits, width: field(symtab).location(symtab.archdef, symtab.archdef.mxlen).size) + Type.new(:bits, width: field(symtab).location(symtab.cfg_arch, symtab.cfg_arch.mxlen).size) end elsif field(symtab).base64_only? - Type.new(:bits, width: field(symtab).location(symtab.archdef, 64).size) + Type.new(:bits, width: field(symtab).location(symtab.cfg_arch, 64).size) elsif field(symtab).base32_only? - Type.new(:bits, width: field(symtab).location(symtab.archdef, 32).size) + Type.new(:bits, width: field(symtab).location(symtab.cfg_arch, 32).size) else internal_error "Unexpected base for field" end @@ -2043,7 +2048,7 @@ def type_check(symtab) type_error "Cannot write to read-only CSR field" end end - # ok, we don't know the type because the archdef isn't configured + # ok, we don't know the type because the cfg_arch isn't configured write_value.type_check(symtab) type_error "Incompatible type in assignment" unless write_value.type(symtab).convertable_to?(type(symtab)) @@ -2270,7 +2275,7 @@ def decl_type(symtab) dtype = Type.new(:array, width: ary_size.value(symtab), sub_type: dtype, qualifiers:) end value_else(value_result) do - type_error "Array size must be known at compile time" if symtab.archdef.fully_configured? + type_error "Array size must be known at compile time" if symtab.cfg_arch.fully_configured? dtype = Type.new(:array, width: :unknown, sub_type: dtype, qualifiers:) end end @@ -2295,8 +2300,8 @@ def type_check(symtab, add_sym = true) ary_size.value(symtab) end value_else(value_result) do - # if this is a fully configured ArchDef, this is an error because all constants are supposed to be known - if symtab.archdef.fully_configured? + # if this is a fully configured ConfiguredArchitecture, this is an error because all constants are supposed to be known + if symtab.cfg_arch.fully_configured? type_error "Array size (#{ary_size.text_value}) must be known at compile time" else # otherwise, it's ok that we don't know the value yet, as long as the value is a const @@ -2404,7 +2409,6 @@ def type_check(symtab) decl_type = lhs_type(symtab) - if decl_type.const? # this is a constant; ensure we are assigning a constant value value_result = value_try do @@ -2412,7 +2416,7 @@ def type_check(symtab) end value_else(value_result) do unless rhs.type(symtab).const? - type_error "Declaring constant with a non-constant value (#{e})" + type_error "Declaring constant with a non-constant value (#{rhs.text_value})" end symtab.add(lhs.text_value, Var.new(lhs.text_value, decl_type.clone)) end @@ -2587,10 +2591,10 @@ def type(symtab) when :enum_ref Type.new(:bits, width: etype.enum_class.width) when :csr - if etype.csr.dynamic_length?(symtab.archdef) + if etype.csr.dynamic_length?(symtab.cfg_arch) Type.new(:bits, width: :unknown) else - Type.new(:bits, width: etype.csr.length(symtab.archdef)) + Type.new(:bits, width: etype.csr.length(symtab.cfg_arch)) end end end @@ -2950,14 +2954,19 @@ def value(symtab) elsif op == "|" # if one side is all ones, we don't need to know the other side + rhs_type = rhs.type(symtab) + value_error("Unknown width") if rhs_type.width == :unknown + lhs_type = lhs.type(symtab) + value_error("unknown width") if lhs_type.width == :unknown + value_result = value_try do - rhs_mask = ((1 << rhs.type(symtab).width) - 1) - return rhs_mask if (rhs.value(symtab) == rhs_mask) && (lhs.type(symtab).width <= rhs.type(symtab).width) + rhs_mask = ((1 << rhs_type.width) - 1) + return rhs_mask if (rhs.value(symtab) == rhs_mask) && (lhs_type.width <= rhs_type.width) end - # ok, trye rhs + # ok, try rhs - lhs_mask = ((1 << lhs.type(symtab).width) - 1) - return lhs_mask if (lhs.value(symtab) == lhs_mask) && (rhs.type(symtab).width <= lhs.type(symtab).width) + lhs_mask = ((1 << lhs_type.width) - 1) + return lhs_mask if (lhs.value(symtab) == lhs_mask) && (rhs_type.width <= lhs_type.width) lhs.value(symtab) | rhs.value(symtab) @@ -2991,6 +3000,10 @@ def value(symtab) v_trunc = if !lhs.type(symtab).const? || !rhs.type(symtab).const? # when both sides are constant, the value is not truncated + width = type(symtab).width + if width == :unknown + value_error("unknown width in op that possibly truncates") + end v & ((1 << type(symtab).width) - 1) else v @@ -3097,15 +3110,21 @@ def type_check(symtab) expressions.each do |exp| exp.type_check(symtab) - type_error "Concatenation only supports Bits<> types" unless exp.type(symtab).kind == :bits + e_type = exp.type(symtab) + type_error "Concatenation only supports Bits<> types" unless e_type.kind == :bits - internal_error "Negative width for element #{exp.text_value}" if exp.type(symtab).width <= 0 + internal_error "Negative width for element #{exp.text_value}" if (e_type.width != :unknown) && (e_type.width <= 0) end end # @!macro type def type(symtab) - total_width = expressions.reduce(0) { |sum, exp| sum + exp.type(symtab).width } + total_width = expressions.reduce(0) do |sum, exp| + e_type = exp.type(symtab) + return BitsUnknownType if e_type.width == :unknown + + sum + e_type.width + end Type.new(:bits, width: total_width) end @@ -3430,7 +3449,7 @@ def initialize(input, interval, class_name, member_name) def freeze_tree(global_symtab) return if frozen? - enum_def_ast = global_symtab.archdef.global_ast.enums.find { |e| e.name == @enum_class_name } + enum_def_ast = global_symtab.cfg_arch.global_ast.enums.find { |e| e.name == @enum_class_name } @enum_def_type = if enum_def_ast.is_a?(BuiltinEnumDefinitionAst) @@ -3452,7 +3471,7 @@ def type_check(symtab) type_error "#{@enum_class_name} has no member '#{@member_name}'" if enum_def_type.value(@member_name).nil? end - # @!macro type_no_archdef + # @!macro type_no_cfg_arch def type(symtab) internal_error "Not frozen?" unless frozen? type_error "No enum named #{@enum_class_name}" if @enum_def_type.nil? @@ -3460,7 +3479,7 @@ def type(symtab) @enum_def_type.ref_type end - # @!macro value_no_archdef + # @!macro value_no_cfg_arch def value(symtab) internal_error "Must call type_check first" if @enum_def_type.nil? @@ -3561,12 +3580,16 @@ def value(symtab) else internal_error "Unhandled unary op #{op}" end - if type(symtab).integral? - val_trunc = val & ((1 << type(symtab).width) - 1) - if type(symtab).signed? && ((((val_trunc >> (type(symtab).width - 1))) & 1) == 1) + t = type(symtab) + if t.integral? + if t.width == :unknown + value_error("Unknown width for truncation") + end + val_trunc = val & ((1 << t.width) - 1) + if t.signed? && ((((val_trunc >> (t.width - 1))) & 1) == 1) # need to make this negative! # take the twos compliment - val_trunc = -((1 << type(symtab).width) - val_trunc) + val_trunc = -((1 << t.width) - val_trunc) end end @@ -3879,7 +3902,7 @@ def type(_symtab) end # @!macro value_no_args - def value(_symtab, _archdef) = internal_error "Why are you calling value for an lval?" + def value(_symtab, _cfg_arch) = internal_error "Why are you calling value for an lval?" def to_idl = "-" end @@ -3992,7 +4015,7 @@ def expected_return_type(symtab) symtab.get("__expected_return_type") else # need to find the type to get the right symbol table - func_type = @func_type_cache[symtab.archdef] + func_type = @func_type_cache[symtab.cfg_arch] return func_type.return_type(EMPTY_ARRAY, self) unless func_type.nil? func_type = symtab.get_global(func_def.name) @@ -4011,7 +4034,7 @@ def expected_return_type(symtab) end func_type.return_type(template_values.sort { |a, b| a.template_index <=> b.template_index }.map(&:value), self) else - @func_type_cache[symtab.archdef]= func_type + @func_type_cache[symtab.cfg_arch]= func_type func_type.return_type(EMPTY_ARRAY, self) end end @@ -4185,7 +4208,7 @@ def type_check(symtab) end end value_else(value_result) do - type_error "Bit width must be known at compile time" if symtab.archdef.fully_configured? + type_error "Bit width must be known at compile time" if symtab.cfg_arch.fully_configured? end end unless ["Bits", "String", "XReg", "Boolean", "U32", "U64"].include?(@type_name) @@ -4297,7 +4320,15 @@ class IntLiteralAst < AstNode def initialize(input, interval) super(input, interval, EMPTY_ARRAY) - @types = [nil, nil] + end + + def freeze_tree(global_symtab) + return if frozen? + + # initialize the cached objects + type(global_symtab) + value(global_symtab) + freeze end # @!macro type_check @@ -4308,8 +4339,7 @@ def type_check(symtab) value_text = ::Regexp.last_match(6) if width.nil? || width == "XLEN" - width = symtab.mxlen - memoize = false + width = symtab.mxlen.nil? ? 32 : symtab.mxlen # 32 is the min width, which is what we care about here end # ensure we actually have enough bits to represent the value @@ -4319,48 +4349,39 @@ def type_check(symtab) # @!macro type def type(symtab) - cache_idx = symtab.mxlen >> 6 # 0 = 32, 1 = 64 - return @types[cache_idx] unless @types[cache_idx].nil? + return @type unless @type.nil? case text_value.delete("_") when /^((XLEN)|([0-9]+))?'(s?)([bodh]?)(.*)$/ # verilog-style literal - width = ::Regexp.last_match(1) signed = ::Regexp.last_match(4) + width = width(symtab) - memoize = true - if width.nil? || width == "XLEN" - width = symtab.mxlen - memoize = false + unless width == :unknown + type_error("integer width must be positive (is #{width})") unless width.is_a?(Integer) && width.positive? end qualifiers = signed == "s" ? [:signed, :const] : [:const] - t = Type.new(:bits, width: width.to_i, qualifiers:) - @types[cache_idx] = t if memoize - t + @type = Type.new(:bits, width:, qualifiers:) when /^0([bdx]?)([0-9a-fA-F]*)(s?)$/ # C++-style literal signed = ::Regexp.last_match(3) qualifiers = signed == "s" ? [:signed, :const] : [:const] - type = Type.new(:bits, width: width(symtab), qualifiers:) - @types[cache_idx] = type - type + @type = Type.new(:bits, width: width(symtab), qualifiers:) when /^([0-9]*)(s?)$/ # basic decimal signed = ::Regexp.last_match(2) qualifiers = signed == "s" ? [:signed, :const] : [:const] - type = Type.new(:bits, width: width(symtab), qualifiers:) - @types[cache_idx] = type - type + @type = Type.new(:bits, width: width(symtab), qualifiers:) else internal_error "Unhandled int value" end end def width(symtab) - # return @width unless @width.nil? + return @width unless @width.nil? text_value_no_underscores = text_value.delete("_") @@ -4368,29 +4389,26 @@ def width(symtab) when /^((XLEN)|([0-9]+))?'(s?)([bodh]?)(.*)$/ # verilog-style literal width = ::Regexp.last_match(1) - memoize = true if width.nil? || width == "XLEN" - width = symtab.mxlen - memoize = false + width = symtab.mxlen.nil? ? :unknown : symtab.mxlen + else + width = width.to_i end - # @width = width if memoize - width + @width = width when /^0([bdx]?)([0-9a-fA-F]*)(s?)$/ signed = ::Regexp.last_match(3) width = signed == "s" ? value(symtab).bit_length + 1 : value(symtab).bit_length width = 1 if width.zero? # happens when the literal is '0' - # @width = width - width + @width = width when /^([0-9]*)(s?)$/ signed = ::Regexp.last_match(3) width = signed == "s" ? value(symtab).bit_length + 1 : value(symtab).bit_length width = 1 if width.zero? # happens when the literal is '0' - # @width = width - width + @width = width else internal_error "No match on int literal" end @@ -4398,65 +4416,64 @@ def width(symtab) # @!macro value def value(symtab) - # return @value unless @value.nil? + return @value unless @value.nil? if text_value.delete("_") =~ /^((XLEN)|([0-9]+))?'(s?)([bodh]?)(.*)$/ # verilog-style literal - width = ::Regexp.last_match(1) signed = ::Regexp.last_match(4) - - memoize = true - if width.nil? || width == "XLEN" - width = symtab.mxlen - memoize = false - end + width = width(symtab) v = - if !signed.empty? && ((unsigned_value >> (width.to_i - 1)) == 1) - -(2**width.to_i - unsigned_value) + if width == :unknown + if !signed.empty? + if unsigned_value > 0x7fff_ffff + value_error("Don't know if value will be negative") + else + if unsigned_value > 0xffff_ffff + value_error("Don't know if value will fit in literal") + end + unsigned_value + end + else + if unsigned_value > 0xffff_ffff + value_error("Don't know if value will fit in literal") + end + unsigned_value + end else - unsigned_value + if unsigned_value.bit_length > width + value_error("Value does not fit in literal") + end + if !signed.empty? && ((unsigned_value >> (width - 1)) == 1) + if unsigned_value.bit_length > (width - 1) + value_error("Value does not fit in literal") + end + -(2**width.to_i - unsigned_value) + else + unsigned_value + end end - # @value = v if memoize - v + @value = v else - # @value = unsigned_value - unsigned_value + @value = unsigned_value end end # @return [Integer] the unsigned value of this literal (i.e., treating it as unsigned even if the signed specifier is present) def unsigned_value - # return @unsigned_value unless @unsigned_value.nil? - - case text_value.delete("_") - when /^((XLEN)|([0-9]+))?'(s?)([bodh]?)(.*)$/ - # verilog-style literal - radix_id = ::Regexp.last_match(5) - value = ::Regexp.last_match(6) + return @unsigned_value unless @unsigned_value.nil? - radix_id = "d" if radix_id.empty? + @unsigned_value = + case text_value.delete("_") + when /^((XLEN)|([0-9]+))?'(s?)([bodh]?)(.*)$/ + # verilog-style literal + radix_id = ::Regexp.last_match(5) + value = ::Regexp.last_match(6) - case radix_id - when "b" - value.to_i(2) - when "o" - value.to_i(8) - when "d" - value.to_i(10) - when "h" - value.to_i(16) - end - when /^0([bdx]?)([0-9a-fA-F]*)(s?)$/ - # C++-style literal - radix_id = ::Regexp.last_match(1) - value = ::Regexp.last_match(2) + radix_id = "d" if radix_id.empty? - radix_id = "o" if radix_id.empty? - - # @unsigned_value = case radix_id when "b" value.to_i(2) @@ -4464,19 +4481,37 @@ def unsigned_value value.to_i(8) when "d" value.to_i(10) - when "x" + when "h" value.to_i(16) end + when /^0([bdx]?)([0-9a-fA-F]*)(s?)$/ + # C++-style literal + radix_id = ::Regexp.last_match(1) + value = ::Regexp.last_match(2) + + radix_id = "o" if radix_id.empty? + + # @unsigned_value = + case radix_id + when "b" + value.to_i(2) + when "o" + value.to_i(8) + when "d" + value.to_i(10) + when "x" + value.to_i(16) + end - when /^([0-9]*)(s?)$/ - # basic decimal - value = ::Regexp.last_match(1) + when /^([0-9]*)(s?)$/ + # basic decimal + value = ::Regexp.last_match(1) - # @unsigned_value = value.to_i(10) - value.to_i(10) - else - internal_error "Unhandled int value '#{text_value}'" - end + # @unsigned_value = value.to_i(10) + value.to_i(10) + else + internal_error "Unhandled int value '#{text_value}'" + end end # @!macro to_idl @@ -4548,7 +4583,7 @@ def arg_nodes end def func_type(symtab) - func_def_type = @func_def_type_cache[symtab.archdef] + func_def_type = @func_def_type_cache[symtab.cfg_arch] return func_def_type unless func_def_type.nil? func_def_type = symtab.get(@name) @@ -4558,14 +4593,14 @@ def func_type(symtab) type_error "#{@name} is not a function (it's a #{func_def_type.class.name})" end - @func_def_type_cache[symtab.archdef] = func_def_type + @func_def_type_cache[symtab.cfg_arch] = func_def_type end # @!macro type_check def type_check(symtab) level = symtab.levels - unknown_ok = symtab.archdef.partially_configured? + unknown_ok = symtab.cfg_arch.partially_configured? tvals = template_values(symtab, unknown_ok:) func_def_type = func_type(symtab) @@ -4617,7 +4652,7 @@ def type_check(symtab) def type(symtab) return ConstBoolType if name == "implemented?" - func_type(symtab).return_type(template_values(symtab, unknown_ok: symtab.archdef.partially_configured?), self) + func_type(symtab).return_type(template_values(symtab, unknown_ok: symtab.cfg_arch.partially_configured?), self) end # @!macro value @@ -4634,9 +4669,9 @@ def value(symtab) extname_ref = arg_nodes[0] type_error "First argument should be a ExtensionName" unless extname_ref.type(symtab).kind == :enum_ref && extname_ref.class_name == "ExtensionName" - return symtab.archdef.ext?(arg_nodes[0].member_name) if symtab.archdef.fully_configured? + return symtab.cfg_arch.ext?(arg_nodes[0].member_name) if symtab.cfg_arch.fully_configured? - if symtab.archdef.ext?(arg_nodes[0].member_name) + if symtab.cfg_arch.ext?(arg_nodes[0].member_name) # we can know if it is implemented, but not if it's not implemented for a partially configured return true end @@ -4697,12 +4732,12 @@ def type_check(symtab) type_error "#{text_value} is not a type" unless type.is_a?(Type) end - # @!macro type_no_archdef + # @!macro type_no_cfg_arch def type(symtab) - typ = @type_cache[symtab.archdef] + typ = @type_cache[symtab.cfg_arch] return typ unless typ.nil? - @type_cache[symtab.archdef] = symtab.get(text_value) + @type_cache[symtab.cfg_arch] = symtab.get(text_value) end # @!macro to_idl @@ -4929,7 +4964,7 @@ def arguments_list_str # return the return type, which may be a tuple of multiple types def return_type(symtab) - cached = @cached_return_type[symtab.archdef] + cached = @cached_return_type[symtab.cfg_arch] return cached unless cached.nil? unless symtab.levels == 2 @@ -4937,12 +4972,12 @@ def return_type(symtab) end if @return_type_nodes.empty? - @cached_return_type[symtab.archdef] = VoidType + @cached_return_type[symtab.cfg_arch] = VoidType return VoidType end unless templated? - # with no templates, the return type does not change for a given arch_def + # with no templates, the return type does not change for a given cfg_arch rtype = if @return_type_nodes.size == 1 rtype = @return_type_nodes[0].type(symtab) @@ -4960,7 +4995,7 @@ def return_type(symtab) raise "??????" if rtype.nil? - return @cached_return_type[symtab.archdef] = rtype + return @cached_return_type[symtab.cfg_arch] = rtype end if templated? @@ -5707,7 +5742,7 @@ def freeze_tree(symtab) @value = nil end @type = calc_type(symtab) - @archdef = symtab.archdef # remember archdef, used in gen_adoc pass + @cfg_arch = symtab.cfg_arch # remember cfg_arch, used in gen_adoc pass freeze end @@ -5721,17 +5756,17 @@ def type_check(symtab) type_error "No CSR named #{csr_name}" if csr_def(symtab).nil? end type_error "CSR[#{csr_name(symtab)}] has no field named #{@field_name}" if field_def(symtab).nil? - type_error "CSR[#{csr_name(symtab)}].#{@field_name} is not defined in RV32" if symtab.archdef.mxlen == 32 && !field_def(symtab).defined_in_base32? - type_error "CSR[#{csr_name(symtab)}].#{@field_name} is not defined in RV64" if symtab.archdef.mxlen == 64 && !field_def(symtab).defined_in_base64? + type_error "CSR[#{csr_name(symtab)}].#{@field_name} is not defined in RV32" if symtab.cfg_arch.mxlen == 32 && !field_def(symtab).defined_in_base32? + type_error "CSR[#{csr_name(symtab)}].#{@field_name} is not defined in RV64" if symtab.cfg_arch.mxlen == 64 && !field_def(symtab).defined_in_base64? end def csr_def(symtab) - archdef = symtab.archdef + cfg_arch = symtab.cfg_arch if @idx.is_a?(IntLiteralAst) - archdef.csrs.find { |c| c.address == @idx.value(symtab) } + cfg_arch.csrs.find { |c| c.address == @idx.value(symtab) } else - archdef.csr(@idx) + cfg_arch.csr(@idx) end end @@ -5771,14 +5806,14 @@ def calc_type(symtab) end end if fd.defined_in_all_bases? - Type.new(:bits, width: symtab.archdef.possible_xlens.map{ |xlen| fd.width(symtab.archdef, xlen) }.max) + Type.new(:bits, width: symtab.cfg_arch.possible_xlens.map{ |xlen| fd.width(symtab.cfg_arch, xlen) }.max) elsif fd.base64_only? - if symtab.archdef.possible_xlens.include?(64) - Type.new(:bits, width: fd.width(symtab.archdef, 64)) + if symtab.cfg_arch.possible_xlens.include?(64) + Type.new(:bits, width: fd.width(symtab.cfg_arch, 64)) end elsif fd.base32_only? - if symtab.archdef.possible_xlens.include?(32) - Type.new(:bits, width: fd.width(symtab.archdef, 32)) + if symtab.cfg_arch.possible_xlens.include?(32) + Type.new(:bits, width: fd.width(symtab.cfg_arch, 32)) end else internal_error "unexpected field base" @@ -5802,7 +5837,7 @@ def calc_value(symtab) value_error "'#{csr_name(symtab)}.#{field_name(symtab)}' is not RO" end - field_def(symtab).reset_value(symtab.archdef) + field_def(symtab).reset_value(symtab.cfg_arch) end end @@ -5842,14 +5877,14 @@ def initialize(input, interval, idx) def freeze_tree(symtab) return if frozen? - @archdef = symtab.archdef # remember archdef, used by gen_adoc pass + @cfg_arch = symtab.cfg_arch # remember cfg_arch, used by gen_adoc pass @idx.freeze_tree(symtab) freeze end # @!macro type def type(symtab) - archdef = symtab.archdef + cfg_arch = symtab.cfg_arch cd = csr_def(symtab) if cd.nil? @@ -5861,16 +5896,16 @@ def type(symtab) Bits64Type end else - CsrType.new(cd, archdef) + CsrType.new(cd, cfg_arch) end end # @!macro type_check def type_check(symtab) - archdef = symtab.archdef + cfg_arch = symtab.cfg_arch idx_text = @idx.is_a?(String) ? @idx : @idx.text_value - if !archdef.csr(idx_text).nil? + if !cfg_arch.csr(idx_text).nil? # this is a known csr name # nothing else to check @@ -5881,7 +5916,7 @@ def type_check(symtab) value_result = value_try do idx_value = @idx.value(symtab) - csr_index = archdef.csrs.index { |csr| csr.address == idx_value } + csr_index = cfg_arch.csrs.index { |csr| csr.address == idx_value } type_error "No csr number '#{idx_value}' was found" if csr_index.nil? :ok end @@ -5890,9 +5925,9 @@ def type_check(symtab) end def csr_def(symtab) - archdef = symtab.archdef + cfg_arch = symtab.cfg_arch idx_text = @idx.is_a?(String) ? @idx : @idx.text_value - csr = archdef.csr(idx_text) + csr = cfg_arch.csr(idx_text) if !csr.nil? # this is a known csr name csr @@ -5900,7 +5935,7 @@ def csr_def(symtab) # this is an expression value_result = value_try do idx_value = @idx.value(symtab) - return archdef.csrs.find { |csr| csr.address == idx_value } + return cfg_arch.csrs.find { |csr| csr.address == idx_value } end # || we don't know at compile time which CSR this is... nil @@ -5921,10 +5956,10 @@ def csr_name(symtab) def value(symtab) cd = csr_def(symtab) value_error "CSR number not knowable" if cd.nil? - if symtab.archdef.fully_configured? - value_error "CSR is not implemented" unless symtab.archdef.implemented_csrs.any? { |icsr| icsr.name == cd.name } + if symtab.cfg_arch.fully_configured? + value_error "CSR is not implemented" unless symtab.cfg_arch.transitive_implemented_csrs.any? { |icsr| icsr.name == cd.name } else - value_error "CSR is not defined" unless symtab.archdef.csrs.any? { |icsr| icsr.name == cd.name } + value_error "CSR is not defined" unless symtab.cfg_arch.csrs.any? { |icsr| icsr.name == cd.name } end cd.fields.each { |f| value_error "#{csr_name(symtab)}.#{f.name} not RO" unless f.type(symtab) == "RO" } @@ -5952,12 +5987,13 @@ def initialize(input, interval, csr, expression) end def type_check(symtab) - archdef = symtab.archdef + cfg_arch = symtab.cfg_arch csr.type_check(symtab) expression.type_check(symtab) - return if expression.type(symtab).kind == :bits && expression.type(symtab).width == symtab.mxlen + e_type = expression.type(symtab) + return if e_type.kind == :bits && ((e_type.width == :unknown || symtab.mxlen.nil?) || (e_type.width == symtab.mxlen)) type_error "CSR value must be an XReg" end @@ -6019,14 +6055,14 @@ def type_check(symtab) end def type(symtab) - archdef = symtab.archdef + cfg_arch = symtab.cfg_arch case function_name when "sw_read" if csr_known?(symtab) - Type.new(:bits, width: archdef.csr(csr.csr_name(symtab)).length(archdef)) + Type.new(:bits, width: cfg_arch.csr(csr.csr_name(symtab)).length(cfg_arch)) else - Type.new(:bits, width: symtab.mxlen) + Type.new(:bits, width: symtab.mxlen.nil? ? :unknown : symtab.mxlen) end when "address" Type.new(:bits, width: 12) @@ -6086,10 +6122,10 @@ def initialize(input, interval, idx) def type_check(symtab) if idx.is_a?(IntLiteralAst) # make sure this value is a defined CSR - index = symtab.archdef.csrs.index { |csr| csr.address == idx.value(symtab) } + index = symtab.cfg_arch.csrs.index { |csr| csr.address == idx.value(symtab) } type_error "No csr number '#{idx.value(symtab)}' was found" if index.nil? else - csr = symtab.archdef.csr(idx.text_value) + csr = symtab.cfg_arch.csr(idx.text_value) type_error "No csr named '#{idx.text_value}' was found" if csr.nil? end end @@ -6097,15 +6133,15 @@ def type_check(symtab) def csr_def(symtab) if idx.is_a?(IntLiteralAst) # make sure this value is a defined CSR - symtab.archdef.csrs.find { |csr| csr.address == idx.text_value.to_i } + symtab.cfg_arch.csrs.find { |csr| csr.address == idx.text_value.to_i } else - symtab.archdef.csr(idx.text_value) + symtab.cfg_arch.csr(idx.text_value) end end # @!macro type def type(symtab) - CsrType.new(csr_def(symtab), symtab.archdef) + CsrType.new(csr_def(symtab), symtab.cfg_arch) end def name(symtab) diff --git a/lib/idl/passes/gen_adoc.rb b/lib/idl/passes/gen_adoc.rb index 21ddb8be1..6213411fd 100644 --- a/lib/idl/passes/gen_adoc.rb +++ b/lib/idl/passes/gen_adoc.rb @@ -301,7 +301,7 @@ def gen_adoc(indent = 0, indent_spaces: 2) if idx_text =~ /[0-9]+/ "#{' '*indent}#{csr_text}" else - if @archdef.csr(csr_text).nil? + if @cfg_arch.csr(csr_text).nil? "#{' '*indent}#{csr_text}" else "#{' '*indent}%%LINK%csr_field;#{idx_text}.#{@field_name};#{csr_text}%%" @@ -324,7 +324,7 @@ def gen_adoc(indent = 0, indent_spaces: 2) # we don't have the symtab to map this to a csr name "#{' '*indent}#{csr_text}" else - if @archdef.csr(csr_text).nil? + if @cfg_arch.csr(csr_text).nil? "#{' '*indent}#{csr_text}" else "#{' '*indent}%%LINK%csr;#{idx_text};#{csr_text}%%" diff --git a/lib/idl/passes/prune.rb b/lib/idl/passes/prune.rb index d0065842f..165a0190b 100644 --- a/lib/idl/passes/prune.rb +++ b/lib/idl/passes/prune.rb @@ -220,16 +220,19 @@ def prune(symtab) BinaryExpressionAst.new(input, interval, lhs.prune(symtab), @op, rhs.prune(symtab)) end elsif op == "|" + rhs_type = rhs.type(symtab) + lhs_type = lhs.type(symtab) + if lhs_value == 0 # rhs idenntity rhs.prune(symtab) - elsif lhs_value == ((1 << rhs.type(symtab).width) - 1) + elsif rhs_type.width != :unknown && lhs_value == ((1 << rhs.type(symtab).width) - 1) # ~0 | anything == ~0 create_literal(lhs_value) elsif rhs_value == 0 # lhs identity lhs.prune(symtab) - elsif rhs_value == (1 << lhs.type(symtab).width - 1) + elsif lhs_type.width != :unknown && rhs_value == (1 << lhs.type(symtab).width - 1) # anything | ~0 == ~0 create_literal(rhs_value) else diff --git a/lib/idl/passes/reachable_functions_unevaluated.rb b/lib/idl/passes/reachable_functions_unevaluated.rb index 059d58b12..22364b168 100644 --- a/lib/idl/passes/reachable_functions_unevaluated.rb +++ b/lib/idl/passes/reachable_functions_unevaluated.rb @@ -4,43 +4,43 @@ module Idl class AstNode - # @param arch_def [ArchDef] Architecture definition + # @param cfg_arch [ConfiguredArchitecture] Architecture definition # @return [Array] List of all functions that can be reached (via function calls) from this node, without considering value evaluation - def reachable_functions_unevaluated(arch_def) + def reachable_functions_unevaluated(cfg_arch) children.reduce([]) do |list, e| - list.concat e.reachable_functions_unevaluated(arch_def) + list.concat e.reachable_functions_unevaluated(cfg_arch) end.uniq(&:name) end end class FunctionCallExpressionAst - def reachable_functions_unevaluated(arch_def) + def reachable_functions_unevaluated(cfg_arch) fns = [] if template? template_arg_nodes.each do |t| - fns.concat(t.reachable_functions_unevaluated(arch_def)) + fns.concat(t.reachable_functions_unevaluated(cfg_arch)) end end arg_nodes.each do |a| - fns.concat(a.reachable_functions_unevaluated(arch_def)) + fns.concat(a.reachable_functions_unevaluated(cfg_arch)) end - func_def_ast = arch_def.function(name) + func_def_ast = cfg_arch.function(name) raise "No function '#{name}' found in Architecture def" if func_def_ast.nil? - fns += func_def_ast.reachable_functions_unevaluated(arch_def) + fns += func_def_ast.reachable_functions_unevaluated(cfg_arch) fns.uniq(&:name) end end class FunctionDefAst - def reachable_functions_unevaluated(arch_def) + def reachable_functions_unevaluated(cfg_arch) fns = [self] unless builtin? body.stmts.each do |stmt| - fns += stmt.reachable_functions_unevaluated(arch_def) + fns += stmt.reachable_functions_unevaluated(cfg_arch) end end diff --git a/lib/idl/symbol_table.rb b/lib/idl/symbol_table.rb index f78acd9a2..32aee4199 100644 --- a/lib/idl/symbol_table.rb +++ b/lib/idl/symbol_table.rb @@ -14,7 +14,7 @@ def initialize(name, type, value = nil, decode_var: false, template_index: nil, @type = type @type.freeze @value = value - raise 'unexpected' unless decode_var.is_a?(TrueClass) || decode_var.is_a?(FalseClass) + raise "unexpected" unless decode_var.is_a?(TrueClass) || decode_var.is_a?(FalseClass) @decode_var = decode_var @template_index = template_index @@ -75,7 +75,7 @@ def value=(new_value) # scoped symbol table holding known symbols at a current point in parsing class SymbolTable - attr_reader :archdef + def cfg_arch = @cfg_arch # @return [Integer] 32 or 64, the XLEN in M-mode attr_reader :mxlen @@ -86,24 +86,20 @@ class DuplicateSymError < StandardError def hash return @frozen_hash unless @frozen_hash.nil? - [@scopes.hash, @archdef.hash].hash + [@scopes.hash, @cfg_arch.hash].hash end - def initialize(arch_def, effective_xlen = nil) - @archdef = arch_def - if arch_def.fully_configured? - raise "effective_xlen should not be set when symbol table is given a fully-configured ArchDef" unless effective_xlen.nil? - else - raise "effective_xlen should be set when symbol table is given an ArchDef" if effective_xlen.nil? && arch_def.mxlen.nil? - end - @mxlen = effective_xlen.nil? ? arch_def.mxlen : effective_xlen + def initialize(cfg_arch) + raise if cfg_arch.nil? + @cfg_arch = cfg_arch + @mxlen = cfg_arch.unconfigured? ? nil : cfg_arch.mxlen @callstack = [nil] @scopes = [{ "X" => Var.new( "X", - Type.new(:array, sub_type: XregType.new(@mxlen), width: 32, qualifiers: [:global]) + Type.new(:array, sub_type: XregType.new(@mxlen.nil? ? :unknown : @mxlen), width: 32, qualifiers: [:global]) ), - "XReg" => XregType.new(@mxlen), + "XReg" => XregType.new(@mxlen.nil? ? :unknown : @mxlen), "Boolean" => Type.new(:boolean), "true" => Var.new( "true", @@ -117,7 +113,7 @@ def initialize(arch_def, effective_xlen = nil) ) }] - arch_def.params_with_value.each do |param_with_value| + cfg_arch.params_with_value.each do |param_with_value| type = Type.from_json_schema(param_with_value.schema).make_const if type.kind == :array && type.width == :unknown type = Type.new(:array, width: param_with_value.value.length, sub_type: type.sub_type, qualifiers: [:const]) @@ -133,15 +129,11 @@ def initialize(arch_def, effective_xlen = nil) end end end + # now add all parameters, even those not implemented - arch_def.params_without_value.each do |param| + cfg_arch.params_without_value.each do |param| if param.exts.size == 1 - if param.name == "XLEN" - # special case: we actually do know XLEN - add!(param.name, Var.new(param.name, param.idl_type.clone.make_const, @mxlen)) - else - add!(param.name, Var.new(param.name, param.idl_type.clone.make_const)) - end + add!(param.name, Var.new(param.name, param.idl_type.clone.make_const)) else # could already be present... existing_sym = get(param.name) @@ -154,32 +146,6 @@ def initialize(arch_def, effective_xlen = nil) end end end - - # add the builtin extensions - # add!( - # "ExtensionName", - # EnumerationType.new( - # "ExtensionName", - # arch_def.extensions.map(&:name), - # Array.new(arch_def.extensions.size) { |i| i + 1 } - # ) - # ) - # add!( - # "ExceptionCode", - # EnumerationType.new( - # "ExceptionCode", - # arch_def.exception_codes.map(&:var), - # arch_def.exception_codes.map(&:num) - # ) - # ) - # add!( - # "InterruptCode", - # EnumerationType.new( - # "InterruptCode", - # arch_def.interrupt_codes.map(&:var), - # arch_def.interrupt_codes.map(&:num) - # ) - # ) end # do a deep freeze to protect the sym table and all its entries from modification @@ -191,7 +157,7 @@ def deep_freeze @scopes.freeze # set frozen_hash so that we can quickly compare symtabs - @frozen_hash = [@scopes.hash, @archdef.hash].hash + @frozen_hash = [@scopes.hash, @cfg_arch.hash].hash # set up the global clone that be used as a mutable table @global_clone_pool = [] @@ -200,7 +166,7 @@ def deep_freeze copy = SymbolTable.allocate copy.instance_variable_set(:@scopes, [@scopes[0]]) copy.instance_variable_set(:@callstack, [@callstack[0]]) - copy.instance_variable_set(:@archdef, @archdef) + copy.instance_variable_set(:@cfg_arch, @cfg_arch) copy.instance_variable_set(:@mxlen, @mxlen) copy.instance_variable_set(:@global_clone_pool, @global_clone_pool) copy.instance_variable_set(:@in_use, false) @@ -213,7 +179,7 @@ def deep_freeze # @return [String] inspection string def inspect - "SymbolTable[#{@archdef.name}]#{frozen? ? ' (frozen)' : ''}" + "SymbolTable[#{@cfg_arch.name}]#{frozen? ? ' (frozen)' : ''}" end # pushes a new scope @@ -376,7 +342,7 @@ def global_clone copy = SymbolTable.allocate copy.instance_variable_set(:@scopes, [@scopes[0]]) copy.instance_variable_set(:@callstack, [@callstack[0]]) - copy.instance_variable_set(:@archdef, @archdef) + copy.instance_variable_set(:@cfg_arch, @cfg_arch) copy.instance_variable_set(:@mxlen, @mxlen) copy.instance_variable_set(:@global_clone_pool, @global_clone_pool) copy.instance_variable_set(:@in_use, false) diff --git a/lib/idl/tests/helpers.rb b/lib/idl/tests/helpers.rb index 8ef30e6c0..3b3132d5d 100644 --- a/lib/idl/tests/helpers.rb +++ b/lib/idl/tests/helpers.rb @@ -13,8 +13,8 @@ def initialize(name) MockExtensionParameter = Struct.new(:name, :desc, :schema, :extra_validation, :exts, :type) MockExtensionParameterWithValue = Struct.new(:name, :desc, :schema, :extra_validation, :exts, :value) -# ArchDef mock that knows about XLEN and extensions -class MockArchDef +# ConfiguredArchitecture mock that knows about XLEN and extensions +class MockConfiguredArchitecture def param_values = { "XLEN" => 32 } def params_with_value = [MockExtensionParameterWithValue.new("XLEN", "mxlen", {"type" => "integer", "enum" => [32, 64]}, nil, nil, 32)] def params_without_value = [] @@ -35,8 +35,8 @@ def name = "mock" module TestMixin def setup - @archdef = MockArchDef.new - @symtab = Idl::SymbolTable.new(@archdef, 32) - @compiler = Idl::Compiler.new(@archdef) + @cfg_arch = MockConfiguredArchitecture.new + @symtab = Idl::SymbolTable.new(@cfg_arch) + @compiler = Idl::Compiler.new end end diff --git a/lib/idl/tests/test_expressions.rb b/lib/idl/tests/test_expressions.rb index 019e06ada..cb16a7d92 100644 --- a/lib/idl/tests/test_expressions.rb +++ b/lib/idl/tests/test_expressions.rb @@ -94,12 +94,10 @@ def test_that_integer_literals_give_correct_values assert_equal(-13, ast.value(@symtab)) idl = "-4'sb1101" - ast = @compiler.compile_expression(idl, @symtab) - assert_equal 3, ast.value(@symtab) + assert_raises(Idl::AstNode::TypeError) { @compiler.compile_expression(idl, @symtab, pass_error: true) } idl = "4'sb1101" - ast = @compiler.compile_expression(idl, @symtab) - assert_equal(-3, ast.value(@symtab)) + assert_raises(Idl::AstNode::TypeError) { @compiler.compile_expression(idl, @symtab, pass_error: true) } idl = "32'h80000000" ast = @compiler.compile_expression(idl, @symtab) @@ -124,7 +122,7 @@ def test_that_integer_literals_give_correct_values assert_equal 13, ast.value(@symtab) # compilation error: 300 does not fit in 8 bits - idl = "'h1_0000_0000" + idl = "8'h1_0000_0000" assert_raises(Idl::AstNode::TypeError) { @compiler.compile_expression(idl, @symtab, pass_error: true) } # 3 decimal: the literal is 13, unsigned, in 4-bits. when negated, the sign bit is lost @@ -144,5 +142,4 @@ def test_that_integer_literals_give_correct_values idl = "4'hff" assert_raises(Idl::AstNode::TypeError) { @compiler.compile_expression(idl, @symtab, pass_error: true) } end - end diff --git a/lib/idl/tests/test_functions.rb b/lib/idl/tests/test_functions.rb index 70917a73d..d079348ca 100644 --- a/lib/idl/tests/test_functions.rb +++ b/lib/idl/tests/test_functions.rb @@ -72,7 +72,7 @@ def test_that_reachable_raise_analysis_respects_transitive_known_values ast = @compiler.compile_file(path) ast.add_global_symbols(@symtab) @symtab.deep_freeze - @archdef.global_ast = ast + @cfg_arch.global_ast = ast ast.freeze_tree(@symtab) test_ast = ast.functions.select { |f| f.name == "test" }[0] @@ -137,7 +137,7 @@ def test_that_reachable_raise_analysis_respects_known_paths_down_an_unknown_path ast = @compiler.compile_file(path) ast.add_global_symbols(@symtab) @symtab.deep_freeze - @archdef.global_ast = ast + @cfg_arch.global_ast = ast ast.freeze_tree(@symtab) test_ast = ast.functions.select { |f| f.name == "test" }[0] diff --git a/lib/idl/type.rb b/lib/idl/type.rb index 03482b9a5..7ad04a4ee 100644 --- a/lib/idl/type.rb +++ b/lib/idl/type.rb @@ -61,10 +61,10 @@ def qualify(qualifier) self end - def self.from_typename(type_name, arch_def) + def self.from_typename(type_name, cfg_arch) case type_name when 'XReg' - return Type.new(:bits, width: arch_def.param_values['XLEN']) + return Type.new(:bits, width: cfg_arch.param_values['XLEN']) when 'FReg' return Type.new(:freg, width: 32) when 'DReg' @@ -609,8 +609,8 @@ def clone class CsrType < Type attr_reader :csr - def initialize(csr, arch_def, qualifiers: []) - super(:csr, name: csr.name, csr: csr, width: csr.max_length(arch_def), qualifiers: qualifiers) + def initialize(csr, cfg_arch, qualifiers: []) + super(:csr, name: csr.name, csr: csr, width: csr.max_length(cfg_arch), qualifiers: qualifiers) end def fields diff --git a/lib/resolver.rb b/lib/resolver.rb deleted file mode 100644 index 494d7c9f5..000000000 --- a/lib/resolver.rb +++ /dev/null @@ -1,20 +0,0 @@ -# given an architecture folder, resolves inheritance and expands some fields - -require "pathname" - -class Resolver - def initialize(arch_folder) - @dir = Pathname.new(arch_folder).realpath - end - - def resolve_all(output_folder) - Dir.glob(@dir / "**" / "*.yaml") do |f| - resolve(f, "#{output_folder}/#{f.gsub("#{@dir.to_s}/", "")}") - end - end - - def resolve(input_file, output_file) - obj = YamlLoader.load(input_file, permitted_classes: [Date]) - File.write(output_file, YAML::dump(obj)) - end -end diff --git a/lib/test/test_yaml_loader.rb b/lib/test/test_yaml_loader.rb index c7285fa44..593a95fd2 100644 --- a/lib/test/test_yaml_loader.rb +++ b/lib/test/test_yaml_loader.rb @@ -1,12 +1,61 @@ +# frozen_string_literal: true +require "English" +require "fileutils" require "minitest/autorun" +require "open3" +require "tmpdir" +require "yaml" -$root ||= (Pathname.new(__FILE__) / ".." / ".." ).realpath - -require_relative "../yaml_loader" +$root ||= (Pathname.new(__FILE__) / ".." / ".." / "..").realpath class TestYamlLoader < Minitest::Test + def resolve_yaml(yaml) + Dir.mktmpdir do |dir| + arch_dir = Pathname.new(dir) / "arch" + resolved_dir = Pathname.new(dir) / "resolved_arch" + test_dir = arch_dir / "test" + FileUtils.mkdir_p test_dir + + File.write(test_dir / "test.yaml", yaml) + + stdout, stderr, status = + Dir.chdir($root) do + Open3.capture3("/bin/bash -c \"source #{$root}/.home/.venv/bin/activate && #{$root}/.home/.venv/bin/python3 #{$root}/lib/yaml_resolver.py resolve --no-progress --no-checks #{arch_dir} #{resolved_dir}\"") + end + # puts stdout + # puts stderr + # puts status + + if status.to_i.zero? + YAML.load_file(resolved_dir / "test" / "test.yaml") + end + end + end + def resolve_multi_yaml(*yamls) + Dir.mktmpdir do |dir| + arch_dir = Pathname.new(dir) / "arch" + resolved_dir = Pathname.new(dir) / "resolved_arch" + test_dir = arch_dir / "test" + FileUtils.mkdir_p test_dir + + yamls.each_index do |i| + yaml = yamls[i] + yamls.size.times do |j| + yaml = yaml.gsub("YAML#{j + 1}_REL_PATH", "test/test#{j + 1}.yaml") + end + File.write(test_dir / "test#{i + 1}.yaml", yaml) + end + + system "/bin/bash -c \"source #{$root}/.home/.venv/bin/activate && #{$root}/.home/.venv/bin/python3 #{$root}/lib/yaml_resolver.py resolve --no-checks #{arch_dir} #{resolved_dir}\"" + # `source #{$root}/.home/.venv/bin/activate && python3 #{$root}/lib/yaml_resolver.py resolve #{arch_dir} #{resolved_dir}` + + if $CHILD_STATUS == 0 + YAML.load_file(resolved_dir / "test" / "test1.yaml") + end + end + end def test_remove yaml = <<~YAML @@ -20,12 +69,8 @@ def test_remove key3: value3 YAML - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - doc = YamlLoader.load(f.path) - assert_equal({ "key1" => "value1", "key3" => "value3" }, doc["child"]) + doc = resolve_yaml(yaml) + assert_equal({ "$child_of" => "#/base", "key1" => "value1", "key3" => "value3" }, doc["child"]) end def test_multiple_remove @@ -43,12 +88,8 @@ def test_multiple_remove key4: value4 YAML - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - doc = YamlLoader.load(f.path) - assert_equal({ "key1" => "value1", "key4" => "value4" }, doc["child"]) + doc = resolve_yaml(yaml) + assert_equal({ "$child_of" => "#/base", "key1" => "value1", "key4" => "value4" }, doc["child"]) end def test_that_inherits_with_nested_replace_works @@ -65,7 +106,6 @@ def test_that_inherits_with_nested_replace_works bottom: $inherits: - - "#/base" - "#/middle" key1: sub_key6: value6 @@ -74,15 +114,11 @@ def test_that_inherits_with_nested_replace_works key5: value5 YAML - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - doc = YamlLoader.load(f.path) - assert_equal({ "key1" => {"sub_key1" => "value1", "sub_key6" => "value6"}, "key2" => "value2_new", "key3" => "value3", "key4" => "value4_new", "key5" => "value5" }, doc["bottom"]) + doc = resolve_yaml(yaml) + assert_equal({ "$child_of" => ["#/middle"], "key1" => { "sub_key1" => "value1", "sub_key6" => "value6" }, "key2" => "value2_new", "key3" => "value3", "key4" => "value4_new", "key5" => "value5" }, doc["bottom"]) end - def test_that_spurious_recursive_inherits_works + def test_that_recursive_inherits_works yaml = <<~YAML base: key1: value1 @@ -94,118 +130,73 @@ def test_that_spurious_recursive_inherits_works key4: value4 bottom: - $inherits: - - "#/base" - - "#/middle" + $inherits: "#/middle" key2: value2_new key4: value4_new key5: value5 YAML - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - doc = YamlLoader.load(f.path) - assert_equal({ "key1" => "value1", "key2" => "value2", "key3" => "value3", "key4" => "value4" }, doc["middle"]) - assert_equal({ "key1" => "value1", "key2" => "value2_new", "key3" => "value3", "key4" => "value4_new", "key5" => "value5" }, doc["bottom"]) + doc = resolve_yaml(yaml) + assert_equal({ "$child_of" => "#/base", "$parent_of" => "test/test.yaml#/bottom", "key1" => "value1", "key2" => "value2", "key3" => "value3", "key4" => "value4" }, doc["middle"]) + assert_equal({ "$child_of" => "#/middle", "key1" => "value1", "key2" => "value2_new", "key3" => "value3", "key4" => "value4_new", "key5" => "value5" }, doc["bottom"]) end - def test_that_recursive_inherits_works + def test_that_nested_inherits_works yaml = <<~YAML - base: - key1: value1 - key2: value2 - - middle: - $inherits: "#/base" - key3: value3 - key4: value4 + top: + base: + key1: value1 + key2: value2 + key3: value3 bottom: - $inherits: "#/middle" - key2: value2_new - key4: value4_new - key5: value5 + child: + $inherits: "#/top/base" + key3: value3_new YAML - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - doc = YamlLoader.load(f.path) - assert_equal({ "key1" => "value1", "key2" => "value2", "key3" => "value3", "key4" => "value4" }, doc["middle"]) - assert_equal({ "key1" => "value1", "key2" => "value2_new", "key3" => "value3", "key4" => "value4_new", "key5" => "value5" }, doc["bottom"]) + doc = resolve_yaml(yaml) + assert_equal({ "$child_of" => "#/top/base", "key1" => "value1", "key2" => "value2", "key3" => "value3_new" }, doc["bottom"]["child"]) end - def test_that_nested_inherits_works + def test_that_inherits_doesnt_delete_keys yaml = <<~YAML - top: base: key1: value1 key2: value2 key3: value3 - bottom: child: - $inherits: "#/top/base" + $inherits: "#/base" key3: value3_new YAML - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - doc = YamlLoader.load(f.path) - assert_equal({ "key1" => "value1", "key2" => "value2", "key3" => "value3_new" }, doc["bottom"]["child"]) + doc = resolve_yaml(yaml) + assert_equal({ "$child_of" => "#/base", "key1" => "value1", "key2" => "value2", "key3" => "value3_new" }, doc["child"]) end - def test_that_inherits_doesnt_delete_keys + def test_that_double_inherits_doesnt_delete_keys yaml = <<~YAML - base: - key1: value1 - key2: value2 - key3: value3 - - child: - $inherits: "#/base" - key3: value3_new - YAML - - f = Tempfile.new("yml") - f.write(yaml) - f.flush + base1: + key1: value1 + key2: value2 + key3: value3 - doc = YamlLoader.load(f.path) - assert_equal({ "key1" => "value1", "key2" => "value2", "key3" => "value3_new" }, doc["child"]) - end + base2: + key4: value4 + key5: value5 + key6: value6 - def test_that_double_inherits_doesnt_delete_keys - yaml = <<~YAML - base1: - key1: value1 - key2: value2 - key3: value3 - - base2: - key4: value4 - key5: value5 - key6: value6 - - child: - $inherits: - - "#/base1" - - "#/base2" - key3: value3_new - key6: value6_new + child: + $inherits: + - "#/base1" + - "#/base2" + key3: value3_new + key6: value6_new YAML - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - doc = YamlLoader.load(f.path) - assert_equal({ "key1" => "value1", "key2" => "value2", "key3" => "value3_new", "key4" => "value4", "key5" => "value5", "key6" => "value6_new" }, doc["child"]) + doc = resolve_yaml(yaml) + assert_equal({ "$child_of" => ["#/base1", "#/base2"], "key1" => "value1", "key2" => "value2", "key3" => "value3_new", "key4" => "value4", "key5" => "value5", "key6" => "value6_new" }, doc["child"]) end def test_inherits_in_the_same_document @@ -227,50 +218,37 @@ def test_inherits_in_the_same_document $inherits: "#/$defs/target2" YAML - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - doc = YamlLoader.load(f.path) - assert_equal({ "a" => "hash" }, doc["obj1"]) - assert_equal({ "a" => "Should take precedence" }, doc["obj2"]) - assert_equal({ "a" => "Should take precedence" }, doc["obj3"]) + doc = resolve_yaml(yaml) + assert_equal({ "$child_of" => "#/$defs/target2", "a" => "hash" }, doc["obj1"]) + assert_equal({ "$child_of" => "#/$defs/target2", "a" => "Should take precedence" }, doc["obj2"]) + assert_equal({ "$child_of" => "#/$defs/target2", "a" => "Should take precedence" }, doc["obj3"]) end def test_inherits_in_the_different_document - yaml1 = <<~YAML + yaml2 = <<~YAML $defs: target1: A string target2: a: hash YAML - f1 = Tempfile.new("yml") - f1.write(yaml1) - f1.flush - f1_path = Pathname.new(f1.path) - - yaml2 = <<~YAML + yaml1 = <<~YAML obj1: - $inherits: "#{f1_path.basename}#/$defs/target2" + $inherits: "YAML2_REL_PATH#/$defs/target2" obj2: - $inherits: "#{f1_path.basename}#/$defs/target2" + $inherits: "YAML2_REL_PATH#/$defs/target2" a: Should take precedence obj3: a: Should take precedence - $inherits: "#{f1_path.basename}#/$defs/target2" + $inherits: "YAML2_REL_PATH#/$defs/target2" YAML - f2 = Tempfile.new("yml") - f2.write(yaml2) - f2.flush - - doc = YamlLoader.load(f2.path) - assert_equal({ "a" => "hash" }, doc["obj1"]) - assert_equal({ "a" => "Should take precedence" }, doc["obj2"]) - assert_equal({ "a" => "Should take precedence" }, doc["obj3"]) + doc = resolve_multi_yaml(yaml1, yaml2) + assert_equal({ "$child_of" => "test/test2.yaml#/$defs/target2", "a" => "hash" }, doc["obj1"]) + assert_equal({ "$child_of" => "test/test2.yaml#/$defs/target2", "a" => "Should take precedence" }, doc["obj2"]) + assert_equal({ "$child_of" => "test/test2.yaml#/$defs/target2", "a" => "Should take precedence" }, doc["obj3"]) end def test_multi_inherits_in_the_same_document @@ -288,12 +266,8 @@ def test_multi_inherits_in_the_same_document YAML - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - doc = YamlLoader.load(f.path) - assert_equal({ "a" => "hash", "b" => "nice" }, doc["obj1"]) + doc = resolve_yaml(yaml) + assert_equal({ "$child_of" => ["#/$defs/target1", "#/$defs/target2"], "a" => "hash", "b" => "nice" }, doc["obj1"]) end def test_that_invalid_inherits_raise @@ -309,123 +283,7 @@ def test_that_invalid_inherits_raise YAML - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - assert_raises(YamlLoader::DereferenceError) { YamlLoader.load(f.path) } + doc = resolve_yaml(yaml) + assert_nil doc end - - def test_that_invalid_refs_raise - yaml = <<~YAML - $defs: - target1: - b: nice - target2: - a: hash - - obj1: - $ref: "#/path/to/nowwhere" - - YAML - - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - assert_raises(YamlLoader::DereferenceError) { YamlLoader.load(f.path) } - end - - def test_copy_in_the_same_document - yaml = <<~YAML - $defs: - target1: A string - target2: - a: hash - target3: Another string - - obj1: - target10: abc - target11: - $copy: "#/$defs/target1" - target12: def - target13: - $copy: "#/$defs/target3" - - YAML - - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - doc = YamlLoader.load(f.path) - assert_equal({ - "target10" => "abc", - "target11" => "A string", - "target12" => "def", - "target13" => "Another string" - }, doc["obj1"]) - end - - def test_copy_in_the_different_document - yaml1 = <<~YAML - $defs: - target1: A string - target2: - a: hash - target3: Another string - YAML - - f1 = Tempfile.new("yml") - f1.write(yaml1) - f1.flush - f1_path = Pathname.new(f1.path) - - yaml2 = <<~YAML - obj1: - target10: abc - target11: - $copy: "#{f1_path.basename}#/$defs/target1" - target12: def - target13: - $copy: "#{f1_path.basename}#/$defs/target3" - YAML - - f2 = Tempfile.new("yml") - f2.write(yaml2) - f2.flush - - doc = YamlLoader.load(f2.path) - assert_equal({ - "target10" => "abc", - "target11" => "A string", - "target12" => "def", - "target13" => "Another string" - }, doc["obj1"]) - end - - def test_multi_inherits_in_the_same_document - yaml = <<~YAML - $defs: - target1: - b: nice - target2: - a: hash - - obj1: - $inherits: - - "#/$defs/target1" - - "#/$defs/target2" - - YAML - - f = Tempfile.new("yml") - f.write(yaml) - f.flush - - doc = YamlLoader.load(f.path) - assert_equal({ "a" => "hash", "b" => "nice" }, doc["obj1"]) - end - - end diff --git a/lib/validate.rb b/lib/validate.rb deleted file mode 100644 index 3375922d2..000000000 --- a/lib/validate.rb +++ /dev/null @@ -1,278 +0,0 @@ -# frozen_string_literal: true - -require_relative "yaml_loader" - -require "date" -require "json" -require "json_schemer" -require "pathname" -require "singleton" -require "yaml" - -$root = Pathname.new(__FILE__).dirname.dirname.realpath if $root.nil? - -# class used to validate schmeas and objects -class Validator - include Singleton - - # map of type to schema filesystem path - SCHEMA_PATHS = { - arch: $root / "schemas" / "arch_schema.json", - inst: $root / "schemas" / "inst_schema.json", - ext: $root / "schemas" / "ext_schema.json", - csr: $root / "schemas" / "csr_schema.json", - cfg_impl_ext: $root / "schemas" / "implemented_exts_schema.json", - manual_version: $root / "schemas" / "manual_version_schema.json", - cert_class: $root / "schemas" / "cert_class_schema.json" - }.freeze - - # types of objects that can be validated - TYPES = SCHEMA_PATHS.keys.freeze - - # Exception raised when there is a problem with a schema file - class SchemaError < ::StandardError - # result from JsonSchemer.validate - attr_reader :result - - def initialize(result) - if result.is_a?(Enumerator) - super(result.to_a.map { |e| "At #{e['schema_pointer']}: #{e['type']}" }) - else - super(result["error"]) - end - @result = result - end - end - - class ValidationError < ::StandardError - def initialize(why) - super(why) - end - end - - # exception raised when an object does not validate against its schema - class SchemaValidationError < ::StandardError - - # result from JsonSchemer.validate - attr_reader :result - - # create a new SchemaValidationError - # - # @param result [JsonSchemer::Result] JsonSchemer result - def initialize(path, result) - msg = "While validating #{path}:\n\n" - nerrors = result.count - msg << "#{nerrors} error(s) during validations\n\n" - result.to_a.each do |r| - msg << - if r["type"] == "required" && !r.dig("details", "missing_keys").nil? - " At '#{r['data_pointer']}': Missing required parameter(s) '#{r['details']['missing_keys']}'\n" - elsif r["type"] == "schema" - if r["schema_pointer"] == "/additionalProperties" - " At #{r['data_pointer']}, there is an unallowed additional key\n" - else - " At #{r['data_pointer']}, endpoint is an invalid key\n" - end - elsif r["type"] == "enum" - " At #{r['data_pointer']}, '#{r['data']}' is not a valid enum value (#{r['schema']['enum']})\n" - elsif r["type"] == "maxProperties" - " Maximum number of properties exceeded\n" - elsif r["type"] == "object" - " At #{r['data_pointer']}, Expecting object, got #{r['data']}\n" - elsif r["type"] == "pattern" - " At #{r['data_pointer']}, RegEx validation failed; '#{r['data']}' does not match '#{r['schema']['pattern']}'\n" - elsif r["type"] == "integer" - " At #{r['data_pointer']}, '#{r['data']}' is not a integer\n" - elsif r["type"] == "array" - " At #{r['data_pointer']}, '#{r['data']}' is not a array\n" - elsif r["type"] == "oneOf" - " At #{r['data_pointer']}, '#{r['data']}' matches more than one of #{r['schema']['oneOf']}\n" - elsif r["type"] == "const" - " At #{r['data_pointer']}, '#{r['data']}' does not match required value '#{r['schema']['const']}'\n" - else - " #{r}\n\n" - end - end - msg << "\n" - # msg << result.to_a.to_s - super(msg) - @result = result - end - end - - # initialize a new Validator - # - # @raise [SchemaError] if a schema is ill-formed - def initialize - @schemas = {} - SCHEMA_PATHS.each do |type, path| - # resolve refs as a relative path from the schema file - ref_resolver = proc do |pattern| - if pattern.to_s =~ /^http/ - JSON.parse(Net::HTTP.get(pattern)) - else - JSON.load_file($root / "schemas" / pattern.to_s) - end - end - - @schemas[type] = - JSONSchemer.schema( - path.read, - regexp_resolver: "ecma", - ref_resolver:, - insert_property_defaults: true - ) - raise SchemaError, @schemas[type].validate_schema unless @schemas[type].valid_schema? - end - end - - # validate a YAML string of a given type - # - # @return [Object] The object represented by str - # @param str [String] A YAML document - # @param type [Symbol] Type of the object (One of TYPES) - # @raise [SchemaValidationError] if the str is not valid against the type schema - # @see TYPES - def validate_str(str, type: nil, schema_path: nil, path: nil) - raise "Invalid type #{type}" unless TYPES.any?(type) || !schema_path.nil? - - begin - obj = YAML.safe_load(str, permitted_classes: [Symbol, Date]) - rescue Psych::SyntaxError => e - warn "While parsing: #{str}\n\n" - raise e - end - # convert through JSON to handle anything supported in YAML but not JSON - # (e.g., integer object keys will be coverted to strings) - jsonified_obj = JSON.parse(JSON.generate(obj)) - - raise "Nothing there?" if jsonified_obj.nil? - - schema = - if schema_path.nil? - @schemas[type] - else - # resolve refs as a relative path from the schema file - ref_resolver = proc do |pattern| - JSON.load_file(schema_path.dirname / pattern.to_s) - end - JSONSchemer.schema( - schema_path.read, - regexp_resolver: "ecma", - ref_resolver:, - insert_property_defaults: true - ) - end - - raise SchemaValidationError.new(path, schema.validate(jsonified_obj)) unless schema.valid?(jsonified_obj) - - jsonified_obj - end - - # validate a YAML file - # - # The type of the file is infered from its path unless type is provided - # - # @param path [#to_s] Path to a YAML document - # @param type [Symbol] Type of the object (One of TYPES). If nil, type will be inferred from path - # @raise [SchemaValidationError] if the str is not valid against the type schema - # @see TYPES - def validate(path, type: nil) - schema_path = nil - if type.nil? - case path.to_s - when %r{.*cfgs/([^/]+)/params\.yaml} - cfg_name = $1.to_s - type = :cfg_params - schema_path = $root / "gen" / cfg_name / "schemas" / "params_schema.json" - when %r{.*cfgs/[^/]+/implemented_exts\.yaml$} - type = :cfg_impl_ext - when %r{.*arch/arch_def\.yaml$} - type = :arch - when %r{.*arch/inst/.*/.*\.yaml$} - type = :inst - when %r{.*arch/ext/.*\.yaml$} - type = :ext - when %r{.*arch/csr/.*\.yaml$} - type = :csr - when %r{.*arch/manual/.*/.*contents\.yaml$} - type = :manual_version - when %r{.*arch/certificate_class/.*\.yaml$} - type = :cert_class - else - warn "Cannot determine type from YAML path '#{path}'; skipping" - return - end - end - begin - obj = validate_str(File.read(path.to_s), path:, type:, schema_path:) - - # check that the name matches the filename - if [:inst, :csr, :ext, :cert_class].include?(type) && obj["name"] != File.basename(path, ".yaml").to_s - raise ValidationError, "In #{path}, object name '#{obj.keys.first}' does not match filename '#{File.basename(path)}'" - end - obj - rescue Psych::SyntaxError => e - warn "While parsing #{path}" - raise e - end - end - - def ary_from_location(location_str_or_int) - return [location_str_or_int] if location_str_or_int.is_a?(Integer) - - bits = [] - parts = location_str_or_int.split("|") - parts.each do |part| - if part.include?("-") - msb, lsb = part.split("-").map(&:to_i) - (lsb..msb).each { |i| bits << i } - else - bits << part.to_i - end - end - bits - end - - def validate_instruction_encoding(inst_name, encoding) - match = encoding["match"] - raise "No match for instruction #{inst_name}?" if match.nil? - - variables = encoding["variables"] - match.size.times do |i| - if match[match.size - 1 - i] == "-" - # make sure exactly one variable covers this bit - vars_match = variables.count { |variable| ary_from_location(variable["location"]).include?(i) } - if vars_match.zero? - raise ValidationError, "In instruction #{inst_name}, no variable or encoding bit covers bit #{i}" - elsif vars_match != 1 - raise ValidationError, "In instruction, #{inst_name}, bit #{i} is covered by more than one variable" - end - else - # make sure no variable covers this bit - unless variables.nil? - unless variables.none? { |variable| ary_from_location(variable["location"]).include?(i) } - raise ValidationError, "In instruction, #{inst_name}, bit #{i} is covered by both a variable and the match string" - end - end - end - end - end - - # @param path [Pathname] Path to an instruction YAML document - # @raise [ValidateError] if there is a problem with the instruction defintion - def validate_instruction(path) - obj = YamlLoader.load(path) - raise "Invalid instruction definition: #{obj}" unless obj.is_a?(Hash) - - inst_name = path.basename('.yaml').to_s - raise "Invalid instruction definition: #{inst_name} #{obj}" unless obj["name"] == inst_name - - if (obj["encoding"]["RV32"].nil?) - validate_instruction_encoding(inst_name, obj["encoding"]) - else - validate_instruction_encoding(inst_name, obj["encoding"]["RV32"]) - validate_instruction_encoding(inst_name, obj["encoding"]["RV64"]) - end - end -end diff --git a/lib/version.rb b/lib/version.rb new file mode 100644 index 000000000..5ea01b048 --- /dev/null +++ b/lib/version.rb @@ -0,0 +1,194 @@ +# frozen_string_literal: true + +# Represents an RVI version specifier +# +# Version specs have the form: +# MAJOR[.MINOR[.PATCH[-pre]]] +# Where MAJOR, MINOR, and PATCH are integers and "pre" is an optional string +# +# Notably, these DO NOT represent a Semantic Version (https://semver.og). +# +# Rather, versions are treated as follows: +# +# * Versions are assumed to be backward compatible by default. +# For example, +# - 2.0 is compatible with 1.0 +# - 1.1 is compatible with 1.0 +# - 0.9 is *not* compatible with 1.0 +# * A version can be explictly marked as "breaking" in the architecture defintion +# Breaking versions are not backward compatible with any smaller versions +# For example, if version 2.2 is Breaking, +# - 3.0 is compatible with 2.2 +# - 2.3 is compatible with 2.2 +# - 3.0 is *not* compatible with 2.0 +# - 2.2 is *not* compatible with 2.0 +# - 2.1 is compatible with 2.0 +# +class VersionSpec + include Comparable + + # MAJOR[.MINOR[.PATCH[-pre]]] + VERSION_REGEX = /([0-9]+)(?:\.([0-9]+)(?:\.([0-9]+)(?:-(pre))?)?)?/ + + # @return [Integer] Major version number + attr_reader :major + + # @return [Integer] Minor version number + attr_reader :minor + + # @return [Integer] Patch version number + attr_reader :patch + + # @return [Boolean] Whether or not this is a pre-release + attr_reader :pre + + def initialize(version_str) + if version_str =~ /^\s*#{VERSION_REGEX}\s*$/ + m = ::Regexp.last_match + @major = m[1].to_i + @minor_given = !m[2].nil? + @minor = @minor_given ? m[2].to_i : 0 + @patch_given = !m[3].nil? + @patch = @patch_given ? m[3].to_i : 0 + @pre = !m[4].nil? + else + raise ArgumentError, "#{version_str} is not a valid Version spec" + end + @version_str = version_str + end + + def inspect + "VersionSpec[str: #{@version_str}; major: #{@major}, minor: #{@minor}, patch: #{@patch}, pre: #{@pre}]" + end + + # @return [String] The version, in canonical form + def canonical + "#{@major}.#{@minor}.#{@patch}#{@pre ? '-pre' : ''}" + end + + # @return [String] The version formatted like RVI docs + # + # @example + # VersionSpec.new("2.2").to_rvi_s #=> "2p2" + def to_rvi_s + s = @major.to_s + s += "p#{@minor}" if @minor_given + s += "p#{@patch}" if @patch_given + s += "-pre" if @pre + s + end + + # @return [String] The exact string used during construction + def to_s = @version_str + + def <=>(other) + if other.is_a?(String) + VersionSpec.new(other) <=> self + elsif other.is_a?(VersionSpec) + if @major != other.major + @major <=> other.major + elsif @minor != other.minor + @minor <=> other.minor + elsif @patch != other.patch + @patch <=> other.patch + elsif @pre != other.pre + @pre ? 1 : -1 + else + 0 + end + else + raise ArgumentError, "Cannot compare VersionSpec with #{other.class.name}" + end + end + + # @param other [VersionSpec] Comparison + # @return [Boolean] Whether or not +other+ is an VersionSpec with the same canonical version + def eql?(other) + if other.is_a?(String) + eql?(ExtensionVersion.new(other)) + elsif other.is_a?(VersionSpec) + other.major == @major && \ + other.minor == @minor && \ + other.patch == @patch && \ + other.pre == @pre + else + raise ArgumentError, "Cannot compare VersionSpec with #{other.class.name}" + end + end +end + +# Represents a version requirement +# +# A requirement is either a logical comparison (>, >=, <, <=, =, !=) +# or a compatible operator (~>). +# +# @example Logical requirement +# # When the requirement is a logical comparison, the extension parameter is not needed +# RequirementSpec.new(">= 0.5").satisfied_by?(VersionSpec.new("1.0"), nil) #=> true +# RequirementSpec.new(">= 0.5").satisfied_by?(VersionSpec.new("0.4"), nil) #=> false +# +# @example Compatible requirement +# s_ext = Extension.new(...) # S extension, which is breaking between 1.11 -> 1.12 +# RequirementSpec.new("~> 1.11").satisfied_by?(VersionSpec.new("1.10"), s_ext) #=> true +# RequirementSpec.new("~> 1.11").satisfied_by?(VersionSpec.new("1.11"), s_ext) #=> true +# RequirementSpec.new("~> 1.11").satisfied_by?(VersionSpec.new("1.12"), s_ext) #=> false +class RequirementSpec + REQUIREMENT_OP_REGEX = /((?:>=)|(?:>)|(?:~>)|(?:<)|(?:<=)|(?:!=)|(?:=))/ + REQUIREMENT_REGEX = /#{REQUIREMENT_OP_REGEX}\s*(#{VersionSpec::VERSION_REGEX})/ + + # @param requirement [String] A requirement string + def initialize(requirement) + unless requirement.is_a?(String) + raise ArgumentError, "requirement must be a string (is a #{requirement.class.name})" + end + + if requirement =~ /^\s*#{REQUIREMENT_REGEX}\s*$/ + m = ::Regexp.last_match + @op = m[1] + @version_str = m[2] + @version_spec = VersionSpec.new(@version_str) + else + raise ArgumentError, "Bad requirement string '#{requirement}'" + end + end + + def to_s + "#{@op} #{@version_str}" + end + + # @param version [String] A version string + # @param version [VersionSpec] A version spec + # @param ext [Extension] An extension, needed to evaluate the compatible (~>) operator + # @return [Boolean] if the version satisfies the requirement + def satisfied_by?(version, ext) + v_spec = + case version + when String + VersionSpec.new(version) + when VersionSpec + version + else + raise ArgumentError, "satisfied_by? expects a String or VersionSpec (got #{version.class.name})" + end + + case @op + when ">=" + v_spec >= @version_spec + when ">" + v_spec > @version_spec + when "<=" + v_spec <= @version_spec + when "<" + v_spec < @version_spec + when "=" + v_spec == @version_spec + when "!=" + v_spec != @version_spec + when "~>" + matching_ver = ext.versions.find { |v| v.version_spec == v_spec } + raise "Can't find version?" if matching_ver.nil? + + matching_ver.compatible?(ExtensionVersion.new(ext.name, v_spec.to_s, ext.cfg_arch)) + end + end +end diff --git a/lib/yaml_loader.rb b/lib/yaml_loader.rb deleted file mode 100644 index 691ab45f3..000000000 --- a/lib/yaml_loader.rb +++ /dev/null @@ -1,215 +0,0 @@ -# frozen_string_literal: true - -require "pathname" -require "yaml" - -# loads a YAML file and expands any $ref/$inherits references -class YamlLoader - @cache = {} - - class DereferenceError < StandardError; end - - def self.expand(filename, obj, yaml_opts = {}) - return obj unless obj.is_a?(Hash) || obj.is_a?(Array) - - return obj.map { |v| expand(filename, v, yaml_opts) } if obj.is_a?(Array) - - new_obj = - if obj.keys.include?("$ref") - # according JSON Reference, all keys except $ref are ignored - relative_path = obj["$ref"].split("#")[0] - if relative_path.empty? - # this is a reference in the same document - obj_doc = YAML.load_file(filename, **yaml_opts) - obj_path = obj["$ref"].split("#")[1].split("/")[1..] - target_obj = obj_doc.dig(*obj_path) - raise DereferenceError, "#{obj['$ref']} cannot be found" if target_obj.nil? - - ref = expand(filename, target_obj, yaml_opts) - if ref.nil? - raise DereferenceError, "JSON Path #{obj['$ref'].split('#')[1]} does not exist in #{filename}" - end - - { "$ref" => obj["$ref"] } # ignore any other keys that might exist - else - target_filename = - if File.exist?(File.join(filename.dirname, relative_path)) - File.realpath(File.join(filename.dirname, relative_path)) - elsif File.exist?(File.join($root, 'arch', relative_path)) - File.join($root, 'arch', relative_path) - else - raise DereferenceError, "#{relative_path} cannot be found" - end - - obj_doc = YamlLoader.load(target_filename, yaml_opts) - file_path, obj_path = obj["$ref"].split("#") - target_obj = - if obj_path.nil? - obj_doc - else - obj_doc.dig(*(obj_path.split("/")[1..])) - - end - raise "#{obj['$ref']} cannot be found" if target_obj.nil? - - ref = expand(target_filename, target_obj, yaml_opts) - if ref.nil? - raise DereferenceError, "JSON Path #{obj['$ref'].split('#')[1]} does not exist in #{target_filename}" - end - - { "$ref" => obj["$ref"] } # ignore any other keys that might exist - end - elsif obj.keys.include?("$inherits") - # we handle the inherits key first so that any override will take priority - inherits = obj["$inherits"] - raise ArgumentError, "Missing reference after $inherits (did you forget to put a relative reference in quotes?)" if inherits.nil? - inherits_targets = inherits.is_a?(String) ? [inherits] : inherits - - new_obj = {} - - inherits_targets.each do |inherits_target| - relative_path = inherits_target.split("#")[0] - target_obj = - if relative_path.empty? - YAML.load_file(filename, **yaml_opts) - else - target_filename = - if File.exist?(File.join(filename.dirname, relative_path)) - File.realpath(File.join(filename.dirname, relative_path)) - elsif File.exist?(File.join($root, 'arch', relative_path)) - File.join($root, 'arch', relative_path) - else - raise DereferenceError, "#{relative_path} cannot be found" - end - - unless File.exist?(target_filename) - raise DereferenceError, "While locating $inherits in #{filename}, #{target_filename} does not exist" - end - - YamlLoader.load(target_filename, yaml_opts) - end - - inherits_target_suffix = inherits_target.split("#/")[1] - inherits_target_path = inherits_target_suffix.split("/") - begin - target_obj = target_obj.dig(*inherits_target_path) - rescue TypeError => e - if e.message == "no implicit conversion of String into Integer" - warn "$inherits: \"#{inherits_target}\" found in file #{filename} references an Array but needs to reference a Hash" - end - raise e - end - - raise DereferenceError, "JSON Path #{inherits_target_suffix} in file #{filename} does not exist in #{relative_path}" if target_obj.nil? - raise ArgumentError, "$inherits: \"#{inherits_target}\" in file #{filename} references a #{target_obj.class} but needs to reference a Hash" unless target_obj.is_a?(Hash) - - target_obj = expand(filename, target_obj, yaml_opts) - target_obj.each do |target_key, target_value| - if (new_obj[target_key].is_a?(Hash)) - raise "Should be a hash" unless target_value.is_a?(Hash) - new_obj[target_key] = target_value.merge(new_obj[target_key]) - else - new_obj[target_key] = target_value - end - end - end - - obj.delete("$inherits") - # now merge target_obj and obj - keys = (obj.keys + new_obj.keys).uniq - final_obj = {} - keys.each do |key| - if !obj.key?(key) - final_obj[key] = new_obj[key] - elsif !new_obj.key?(key) - final_obj[key] = expand(filename, obj[key], yaml_opts) - else - value = obj[key] - - if new_obj[key].is_a?(Hash) - raise "should be a hash" unless new_obj[key].is_a?(Hash) - final_obj[key] = new_obj[key].merge(obj[key]) - else - final_obj[key] = expand(filename, obj[key], yaml_opts) - end - end - end - - final_obj - elsif obj.keys.include?("$copy") - self.get_copy_target_obj(filename, obj["$copy"], yaml_opts) - else - # Go through each hash entry. - obj.each do |key, value| - obj[key] = expand(filename, value, yaml_opts) - end - obj - end - - if new_obj.is_a?(Hash) - obj_keys = new_obj.keys - if obj_keys.include? "$remove" - remove_keys = obj["$remove"].is_a?(Array) ? obj["$remove"] : [obj["$remove"]] - remove_keys.each do |key| - new_obj.delete(key) - end - end - new_obj.delete("$remove") - end - - new_obj - end - - # @param filename [String,Pathname] path to the YAML file - # @param copy_target [String] - # @param yaml_opts [Hash] options to pass to YAML.load_file - # @return [Object] - def self.get_copy_target_obj(filename, copy_target, yaml_opts) - relative_path = copy_target.split("#")[0] - if relative_path.empty? - # this is a reference in the same document - obj_doc = YAML.load_file(filename, **yaml_opts) - obj_path = copy_target.split("#")[1].split("/")[1..] - target_obj = obj_doc.dig(*obj_path) - raise DereferenceError, "$copy: #{obj_path} referenced to same file cannot be found in file #{filename}" if target_obj.nil? - - ref = expand(filename, target_obj, yaml_opts) - if ref.nil? - raise DereferenceError, "$copy: JSON Path #{obj_path} referenced to same file does not exist in file #{filename}" - end - - ref - else - target_filename = File.realpath(File.join(filename.dirname, relative_path)) - - obj_doc = YamlLoader.load(target_filename, yaml_opts) - obj_path = copy_target.split("#")[1].split("/")[1..] - target_obj = obj_doc.dig(*obj_path) - raise DereferenceError, "$copy: #{obj_path} referenced from file #{filename} cannot be found in file #{target_filename}" if target_obj.nil? - - ref = expand(target_filename, target_obj, yaml_opts) - if ref.nil? - raise DereferenceError, "$copy: JSON Path #{obj_path} referenced from file #{filename} does not exist in file #{target_filename}" - end - - ref - end - end - - # load a YAML file and expand any $ref/$inherits references - # @param filename [String,Pathname] path to the YAML file - # @param yaml_opts [Hash] options to pass to YAML.load_file - # @return [Object] the loaded YAML file - def self.load(filename, yaml_opts = {}) - filename = Pathname.new(filename) - raise ArgumentError, "Cannot find file #{filename}" unless filename.exist? - - filename = filename.realpath - return @cache[filename] if @cache.key?(filename) - - obj = YAML.load_file(filename, **yaml_opts) - obj = expand(filename, obj, yaml_opts) if obj.is_a?(Hash) - - # @cache[filename] = obj - end -end diff --git a/lib/yaml_resolver.py b/lib/yaml_resolver.py index 3b9b1ab5c..f25fd9ad5 100644 --- a/lib/yaml_resolver.py +++ b/lib/yaml_resolver.py @@ -1,28 +1,180 @@ import glob, os +import argparse +import shutil +import json +import sys + +from pathlib import Path from copy import deepcopy from tqdm import tqdm from ruamel.yaml import YAML from mergedeep import merge, Strategy +from jsonschema import Draft7Validator, validators +from jsonschema.exceptions import best_match +from jsonschema.exceptions import ValidationError + +from referencing import Registry, Resource +from referencing.exceptions import NoSuchResource + +# cahce of Schema valiators +schemas = {} + +SCHEMAS_PATH = Path(os.path.join(os.path.dirname(os.path.dirname(__file__)), "schemas")) + +def retrieve_from_filesystem(uri: str): + path = SCHEMAS_PATH / Path(uri) + contents = json.loads(path.read_text()) + return Resource.from_contents(contents) + +registry = Registry(retrieve=retrieve_from_filesystem) + +# extend the validator to support default values +# https://python-jsonschema.readthedocs.io/en/stable/faq/#why-doesn-t-my-schema-s-default-property-set-the-default-on-my-instance +def extend_with_default(validator_class): + """Extends the jsonschema validator to support default values. + + Parameters + ---------- + validator_class : jsonschema.Draft7Validator + The validator class to extend. + + Returns + ------- + jsonschema.Draft7Validator + The extended validator class that will fill in default values + """ + + validate_properties = validator_class.VALIDATORS["properties"] + + + def set_defaults(validator, properties, instance, schema): + for property, subschema in properties.items(): + if not isinstance(subschema, dict): + continue + if "default" in subschema: + instance.setdefault(property, subschema["default"]) + + for error in validate_properties( + validator, properties, instance, schema, + ): + yield error + + return validators.extend( + validator_class, {"properties" : set_defaults}, + ) + + +DefaultValidatingValidator = extend_with_default(Draft7Validator) -OUT_DIR="arch_resolved" UDB_ROOT=os.path.dirname(os.path.dirname(os.path.realpath(__file__))) yaml = YAML(typ="rt") yaml.default_flow_style = False yaml.preserve_quotes = True +def _merge_patch(base: dict, patch: dict, path_so_far = []) -> None: + """merges patch into base according to JSON Merge Patch (RFC 7386) -def read_yaml(file_path): + Parameters + ---------- + base : dict + The base object, which will be altered by the patch + patch : dict + The patch object + path_so_far : list + The current dict key path within patch + """ + + patch_obj = patch if len(path_so_far) == 0 else dig(patch, *path_so_far) + for key, patch_value in patch_obj.items(): + if isinstance(patch_value, dict): + # continue to dig + _merge_patch(base, patch, (path_so_far + [key])) + else: + base_ptr = dig(base, *path_so_far) + base_value = dig(base_ptr, key) + if patch_value == None: + # remove from base, if it exists + if base_value != None: + base_ptr.pop(key) + else: + if base_ptr == None: + # add or overwrite value in base + base_ptr = base + for k in path_so_far: + if not k in base_ptr: + base_ptr[k] = {} + base_ptr = base_ptr[k] + base_ptr = dig(base, *path_so_far) + base_ptr[key] = patch_value + +def json_merge_patch(base_obj: dict, patch: dict) -> dict: + """merges patch into base according to JSON Merge Patch (RFC 7386) + + Parameters + ---------- + base : dict + The base object, which will be altered by the patch + patch : dict + The patch object + + Returns + ------- + dict + base_obj, now with the patch applied + """ + _merge_patch(base_obj, patch, []) + return base_obj + +def read_yaml(file_path : str | Path): + """Read a YAML file from file_path and return the parsed content + + Parameters + ---------- + file_path : str, Path + Filesystem path to the YAML file + + Returns + ------- + dict, list + The object represented in the YAML file + """ with open(file_path, 'r') as file: data = yaml.load(file) return data -def write_yaml(file_path, data): +def write_yaml(file_path : str | Path, data): + """Write data as YAML to file_path + + Parameters + ---------- + file_path : str, Path + Filesystem path to the YAML file + data : dict, list + The object to write as YAML + """ with open(file_path, 'w') as file: yaml.dump(data, file) -def dig(obj, *keys): +def dig(obj : dict, *keys): + """Digs data out of dictionary obj + + Parameters + ---------- + obj : dict + A dictionary + *keys + A list of obj keys + + Returns + ------- + Any + The value of obj[keys[0]][keys[1]]...[keys[-1]] + """ + if obj == None: + return None + if len(keys) == 0: return obj @@ -31,25 +183,48 @@ def dig(obj, *keys): if len(keys) == 1: return next_obj else: + if not isinstance(next_obj, dict): + raise ValueError(f"Not a hash: {keys}") return dig(next_obj, *keys[1:]) except KeyError: return None resolved_objs = {} -def resolve(path, rel_path, arch_root): - if path in resolved_objs: - return resolved_objs[path] +def resolve(rel_path : str | Path, arch_root : str | Path, do_checks: bool) -> dict: + """Resolve the file at arch_root/rel_path by expanding operators and applying defaults + + Parameters + ---------- + rel_path : str, Path + The relative path, from arch_root, to the file to resolve + arch_root : str, Path + The root of the architecture + + Returns + ------- + dict + The resolved object + """ + if str(rel_path) in resolved_objs: + return resolved_objs[str(rel_path)] else: - unresolved_data = read_yaml(path) - resolved_objs[path] = _resolve(unresolved_data, [], rel_path, unresolved_data, arch_root) - return resolved_objs[path] + unresolved_arch_data = read_yaml(os.path.join(arch_root, rel_path)) + if do_checks and (not "name" in unresolved_arch_data): + print(f"ERROR: Missing 'name' key in {arch_root}/{rel_path}", file=sys.stderr) + exit(1) + fn_name = Path(rel_path).stem + if do_checks and (fn_name != unresolved_arch_data["name"]): + print(f"ERROR: 'name' key ({unresolved_arch_data["name"]}) must match filename ({fn_name} in {arch_root}/{rel_path}", file=sys.stderr) + exit(1) + resolved_objs[str(rel_path)] = _resolve(unresolved_arch_data, [], rel_path, unresolved_arch_data, arch_root, do_checks) + return resolved_objs[str(rel_path)] -def _resolve(obj, obj_path, obj_file_path, doc_obj, arch_root): +def _resolve(obj, obj_path, obj_file_path, doc_obj, arch_root, do_checks): if not (isinstance(obj, list) or isinstance(obj, dict)): return obj if isinstance(obj, list): - obj = list(map(lambda o: _resolve(o, obj_path, obj_file_path, doc_obj, arch_root), obj)) + obj = list(map(lambda o: _resolve(o, obj_path, obj_file_path, doc_obj, arch_root, do_checks), obj)) return obj if "$inherits" in obj: @@ -57,7 +232,7 @@ def _resolve(obj, obj_path, obj_file_path, doc_obj, arch_root): inherits_targets = [obj["$inherits"]] if isinstance(obj["$inherits"], str) else obj["$inherits"] obj["$child_of"] = obj["$inherits"] - new_obj = yaml.load("{}") + parent_obj = yaml.load("{}") for inherits_target in inherits_targets: ref_file_path = inherits_target.split("#")[0] @@ -70,62 +245,235 @@ def _resolve(obj, obj_path, obj_file_path, doc_obj, arch_root): ref_obj = dig(doc_obj, *ref_obj_path) if ref_obj == None: raise ValueError(f"{ref_obj_path} cannot be found in #{doc_obj}") - ref_obj = _resolve(ref_obj, ref_obj_path, ref_file_path, doc_obj, arch_root) + ref_obj = _resolve(ref_obj, ref_obj_path, ref_file_path, doc_obj, arch_root, do_checks) else: # this is a reference to another doc - if not os.path.exists(os.path.join(UDB_ROOT, arch_root, ref_file_path)): + if not os.path.exists(os.path.join(arch_root, ref_file_path)): raise ValueError(f"{ref_file_path} does not exist in {arch_root}/") - ref_file_full_path = os.path.join(UDB_ROOT, arch_root, ref_file_path) - ref_doc_obj = resolve(ref_file_full_path, ref_file_path, arch_root) + ref_doc_obj = resolve(ref_file_path, arch_root, do_checks) ref_obj = dig(ref_doc_obj, *ref_obj_path) - ref_obj = _resolve(ref_obj, ref_obj_path, ref_file_path, ref_doc_obj, arch_root) + ref_obj = _resolve(ref_obj, ref_obj_path, ref_file_path, ref_doc_obj, arch_root, do_checks) for key in ref_obj: - if isinstance(new_obj.get(key), dict): - merge(new_obj[key], ref_obj, strategy=Strategy.REPLACE) + if key == "$parent_of" or key == "$child_of": + continue # we don't propagate $parent_of / $child_of + if isinstance(parent_obj.get(key), dict): + merge(parent_obj[key], ref_obj[key], strategy=Strategy.REPLACE) else: - new_obj[key] = deepcopy(ref_obj[key]) + parent_obj[key] = deepcopy(ref_obj[key]) - print(f"{obj_file_path} {obj_path} inherits {ref_file_path} {ref_obj_path}") - ref_obj["$parent_of"] = f"{obj_file_path}#/{"/".join(obj_path)}" + if "$parent_of" in ref_obj: + if isinstance(ref_obj["$parent_of"], list): + ref_obj["$parent_of"].append(f"{obj_file_path}#/{"/".join(obj_path)}") + else: + ref_obj["$parent_of"] = [ref_obj["$parent_of"], f"{obj_file_path}#/{"/".join(obj_path)}"] + else: + ref_obj["$parent_of"] = f"{obj_file_path}#/{"/".join(obj_path)}" del obj["$inherits"] - # now new_obj is the child and obj is the parent + # now parent_obj is the child and obj is the parent # merge them keys = [] for key in obj.keys(): keys.append(key) - for key in new_obj.keys(): + for key in parent_obj.keys(): if keys.count(key) == 0: keys.append(key) final_obj = yaml.load('{}') for key in keys: if not key in obj: - final_obj[key] = new_obj[key] - elif not key in new_obj: - final_obj[key] = _resolve(obj[key], obj_path + [key], obj_file_path, doc_obj, arch_root) + final_obj[key] = parent_obj[key] + elif not key in parent_obj: + final_obj[key] = _resolve(obj[key], obj_path + [key], obj_file_path, doc_obj, arch_root, do_checks) else: - if isinstance(new_obj[key], dict): - if not isinstance(new_obj[key], dict): - raise ValueError("should be a hash") - final_obj[key] = merge(yaml.load('{}'), new_obj[key], obj[key], strategy=Strategy.REPLACE) + if isinstance(parent_obj[key], dict): + final_obj[key] = merge(yaml.load('{}'), parent_obj[key], obj[key], strategy=Strategy.REPLACE) else: - final_obj[key] = _resolve(obj[key], obj_path + [key], obj_file_path, doc_obj, arch_root) + final_obj[key] = _resolve(obj[key], obj_path + [key], obj_file_path, doc_obj, arch_root, do_checks) + if "$remove" in final_obj: + if isinstance(final_obj["$remove"], list): + for key in final_obj["$remove"]: + if key in final_obj: + del final_obj[key] + else: + if final_obj["$remove"] in final_obj: + del final_obj[final_obj["$remove"]] + del final_obj["$remove"] return final_obj else: for key in obj: - obj[key] = _resolve(obj[key], obj_path + [key], obj_file_path, doc_obj, arch_root) + obj[key] = _resolve(obj[key], obj_path + [key], obj_file_path, doc_obj, arch_root, do_checks) + + if "$remove" in obj: + if isinstance(obj["$remove"], list): + for key in obj["$remove"]: + if key in obj: + del obj[key] + else: + if obj["$remove"] in obj: + del obj[obj["$remove"]] + del obj["$remove"] return obj -arch_paths = glob.glob("arch/**/*.yaml", recursive=True, root_dir=UDB_ROOT) -for arch_path in tqdm(arch_paths): - resolved_arch_path = f"{UDB_ROOT}/{OUT_DIR}/{arch_path}" - os.makedirs(os.path.dirname(resolved_arch_path), exist_ok=True) - write_yaml(resolved_arch_path, resolve(arch_path, os.path.join(*arch_path.split("/")[1:]), "arch")) +def merge_file(rel_path : str | Path, arch_dir : str | Path, overlay_dir : str | Path | None, merge_dir : str | Path) -> None: + """ pick the right file(s) to merge, and write the result to merge_dir + + Parameters + ---------- + rel_path : str, Path + Relative path, from arch_dir, to base file + arch_dir : str, Path + Absolute path to arch dir with base files + overlay_dir : str, Path, None + Absolute path to overlay dir with overlay files + merge_dir : str, Path + Absolute path to merge dir, where the merged file will be written + """ + arch_path = overlay_path = None + + if arch_dir != None: + arch_path = os.path.join(arch_dir, rel_path) + if overlay_dir != None: + overlay_path = os.path.join(overlay_dir, rel_path) + merge_path = os.path.join(merge_dir, rel_path) + if not os.path.exists(arch_path) and (overlay_path == None or not os.path.exists(overlay_path)): + # neither exist + if not os.path.exists(merge_path): + raise "Script error: no path exists" + + # remove the merged file + os.remove(merge_path) + elif overlay_path == None or not os.path.exists(overlay_path): + if arch_path == None: + raise "Must supply with arch_path or overlay_path" + + # no overlay, just copy arch + if not os.path.exists(merge_path) or (os.path.getmtime(arch_path) > os.path.getmtime(merge_path)): + shutil.copyfile(os.path.join(arch_dir, rel_path), merge_path) + elif not os.path.exists(arch_path): + if overlay_path == None or not os.path.exists(overlay_path): + raise "Must supply with arch_path or overlay_path" + + # no arch, just copy overlay + if not os.path.exists(merge_path) or (os.path.getmtime(overlay_path) > os.path.getmtime(merge_path)): + shutil.copyfile(os.path.join(overlay_dir, rel_path), merge_path) + else: + # both exist, merge + if not os.path.exists(merge_path) or (os.path.getmtime(overlay_path) > os.path.getmtime(merge_path)) or (os.path.getmtime(arch_path) > os.path.getmtime(merge_path)): + arch_obj = read_yaml(os.path.join(arch_dir, rel_path)) + overlay_obj = read_yaml(os.path.join(overlay_dir, rel_path)) + + write_yaml(os.path.join(merge_dir, rel_path), json_merge_patch(arch_obj, overlay_obj)) + +class SchemaNotFoundException(Exception): + pass + +def _get_schema(uri): + rel_path= uri.split("#")[0] + + if rel_path in schemas: + return schemas[rel_path] + + abs_path = os.path.join(SCHEMAS_PATH, rel_path) + if not os.path.exists(abs_path): + raise SchemaNotFoundException(f"Schema not found: {uri}") + + # Open the JSON file + with open(abs_path, 'r') as f: + # Load the JSON data into a Python dictionary + schema_obj = json.load(f) + + schemas[rel_path] = DefaultValidatingValidator(schema_obj, registry=registry) + return schemas[rel_path] + + +def resolve_file(rel_path : str | Path, arch_dir: str | Path, resolved_dir: str | Path, do_checks: bool): + """Read object at arch_dir/rel_path, resolve it, and write it as YAML to resolved_dir/rel_path + + Parameters + ---------- + rel_path : str | Path + Path to file relative to arch_dir + arch_dir : str | Path + Absolute path to arch directory + resolved_dir : str | Path + Directory to write the resolved file to + """ + arch_path = os.path.join(arch_dir, rel_path) + resolved_path = os.path.join(resolved_dir, rel_path) + if not os.path.exists(arch_path): + if os.path.exists(resolved_path): + os.remove(resolved_path) + elif not os.path.exists(resolved_path) or (os.path.getmtime(arch_path) > os.path.getmtime(resolved_path)) or (os.path.getmtime(__file__) > os.path.getmtime(resolved_path)): + if os.path.exists(resolved_path): + os.remove(resolved_path) + resolved_obj = resolve(rel_path, args.arch_dir, do_checks) + resolved_obj["$source"] = os.path.join(args.arch_dir, rel_path) + + if do_checks and ("$schema" in resolved_obj): + schema = _get_schema(resolved_obj["$schema"]) + try: + schema.validate(instance=resolved_obj) + except ValidationError as e: + print(f"JSON Schema Validation Error for {rel_path}:") + print(best_match(schema.iter_errors(resolved_obj)).message) + exit(1) + + write_yaml(resolved_path, resolved_obj) + os.chmod(resolved_path, 0o444) + +if __name__ == '__main__': + cmdparser = argparse.ArgumentParser( + prog="yaml_resolver.py", + description="Resolves/overlays UDB architecture YAML files") + subparsers = cmdparser.add_subparsers(dest='command', help='sub-command help') + merge_parser = subparsers.add_parser('merge', help='Merge overlay on top of architecture files') + merge_parser.add_argument("arch_dir", type=str, help="Unresolved architecture (input) directory") + merge_parser.add_argument("overlay_dir", type=str, help="Overlay directory") + merge_parser.add_argument("merged_dir", type=str, help="Merged architecture (output) directory") + + all_parser = subparsers.add_parser('resolve', help='Resolve all architecture files') + all_parser.add_argument("arch_dir", type=str, help="Unresolved architecture (input) directory") + all_parser.add_argument("resolved_dir", type=str, help="Resolved architecture (output) directory") + all_parser.add_argument("--no-progress", action="store_true", help="Don't display progress bar") + all_parser.add_argument("--no-checks", action="store_true", help="Don't verify schema") + + args = cmdparser.parse_args() + + if args.command == 'merge': + arch_paths = glob.glob(f"**/*.yaml", recursive=True, root_dir=args.arch_dir) + if args.overlay_dir != None: + overlay_paths = glob.glob(f"**/*.yaml", recursive=True, root_dir=args.overlay_dir) + arch_paths.extend(overlay_paths) + arch_paths = list(set(arch_paths)) + merged_paths = glob.glob(f"**/*.yaml", recursive=True, root_dir=args.merged_dir) + arch_paths.extend(merged_paths) + arch_paths = list(set(arch_paths)) + + for arch_path in tqdm(arch_paths, ascii=True, desc="Merging arch"): + merged_arch_path = f"{UDB_ROOT}/{args.merged_dir}/{arch_path}" + os.makedirs(os.path.dirname(merged_arch_path), exist_ok=True) + merge_file(arch_path, args.arch_dir, args.overlay_dir, args.merged_dir) + + print(f"[INFO] Merged architecture files written to {args.merged_dir}") + + elif args.command == 'resolve': + arch_paths = glob.glob(f"**/*.yaml", recursive=True, root_dir=args.arch_dir) + if os.path.exists(args.resolved_dir): + resolved_paths = glob.glob(f"**/*.yaml", recursive=True, root_dir=args.resolved_dir) + arch_paths.extend(resolved_paths) + arch_paths = list(set(arch_paths)) + iter = arch_paths if args.no_progress else tqdm(arch_paths, ascii=True, desc="Resolving arch") + for arch_path in iter: + resolved_arch_path = f"{UDB_ROOT}/{args.resolved_dir}/{arch_path}" if not os.path.isabs(args.resolved_dir) else f"{args.resolved_dir}/{arch_path}" + os.makedirs(os.path.dirname(resolved_arch_path), exist_ok=True) + resolve_file(arch_path, args.arch_dir, args.resolved_dir, not args.no_checks) + + print(f"[INFO] Resolved architecture files written to {args.resolved_dir}") diff --git a/package-lock.json b/package-lock.json index 4f01bd6b8..a51844f2e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -11,6 +11,7 @@ "@antora/site-generator": "3.1.7", "@asciidoctor/tabs": "v1.0.0-beta.6", "asciidoctor-kroki": "0.17.0", + "prettier": "3.4.2", "wavedrom-cli": "^3.1.1" } }, @@ -2889,6 +2890,20 @@ "node": ">=4.0.0" } }, + "node_modules/prettier": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.4.2.tgz", + "integrity": "sha512-e9MewbtFo+Fevyuxn/4rrcDAaq0IYxPGLvObpQjiZBMAzB9IGmzlnG9RZy3FFas+eBMu2vA0CszMeduow5dIuQ==", + "bin": { + "prettier": "bin/prettier.cjs" + }, + "engines": { + "node": ">=14" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" + } + }, "node_modules/process": { "version": "0.11.10", "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", diff --git a/package.json b/package.json index 1181d44f5..a3e34026d 100644 --- a/package.json +++ b/package.json @@ -6,6 +6,7 @@ "@antora/site-generator": "3.1.7", "asciidoctor-kroki": "0.17.0", "@asciidoctor/tabs": "v1.0.0-beta.6", + "prettier": "3.4.2", "wavedrom-cli": "^3.1.1" } } diff --git a/requirements.txt b/requirements.txt index 2d2633076..37293bfb9 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,2 +1,5 @@ pre_commit==4.0.1 -PyYAML==6.0.2 +jsonschema==4.23.0 +tqdm==4.67.1 +ruamel.yaml==0.18.6 +mergedeep==1.3.4 diff --git a/schemas/arch_schema.json b/schemas/arch_schema.json deleted file mode 100644 index 52871d719..000000000 --- a/schemas/arch_schema.json +++ /dev/null @@ -1,129 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - - "type": "object", - "title": "Unified Architecture Specification", - "required": ["type", "extensions", "csrs", "instructions"], - "allOf": [ - { - "if": { - "properties": { - "type": { "enum": ["fully configured"] } - } - }, - "then": { - "required": [ - "implemented_instructions", - "implemented_extensions", - "implemented_csrs", - "params" - ] - } - } - ], - "properties": { - "type": { - "type": "string", - "description": "Type of the arch", - "enum": ["unconfigured", "partially configured", "fully configured"] - }, - "params": { - "type": "object" - }, - "implemented_extensions": { - "description": "Extensions implemented by this architecture", - "type": "array", - "items": { - "type": "object", - "required": ["name", "version"], - "properties": { - "name": { - "type": "string", - "pattern": "^([A-WY]|([SXZ][a-z0-9]+))$", - "description": "Extension name" - }, - "version": { - "oneOf": [ - { - "type": "string", - "description": "Extension version" - }, - { - "type": "number" - } - ] - } - }, - "additionalProperties": false - } - }, - "mandatory_extensions": { - "description": "Extensions mandatory in this architecture", - "type": "array", - "items": { - "$ref": "schema_defs.json#/$defs/extension_requirement" - } - }, - "extensions": { - "type": "array", - "items": { - "$ref": "ext_schema.json#/$defs/ext_data" - } - }, - "csrs": { - "type": "array", - "items": { - "$ref": "csr_schema.json#/$defs/csr_register" - } - }, - "implemented_csrs": { - "description": "CSRs implemented by this architecture", - "type": "array", - "items": { - "type": "string", - "pattern": "^[a-z][a-zA-Z0-9_]+$", - "description": "CSR name" - } - }, - "instructions": { - "type": "array", - "items": { - "$ref": "inst_schema.json#" - } - }, - "implemented_instructions": { - "description": "Instructions implemented by this architecture", - "type": "array", - "items": { - "type": "string", - "pattern": "^[a-z][a-zA-Z0-9.]+$", - "description": "Instruction name" - } - }, - "profile_classes": { - "type": "object" - }, - "profile_releases": { - "type": "object", - "items": { - "type": "string" - } - }, - "manuals": { - "type": "object" - }, - "certificate_classes": { - "type": "array", - "items": { - "$ref": "cert_class_schema.json#" - } - }, - "certificate_models": { - "type": "array", - "items": { - "$ref": "cert_model_schema.json#" - } - } - }, - "additionalProperties": false -} diff --git a/schemas/cert_class_schema.json b/schemas/cert_class_schema.json index c7cd83438..8fc5a4c9d 100644 --- a/schemas/cert_class_schema.json +++ b/schemas/cert_class_schema.json @@ -40,8 +40,8 @@ "minItems": 1, "description": "List of mandatory privilege modes for the class" }, - "__source": { - "$ref": "schema_defs.json#/$defs/__source" + "$source": { + "$ref": "schema_defs.json#/$defs/$source" } } } diff --git a/schemas/cert_model_schema.json b/schemas/cert_model_schema.json index 65d514106..1da6fac39 100644 --- a/schemas/cert_model_schema.json +++ b/schemas/cert_model_schema.json @@ -100,18 +100,26 @@ "oneOf": [ { "type": "string", - "pattern": "^profile_release/.*\\.yaml#.*" + "pattern": "^profile/.*\\.yaml#.*" }, { "type": "array", "items": { "type": "string", - "pattern": "^profile_release/.*\\.yaml#.*" + "pattern": "^profile/.*\\.yaml#.*" }, "uniqueItems": true } ] }, + "^\\$child_of": { + "type": "array", + "items": { + "type": "string", + "pattern": "^profile/.*\\.yaml#.*" + }, + "uniqueItems": true + }, "^([A-WY])|([SXZ][a-z0-9]+)$": { "type": "object", "properties": { @@ -192,8 +200,8 @@ } } }, - "__source": { - "$ref": "schema_defs.json#/$defs/__source" + "$source": { + "$ref": "schema_defs.json#/$defs/$source" } } } diff --git a/schemas/config_schema.json b/schemas/config_schema.json index 096cd0551..f6f4f415d 100644 --- a/schemas/config_schema.json +++ b/schemas/config_schema.json @@ -1,538 +1,165 @@ { "$schema": "http://json-schema.org/draft-07/schema#", - "$defs": { - "params": { - "type": "object", - "required": [ - "NAME", - "XLEN", - "M_MODE_ENDIANESS", - "NUM_PMP_ENTRIES", - "ARCH_ID", - "VENDOR_ID_BANK", - "VENDOR_ID_OFFSET", - "IMP_ID", - "MISALIGNED_LDST", - "NUM_HPM_COUNTERS", - "REPORT_VA_IN_MTVAL_ON_BREAKPOINT", - "REPORT_VA_IN_MTVAL_ON_LOAD_MISALIGNED", - "REPORT_VA_IN_MTVAL_ON_STORE_AMO_MISALIGNED", - "REPORT_VA_IN_MTVAL_ON_INSTRUCTION_MISALIGNED", - "REPORT_VA_IN_MTVAL_ON_LOAD_ACCESS_FAULT", - "REPORT_VA_IN_MTVAL_ON_STORE_AMO_ACCESS_FAULT", - "REPORT_VA_IN_MTVAL_ON_INSTRUCTION_ACCESS_FAULT", - "REPORT_VA_IN_MTVAL_ON_LOAD_PAGE_FAULT", - "REPORT_VA_IN_MTVAL_ON_STORE_AMO_PAGE_FAULT", - "REPORT_VA_IN_MTVAL_ON_INSTRUCTION_PAGE_FAULT", - "REPORT_ENCODING_IN_MTVAL_ON_ILLEGAL_INSTRUCTION", - "TRAP_ON_ILLEGAL_WLRL", - "CONFIG_PTR_ADDRESS", - "PHYS_ADDR_WIDTH", - "CACHE_BLOCK_SIZE", - "MISALIGNED_SPLIT_STRATEGY" - ], - "properties": { - "XLEN": { - "type": "integer", - "enum": [32, 64], - "description": "Base instruction set datapath width (e.g., RV32 or RV64)\n" - }, - "NAME": { - "type": "string", - "description": "Name of the configuration.\n*Must* match the directory name under cfgs/ where params.yaml is stored.\n" - }, - "MISALIGNED_SPLIT_STRATEGY": { - "type": "string", - "enum": ["by_byte"], - "description": "How the implementation splits misaligned loads/stores." - }, - "M_MODE_ENDIANESS": { - "type": "integer", - "enum": [0, 1, 2], - "description": "Endianess of data in M-mode. Can be one of:\n\n * 0: M-mode data is always little endian\n * 1: M-mode data is always big endian\n * 2: M-mode data can be either little or big endian, depending on the RW CSR field mstatus.MBE\n" - }, - "S_MODE_ENDIANESS": { - "type": "integer", - "enum": [0, 1, 2], - "description": "Endianess of data in S-mode. Can be one of:\n\n * 0: S-mode data is always little endian\n * 1: S-mode data is always big endian\n * 2: S-mode data can be either little or big endian, depending on the RW CSR field mstatus.SBE\n" - }, - "U_MODE_ENDIANESS": { - "type": "integer", - "enum": [0, 1, 2], - "description": "Endianess of data in U-mode. Can be one of:\n\n * 0: U-mode data is always little endian\n * 1: U-mode data is always big endian\n * 2: U-mode data can be either little or big endian, depending on the RW CSR field mstatus.UBE\n" - }, - "VS_MODE_ENDIANESS": { - "type": "integer", - "enum": [0, 1, 2], - "description": "Endianess of data in VS-mode. Can be one of:\n\n * 0: VS-mode data is always little endian\n * 1: VS-mode data is always big endian\n * 2: VS-mode data can be either little or big endian, depending on the RW CSR field hstatus.VSBE\n" - }, - "VU_MODE_ENDIANESS": { - "type": "integer", - "enum": [0, 1, 2], - "description": "Endianess of data in VU-mode. Can be one of:\n\n * 0: VU-mode data is always little endian\n * 1: VU-mode data is always big endian\n * 2: VU-mode data can be either little or big endian, depending on the RW CSR field vsstatus.UBE\n" - }, - "SXLEN": { - "description": "XLENs supported in S-mode. Can be one of:\n\n * 32: SXLEN is always 32\n * 64: SXLEN is always 64 * 3264: SXLEN can be changed (via mstatus.SXL) between 32 and 64", - "type": "integer", - "enum": [32, 64, 3264] - }, - "UXLEN": { - "description": "XLENs supported in U-mode. Can be one of:\n\n * 32: UXLEN is always 32\n * 64: UXLEN is always 64 * 3264: UXLEN can be changed (via mstatus.UXL) between 32 and 64", - "type": "integer", - "enum": [32, 64, 3264] - }, - "VSXLEN": { - "description": "XLENs supported in VS-mode. Can be one of:\n\n * 32: VSXLEN is always 32\n * 64: VSXLEN is always 64 * 3264: VSXLEN can be changed (via hstatus.VSXL) between 32 and 64", - "type": "integer", - "enum": [32, 64, 3264] - }, - "VUXLEN": { - "description": "XLENs supported in VU-mode. Can be one of:\n\n * 32: UXLEN is always 32\n * 64: VUXLEN is always 64 * 3264: VUXLEN can be changed (via vsstatus.SXL) between 32 and 64", - "type": "integer", - "enum": [32, 64, 3264] - }, - "ASID_WIDTH": { - "type": "integer", - "maximum": 16, - "minimum": 0, - "description": "Number of implemented ASID bits" - }, - "NUM_PMP_ENTRIES": { - "type": "integer", - "maximum": 64, - "minimum": 0, - "description": "Number of implemented PMP entries" - }, - "ARCH_ID": { - "type": "integer", - "minimum": 0, - "maximum": 18446744073709551615, - "description": "Vendor-specific architecture ID presented in `marchid`" - }, - "VENDOR_ID_BANK": { - "type": "integer", - "minimum": 0, - "maximum": 33554431, - "description": "Vendor JEDEC code, bank" - }, - "VENDOR_ID_OFFSET": { - "type": "integer", - "minimum": 0, - "maximum": 127, - "description": "Vendor JEDEC code, offset" - }, - "IMP_ID": { - "type": "integer", - "minimum": 0, - "maximum": 18446744073709551615, - "description": "Vendor-specific implementation ID present in `mimpid`" - }, - "MISALIGNED_LDST": { - "type": "boolean", - "description": " whether or not the implementation supports misaligned loads and stores in main memory (not including atomics). Must be true when extension Zicclsm is implemented." - }, - "MISALIGNED_AMO": { - "type": "boolean", - "description": " whether or not the implementation supports misaligned atomics.", - "default": false - }, - "NUM_HPM_COUNTERS": { - "type": "integer", - "minimum": 0, - "maximum": 29, - "description": "Number of implemented programmable hardware counters (not including cycle, time, and instret)" - }, - "HPM_EVENTS": { - "type": "array", - "items": { - "type": "integer", - "minimum": 0 + "type": "object", + "title": "Architecture configuration", + "required": ["$schema", "kind", "type", "name", "description"], + "allOf": [ + { + "if": { + "properties": { + "type": { "const": "fully configured" } + } + }, + "then": { + "required": ["implemented_extensions", "params"], + "properties": { + "params": { + "type": "object" }, - "uniqueItems": true, - "description": "List of defined HPM events that can be programmed into CSR[mhpmevent*]" - }, - "COUNTINHIBIT_EN": { - "type": "array", - "description": "Indicates which counters can be disabled from mcountinhibit\n\n Formatted as a one-hot enable vector so that, for example, COUNTINHIBIT_EN[0] is for CY and COUNTINHIBIT_EN[3] is for HPM3", - "items": [ - { - "type": "boolean" + "mandatory_extensions": { + "type": "null" + }, + "non_mandatory_extensions": { + "type": "null" + }, + "prohbited_extensions": { + "type": "null" + }, + "implemented_extensions": { + "description": "Extensions implemented by this architecture", + "type": "array", + "items": { + "type": "object", + "required": ["name", "version"], + "properties": { + "name": { + "type": "string", + "pattern": "^([A-WY]|([SXZ][a-z0-9]+))$", + "description": "Extension name" + }, + "version": { + "oneOf": [ + { + "type": "string", + "description": "Extension version" + }, + { + "type": "number" + } + ] + } + }, + "additionalProperties": false + } + } + } + } + }, + { + "if": { + "properties": { + "type": { "const": "partially configured" } + } + }, + "then": { + "anyOf": [ + { + "required": ["mandatory_extensions"] + }, + { + "required": ["params"] + } + ], + "properties": { + "params": { + "type": "object" + }, + "mandatory_extensions": { + "description": "Extensions mandatory in this architecture", + "type": "array", + "items": { + "$ref": "schema_defs.json#/$defs/extension_requirement" }, - { - "const": false, - "$comment": "There is no counter at index 1" + "default": { + "const": [] } - ], - "additionalItems": { - "type": "boolean" }, - "minItems": 32, - "maxItems": 32 - }, - "COUNTENABLE_EN": { - "type": "array", - "description": "Indicates which counters can be delegate from mcounteren\n\n Formatted as a one-hot enable vector so that, for example, COUNTENABLE_EN[0] is for CY and COUNTENABLE_EN[3] is for HPM3", - "items": [ - { - "type": "boolean" + "non_mandatory_extensions": { + "description": "Extensions that are not mandatory but are still _special_ in this architecture. This could mean different things depending on the context: for certificates or generated IP, this would correspond to _optional supported_, and extensions not in non_mandatory are not possible. For profiles, this corresponds to some type of _profile optional_, but extensions in non_mandatory are still possible.", + "type": "array", + "items": { + "$ref": "schema_defs.json#/$defs/extension_requirement" }, - { - "const": false, - "$comment": "There is no counter at index 1" + "default": { + "const": [] } - ], - "additionalItems": { - "type": "boolean" }, - "minItems": 32, - "maxItems": 32 - }, - "TRAP_ON_ILLEGAL_WLRL": { - "type": "boolean", - "default": true, - "description": "When true, writing an illegal value to a WLRL CSR field raises an Illegal Instruction exception.\nWhen false, writing an illegal value to a WLRL CSR field is ignored." - }, - - "REPORT_VA_IN_MTVAL_ON_BREAKPOINT": { - "type": "boolean", - "default": false, - "description": "When true, mtval is written with the virtual PC of the EBREAK instruction (same information as mepc).\nWhen false, mtval is written with 0 on an EBREAK instruction\n\nregardless, mtval is always written with a virtual PC when an external breakpoint is generated." - }, - "REPORT_VA_IN_MTVAL_ON_LOAD_MISALIGNED": { - "type": "boolean", - "default": false, - "description": "When true, mtval is written with the virtual address when a misaligned load causes a LoadAddressMisaligned exception.\nWhen false, mtval is written with 0" - }, - "REPORT_VA_IN_MTVAL_ON_STORE_AMO_MISALIGNED": { - "type": "boolean", - "default": false, - "description": "When true, mtval is written with the virtual address when a misaligned store or atomic causes a StoreAmoAddressMisaligned exception.\nWhen false, mtval is written with 0" - }, - "REPORT_VA_IN_MTVAL_ON_INSTRUCTION_MISALIGNED": { - "type": "boolean", - "default": false, - "description": "When true, mtval is written with the virtual address when a misaligned store or atomic causes a StoreAmoAddressMisaligned exception.\nWhen false, mtval is written with 0" - }, - "REPORT_VA_IN_MTVAL_ON_LOAD_ACCESS_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, mtval is written with the virtual address when a load causes a LoadAccessFault exception.\nWhen false, mtval is written with 0" - }, - "REPORT_VA_IN_MTVAL_ON_STORE_AMO_ACCESS_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, mtval is written with the virtual address when a store or atomic causes a StoreAmoAccessFault exception.\nWhen false, mtval is written with 0" - }, - "REPORT_VA_IN_MTVAL_ON_INSTRUCTION_ACCESS_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, mtval is written with the virtual address when a store or atomic causes a StoreAmoAccessFault exception.\nWhen false, mtval is written with 0" - }, - "REPORT_VA_IN_MTVAL_ON_LOAD_PAGE_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, mtval is written with the virtual address when a load causes a LoadPageFault exception.\nWhen false, mtval is written with 0" - }, - "REPORT_VA_IN_MTVAL_ON_STORE_AMO_PAGE_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, mtval is written with the virtual address when a store or atomic causes a StoreAmoPageFault exception.\nWhen false, mtval is written with 0" - }, - "REPORT_VA_IN_MTVAL_ON_INSTRUCTION_PAGE_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, mtval is written with the virtual address when a store or atomic causes a StoreAmoPageFault exception.\nWhen false, mtval is written with 0" - }, - "REPORT_ENCODING_IN_MTVAL_ON_ILLEGAL_INSTRUCTION": { - "type": "boolean", - "default": false, - "description": "When true, mtval is written with the faulting instruciton encoding when a IllegalInstruction exception occurs.\nWhen false, mtval is written with 0" - }, - "REPORT_CAUSE_IN_MTVAL_ON_SOFTWARE_CHECK": { - "type": "boolean", - "default": false, - "description": "When true, mtval is written with the casue when a SoftwareCheck exception occurs.\nWhen false, mtval is written with 0" - }, - "MTVAL_WIDTH": { - "type": "integer", - "maximum": 64, - "description": "Number of implemented bits in MTVAL. Must be >= largest virtual address size if a VA is ever written to mtval by hardware or if Sdext is implemented. Must also be able to hold the minimum of MXLEN or ILEN if encodings are reported on IllegalInstruction exceptions" - }, - - "REPORT_VA_IN_STVAL_ON_BREAKPOINT": { - "type": "boolean", - "default": false, - "description": "When true, stval is written with the virtual PC of the EBREAK instruction (same information as mepc).\nWhen false, stval is written with 0 on an EBREAK instruction\n\nregardless, stval is always written with a virtual PC when an external breakpoint is generated." - }, - "REPORT_VA_IN_STVAL_ON_LOAD_MISALIGNED": { - "type": "boolean", - "default": false, - "description": "When true, stval is written with the virtual address when a misaligned load causes a LoadAddressMisaligned exception.\nWhen false, stval is written with 0" - }, - "REPORT_VA_IN_STVAL_ON_STORE_AMO_MISALIGNED": { - "type": "boolean", - "default": false, - "description": "When true, stval is written with the virtual address when a misaligned store or atomic causes a StoreAmoAddressMisaligned exception.\nWhen false, stval is written with 0" - }, - "REPORT_VA_IN_STVAL_ON_INSTRUCTION_MISALIGNED": { - "type": "boolean", - "default": false, - "description": "When true, stval is written with the virtual address when a misaligned store or atomic causes a StoreAmoAddressMisaligned exception.\nWhen false, stval is written with 0" - }, - "REPORT_VA_IN_STVAL_ON_LOAD_ACCESS_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, stval is written with the virtual address when a load causes a LoadAccessFault exception.\nWhen false, stval is written with 0" - }, - "REPORT_VA_IN_STVAL_ON_STORE_AMO_ACCESS_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, stval is written with the virtual address when a store or atomic causes a StoreAmoAccessFault exception.\nWhen false, stval is written with 0" - }, - "REPORT_VA_IN_STVAL_ON_INSTRUCTION_ACCESS_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, stval is written with the virtual address when a store or atomic causes a StoreAmoAccessFault exception.\nWhen false, stval is written with 0" - }, - "REPORT_VA_IN_STVAL_ON_LOAD_PAGE_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, stval is written with the virtual address when a load causes a LoadPageFault exception.\nWhen false, stval is written with 0" - }, - "REPORT_VA_IN_STVAL_ON_STORE_AMO_PAGE_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, stval is written with the virtual address when a store or atomic causes a StoreAmoPageFault exception.\nWhen false, stval is written with 0" - }, - "REPORT_VA_IN_STVAL_ON_INSTRUCTION_PAGE_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, stval is written with the virtual address when a store or atomic causes a StoreAmoPageFault exception.\nWhen false, stval is written with 0" - }, - "REPORT_ENCODING_IN_STVAL_ON_ILLEGAL_INSTRUCTION": { - "type": "boolean", - "default": false, - "description": "When true, stval is written with the faulting instruction encoding when a IllegalInstruction exception occurs.\nWhen false, stval is written with 0" - }, - "REPORT_CAUSE_IN_STVAL_ON_SOFTWARE_CHECK": { - "type": "boolean", - "default": false, - "description": "When true, stval is written with the cause when a SoftwareCheck exception occurs.\nWhen false, stval is written with 0" - }, - "STVAL_WIDTH": { - "type": "integer", - "maximum": 64, - "description": "Number of implemented bits in STVAL. Must be >= largest virtual address size if a VA is ever written to stval by hardware or if Sdext is implemented. Must also be able to hold the minimum of SXLEN or ILEN if encodings are reported on IllegalInstruction exceptions" - }, - - "REPORT_VA_IN_VSTVAL_ON_BREAKPOINT": { - "type": "boolean", - "default": false, - "description": "When true, vstval is written with the virtual PC of the EBREAK instruction (same information as mepc).\nWhen false, vstval is written with 0 on an EBREAK instruction\n\nregardless, vstval is always written with a virtual PC when an external breakpoint is generated." - }, - "REPORT_VA_IN_VSTVAL_ON_LOAD_MISALIGNED": { - "type": "boolean", - "default": false, - "description": "When true, vstval is written with the virtual address when a misaligned load causes a LoadAddressMisaligned exception.\nWhen false, vstval is written with 0" - }, - "REPORT_VA_IN_VSTVAL_ON_STORE_AMO_MISALIGNED": { - "type": "boolean", - "default": false, - "description": "When true, vstval is written with the virtual address when a misaligned store or atomic causes a StoreAmoAddressMisaligned exception.\nWhen false, vstval is written with 0" - }, - "REPORT_VA_IN_VSTVAL_ON_INSTRUCTION_MISALIGNED": { - "type": "boolean", - "default": false, - "description": "When true, vstval is written with the virtual address when a misaligned store or atomic causes a StoreAmoAddressMisaligned exception.\nWhen false, vstval is written with 0" - }, - "REPORT_VA_IN_VSTVAL_ON_LOAD_ACCESS_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, vstval is written with the virtual address when a load causes a LoadAccessFault exception.\nWhen false, vstval is written with 0" - }, - "REPORT_VA_IN_VSTVAL_ON_STORE_AMO_ACCESS_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, vstval is written with the virtual address when a store or atomic causes a StoreAmoAccessFault exception.\nWhen false, vstval is written with 0" - }, - "REPORT_VA_IN_VSTVAL_ON_INSTRUCTION_ACCESS_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, vstval is written with the virtual address when a store or atomic causes a StoreAmoAccessFault exception.\nWhen false, vstval is written with 0" - }, - "REPORT_VA_IN_VSTVAL_ON_LOAD_PAGE_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, vstval is written with the virtual address when a load causes a LoadPageFault exception.\nWhen false, vstval is written with 0" - }, - "REPORT_VA_IN_VSTVAL_ON_STORE_AMO_PAGE_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, vstval is written with the virtual address when a store or atomic causes a StoreAmoPageFault exception.\nWhen false, vstval is written with 0" - }, - "REPORT_VA_IN_VSTVAL_ON_INSTRUCTION_PAGE_FAULT": { - "type": "boolean", - "default": false, - "description": "When true, vstval is written with the virtual address when a store or atomic causes a StoreAmoPageFault exception.\nWhen false, vstval is written with 0" - }, - "REPORT_ENCODING_IN_VSTVAL_ON_ILLEGAL_INSTRUCTION": { - "type": "boolean", - "default": false, - "description": "When true, vstval is written with the faulting instruciton encoding when a IllegalInstruction exception occurs.\nWhen false, vstval is written with 0" - }, - "REPORT_CAUSE_IN_VSTVAL_ON_SOFTWARE_CHECK": { - "type": "boolean", - "default": false, - "description": "When true, vstval is written with the casue when a SoftwareCheck exception occurs.\nWhen false, vstval is written with 0" - }, - - "CONFIG_PTR_ADDRESS": { - "type": "integer", - "minimum": 0, - "maximum": 18446744073709551615, - "description": "Physical address of the unified discovery configuration data structure (reported in `mconfigptr`)." - }, - "PMP_GRANULARITY": { - "type": "integer", - "minimum": 2, - "maximum": 66, - "description": "log2 of the smallest supported PMP region." - }, - "PMA_GRANULARITY": { - "type": "integer", - "minimum": 2, - "maximum": 66, - "description": "log2 of the smallest supported PMA region." - }, - "PHYS_ADDR_WIDTH": { - "type": "integer", - "minimum": 1, - "maximum": 36028797018963968, - "description": "Number of bits in the physical address space" - }, - "MUTABLE_MISA_A": { - "type": "boolean", - "default": false, - "description": "When A extension is supported, wether or not it can be dynamically disabled by writing the `misa.A` bit." - }, - "MUTABLE_MISA_B": { - "type": "boolean", - "default": false, - "description": "When B extension is supported, wether or not it can be dynamically disabled by writing the `misa.B` bit." - }, - "MUTABLE_MISA_C": { - "type": "boolean", - "default": false, - "description": "When C extension is supported, wether or not it can be dynamically disabled by writing the `misa.C` bit." - }, - "MUTABLE_MISA_D": { - "type": "boolean", - "default": false, - "description": "When D extension is supported, wether or not it can be dynamically disabled by writing the `misa.D` bit." - }, - "MUTABLE_MISA_F": { - "type": "boolean", - "default": false, - "description": "When F extension is supported, wether or not it can be dynamically disabled by writing the `misa.F` bit." - }, - "MUTABLE_MISA_H": { - "type": "boolean", - "default": false, - "description": "When H extension is supported, wether or not it can be dynamically disabled by writing the `misa.H` bit." - }, - "MUTABLE_MISA_M": { - "type": "boolean", - "default": false, - "description": "When M extension is supported, wether or not it can be dynamically disabled by writing the `misa.M` bit." - }, - "MUTABLE_MISA_S": { - "type": "boolean", - "default": false, - "description": "When S extension is supported, wether or not it can be dynamically disabled by writing the `misa.S` bit." - }, - "MUTABLE_MISA_U": { - "type": "boolean", - "default": false, - "description": "When U extension is supported, wether or not it can be dynamically disabled by writing the `misa.U` bit." - }, - "MUTABLE_MISA_V": { - "type": "boolean", - "default": false, - "description": "When V extension is supported, wether or not it can be dynamically disabled by writing the `misa.V` bit." - }, - "CACHE_BLOCK_SIZE": { - "type": "integer", - "minimum": 1, - "maximum": 65536, - "description": "Size, in bytes of a cache block (as seen by cache maintence operations)" - }, - "NUM_EXTERNAL_GUEST_INTERRUPTS": { - "type": "integer", - "minimum": 1, - "maximum": 63, - "description": "Number of supported virtualized guest external interrupts.\nCorresponds to the GEILEN parameter in RISC-V specifications." - }, - "LRSC_RESERVATION_STRATEGY": { - "type": "string", - "enum": [ - "reserve naturally-aligned 64-byte region", - "reserve naturally-aligned 128-byte region", - "reserve exactly enough to cover the access", - "custom" - ], - "description": "Strategy used to handle reservation sets\n\n * 'reserve naturally-aligned 64-byte region': Always reserve the 64-byte block containing the LR/SC address\n * 'reserve naturally-aligned 128-byte region': Always reserve the 128-byte block containing the LR/SC address\n * 'reserve exactly enough to cover the access': Always reserve exactly the LR/SC access, and no more\n * 'custom': Custom behavior, leading to an 'unpredictable' call on any LR/SC" + "prohibited_extensions": { + "description": "Extensions explicitly prohibited in this architecture. Does *not* need to include extensions that are excluded because of a conflict-by-definition with a mandatory extension, as those will be calculated automatically", + "type": "array", + "items": { + "$ref": "schema_defs.json#/$defs/extension_requirement" + }, + "default": { + "const": [] + } + }, + "implemented_extensions": { + "type": "null" + } + } + } + }, + { + "if": { + "properties": { + "type": { "const": "unconfigured" } + } + }, + "then": { + "mandatory_extensions": { + "type": "null" }, - "LRSC_FAIL_ON_VA_SYNONYM": { - "type": "boolean", - "description": "whether or not an SC will fail if its VA does not match the VA of the prior LR, even if the physical address of the SC and LR are the same" + "non_mandatory_extensions": { + "type": "null" }, - "LRSC_FAIL_ON_NON_EXACT_LRSC": { - "type": "boolean", - "description": "whether or not a Store Conditional fails if its physical address and size do not\nexactly match the physical address and size of the last Load Reserved in program order\n(independent of whether or not the SC is in the current reservation set)\n" + "prohbited_extensions": { + "type": "null" }, - "LRSC_MISALIGNED_BEHAVIOR": { - "type": "string", - "enum": [ - "always raise misaligned exception", - "always raise access fault", - "custom" - ], - "description": "what to do when an LR/SC address is misaligned:\n\n * 'always raise misaligned exception': self-explainitory\n * 'always raise access fault': self-explainitory\n * 'custom': Custom behavior; misaligned LR/SC may sometimes raise a misaligned exception and sometimes raise a access fault. Will lead to an 'unpredictable' call on any misaligned LR/SC access" + "params": { + "type": "null" } - }, - "additionalProperties": false + } } - }, - "type": "object", - "required": ["params", "extensions"], + ], "properties": { - "params": { - "$ref": "#/$defs/params" + "type": { + "type": "string", + "description": "Type of the arch", + "enum": ["unconfigured", "partially configured", "fully configured"] }, - "extensions": { - "type": "array", - "description": "Extension names and versions", - "items": { - "type": "array", - "prefixItems": [ - { - "type": "string" - }, - { - "oneOf": [ - { - "type": "number" - }, - { - "type": "string", - "pattern": "^[0-9]+(\\.[0-9]+(\\.[0-9]+(-[a-fA-F0-9]+)?)?)?$" - } - ] - } - ], - "additionalItems": false - } - } - } + "$schema": { + "type": "string", + "format": "uri-reference", + "const": "config_schema.json#" + }, + "kind": { + "type": "string", + "const": "architecture configuration" + }, + "name": { + "type": "string" + }, + "description": { + "type": "string", + "description": "An asciidoc description of the configuration" + }, + "params": true, + "mandatory_extensions": true, + "non_mandatory_extensions": true, + "prohibited_extensions": true, + "implemented_extensions": true + }, + "additionalProperties": false } diff --git a/schemas/csr_schema.json b/schemas/csr_schema.json index 92b53160b..8110e61e1 100644 --- a/schemas/csr_schema.json +++ b/schemas/csr_schema.json @@ -266,7 +266,7 @@ "type": "string", "description": "Function that returns the value of the CSR when read by software (i.e., a Zicsr instruction). If not specified, the value last written (through hw_write) is returned." }, - "__source": { + "$source": { "description": "Path to the source file this definition came from; used by downstream tooling -- not expected to be in handwritten files", "type": "string" } diff --git a/schemas/ext_schema.json b/schemas/ext_schema.json index 222fd2e18..40d6d485c 100644 --- a/schemas/ext_schema.json +++ b/schemas/ext_schema.json @@ -74,34 +74,10 @@ }, "company": { "description": "The company that developed this extension", - "type": "object", - "properties": { - "name": { - "type": "string", - "description": "Name of the company that developed this extension. Should be \"RISC-V International\" for standard extensions" - }, - "url": { - "type": "string", - "format": "uri", - "description": "Website of the company that developed this extension. Should be \"https://riscv.org\" for standard extensions" - } - } + "$ref": "schema_defs.json#/$defs/company" }, "doc_license": { - "description": "License that applies to the textual documentation for this extension", - "type": "object", - "properties": { - "name": { - "type": "string", - "description": "License name" - }, - "url": { - "type": "string", - "format": "uri", - "description": "Link to license text" - } - }, - "additionalProperties": false + "$ref": "schema_defs.json#/$defs/license" }, "type": { "enum": ["unprivileged", "privileged"] }, "conflicts": { @@ -311,7 +287,7 @@ }, "additionalProperties": false }, - "__source": { + "$source": { "type": "string", "description": "Source file where this extension was defined" } diff --git a/schemas/implemented_exts_schema.json b/schemas/implemented_exts_schema.json deleted file mode 100644 index cf4d6dd29..000000000 --- a/schemas/implemented_exts_schema.json +++ /dev/null @@ -1,14 +0,0 @@ -{ - "$schema": "http://json-schema.org/draft-07/schema#", - - "type": "object", - "properties": { - "implemented_extensions": { - "type": "array", - "items": { - "$ref": "schema_defs.json#/$defs/extension_name_and_version" - } - } - }, - "additionalProperties": false -} diff --git a/schemas/inst_schema.json b/schemas/inst_schema.json index 6a54ffd9b..637543623 100644 --- a/schemas/inst_schema.json +++ b/schemas/inst_schema.json @@ -75,6 +75,11 @@ "name": { "type": "string" }, + "$child_of": { + "type": "string", + "pattern": "^common/inst_variable_types\\.yaml#/[a-zA-Z0-9_]+", + "description": "Cookie crumb of the reference to variable metadata" + }, "location": { "$ref": "#/$defs/field_location" }, @@ -279,7 +284,7 @@ "additionalProperties": false } }, - "__source": { + "$source": { "description": "Path to the source file. Used by downstream tooling; not expected to be found in handwritten files", "type": "string" } diff --git a/schemas/json-schema-draft-07.json b/schemas/json-schema-draft-07.json index 3f39fddaf..252199ee5 100644 --- a/schemas/json-schema-draft-07.json +++ b/schemas/json-schema-draft-07.json @@ -3,243 +3,240 @@ "$id": "http://json-schema.org/draft-07/schema#", "title": "Core schema meta-schema", "definitions": { - "schemaArray": { - "type": "array", - "minItems": 1, - "items": { - "$ref": "#" - } - }, - "nonNegativeInteger": { - "type": "integer", - "minimum": 0 - }, - "nonNegativeIntegerDefault0": { - "allOf": [ - { - "$ref": "#/definitions/nonNegativeInteger" - }, - { - "default": 0 - } - ] - }, - "simpleTypes": { - "enum": [ - "array", - "boolean", - "integer", - "null", - "number", - "object", - "string" - ] - }, - "stringArray": { - "type": "array", - "items": { - "type": "string" - }, - "uniqueItems": true, - "default": [] + "schemaArray": { + "type": "array", + "minItems": 1, + "items": { + "$ref": "#" } - }, - "type": [ - "object", - "boolean" - ], - "properties": { - "$id": { - "type": "string", - "format": "uri-reference" - }, - "$schema": { - "type": "string", - "format": "uri" - }, - "$ref": { - "type": "string", - "format": "uri-reference" - }, - "$comment": { - "type": "string" - }, - "title": { - "type": "string" - }, - "description": { - "type": "string" - }, - "default": true, - "readOnly": { - "type": "boolean", - "default": false - }, - "writeOnly": { - "type": "boolean", - "default": false - }, - "examples": { - "type": "array", - "items": true - }, - "multipleOf": { - "type": "number", - "exclusiveMinimum": 0 - }, - "maximum": { - "type": "number" - }, - "exclusiveMaximum": { - "type": "number" - }, - "minimum": { - "type": "number" - }, - "exclusiveMinimum": { - "type": "number" - }, - "maxLength": { + }, + "nonNegativeInteger": { + "type": "integer", + "minimum": 0 + }, + "nonNegativeIntegerDefault0": { + "allOf": [ + { "$ref": "#/definitions/nonNegativeInteger" - }, - "minLength": { - "$ref": "#/definitions/nonNegativeIntegerDefault0" - }, - "pattern": { - "type": "string", - "format": "regex" - }, - "additionalItems": { - "$ref": "#" - }, + }, + { + "default": 0 + } + ] + }, + "simpleTypes": { + "enum": [ + "array", + "boolean", + "integer", + "null", + "number", + "object", + "string" + ] + }, + "stringArray": { + "type": "array", "items": { - "anyOf": [ - { - "$ref": "#" - }, - { - "$ref": "#/definitions/schemaArray" - } - ], - "default": true - }, - "maxItems": { - "$ref": "#/definitions/nonNegativeInteger" - }, - "minItems": { - "$ref": "#/definitions/nonNegativeIntegerDefault0" - }, - "uniqueItems": { - "type": "boolean", - "default": false + "type": "string" }, - "contains": { + "uniqueItems": true, + "default": [] + } + }, + "type": ["object", "boolean"], + "properties": { + "$id": { + "type": "string", + "format": "uri-reference" + }, + "$schema": { + "type": "string", + "format": "uri" + }, + "$ref": { + "type": "string", + "format": "uri-reference" + }, + "$comment": { + "type": "string" + }, + "title": { + "type": "string" + }, + "description": { + "type": "string" + }, + "default": true, + "readOnly": { + "type": "boolean", + "default": false + }, + "writeOnly": { + "type": "boolean", + "default": false + }, + "examples": { + "type": "array", + "items": true + }, + "multipleOf": { + "type": "number", + "exclusiveMinimum": 0 + }, + "maximum": { + "type": "number" + }, + "exclusiveMaximum": { + "type": "number" + }, + "minimum": { + "type": "number" + }, + "exclusiveMinimum": { + "type": "number" + }, + "maxLength": { + "$ref": "#/definitions/nonNegativeInteger" + }, + "minLength": { + "$ref": "#/definitions/nonNegativeIntegerDefault0" + }, + "pattern": { + "type": "string", + "format": "regex" + }, + "additionalItems": { + "$ref": "#" + }, + "items": { + "anyOf": [ + { "$ref": "#" + }, + { + "$ref": "#/definitions/schemaArray" + } + ], + "default": true + }, + "maxItems": { + "$ref": "#/definitions/nonNegativeInteger" + }, + "minItems": { + "$ref": "#/definitions/nonNegativeIntegerDefault0" + }, + "uniqueItems": { + "type": "boolean", + "default": false + }, + "contains": { + "$ref": "#" + }, + "maxProperties": { + "$ref": "#/definitions/nonNegativeInteger" + }, + "minProperties": { + "$ref": "#/definitions/nonNegativeIntegerDefault0" + }, + "required": { + "$ref": "#/definitions/stringArray" + }, + "additionalProperties": { + "$ref": "#" + }, + "definitions": { + "type": "object", + "additionalProperties": { + "$ref": "#" }, - "maxProperties": { - "$ref": "#/definitions/nonNegativeInteger" - }, - "minProperties": { - "$ref": "#/definitions/nonNegativeIntegerDefault0" - }, - "required": { - "$ref": "#/definitions/stringArray" - }, + "default": {} + }, + "properties": { + "type": "object", "additionalProperties": { - "$ref": "#" + "$ref": "#" }, - "definitions": { - "type": "object", - "additionalProperties": { - "$ref": "#" - }, - "default": {} + "default": {} + }, + "patternProperties": { + "type": "object", + "additionalProperties": { + "$ref": "#" }, - "properties": { - "type": "object", - "additionalProperties": { - "$ref": "#" - }, - "default": {} + "propertyNames": { + "format": "regex" }, - "patternProperties": { - "type": "object", - "additionalProperties": { - "$ref": "#" - }, - "propertyNames": { - "format": "regex" + "default": {} + }, + "dependencies": { + "type": "object", + "additionalProperties": { + "anyOf": [ + { + "$ref": "#" }, - "default": {} - }, - "dependencies": { - "type": "object", - "additionalProperties": { - "anyOf": [ - { - "$ref": "#" - }, - { - "$ref": "#/definitions/stringArray" - } - ] + { + "$ref": "#/definitions/stringArray" } - }, - "propertyNames": { - "$ref": "#" - }, - "const": true, - "enum": { + ] + } + }, + "propertyNames": { + "$ref": "#" + }, + "const": true, + "enum": { + "type": "array", + "items": true, + "minItems": 1, + "uniqueItems": true + }, + "type": { + "anyOf": [ + { + "$ref": "#/definitions/simpleTypes" + }, + { "type": "array", - "items": true, + "items": { + "$ref": "#/definitions/simpleTypes" + }, "minItems": 1, "uniqueItems": true - }, - "type": { - "anyOf": [ - { - "$ref": "#/definitions/simpleTypes" - }, - { - "type": "array", - "items": { - "$ref": "#/definitions/simpleTypes" - }, - "minItems": 1, - "uniqueItems": true - } - ] - }, - "format": { - "type": "string" - }, - "contentMediaType": { - "type": "string" - }, - "contentEncoding": { - "type": "string" - }, - "if": { - "$ref": "#" - }, - "then": { - "$ref": "#" - }, - "else": { - "$ref": "#" - }, - "allOf": { - "$ref": "#/definitions/schemaArray" - }, - "anyOf": { - "$ref": "#/definitions/schemaArray" - }, - "oneOf": { - "$ref": "#/definitions/schemaArray" - }, - "not": { - "$ref": "#" - } + } + ] + }, + "format": { + "type": "string" + }, + "contentMediaType": { + "type": "string" + }, + "contentEncoding": { + "type": "string" + }, + "if": { + "$ref": "#" + }, + "then": { + "$ref": "#" + }, + "else": { + "$ref": "#" + }, + "allOf": { + "$ref": "#/definitions/schemaArray" + }, + "anyOf": { + "$ref": "#/definitions/schemaArray" + }, + "oneOf": { + "$ref": "#/definitions/schemaArray" + }, + "not": { + "$ref": "#" + } }, "default": true } diff --git a/schemas/manual_schema.json b/schemas/manual_schema.json new file mode 100644 index 000000000..2ff45450d --- /dev/null +++ b/schemas/manual_schema.json @@ -0,0 +1,42 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + + "type": "object", + "required": ["$schema", "kind", "name"], + "properties": { + "$schema": { + "type": "string", + "const": "manual_schema.json#" + }, + "kind": { + "type": "string", + "const": "manual" + }, + "name": { + "type": "string", + "description": "Name (database key) of this manual" + }, + "marketing_name": { + "type": "string", + "description": "The publicly displayed manual name" + }, + "state": { + "$ref": "schema_defs.json#/$defs/spec_state", + "description": "State of this version" + }, + "url": { + "type": "string", + "format": "uri", + "description": "URL to the repository" + }, + "license": { + "$ref": "schema_defs.json#/$defs/license" + }, + "$source": { + "type": "string", + "format": "uri-reference", + "description": "Relative (from arch/) path to the original source file" + } + }, + "additionalProperties": false +} diff --git a/schemas/manual_version_schema.json b/schemas/manual_version_schema.json index e9de71045..98815be9d 100644 --- a/schemas/manual_version_schema.json +++ b/schemas/manual_version_schema.json @@ -51,6 +51,8 @@ "type": "object", "required": [ + "$schema", + "kind", "manual", "version", "name", @@ -59,13 +61,28 @@ "volumes" ], "properties": { + "$schema": { + "type": "string", + "const": "manual_version_schema.json#" + }, + "kind": { + "type": "string", + "const": "manual version" + }, "name": { "type": "string", "description": "Name (database key) of this version" }, "manual": { - "type": "string", - "description": "Name (database key) of the manual this version belongs to" + "type": "object", + "properties": { + "$ref": { + "type": "string", + "format": "uri-reference", + "pattern": "^manual/.*\\.yaml#$", + "description": "Pointer to the manual" + } + } }, "version": { "$ref": "schema_defs.json#/$defs/semantic_version", @@ -99,6 +116,11 @@ "$ref": "#/$defs/volume" }, "description": "List of volumes in this version" + }, + "$source": { + "type": "string", + "format": "uri-reference", + "description": "Relative (from arch/) path to the original source file" } }, "additionalProperties": false diff --git a/schemas/profile_class_schema.json b/schemas/profile_class_schema.json new file mode 100644 index 000000000..e3163b1b9 --- /dev/null +++ b/schemas/profile_class_schema.json @@ -0,0 +1,48 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + + "type": "object", + "required": ["$schema", "kind", "name"], + "properties": { + "$schema": { + "type": "string", + "const": "profile_class_schema.json#" + }, + "kind": { + "type": "string", + "const": "profile class" + }, + "name": { + "type": "string", + "description": "Name (database key) of this Profile Class" + }, + "marketing_name": { + "type": "string", + "description": "The publicly displayed profile class name" + }, + "introduction": { + "type": "string", + "description": "Asciidoc introduction to this Profile Class" + }, + "description": { + "type": "string", + "description": "Prose introduction, in asciidoc" + }, + "naming_scheme": { + "type": "string", + "description": "Commentary on how profile releases in the class are named" + }, + "company": { + "$ref": "schema_defs.json#/$defs/company" + }, + "doc_license": { + "$ref": "schema_defs.json#/$defs/license" + }, + "$source": { + "type": "string", + "format": "uri-refencence", + "description": "Realtive (from arch/) path to the original YAML file" + } + }, + "additionalProperties": false +} diff --git a/schemas/profile_release_schema.json b/schemas/profile_release_schema.json new file mode 100644 index 000000000..9d784c6d3 --- /dev/null +++ b/schemas/profile_release_schema.json @@ -0,0 +1,20 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + + "type": "object", + "required": ["$schema", "kind", "name"], + "properties": { + "$schema": { + "type": "string", + "const": "profile_release_schema.json#" + }, + "kind": { + "type": "string", + "const": "profile release" + }, + "name": { + "type": "string", + "description": "Name (database key) of this Profile Release" + } + } +} diff --git a/schemas/profile_schema.json b/schemas/profile_schema.json new file mode 100644 index 000000000..40e6f2e88 --- /dev/null +++ b/schemas/profile_schema.json @@ -0,0 +1,20 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + + "type": "object", + "required": ["$schema", "kind", "name"], + "properties": { + "$schema": { + "type": "string", + "const": "profile_schema.json#" + }, + "kind": { + "type": "string", + "const": "profile" + }, + "name": { + "type": "string", + "description": "Name (database key) of this Profile" + } + } +} diff --git a/schemas/schema_defs.json b/schemas/schema_defs.json index df5f06eab..b4d7a22ad 100644 --- a/schemas/schema_defs.json +++ b/schemas/schema_defs.json @@ -4,7 +4,7 @@ "title": "Common patterns used by all schemas", "$defs": { - "__source": { + "$source": { "type": "string", "format": "uri-reference", "description": "Path to the source file containing this object" @@ -58,6 +58,46 @@ "nonstandard-released" ] }, + "license": { + "description": "License that applies to the textual documentation for this extension", + "type": "object", + "properties": { + "name": { + "type": "string", + "description": "License name" + }, + "id": { + "type": "string", + "description": "License identifier" + }, + "url": { + "type": "string", + "format": "uri", + "description": "Link to license text" + }, + "text_url": { + "type": "string", + "format": "uri", + "description": "Link to license text" + } + }, + "additionalProperties": false + }, + "company": { + "description": "A company", + "type": "object", + "properties": { + "name": { + "type": "string", + "description": "Name of the company. Should be \"RISC-V International\" for standard extensions" + }, + "url": { + "type": "string", + "format": "uri", + "description": "Website of the company. Should be \"https://riscv.org\" for standard extensions" + } + } + }, "extension_presence": { "oneOf": [ { @@ -85,7 +125,7 @@ }, "extension_name": { "type": "string", - "pattern": "^([A-WY])|([SXZ][a-z0-9]+)$" + "pattern": "^(([A-WY])|([SXZ][a-z0-9]+))$" }, "extension_version": { "$ref": "#/$defs/semantic_version"