Compare commits
	
		
			192 Commits
		
	
	
		
			first-work
			...
			develop
		
	
	| Author | SHA1 | Date | 
|---|---|---|
|  | da9f2940ba | |
|  | 30db2405ef | |
|  | a9a97d132d | |
|  | f4418c3164 | |
|  | 637304bdb3 | |
|  | a95d4e3fb0 | |
|  | 162592beaa | |
|  | cfe4a230de | |
|  | 3c57c6e2df | |
|  | 2d9da8c5db | |
|  | 0234f2136e | |
|  | e5a3ea6998 | |
|  | 7f4e9de6d5 | |
|  | 8f26c6fa44 | |
|  | f76fab3fff | |
|  | 9cbc6ce80f | |
|  | 8b6a585250 | |
|  | ea97d29c3c | |
|  | 61eb2f9538 | |
|  | 16ec45b97c | |
|  | 33dbf79041 | |
|  | 1e17e03a8d | |
|  | 721af870bf | |
|  | 40782ea7cc | |
|  | 4cdeeb8556 | |
|  | b26d61cb67 | |
|  | cc4170200f | |
|  | e3f199f89f | |
|  | af3dc82c76 | |
|  | 7d03ea85a2 | |
|  | ca263b33a1 | |
|  | 90389705a7 | |
|  | 5e1bbf9758 | |
|  | 503f43d983 | |
|  | 5a304557a7 | |
|  | f6da27062f | |
|  | e6ddbb76e7 | |
|  | d706b2edac | |
|  | 115c93e3f8 | |
|  | 062591fb48 | |
|  | 36b2d92515 | |
|  | 7af16897ec | |
|  | a9b9aac50d | |
|  | 155ae413b4 | |
|  | e4b12e2c45 | |
|  | 79f7ebcaab | |
|  | 77e3423273 | |
|  | 0f4824afd3 | |
|  | 1644f299fe | |
|  | 82d44a9c62 | |
|  | 856c4a9e68 | |
|  | 6281204062 | |
|  | d3f561d214 | |
|  | 01906a02bc | |
|  | 50e955c6a7 | |
|  | 601cf7a784 | |
|  | 147a88acb4 | |
|  | 8a5ef112be | |
|  | e6cea882dd | |
|  | ed568f8ca4 | |
|  | a267ad83ad | |
|  | 73bc383e23 | |
|  | 2c43585286 | |
|  | 20363a7e25 | |
|  | a53ede11af | |
|  | 7e9d325b14 | |
|  | 74303ac7c1 | |
|  | c20faca3bd | |
|  | 7def43ad97 | |
|  | 32a738994d | |
|  | c174ad204b | |
|  | 1c37e0dfba | |
|  | f3103126ab | |
|  | 42c3ace1fa | |
|  | b4205f5ce6 | |
|  | d85d824bfd | |
|  | 1de52fa109 | |
|  | 696307f30c | |
|  | 4cb6f29e0d | |
|  | 8265b06474 | |
|  | 43d5e4ea21 | |
|  | 3147e5327a | |
|  | e72fac01b0 | |
|  | 6896b7a1cc | |
|  | 2ac804793b | |
|  | 6127b5d940 | |
|  | 6050ca9e9d | |
|  | 4e45358339 | |
|  | 367ad5bbb2 | |
|  | 53e68bf7b3 | |
|  | 02124ed342 | |
|  | 5dffa9eb8e | |
|  | d124b02e7c | |
|  | de8b24d504 | |
|  | e8da38e04f | |
|  | 91e8652abc | |
|  | 7604f7c166 | |
|  | f07f130132 | |
|  | 09003a7da2 | |
|  | 373f4753dc | |
|  | 482c62d341 | |
|  | df0510266e | |
|  | dc2a73f6c6 | |
|  | bfba78118f | |
|  | aac05a8776 | |
|  | 50d9caa5f6 | |
|  | 20c7c8bd7e | |
|  | 79028dba96 | |
|  | 6da44a69ac | |
|  | 1776dac439 | |
|  | a5eeacf19b | |
|  | d4356434a0 | |
|  | d38661b63f | |
|  | 2a4eb20131 | |
|  | 3415ccd68c | |
|  | c378250aba | |
|  | 1fedc95d42 | |
|  | ec27898830 | |
|  | d342031331 | |
|  | ab696eca8e | |
|  | fef66a0191 | |
|  | cecddea41e | |
|  | 7a46d7d993 | |
|  | 9f06e2f1a5 | |
|  | 201cf1e4af | |
|  | ea75413df0 | |
|  | 7adf3e1629 | |
|  | 41eed8e412 | |
|  | 0def5a4687 | |
|  | 79baa20eb6 | |
|  | 149245f411 | |
|  | e19177da52 | |
|  | 082a1c29e7 | |
|  | 1f9565b239 | |
|  | 4da36b8d5e | |
|  | c3d8424a46 | |
|  | 6ae59124a6 | |
|  | e6d4bf6095 | |
|  | d572b2c297 | |
|  | 6c5d7a13fc | |
|  | fb7f86abf0 | |
|  | e48316cc4b | |
|  | 97e2c09871 | |
|  | 4672ddace6 | |
|  | 4083e955d4 | |
|  | 88b0af44cb | |
|  | c823b42447 | |
|  | 07aef5f2cc | |
|  | 0692c324af | |
|  | 504dcd1f65 | |
|  | 2a4cfd4184 | |
|  | 458aba6652 | |
|  | 408857fcb6 | |
|  | 4898ad3139 | |
|  | 2f663f61de | |
|  | ded53c16a7 | |
|  | f85020ef4e | |
|  | 2ef8999daf | |
|  | 44ef31ff77 | |
|  | 589696a500 | |
|  | 7e4bf75495 | |
|  | 44037398d4 | |
|  | dd2938fa2f | |
|  | bd428133d9 | |
|  | 684ed4e842 | |
|  | 4f616302d5 | |
|  | 91a14f2792 | |
|  | d453d60e19 | |
|  | 041c4add0d | |
|  | c284b4bd48 | |
|  | 4cc5305dc8 | |
|  | 50b0613f5c | |
|  | af1cc83530 | |
|  | 33722e895f | |
|  | 9fd9612b77 | |
|  | 34115db6f6 | |
|  | 310e59f9b3 | |
|  | cd16b733e9 | |
|  | f485629fb6 | |
|  | be3131f666 | |
|  | 8d4c96e437 | |
|  | 668296b8e0 | |
|  | a4a3289712 | |
|  | ecec3c96e1 | |
|  | 02107bf297 | |
|  | 57e4b81e1c | |
|  | 8bacc7cfb7 | |
|  | 7ad3329978 | |
|  | 57fc6e5aa5 | |
|  | cb814b353b | |
|  | 1e2878dca6 | |
|  | a57833d2b1 | 
|  | @ -1,5 +0,0 @@ | ||||||
| [target.aarch64-vesper-metta] |  | ||||||
| rustflags = [ |  | ||||||
|   "-C", "target-feature=-fp-armv8", |  | ||||||
|   "-C", "target-cpu=cortex-a53", |  | ||||||
| ] |  | ||||||
|  | @ -0,0 +1,12 @@ | ||||||
|  | [build] | ||||||
|  | # https://internals.rust-lang.org/t/evaluating-pipelined-rustc-compilation/10199/12 | ||||||
|  | pipelining = true | ||||||
|  | 
 | ||||||
|  | [target.aarch64-vesper-metta] | ||||||
|  | rustflags = [ | ||||||
|  |     "-C", "target-feature=-fp-armv8", | ||||||
|  |     "-C", "target-cpu=cortex-a53", | ||||||
|  |     "-C", "embed-bitcode=yes", | ||||||
|  |     "-Z", "macro-backtrace", | ||||||
|  | ] | ||||||
|  | runner = "cargo make test-runner" | ||||||
|  | @ -0,0 +1,21 @@ | ||||||
|  | version: 2 | ||||||
|  | updates: | ||||||
|  | - package-ecosystem: cargo | ||||||
|  |   directory: "/" | ||||||
|  |   schedule: | ||||||
|  |     interval: daily | ||||||
|  |   open-pull-requests-limit: 10 | ||||||
|  |   ignore: | ||||||
|  |   - dependency-name: cortex-a | ||||||
|  |     versions: | ||||||
|  |     - 5.1.2 | ||||||
|  |     - 5.1.3 | ||||||
|  |     - 5.1.4 | ||||||
|  |     - 5.1.5 | ||||||
|  |     - 5.1.6 | ||||||
|  |   - dependency-name: qemu-exit | ||||||
|  |     versions: | ||||||
|  |     - 1.0.2 | ||||||
|  |   - dependency-name: register | ||||||
|  |     versions: | ||||||
|  |     - 1.0.2 | ||||||
|  | @ -0,0 +1,150 @@ | ||||||
|  | name: Build | ||||||
|  | 
 | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - '*' | ||||||
|  |   pull_request: | ||||||
|  | 
 | ||||||
|  | jobs: | ||||||
|  |   test: | ||||||
|  |     name: Test | ||||||
|  | 
 | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         platform: [ | ||||||
|  |           ubuntu-latest, | ||||||
|  |           macos-latest, | ||||||
|  |           windows-latest | ||||||
|  |         ] | ||||||
|  | 
 | ||||||
|  |     runs-on: ${{ matrix.platform }} | ||||||
|  |     timeout-minutes: 30 | ||||||
|  | 
 | ||||||
|  |     steps: | ||||||
|  |     - name: "Checkout Repository" | ||||||
|  |       uses: actions/checkout@v1 | ||||||
|  | 
 | ||||||
|  |     - name: Install Rustup (macOS) | ||||||
|  |       run: | | ||||||
|  |         curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y | ||||||
|  |         echo "$HOME/.cargo/bin" >> $GITHUB_PATH | ||||||
|  |       if: runner.os == 'macOS' | ||||||
|  | 
 | ||||||
|  |     - name: Set Rustup profile to minimal | ||||||
|  |       run: rustup set profile minimal | ||||||
|  | 
 | ||||||
|  |     - name: "Switch to Rust nightly" | ||||||
|  |       run: rustup default nightly | ||||||
|  | 
 | ||||||
|  |     - name: "Print Rust Version" | ||||||
|  |       run: | | ||||||
|  |         rustc -Vv | ||||||
|  |         cargo -Vv | ||||||
|  | 
 | ||||||
|  |     - name: "Install build tools" | ||||||
|  |       run: cargo install cargo-make cargo-binutils | ||||||
|  | 
 | ||||||
|  |     - name: "Prepare packages (Linux)" | ||||||
|  |       run: | | ||||||
|  |         sudo apt install software-properties-common | ||||||
|  |         sudo add-apt-repository ppa:jacob/virtualisation | ||||||
|  |         sudo apt update | ||||||
|  |       if: runner.os == 'Linux' | ||||||
|  | 
 | ||||||
|  |     - name: "Install dev libraries (Linux)" | ||||||
|  |       run: sudo apt install libudev-dev | ||||||
|  |       if: runner.os == 'Linux' | ||||||
|  | 
 | ||||||
|  |     - name: "Validate rust-lld" | ||||||
|  |       run: | | ||||||
|  |         which rust-lld || echo "Not found" | ||||||
|  |         otool -L ~/.cargo/bin/rust-lld | ||||||
|  |       if: runner.os == 'macOS' | ||||||
|  | 
 | ||||||
|  |     - name: "Print Tools Version" | ||||||
|  |       run: | | ||||||
|  |         cargo make --version | ||||||
|  |         cargo objcopy --version | ||||||
|  | 
 | ||||||
|  |     - name: "Deny Warnings" | ||||||
|  |       run: cargo make build | ||||||
|  |       env: | ||||||
|  |         RUSTFLAGS: "-D warnings" | ||||||
|  | 
 | ||||||
|  |     - name: Install QEMU (Linux) | ||||||
|  |       run: | | ||||||
|  |         sudo apt install qemu-system-aarch64 | ||||||
|  |       if: runner.os == 'Linux' | ||||||
|  | 
 | ||||||
|  |     - name: Install QEMU (macOS) | ||||||
|  |       run: brew install qemu | ||||||
|  |       if: runner.os == 'macOS' | ||||||
|  |       env: | ||||||
|  |         HOMEBREW_NO_AUTO_UPDATE: 1 | ||||||
|  |         HOMEBREW_NO_BOTTLE_SOURCE_FALLBACK: 1 | ||||||
|  |         HOMEBREW_NO_INSTALL_CLEANUP: 1 | ||||||
|  | 
 | ||||||
|  |     - name: Install Scoop (Windows) | ||||||
|  |       run: | | ||||||
|  |         Invoke-Expression (New-Object System.Net.WebClient).DownloadString('https://get.scoop.sh') | ||||||
|  |         echo "$HOME\scoop\shims" | Out-File -FilePath $env:GITHUB_PATH -Encoding utf8 -Append | ||||||
|  |       if: runner.os == 'Windows' | ||||||
|  |       shell: pwsh | ||||||
|  | 
 | ||||||
|  |     - name: Add custom Scoop bucket (Windows) | ||||||
|  |       run: | | ||||||
|  |         scoop bucket add scoop-for-ci https://github.com/metta-systems/scoop-for-ci | ||||||
|  |       if: runner.os == 'Windows' | ||||||
|  |       shell: pwsh | ||||||
|  | 
 | ||||||
|  |     - name: Install QEMU (Windows) | ||||||
|  |       run: scoop install qemu-510 | ||||||
|  |       if: runner.os == 'Windows' | ||||||
|  |       shell: pwsh | ||||||
|  | 
 | ||||||
|  |     - name: "Print QEMU Version" | ||||||
|  |       run: qemu-system-aarch64 --version | ||||||
|  | 
 | ||||||
|  |     - name: 'Build kernel' | ||||||
|  |       run: cargo make build | ||||||
|  | 
 | ||||||
|  |     - name: 'Run tests (macOS)' | ||||||
|  |       run: cargo make test | ||||||
|  |       if: runner.os == 'macOS' | ||||||
|  | 
 | ||||||
|  |     - name: 'Run tests (other OSes)' | ||||||
|  |       run: env QEMU_MACHINE=raspi3 cargo make test | ||||||
|  |       if: runner.os != 'macOS' | ||||||
|  | 
 | ||||||
|  |   check_formatting: | ||||||
|  |     name: "Check Formatting" | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     timeout-minutes: 2 | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v1 | ||||||
|  |     - run: rustup toolchain install nightly --profile minimal --component rustfmt | ||||||
|  |     - run: cargo +nightly fmt -- --check | ||||||
|  | 
 | ||||||
|  |   clippy: | ||||||
|  |     name: "Clippy" | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         features: [ | ||||||
|  |           "", | ||||||
|  |           "noserial", | ||||||
|  |           "qemu", | ||||||
|  |           "noserial,qemu", | ||||||
|  |           "jtag", | ||||||
|  |           "noserial,jtag", | ||||||
|  |           # jtag and qemu together don't make much sense | ||||||
|  |         ] | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     timeout-minutes: 10 | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v1 | ||||||
|  |     - run: sudo apt update | ||||||
|  |     - run: sudo apt install libudev-dev | ||||||
|  |     - run: rustup toolchain install nightly | ||||||
|  |     - run: cargo install cargo-make | ||||||
|  |     - run: env CLIPPY_FEATURES=${{ matrix.features }} cargo make clippy | ||||||
|  | @ -1,8 +1,7 @@ | ||||||
| target/ |  | ||||||
| *.lock |  | ||||||
| .idea/ | .idea/ | ||||||
| .ninja_* | *.iml | ||||||
| kernel8.img | .nova/ | ||||||
| kernel8 | .vscode/ | ||||||
| gdb-connect | target/ | ||||||
|  | kernel8* | ||||||
| .gdb_history | .gdb_history | ||||||
|  |  | ||||||
|  | @ -0,0 +1,27 @@ | ||||||
|  | FROM gitpod/workspace-full | ||||||
|  | 
 | ||||||
|  | USER gitpod | ||||||
|  | 
 | ||||||
|  | RUN sudo apt-get update \ | ||||||
|  |     && sudo apt-get install -yq \ | ||||||
|  |         pkg-config \ | ||||||
|  |         libpython3.6 \ | ||||||
|  |         rust-lldb \ | ||||||
|  |         qemu-system-aarch64 \ | ||||||
|  |         gdb-multiarch \ | ||||||
|  |     && sudo apt-get clean \ | ||||||
|  |     && sudo rm -rf /tmp/* | ||||||
|  | 
 | ||||||
|  | RUN .cargo/bin/rustup toolchain install nightly \ | ||||||
|  |     && .cargo/bin/rustup default nightly \ | ||||||
|  |     && .cargo/bin/rustup toolchain uninstall 1.48.0 \ | ||||||
|  |     && .cargo/bin/rustup component add clippy llvm-tools-preview rls rust-analysis rust-src rustfmt \ | ||||||
|  |     && .cargo/bin/rustup target add aarch64-unknown-none-softfloat | ||||||
|  | 
 | ||||||
|  | RUN bash -lc "cargo install cargo-asm cargo-binutils cargo-bloat cargo-cache cargo-expand cargo-fmt cargo-geiger cargo-graph cargo-make just" | ||||||
|  | 
 | ||||||
|  | RUN bash -lc "cargo cache -i && cargo cache -e && cargo cache -i" | ||||||
|  | 
 | ||||||
|  | ENV RUST_LLDB=/usr/bin/lldb-9 | ||||||
|  | ENV GDB=/usr/bin/gdb-multiarch | ||||||
|  | ENV QEMU=/usr/bin/qemu-system-aarch64 | ||||||
|  | @ -0,0 +1,29 @@ | ||||||
|  | image: | ||||||
|  |   file: .gitpod.Dockerfile | ||||||
|  | 
 | ||||||
|  | tasks: | ||||||
|  |   - name: Clippy | ||||||
|  |     command: just clippy | ||||||
|  |   - name: Testing | ||||||
|  |     command: just test | ||||||
|  |   - name: Build | ||||||
|  |     command: just build | ||||||
|  | 
 | ||||||
|  | github: | ||||||
|  |   prebuilds: | ||||||
|  |     master: true | ||||||
|  |     branches: true | ||||||
|  |     pullRequests: true | ||||||
|  |     addCheck: false | ||||||
|  |     addComment: true | ||||||
|  |     addBadge: true | ||||||
|  |     addLabel: prebuilt-in-gitpod | ||||||
|  | 
 | ||||||
|  | vscode: | ||||||
|  |   extensions: | ||||||
|  |     - hbenl.vscode-test-explorer@2.15.0:koqDUMWDPJzELp/hdS/lWw== | ||||||
|  |     - Swellaby.vscode-rust-test-adapter@0.11.0:Xg+YeZZQiVpVUsIkH+uiiw== | ||||||
|  |     - serayuzgur.crates@0.4.7:HMkoguLcXp9M3ud7ac3eIw== | ||||||
|  |     - belfz.search-crates-io@1.2.1:kSLnyrOhXtYPjQpKnMr4eQ== | ||||||
|  |     - bungcip.better-toml@0.3.2:3QfgGxxYtGHfJKQU7H0nEw== | ||||||
|  |     - webfreak.debug@0.24.0:1zVcRsAhewYEX3/A9xjMNw== | ||||||
|  | @ -0,0 +1,24 @@ | ||||||
|  | # Vesper Changelog | ||||||
|  | 
 | ||||||
|  | All notable changes to this project will be documented in this file. | ||||||
|  | 
 | ||||||
|  | The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | ||||||
|  | 
 | ||||||
|  | ## [Unreleased] | ||||||
|  | 
 | ||||||
|  | ## [0.0.1] - 2020-08-03 | ||||||
|  | ### Added | ||||||
|  | 
 | ||||||
|  | [Unreleased]: https://github.com/metta-systems/vesper/compare/v0.0.1...HEAD | ||||||
|  | [0.0.1]: https://github.com/metta-systems/vesper/releases/tag/v0.0.1 | ||||||
|  | 
 | ||||||
|  | ---- | ||||||
|  | 
 | ||||||
|  | #### Types of changes | ||||||
|  | 
 | ||||||
|  | * `Added` for new features. | ||||||
|  | * `Changed` for changes in existing functionality. | ||||||
|  | * `Deprecated` for soon-to-be removed features. | ||||||
|  | * `Removed` for now removed features. | ||||||
|  | * `Fixed` for any bug fixes. | ||||||
|  | * `Security` in case of vulnerabilities. | ||||||
|  | @ -0,0 +1,918 @@ | ||||||
|  | # This file is automatically @generated by Cargo. | ||||||
|  | # It is not intended for manual editing. | ||||||
|  | version = 3 | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "CoreFoundation-sys" | ||||||
|  | version = "0.1.4" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d0e9889e6db118d49d88d84728d0e964d973a5680befb5f85f55141beea5c20b" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  |  "mach 0.1.2", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "IOKit-sys" | ||||||
|  | version = "0.1.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "99696c398cbaf669d2368076bdb3d627fb0ce51a26899d7c61228c5c0af3bf4a" | ||||||
|  | dependencies = [ | ||||||
|  |  "CoreFoundation-sys", | ||||||
|  |  "libc", | ||||||
|  |  "mach 0.1.2", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "aho-corasick" | ||||||
|  | version = "0.7.18" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "1e37cfd5e7657ada45f742d6e99ca5788580b5c529dc78faf11ece6dc702656f" | ||||||
|  | dependencies = [ | ||||||
|  |  "memchr", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "anyhow" | ||||||
|  | version = "1.0.53" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "94a45b455c14666b85fc40a019e8ab9eb75e3a124e05494f5397122bc9eb06e0" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "atty" | ||||||
|  | version = "0.2.14" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d9b39be18770d11421cdb1b9947a45dd3f37e93092cbf377614828a319d5fee8" | ||||||
|  | dependencies = [ | ||||||
|  |  "hermit-abi", | ||||||
|  |  "libc", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "autocfg" | ||||||
|  | version = "1.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "bit_field" | ||||||
|  | version = "0.10.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "dcb6dd1c2376d2e096796e234a70e17e94cc2d5d54ff8ce42b28cef1d0d359a4" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "bitflags" | ||||||
|  | version = "1.3.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "bytes" | ||||||
|  | version = "1.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "c4872d67bab6358e59559027aa3b9157c53d9358c51423c17554809a8858e0f8" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "cc" | ||||||
|  | version = "1.0.72" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "22a9137b95ea06864e018375b72adfb7db6e6f68cfc8df5a04d00288050485ee" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "cfg-if" | ||||||
|  | version = "1.0.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "chainboot" | ||||||
|  | version = "0.0.1" | ||||||
|  | dependencies = [ | ||||||
|  |  "bit_field", | ||||||
|  |  "bitflags", | ||||||
|  |  "cfg-if", | ||||||
|  |  "cortex-a", | ||||||
|  |  "machine", | ||||||
|  |  "r0", | ||||||
|  |  "seahash", | ||||||
|  |  "snafu", | ||||||
|  |  "tock-registers", | ||||||
|  |  "usize_conversions", | ||||||
|  |  "ux", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "chainofcommand" | ||||||
|  | version = "0.0.1" | ||||||
|  | dependencies = [ | ||||||
|  |  "anyhow", | ||||||
|  |  "bytes", | ||||||
|  |  "clap", | ||||||
|  |  "crossterm", | ||||||
|  |  "defer", | ||||||
|  |  "fehler", | ||||||
|  |  "futures", | ||||||
|  |  "seahash", | ||||||
|  |  "tokio", | ||||||
|  |  "tokio-serial", | ||||||
|  |  "tokio-util", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "clap" | ||||||
|  | version = "3.0.14" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "b63edc3f163b3c71ec8aa23f9bd6070f77edbf3d1d198b164afa90ff00e4ec62" | ||||||
|  | dependencies = [ | ||||||
|  |  "atty", | ||||||
|  |  "bitflags", | ||||||
|  |  "indexmap", | ||||||
|  |  "os_str_bytes", | ||||||
|  |  "strsim", | ||||||
|  |  "termcolor", | ||||||
|  |  "textwrap", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "cortex-a" | ||||||
|  | version = "7.0.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "5bd95fd055d118f77d4e4d527201b6ceccd13586b19b4dac1270f7081fef0f98" | ||||||
|  | dependencies = [ | ||||||
|  |  "tock-registers", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "crossterm" | ||||||
|  | version = "0.23.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "77b75a27dc8d220f1f8521ea69cd55a34d720a200ebb3a624d9aa19193d3b432" | ||||||
|  | dependencies = [ | ||||||
|  |  "bitflags", | ||||||
|  |  "crossterm_winapi", | ||||||
|  |  "futures-core", | ||||||
|  |  "libc", | ||||||
|  |  "mio", | ||||||
|  |  "parking_lot 0.12.0", | ||||||
|  |  "signal-hook", | ||||||
|  |  "signal-hook-mio", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "crossterm_winapi" | ||||||
|  | version = "0.9.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "2ae1b35a484aa10e07fe0638d02301c5ad24de82d310ccbd2f3693da5f09bf1c" | ||||||
|  | dependencies = [ | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "defer" | ||||||
|  | version = "0.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "647605a6345d5e89c3950a36a638c56478af9b414c55c6f2477c73b115f9acde" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "doc-comment" | ||||||
|  | version = "0.3.3" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "fea41bba32d969b513997752735605054bc0dfa92b4c56bf1189f2e174be7a10" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "fehler" | ||||||
|  | version = "1.0.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d5729fe49ba028cd550747b6e62cd3d841beccab5390aa398538c31a2d983635" | ||||||
|  | dependencies = [ | ||||||
|  |  "fehler-macros", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "fehler-macros" | ||||||
|  | version = "1.0.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "ccb5acb1045ebbfa222e2c50679e392a71dd77030b78fb0189f2d9c5974400f9" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "futures" | ||||||
|  | version = "0.3.21" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "f73fe65f54d1e12b726f517d3e2135ca3125a437b6d998caf1962961f7172d9e" | ||||||
|  | dependencies = [ | ||||||
|  |  "futures-channel", | ||||||
|  |  "futures-core", | ||||||
|  |  "futures-executor", | ||||||
|  |  "futures-io", | ||||||
|  |  "futures-sink", | ||||||
|  |  "futures-task", | ||||||
|  |  "futures-util", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "futures-channel" | ||||||
|  | version = "0.3.21" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "c3083ce4b914124575708913bca19bfe887522d6e2e6d0952943f5eac4a74010" | ||||||
|  | dependencies = [ | ||||||
|  |  "futures-core", | ||||||
|  |  "futures-sink", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "futures-core" | ||||||
|  | version = "0.3.21" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "0c09fd04b7e4073ac7156a9539b57a484a8ea920f79c7c675d05d289ab6110d3" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "futures-executor" | ||||||
|  | version = "0.3.21" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "9420b90cfa29e327d0429f19be13e7ddb68fa1cccb09d65e5706b8c7a749b8a6" | ||||||
|  | dependencies = [ | ||||||
|  |  "futures-core", | ||||||
|  |  "futures-task", | ||||||
|  |  "futures-util", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "futures-io" | ||||||
|  | version = "0.3.21" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "fc4045962a5a5e935ee2fdedaa4e08284547402885ab326734432bed5d12966b" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "futures-macro" | ||||||
|  | version = "0.3.21" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "33c1e13800337f4d4d7a316bf45a567dbcb6ffe087f16424852d97e97a91f512" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "futures-sink" | ||||||
|  | version = "0.3.21" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "21163e139fa306126e6eedaf49ecdb4588f939600f0b1e770f4205ee4b7fa868" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "futures-task" | ||||||
|  | version = "0.3.21" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "57c66a976bf5909d801bbef33416c41372779507e7a6b3a5e25e4749c58f776a" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "futures-util" | ||||||
|  | version = "0.3.21" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d8b7abd5d659d9b90c8cba917f6ec750a74e2dc23902ef9cd4cc8c8b22e6036a" | ||||||
|  | dependencies = [ | ||||||
|  |  "futures-channel", | ||||||
|  |  "futures-core", | ||||||
|  |  "futures-io", | ||||||
|  |  "futures-macro", | ||||||
|  |  "futures-sink", | ||||||
|  |  "futures-task", | ||||||
|  |  "memchr", | ||||||
|  |  "pin-project-lite", | ||||||
|  |  "pin-utils", | ||||||
|  |  "slab", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "hashbrown" | ||||||
|  | version = "0.11.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "ab5ef0d4909ef3724cc8cce6ccc8572c5c817592e9285f5464f8e86f8bd3726e" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "heck" | ||||||
|  | version = "0.3.3" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "6d621efb26863f0e9924c6ac577e8275e5e6b77455db64ffa6c65c904e9e132c" | ||||||
|  | dependencies = [ | ||||||
|  |  "unicode-segmentation", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "hermit-abi" | ||||||
|  | version = "0.1.19" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "indexmap" | ||||||
|  | version = "1.8.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "282a6247722caba404c065016bbfa522806e51714c34f5dfc3e4a3a46fcb4223" | ||||||
|  | dependencies = [ | ||||||
|  |  "autocfg", | ||||||
|  |  "hashbrown", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "instant" | ||||||
|  | version = "0.1.12" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "7a5bbe824c507c5da5956355e86a746d82e0e1464f65d862cc5e71da70e94b2c" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "libc" | ||||||
|  | version = "0.2.117" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e74d72e0f9b65b5b4ca49a346af3976df0f9c61d550727f349ecd559f251a26c" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "lock_api" | ||||||
|  | version = "0.4.6" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "88943dd7ef4a2e5a4bfa2753aaab3013e34ce2533d1996fb18ef591e315e2b3b" | ||||||
|  | dependencies = [ | ||||||
|  |  "scopeguard", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "log" | ||||||
|  | version = "0.4.14" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "51b9bbe6c47d51fc3e1a9b945965946b4c44142ab8792c50835a980d362c2710" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "mach" | ||||||
|  | version = "0.1.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "2fd13ee2dd61cc82833ba05ade5a30bb3d63f7ced605ef827063c63078302de9" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "mach" | ||||||
|  | version = "0.2.3" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "86dd2487cdfea56def77b88438a2c915fb45113c5319bfe7e14306ca4cd0b0e1" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "machine" | ||||||
|  | version = "0.0.1" | ||||||
|  | dependencies = [ | ||||||
|  |  "bit_field", | ||||||
|  |  "bitflags", | ||||||
|  |  "cfg-if", | ||||||
|  |  "cortex-a", | ||||||
|  |  "qemu-exit", | ||||||
|  |  "r0", | ||||||
|  |  "snafu", | ||||||
|  |  "tock-registers", | ||||||
|  |  "usize_conversions", | ||||||
|  |  "ux", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "memchr" | ||||||
|  | version = "2.4.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "308cc39be01b73d0d18f82a0e7b2a3df85245f84af96fdddc5d202d27e47b86a" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "memoffset" | ||||||
|  | version = "0.6.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "5aa361d4faea93603064a027415f07bd8e1d5c88c9fbf68bf56a285428fd79ce" | ||||||
|  | dependencies = [ | ||||||
|  |  "autocfg", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "mio" | ||||||
|  | version = "0.7.14" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "8067b404fe97c70829f082dec8bcf4f71225d7eaea1d8645349cb76fa06205cc" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  |  "log", | ||||||
|  |  "miow", | ||||||
|  |  "ntapi", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "mio-serial" | ||||||
|  | version = "5.0.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d6f47dcdec193b77e2166faa0e3be4ee04f324ce6780c54efe7af6c3ee7360e4" | ||||||
|  | dependencies = [ | ||||||
|  |  "log", | ||||||
|  |  "mio", | ||||||
|  |  "nix 0.22.3", | ||||||
|  |  "serialport", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "miow" | ||||||
|  | version = "0.3.7" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "b9f1c5b025cda876f66ef43a113f91ebc9f4ccef34843000e0adf6ebbab84e21" | ||||||
|  | dependencies = [ | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "nix" | ||||||
|  | version = "0.22.3" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e4916f159ed8e5de0082076562152a76b7a1f64a01fd9d1e0fea002c37624faf" | ||||||
|  | dependencies = [ | ||||||
|  |  "bitflags", | ||||||
|  |  "cc", | ||||||
|  |  "cfg-if", | ||||||
|  |  "libc", | ||||||
|  |  "memoffset", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "nix" | ||||||
|  | version = "0.23.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "9f866317acbd3a240710c63f065ffb1e4fd466259045ccb504130b7f668f35c6" | ||||||
|  | dependencies = [ | ||||||
|  |  "bitflags", | ||||||
|  |  "cc", | ||||||
|  |  "cfg-if", | ||||||
|  |  "libc", | ||||||
|  |  "memoffset", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "ntapi" | ||||||
|  | version = "0.3.7" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "c28774a7fd2fbb4f0babd8237ce554b73af68021b5f695a3cebd6c59bac0980f" | ||||||
|  | dependencies = [ | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "nucleus" | ||||||
|  | version = "0.0.1" | ||||||
|  | dependencies = [ | ||||||
|  |  "bit_field", | ||||||
|  |  "bitflags", | ||||||
|  |  "cfg-if", | ||||||
|  |  "cortex-a", | ||||||
|  |  "machine", | ||||||
|  |  "r0", | ||||||
|  |  "snafu", | ||||||
|  |  "tock-registers", | ||||||
|  |  "usize_conversions", | ||||||
|  |  "ux", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "num_cpus" | ||||||
|  | version = "1.13.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "19e64526ebdee182341572e50e9ad03965aa510cd94427a4549448f285e957a1" | ||||||
|  | dependencies = [ | ||||||
|  |  "hermit-abi", | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "once_cell" | ||||||
|  | version = "1.9.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "da32515d9f6e6e489d7bc9d84c71b060db7247dc035bbe44eac88cf87486d8d5" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "os_str_bytes" | ||||||
|  | version = "6.0.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "8e22443d1643a904602595ba1cd8f7d896afe56d26712531c5ff73a15b2fbf64" | ||||||
|  | dependencies = [ | ||||||
|  |  "memchr", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "parking_lot" | ||||||
|  | version = "0.11.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "7d17b78036a60663b797adeaee46f5c9dfebb86948d1255007a1d6be0271ff99" | ||||||
|  | dependencies = [ | ||||||
|  |  "instant", | ||||||
|  |  "lock_api", | ||||||
|  |  "parking_lot_core 0.8.5", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "parking_lot" | ||||||
|  | version = "0.12.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "87f5ec2493a61ac0506c0f4199f99070cbe83857b0337006a30f3e6719b8ef58" | ||||||
|  | dependencies = [ | ||||||
|  |  "lock_api", | ||||||
|  |  "parking_lot_core 0.9.1", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "parking_lot_core" | ||||||
|  | version = "0.8.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d76e8e1493bcac0d2766c42737f34458f1c8c50c0d23bcb24ea953affb273216" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  |  "instant", | ||||||
|  |  "libc", | ||||||
|  |  "redox_syscall", | ||||||
|  |  "smallvec", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "parking_lot_core" | ||||||
|  | version = "0.9.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "28141e0cc4143da2443301914478dc976a61ffdb3f043058310c70df2fed8954" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  |  "libc", | ||||||
|  |  "redox_syscall", | ||||||
|  |  "smallvec", | ||||||
|  |  "windows-sys", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "pin-project-lite" | ||||||
|  | version = "0.2.8" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e280fbe77cc62c91527259e9442153f4688736748d24660126286329742b4c6c" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "pin-utils" | ||||||
|  | version = "0.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "proc-macro2" | ||||||
|  | version = "1.0.36" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "c7342d5883fbccae1cc37a2353b09c87c9b0f3afd73f5fb9bba687a1f733b029" | ||||||
|  | dependencies = [ | ||||||
|  |  "unicode-xid", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "qemu-exit" | ||||||
|  | version = "3.0.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "9ff023245bfcc73fb890e1f8d5383825b3131cc920020a5c487d6f113dfc428a" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "quote" | ||||||
|  | version = "1.0.15" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "864d3e96a899863136fc6e99f3d7cae289dafe43bf2c5ac19b70df7210c0a145" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "r0" | ||||||
|  | version = "1.0.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "bd7a31eed1591dcbc95d92ad7161908e72f4677f8fabf2a32ca49b4237cbf211" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "redox_syscall" | ||||||
|  | version = "0.2.10" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "8383f39639269cde97d255a32bdb68c047337295414940c68bdd30c2e13203ff" | ||||||
|  | dependencies = [ | ||||||
|  |  "bitflags", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "regex" | ||||||
|  | version = "1.5.4" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d07a8629359eb56f1e2fb1652bb04212c072a87ba68546a04065d525673ac461" | ||||||
|  | dependencies = [ | ||||||
|  |  "aho-corasick", | ||||||
|  |  "memchr", | ||||||
|  |  "regex-syntax", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "regex-syntax" | ||||||
|  | version = "0.6.25" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "f497285884f3fcff424ffc933e56d7cbca511def0c9831a7f9b5f6153e3cc89b" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "scopeguard" | ||||||
|  | version = "1.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d29ab0c6d3fc0ee92fe66e2d99f700eab17a8d57d1c1d3b748380fb20baa78cd" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "seahash" | ||||||
|  | version = "4.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "1c107b6f4780854c8b126e228ea8869f4d7b71260f962fefb57b996b8959ba6b" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "serialport" | ||||||
|  | version = "4.0.1" | ||||||
|  | source = "git+https://github.com/metta-systems/serialport-rs?branch=macos-ENOTTY-fix#91ee1f740ee7d7c9506df8af082860a478656e18" | ||||||
|  | dependencies = [ | ||||||
|  |  "CoreFoundation-sys", | ||||||
|  |  "IOKit-sys", | ||||||
|  |  "bitflags", | ||||||
|  |  "cfg-if", | ||||||
|  |  "mach 0.2.3", | ||||||
|  |  "nix 0.23.1", | ||||||
|  |  "regex", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "signal-hook" | ||||||
|  | version = "0.3.13" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "647c97df271007dcea485bb74ffdb57f2e683f1306c854f468a0c244badabf2d" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  |  "signal-hook-registry", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "signal-hook-mio" | ||||||
|  | version = "0.2.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "29fd5867f1c4f2c5be079aee7a2adf1152ebb04a4bc4d341f504b7dece607ed4" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  |  "mio", | ||||||
|  |  "signal-hook", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "signal-hook-registry" | ||||||
|  | version = "1.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e51e73328dc4ac0c7ccbda3a494dfa03df1de2f46018127f60c693f2648455b0" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "slab" | ||||||
|  | version = "0.4.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "9def91fd1e018fe007022791f865d0ccc9b3a0d5001e01aabb8b40e46000afb5" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "smallvec" | ||||||
|  | version = "1.8.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "f2dd574626839106c320a323308629dcb1acfc96e32a8cba364ddc61ac23ee83" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "snafu" | ||||||
|  | version = "0.7.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "2eba135d2c579aa65364522eb78590cdf703176ef71ad4c32b00f58f7afb2df5" | ||||||
|  | dependencies = [ | ||||||
|  |  "doc-comment", | ||||||
|  |  "snafu-derive", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "snafu-derive" | ||||||
|  | version = "0.7.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "7a7fe9b0669ef117c5cabc5549638528f36771f058ff977d7689deb517833a75" | ||||||
|  | dependencies = [ | ||||||
|  |  "heck", | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "strsim" | ||||||
|  | version = "0.10.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "73473c0e59e6d5812c5dfe2a064a6444949f089e20eec9a2e5506596494e4623" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "syn" | ||||||
|  | version = "1.0.86" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "8a65b3f4ffa0092e9887669db0eae07941f023991ab58ea44da8fe8e2d511c6b" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "unicode-xid", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "termcolor" | ||||||
|  | version = "1.1.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "2dfed899f0eb03f32ee8c6a0aabdb8a7949659e3466561fc0adf54e26d88c5f4" | ||||||
|  | dependencies = [ | ||||||
|  |  "winapi-util", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "textwrap" | ||||||
|  | version = "0.14.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "0066c8d12af8b5acd21e00547c3797fde4e8677254a7ee429176ccebbe93dd80" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "tock-registers" | ||||||
|  | version = "0.7.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "4ee8fba06c1f4d0b396ef61a54530bb6b28f0dc61c38bc8bc5a5a48161e6282e" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "tokio" | ||||||
|  | version = "1.16.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "0c27a64b625de6d309e8c57716ba93021dccf1b3b5c97edd6d3dd2d2135afc0a" | ||||||
|  | dependencies = [ | ||||||
|  |  "bytes", | ||||||
|  |  "libc", | ||||||
|  |  "memchr", | ||||||
|  |  "mio", | ||||||
|  |  "num_cpus", | ||||||
|  |  "once_cell", | ||||||
|  |  "parking_lot 0.11.2", | ||||||
|  |  "pin-project-lite", | ||||||
|  |  "signal-hook-registry", | ||||||
|  |  "tokio-macros", | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "tokio-macros" | ||||||
|  | version = "1.7.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "b557f72f448c511a979e2564e55d74e6c4432fc96ff4f6241bc6bded342643b7" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "tokio-serial" | ||||||
|  | version = "5.4.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "c75b04fbb029a3f77272b3356453c97c5dbdebcf28aa248a34fd80a442f2dda1" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  |  "futures", | ||||||
|  |  "log", | ||||||
|  |  "mio-serial", | ||||||
|  |  "tokio", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "tokio-util" | ||||||
|  | version = "0.7.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "64910e1b9c1901aaf5375561e35b9c057d95ff41a44ede043a03e09279eabaf1" | ||||||
|  | dependencies = [ | ||||||
|  |  "bytes", | ||||||
|  |  "futures-core", | ||||||
|  |  "futures-sink", | ||||||
|  |  "log", | ||||||
|  |  "pin-project-lite", | ||||||
|  |  "tokio", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "unicode-segmentation" | ||||||
|  | version = "1.9.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "7e8820f5d777f6224dc4be3632222971ac30164d4a258d595640799554ebfd99" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "unicode-xid" | ||||||
|  | version = "0.2.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "8ccb82d61f80a663efe1f787a51b16b5a51e3314d6ac365b08639f52387b33f3" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "usize_conversions" | ||||||
|  | version = "0.2.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "f70329e2cbe45d6c97a5112daad40c34cd9a4e18edb5a2a18fefeb584d8d25e5" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "ux" | ||||||
|  | version = "0.1.3" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "88dfeb711b61ce620c0cb6fd9f8e3e678622f0c971da2a63c4b3e25e88ed012f" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "winapi" | ||||||
|  | version = "0.3.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419" | ||||||
|  | dependencies = [ | ||||||
|  |  "winapi-i686-pc-windows-gnu", | ||||||
|  |  "winapi-x86_64-pc-windows-gnu", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "winapi-i686-pc-windows-gnu" | ||||||
|  | version = "0.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "winapi-util" | ||||||
|  | version = "0.1.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "70ec6ce85bb158151cae5e5c87f95a8e97d2c0c4b001223f33a334e3ce5de178" | ||||||
|  | dependencies = [ | ||||||
|  |  "winapi", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "winapi-x86_64-pc-windows-gnu" | ||||||
|  | version = "0.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "windows-sys" | ||||||
|  | version = "0.32.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "3df6e476185f92a12c072be4a189a0210dcdcf512a1891d6dff9edb874deadc6" | ||||||
|  | dependencies = [ | ||||||
|  |  "windows_aarch64_msvc", | ||||||
|  |  "windows_i686_gnu", | ||||||
|  |  "windows_i686_msvc", | ||||||
|  |  "windows_x86_64_gnu", | ||||||
|  |  "windows_x86_64_msvc", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "windows_aarch64_msvc" | ||||||
|  | version = "0.32.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d8e92753b1c443191654ec532f14c199742964a061be25d77d7a96f09db20bf5" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "windows_i686_gnu" | ||||||
|  | version = "0.32.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "6a711c68811799e017b6038e0922cb27a5e2f43a2ddb609fe0b6f3eeda9de615" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "windows_i686_msvc" | ||||||
|  | version = "0.32.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "146c11bb1a02615db74680b32a68e2d61f553cc24c4eb5b4ca10311740e44172" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "windows_x86_64_gnu" | ||||||
|  | version = "0.32.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "c912b12f7454c6620635bbff3450962753834be2a594819bd5e945af18ec64bc" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "windows_x86_64_msvc" | ||||||
|  | version = "0.32.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "504a2476202769977a040c6364301a3f65d0cc9e3fb08600b2bda150a0488316" | ||||||
							
								
								
									
										60
									
								
								Cargo.toml
								
								
								
								
							
							
						
						
									
										60
									
								
								Cargo.toml
								
								
								
								
							|  | @ -1,51 +1,25 @@ | ||||||
| [package] | [workspace] | ||||||
| name = "vesper" | members = [ | ||||||
| version = "1.0.0" |     "nucleus", | ||||||
| authors = ["Berkus Decker <berkus+cargo@metta.systems>"] |     "bin/chainboot", | ||||||
| description = "Vesper exokernel" |     "bin/chainofcommand" | ||||||
| documentation = "https://docs.metta.systems/vesper" | ] | ||||||
| homepage = "https://github.com/metta-systems/vesper" |  | ||||||
| repository = "https://github.com/metta-systems/vesper" |  | ||||||
| readme = "README.md" |  | ||||||
| license = "BSL-1.0" |  | ||||||
| categories = ["no-std", "embedded", "os"] |  | ||||||
| publish = false |  | ||||||
| edition = "2018" |  | ||||||
| 
 | 
 | ||||||
| [package.metadata.cargo-xbuild] | [patch.crates-io] | ||||||
| memcpy = true | serialport = { git = "https://github.com/metta-systems/serialport-rs", branch = "macos-ENOTTY-fix" } | ||||||
| 
 |  | ||||||
| [package.metadata.bootimage] |  | ||||||
| default-target = "targets/aarch64-vesper-metta.json" |  | ||||||
| 
 |  | ||||||
| [features] |  | ||||||
| unstable = [] |  | ||||||
| realtime = [] |  | ||||||
| noserial = [] |  | ||||||
| jlink = [] #'jlink_rtt' |  | ||||||
| 
 |  | ||||||
| #[lib] |  | ||||||
| #name = "nucleus" |  | ||||||
| #path = "src/lib.rs" |  | ||||||
| #crate-type = ["staticlib"] |  | ||||||
| 
 |  | ||||||
| [dependencies] |  | ||||||
| r0 = "0.2.2" |  | ||||||
| rlibc = "1.0.0" |  | ||||||
| panic-abort = "0.3.1" |  | ||||||
| bitflags = "1.0.1" |  | ||||||
| register = "0.3.2" |  | ||||||
| cortex-a = "2.4" |  | ||||||
| #embedded-serial = "0.5.0" |  | ||||||
| # jlink_rtt = { version = "0.1.0", optional = true } |  | ||||||
| static_assertions = { version = "0.3.1", features = ["nightly"] } |  | ||||||
| 
 | 
 | ||||||
| [profile.dev] | [profile.dev] | ||||||
| panic = "abort" # @todo try panic_rtt when feature jlink | # See https://github.com/rust-lang/cargo/issues/7359 about why panic=abort is not working here. | ||||||
|  | # It is still defined in the target JSON so not stricly necessary to specify it here anyway. | ||||||
|  | #panic = "abort" # @todo try panic_rtt with feature jlink | ||||||
|  | opt-level = 's' # otherwise boot code will be mangled | ||||||
| 
 | 
 | ||||||
| [profile.release] | [profile.release] | ||||||
| panic = "abort" | panic = "abort" | ||||||
|  | opt-level = 's' | ||||||
| debug = true | debug = true | ||||||
| lto = true | lto = true | ||||||
| # https://github.com/rust-lang/cargo/pull/6564#issuecomment-457971499 | 
 | ||||||
| incremental = false | [profile.test] | ||||||
|  | opt-level = 's' | ||||||
|  | debug = true | ||||||
|  |  | ||||||
|  | @ -0,0 +1,111 @@ | ||||||
|  | _default: | ||||||
|  |     @just --list | ||||||
|  | 
 | ||||||
|  | # Build default hw kernel and run chainofcommand to boot this kernel onto the board
 | ||||||
|  | boot: chainofcommand | ||||||
|  |     cargo make chainboot | ||||||
|  | 
 | ||||||
|  | # Build and run kernel in QEMU with serial port emulation
 | ||||||
|  | zellij: | ||||||
|  |     cargo make zellij-nucleus | ||||||
|  |     zellij --layout-path emulation/layout.zellij | ||||||
|  | 
 | ||||||
|  | # Build and run chainboot in QEMU with serial port emulation
 | ||||||
|  | zellij-cb: | ||||||
|  |     # Connect to it via chainofcommand to load an actual kernel | ||||||
|  |     # TODO: actually run chainofcommand in a zellij session too | ||||||
|  |     cargo make zellij-cb | ||||||
|  |     zellij --layout-path emulation/layout.zellij | ||||||
|  | 
 | ||||||
|  | # Build chainofcommand serial loader
 | ||||||
|  | chainofcommand: | ||||||
|  |     cd bin/chainofcommand | ||||||
|  |     cargo make build | ||||||
|  | 
 | ||||||
|  | # Build and run kernel in QEMU
 | ||||||
|  | qemu: | ||||||
|  |     cargo make qemu | ||||||
|  | 
 | ||||||
|  | # Build and run kernel in QEMU with GDB port enabled
 | ||||||
|  | qemu-gdb: | ||||||
|  |     cargo make qemu-gdb | ||||||
|  | 
 | ||||||
|  | # Build and run chainboot in QEMU
 | ||||||
|  | qemu-cb: | ||||||
|  |     # Connect to it via chainofcommand to load an actual kernel | ||||||
|  |     cargo make qemu-cb | ||||||
|  | 
 | ||||||
|  | # Build and write kernel to an SD Card
 | ||||||
|  | device: | ||||||
|  |     cargo make sdcard | ||||||
|  | 
 | ||||||
|  | # Build and write kernel to an SD Card, then eject the SD Card volume
 | ||||||
|  | device-eject: | ||||||
|  |     cargo make sdeject | ||||||
|  | 
 | ||||||
|  | # Build and write chainboot to an SD Card, then eject the SD Card volume
 | ||||||
|  | cb-eject: | ||||||
|  |     cd bin/chainboot | ||||||
|  |     cargo make cb-eject | ||||||
|  | 
 | ||||||
|  | # Build default hw kernel
 | ||||||
|  | build: | ||||||
|  |     cargo make build | ||||||
|  |     cargo make kernel-binary | ||||||
|  | 
 | ||||||
|  | # Clean project
 | ||||||
|  | clean: | ||||||
|  |     cargo make clean | ||||||
|  | 
 | ||||||
|  | # Run clippy checks
 | ||||||
|  | clippy: | ||||||
|  |     # TODO: use cargo-hack | ||||||
|  |     cargo make clippy | ||||||
|  |     env CLIPPY_FEATURES=noserial cargo make clippy | ||||||
|  |     env CLIPPY_FEATURES=qemu cargo make clippy | ||||||
|  |     env CLIPPY_FEATURES=noserial,qemu cargo make clippy | ||||||
|  |     env CLIPPY_FEATURES=jtag cargo make clippy | ||||||
|  |     env CLIPPY_FEATURES=noserial,jtag cargo make clippy | ||||||
|  | 
 | ||||||
|  | # Run tests in QEMU
 | ||||||
|  | test: | ||||||
|  |     cargo make test | ||||||
|  | 
 | ||||||
|  | alias disasm := hopper | ||||||
|  | 
 | ||||||
|  | # Build and disassemble kernel
 | ||||||
|  | hopper: | ||||||
|  |     cargo make hopper | ||||||
|  | 
 | ||||||
|  | alias ocd := openocd | ||||||
|  | 
 | ||||||
|  | # Start openocd (by default connected via JTAG to a target device)
 | ||||||
|  | openocd: | ||||||
|  |     cargo make openocd | ||||||
|  | 
 | ||||||
|  | # Build and run kernel in GDB using openocd or QEMU as target (gdb port 5555)
 | ||||||
|  | gdb: | ||||||
|  |     cargo make gdb | ||||||
|  | 
 | ||||||
|  | # Build and run chainboot in GDB using openocd or QEMU as target (gdb port 5555)
 | ||||||
|  | gdb-cb: | ||||||
|  |     cargo make gdb-cb | ||||||
|  | 
 | ||||||
|  | # Build and print all symbols in the kernel
 | ||||||
|  | nm: | ||||||
|  |     cargo make nm | ||||||
|  | 
 | ||||||
|  | # Check formatting
 | ||||||
|  | fmt-check: | ||||||
|  |     cargo fmt -- --check | ||||||
|  | 
 | ||||||
|  | # Run `cargo expand` on nucleus
 | ||||||
|  | expand: | ||||||
|  |     cargo make expand -- nucleus | ||||||
|  | 
 | ||||||
|  | # Generate and open documentation
 | ||||||
|  | doc: | ||||||
|  |     cargo make docs-flow | ||||||
|  | 
 | ||||||
|  | # Run CI tasks
 | ||||||
|  | ci: clean build test clippy fmt-check | ||||||
|  | @ -0,0 +1,55 @@ | ||||||
|  | # Blue Oak Model License | ||||||
|  | 
 | ||||||
|  | Version 1.0.0 | ||||||
|  | 
 | ||||||
|  | ## Purpose | ||||||
|  | 
 | ||||||
|  | This license gives everyone as much permission to work with | ||||||
|  | this software as possible, while protecting contributors | ||||||
|  | from liability. | ||||||
|  | 
 | ||||||
|  | ## Acceptance | ||||||
|  | 
 | ||||||
|  | In order to receive this license, you must agree to its | ||||||
|  | rules.  The rules of this license are both obligations | ||||||
|  | under that agreement and conditions to your license. | ||||||
|  | You must not do anything with this software that triggers | ||||||
|  | a rule that you cannot or will not follow. | ||||||
|  | 
 | ||||||
|  | ## Copyright | ||||||
|  | 
 | ||||||
|  | Each contributor licenses you to do everything with this | ||||||
|  | software that would otherwise infringe that contributor's | ||||||
|  | copyright in it. | ||||||
|  | 
 | ||||||
|  | ## Notices | ||||||
|  | 
 | ||||||
|  | You must ensure that everyone who gets a copy of | ||||||
|  | any part of this software from you, with or without | ||||||
|  | changes, also gets the text of this license or a link to | ||||||
|  | <https://blueoakcouncil.org/license/1.0.0>. | ||||||
|  | 
 | ||||||
|  | ## Excuse | ||||||
|  | 
 | ||||||
|  | If anyone notifies you in writing that you have not | ||||||
|  | complied with [Notices](#notices), you can keep your | ||||||
|  | license by taking all practical steps to comply within 30 | ||||||
|  | days after the notice.  If you do not do so, your license | ||||||
|  | ends immediately. | ||||||
|  | 
 | ||||||
|  | ## Patent | ||||||
|  | 
 | ||||||
|  | Each contributor licenses you to do everything with this | ||||||
|  | software that would otherwise infringe any patent claims | ||||||
|  | they can license or become able to license. | ||||||
|  | 
 | ||||||
|  | ## Reliability | ||||||
|  | 
 | ||||||
|  | No contributor can revoke this license. | ||||||
|  | 
 | ||||||
|  | ## No Liability | ||||||
|  | 
 | ||||||
|  | ***As far as the law allows, this software comes as is, | ||||||
|  | without any warranty or condition, and no contributor | ||||||
|  | will be liable to anyone for any damages related to this | ||||||
|  | software or this license, under any kind of legal claim.*** | ||||||
|  | @ -1,23 +0,0 @@ | ||||||
| Boost Software License - Version 1.0 - August 17th, 2003 |  | ||||||
| 
 |  | ||||||
| Permission is hereby granted, free of charge, to any person or organization |  | ||||||
| obtaining a copy of the software and accompanying documentation covered by |  | ||||||
| this license (the "Software") to use, reproduce, display, distribute, |  | ||||||
| execute, and transmit the Software, and to prepare derivative works of the |  | ||||||
| Software, and to permit third-parties to whom the Software is furnished to |  | ||||||
| do so, all subject to the following: |  | ||||||
| 
 |  | ||||||
| The copyright notices in the Software and this entire statement, including |  | ||||||
| the above license grant, this restriction and the following disclaimer, |  | ||||||
| must be included in all copies of the Software, in whole or in part, and |  | ||||||
| all derivative works of the Software, unless such copies or derivative |  | ||||||
| works are solely in the form of machine-executable object code generated by |  | ||||||
| a source language processor. |  | ||||||
| 
 |  | ||||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |  | ||||||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |  | ||||||
| FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT |  | ||||||
| SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE |  | ||||||
| FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE, |  | ||||||
| ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |  | ||||||
| DEALINGS IN THE SOFTWARE. |  | ||||||
							
								
								
									
										104
									
								
								Makefile
								
								
								
								
							
							
						
						
									
										104
									
								
								Makefile
								
								
								
								
							|  | @ -1,104 +0,0 @@ | ||||||
| #
 |  | ||||||
| # MIT License
 |  | ||||||
| #
 |  | ||||||
| # Copyright (c) 2018 Andre Richter <andre.o.richter@gmail.com>
 |  | ||||||
| # Copyright (c) 2019 Berkus Decker <berkus+github@metta.systems>
 |  | ||||||
| #
 |  | ||||||
| # Permission is hereby granted, free of charge, to any person obtaining a copy
 |  | ||||||
| # of this software and associated documentation files (the "Software"), to deal
 |  | ||||||
| # in the Software without restriction, including without limitation the rights
 |  | ||||||
| # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 |  | ||||||
| # copies of the Software, and to permit persons to whom the Software is
 |  | ||||||
| # furnished to do so, subject to the following conditions:
 |  | ||||||
| #
 |  | ||||||
| # The above copyright notice and this permission notice shall be included in all
 |  | ||||||
| # copies or substantial portions of the Software.
 |  | ||||||
| #
 |  | ||||||
| # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 |  | ||||||
| # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 |  | ||||||
| # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 |  | ||||||
| # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 |  | ||||||
| # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 |  | ||||||
| # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 |  | ||||||
| # SOFTWARE.
 |  | ||||||
| #
 |  | ||||||
| 
 |  | ||||||
| TARGET = aarch64-vesper-metta |  | ||||||
| TARGET_JSON = targets/$(TARGET).json |  | ||||||
| 
 |  | ||||||
| SOURCES = $(shell find src -name "*.rs" -o -name "*.S") $(shell find linker -name "*.ld") |  | ||||||
| 
 |  | ||||||
| DEVICE_FEATURES = --features "noserial" |  | ||||||
| QEMU_FEATURES = |  | ||||||
| 
 |  | ||||||
| OBJCOPY = cargo objcopy -- |  | ||||||
| OBJCOPY_PARAMS = --strip-all -O binary |  | ||||||
| 
 |  | ||||||
| UTILS_CONTAINER = andrerichter/raspi3-utils |  | ||||||
| DOCKER_CMD = docker run -it --rm -v $(shell pwd):/work -w /work -p 5900:5900 |  | ||||||
| QEMU_CMD = qemu-system-aarch64 |  | ||||||
| 
 |  | ||||||
| # -d in_asm,unimp,int -S
 |  | ||||||
| QEMU_OPTS = -M raspi3 -d int |  | ||||||
| QEMU_SERIAL = -serial null -serial stdio |  | ||||||
| QEMU = /usr/local/Cellar/qemu/HEAD-3365de01b5-custom/bin/qemu-system-aarch64 |  | ||||||
| 
 |  | ||||||
| GDB = /usr/local/opt/gdb-8.2.1-aarhc64/bin/aarch64-linux-elf-gdb |  | ||||||
| 
 |  | ||||||
| OPENOCD = /usr/local/openocd-aeb7b327-rtt/bin/openocd |  | ||||||
| 
 |  | ||||||
| .PHONY: all qemu clippy clean objdump nm |  | ||||||
| 
 |  | ||||||
| all: kernel8.img |  | ||||||
| 
 |  | ||||||
| target/$(TARGET)/release/vesper: $(SOURCES) |  | ||||||
| 	cargo xbuild --target=$(TARGET_JSON) --release --features="jlink" |  | ||||||
| 
 |  | ||||||
| kernel8.img: target/$(TARGET)/release/vesper $(SOURCES) |  | ||||||
| 	cp $< ./kernel8 |  | ||||||
| 	$(OBJCOPY) $(OBJCOPY_PARAMS) $< kernel8.img |  | ||||||
| 
 |  | ||||||
| docker_qemu: all |  | ||||||
| 	$(DOCKER_CMD) $(UTILS_CONTAINER) $(QEMU_CMD) $(QEMU_OPTS) -serial stdio -kernel kernel8.img |  | ||||||
| 
 |  | ||||||
| qemu: all |  | ||||||
| 	$(QEMU) $(QEMU_OPTS) $(QEMU_SERIAL) -kernel kernel8.img |  | ||||||
| 
 |  | ||||||
| sdcard: all |  | ||||||
| 	cp kernel8.img /Volumes/BOOT/ |  | ||||||
| 
 |  | ||||||
| sdeject: sdcard |  | ||||||
| 	diskutil unmount /Volumes/BOOT/ |  | ||||||
| 
 |  | ||||||
| clippy: |  | ||||||
| 	cargo xclippy --target=$(TARGET_JSON) |  | ||||||
| 
 |  | ||||||
| clean: |  | ||||||
| 	cargo clean |  | ||||||
| 
 |  | ||||||
| objdump: |  | ||||||
| 	cargo objdump --target $(TARGET_JSON) -- -disassemble -print-imm-hex kernel8 |  | ||||||
| 
 |  | ||||||
| nm: |  | ||||||
| 	cargo nm --target $(TARGET_JSON) -- kernel8 | sort |  | ||||||
| 
 |  | ||||||
| hopper: all |  | ||||||
| 	hopperv4 -l RAW --base-address 0x80000 --entrypoint 0x80000 --file-offset 0 --aarch64 -e kernel8.img |  | ||||||
| 
 |  | ||||||
| openocd: |  | ||||||
| 	$(OPENOCD) -f interface/jlink.cfg -f ./doc/rpi3_jlink_suse.cfg |  | ||||||
| 
 |  | ||||||
| openocd_naotako: |  | ||||||
| 	$(OPENOCD) -f interface/jlink.cfg -f ./doc/rpi3_jlink_naotako.cfg |  | ||||||
| 
 |  | ||||||
| gdb: kernel8.img |  | ||||||
| 	make nm | grep _SEGGER_RTT | awk '{print $$1}' | ./make-gdb-connect.sh |  | ||||||
| 	env RUST_GDB=$(GDB) rust-gdb -x gdb-connect kernel8 |  | ||||||
| 
 |  | ||||||
| gdbdash: kernel8.img |  | ||||||
| 	make nm | grep _SEGGER_RTT | awk '{print $$1}' | ./make-gdb-connect.sh |  | ||||||
| 	env RUST_GDB=$(GDB) rust-gdb -x gdb-connect -x ~/.gdbinit_dashboard kernel8 |  | ||||||
| 
 |  | ||||||
| gdbgui: |  | ||||||
| 	gdbgui -g $(GDB) --gdb-args='--init-eval-command="set startup-with-shell off"' |  | ||||||
| 
 |  | ||||||
|  | @ -0,0 +1,199 @@ | ||||||
|  | #
 | ||||||
|  | # SPDX-License-Identifier: BlueOak-1.0.0
 | ||||||
|  | #
 | ||||||
|  | # Copyright (c) Berkus Decker <berkus+vesper@metta.systems>
 | ||||||
|  | #
 | ||||||
|  | [config] | ||||||
|  | min_version = "0.32.0" | ||||||
|  | default_to_workspace = true | ||||||
|  | 
 | ||||||
|  | [env] | ||||||
|  | DEFAULT_TARGET = "aarch64-vesper-metta" | ||||||
|  | 
 | ||||||
|  | #
 | ||||||
|  | # === User-configurable ===
 | ||||||
|  | #
 | ||||||
|  | 
 | ||||||
|  | # Pass TARGET env var if it does not match the default target above.
 | ||||||
|  | TARGET = { value = "${DEFAULT_TARGET}", condition = { env_not_set = ["TARGET"] } } | ||||||
|  | 
 | ||||||
|  | # Name of the target board "rpi3" or "rpi4"
 | ||||||
|  | TARGET_BOARD = { value = "rpi4", condition = { env_not_set = ["TARGET_BOARD"] } } | ||||||
|  | # Name of the DTB file for target board configuration, use bcm2710-rpi-3-b-plus.dtb for RasPi3B+
 | ||||||
|  | TARGET_DTB = { value = "${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/targets/bcm2711-rpi-4-b.dtb", condition = { env_not_set = ["TARGET_DTB"] } } | ||||||
|  | 
 | ||||||
|  | # AArch64 QEMU binary
 | ||||||
|  | QEMU = { value = "qemu-system-aarch64", condition = { env_not_set = ["QEMU"] } } | ||||||
|  | # QEMU machine type, defaults to raspi3b but CI runners override it due to ancient QEMU versions they use.
 | ||||||
|  | QEMU_MACHINE = { value = "raspi3b", condition = { env_not_set = ["QEMU_MACHINE"] } } | ||||||
|  | 
 | ||||||
|  | # An aarch64-enabled GDB
 | ||||||
|  | GDB = { value = "/usr/local/opt/gdb/HEAD-a2c58332-aarch64/bin/aarch64-unknown-elf-gdb", condition = { env_not_set = ["GDB"] } } | ||||||
|  | 
 | ||||||
|  | # OpenOCD with JLink support
 | ||||||
|  | # (RTT patch from http://openocd.zylin.com/#/c/4055/11 has already been merged into main line)
 | ||||||
|  | OPENOCD = { value = "/usr/local/opt/openocd/4d6519593-rtt/bin/openocd", condition = { env_not_set = ["OPENOCD"] } } | ||||||
|  | 
 | ||||||
|  | # Mounted sdcard partition path
 | ||||||
|  | VOLUME = { value = "/Volumes/BOOT", condition = { env_not_set = ["VOLUME"] } } | ||||||
|  | 
 | ||||||
|  | #
 | ||||||
|  | # === Automatic ===
 | ||||||
|  | #
 | ||||||
|  | CARGO_MAKE_EXTEND_WORKSPACE_MAKEFILE = true | ||||||
|  | 
 | ||||||
|  | RUST_LIBS = "-Z build-std=compiler_builtins,core,alloc -Z build-std-features=compiler-builtins-mem" | ||||||
|  | TARGET_JSON = "${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/targets/${TARGET}.json" | ||||||
|  | PLATFORM_TARGET="--target=${TARGET_JSON} --features=${TARGET_FEATURES} ${RUST_LIBS}" | ||||||
|  | 
 | ||||||
|  | DEVICE_FEATURES = "noserial" | ||||||
|  | QEMU_FEATURES = "qemu,rpi3" | ||||||
|  | 
 | ||||||
|  | OBJCOPY = "rust-objcopy" # Part of `cargo objcopy` in cargo-binutils | ||||||
|  | OBJCOPY_PARAMS = "--strip-all -O binary" | ||||||
|  | NM = "rust-nm" # Part of `cargo nm` in cargo-binutils | ||||||
|  | 
 | ||||||
|  | UTILS_CONTAINER = "andrerichter/raspi3-utils" | ||||||
|  | DOCKER_CMD = "docker run -it --rm -v ${PWD}:/work -w /work -p 5900:5900" | ||||||
|  | QEMU_CONTAINER_CMD = "qemu-system-aarch64" | ||||||
|  | 
 | ||||||
|  | #
 | ||||||
|  | # Could additionally use -nographic to disable GUI -- this shall be useful for automated tests.
 | ||||||
|  | #
 | ||||||
|  | # QEMU has renamed the RasPi machines since version 6.2.0, use just `raspi3` for previous versions.
 | ||||||
|  | QEMU_OPTS = "-M ${QEMU_MACHINE} -d int -semihosting" | ||||||
|  | QEMU_DISASM_OPTS = "-d in_asm,unimp,int" | ||||||
|  | QEMU_SERIAL_OPTS = "-serial pty -serial stdio" | ||||||
|  | QEMU_TESTS_OPTS = "-nographic" | ||||||
|  | # For gdb connection:
 | ||||||
|  | # - if this is set, MUST have gdb attached for SYS_WRITE0 to work, otherwise QEMU will crash.
 | ||||||
|  | # - port 5555 used to match JLink configuration, so we can reuse the same GDB command for both QEMU and JTAG.
 | ||||||
|  | QEMU_GDB_OPTS = "-gdb tcp::5555 -S" | ||||||
|  | 
 | ||||||
|  | GDB_CONNECT_FILE = "${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/target/${TARGET}/gdb-connect" | ||||||
|  | 
 | ||||||
|  | KERNEL_ELF = "${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/target/${TARGET}/release/nucleus" | ||||||
|  | KERNEL_BIN = "${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/target/nucleus.bin" | ||||||
|  | 
 | ||||||
|  | CHAINBOOT_SERIAL = "/dev/tty.SLAB_USBtoUART" | ||||||
|  | CHAINBOOT_BAUD = 115200 | ||||||
|  | 
 | ||||||
|  | [tasks.default] | ||||||
|  | alias = "all" | ||||||
|  | 
 | ||||||
|  | [tasks.all] | ||||||
|  | dependencies = ["kernel-binary"] | ||||||
|  | 
 | ||||||
|  | [tasks.modules] | ||||||
|  | command = "cargo" | ||||||
|  | args = ["modules", "tree"] | ||||||
|  | 
 | ||||||
|  | [tasks.build] | ||||||
|  | env = { "TARGET_FEATURES" = "${TARGET_BOARD}" } | ||||||
|  | command = "cargo" | ||||||
|  | args = ["build", "@@split(PLATFORM_TARGET, )", "--release"] | ||||||
|  | 
 | ||||||
|  | [tasks.build-qemu] | ||||||
|  | env = { "TARGET_FEATURES" = "${QEMU_FEATURES}" } | ||||||
|  | command = "cargo" | ||||||
|  | args = ["build", "@@split(PLATFORM_TARGET, )", "--release"] | ||||||
|  | 
 | ||||||
|  | [tasks.qemu-runner] | ||||||
|  | dependencies = ["build-qemu", "kernel-binary"] | ||||||
|  | env = { "TARGET_FEATURES" = "${QEMU_FEATURES}" } | ||||||
|  | script = [ | ||||||
|  |     "echo Run QEMU ${QEMU_OPTS} ${QEMU_RUNNER_OPTS} with ${KERNEL_BIN}", | ||||||
|  |     "${QEMU} ${QEMU_OPTS} ${QEMU_RUNNER_OPTS} -dtb ${TARGET_DTB} -kernel ${KERNEL_BIN}" | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [tasks.expand] | ||||||
|  | env = { "TARGET_FEATURES" = "" } | ||||||
|  | command = "cargo" | ||||||
|  | args = ["expand", "@@split(PLATFORM_TARGET, )", "--release"] | ||||||
|  | 
 | ||||||
|  | [tasks.test] | ||||||
|  | env = { "TARGET_FEATURES" = "${QEMU_FEATURES}" } | ||||||
|  | command = "cargo" | ||||||
|  | args = ["test", "@@split(PLATFORM_TARGET, )"] | ||||||
|  | 
 | ||||||
|  | [tasks.docs] | ||||||
|  | env = { "TARGET_FEATURES" = "" } | ||||||
|  | command = "cargo" | ||||||
|  | args = ["doc", "--open", "--no-deps", "@@split(PLATFORM_TARGET, )"] | ||||||
|  | 
 | ||||||
|  | [tasks.clippy] | ||||||
|  | env = { "TARGET_FEATURES" = "rpi3", "CLIPPY_FEATURES" = { value = "--features=${CLIPPY_FEATURES}", condition = { env_set = ["CLIPPY_FEATURES"] } } } | ||||||
|  | command = "cargo" | ||||||
|  | args = ["clippy", "@@split(PLATFORM_TARGET, )", "@@remove-empty(CLIPPY_FEATURES)", "--", "-D", "warnings"] | ||||||
|  | 
 | ||||||
|  | # These tasks are written in cargo-make's own script to make it portable across platforms (no `basename` on Windows)
 | ||||||
|  | [tasks.custom-binary] | ||||||
|  | env = { "BINARY_FILE" = "${BINARY_FILE}" } | ||||||
|  | script_runner = "@duckscript" | ||||||
|  | script = [ | ||||||
|  | ''' | ||||||
|  |     binaryFile = basename ${BINARY_FILE} | ||||||
|  |     outElf = set ${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/target/${binaryFile}.elf | ||||||
|  |     outBin = set ${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/target/${binaryFile}.bin | ||||||
|  |     cp ${BINARY_FILE} ${outElf} | ||||||
|  |     exec --fail-on-error ${OBJCOPY} %{OBJCOPY_PARAMS} ${BINARY_FILE} ${outBin} | ||||||
|  |     echo Copied ${binaryFile} to ${outElf} | ||||||
|  |     echo Converted ${binaryFile} to ${outBin} | ||||||
|  | ''' | ||||||
|  | ] | ||||||
|  | install_crate = { crate_name = "cargo-binutils", binary = "rust-objcopy", test_arg = ["--help"] } | ||||||
|  | 
 | ||||||
|  | [tasks.test-binary] | ||||||
|  | env = { "BINARY_FILE" = "${CARGO_MAKE_TASK_ARGS}" } | ||||||
|  | run_task = "custom-binary" | ||||||
|  | 
 | ||||||
|  | [tasks.test-runner] | ||||||
|  | dependencies = ["test-binary"] | ||||||
|  | script_runner = "@duckscript" | ||||||
|  | script = [ | ||||||
|  | ''' | ||||||
|  |     binaryFile = basename ${CARGO_MAKE_TASK_ARGS} | ||||||
|  |     exec --fail-on-error ${QEMU} %{QEMU_OPTS} %{QEMU_TESTS_OPTS} -dtb ${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/targets/bcm2710-rpi-3-b-plus.dtb -kernel ${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/target/${binaryFile}.bin | ||||||
|  | ''' | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [tasks.gdb-config] | ||||||
|  | script_runner = "@duckscript" | ||||||
|  | script = [ | ||||||
|  | ''' | ||||||
|  |     writefile ${GDB_CONNECT_FILE} "target extended-remote :5555\n" | ||||||
|  |     appendfile ${GDB_CONNECT_FILE} "break 0x80000\n" | ||||||
|  | ''' | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [tasks.zellij-config] | ||||||
|  | dependencies = ["build-qemu", "kernel-binary"] | ||||||
|  | script_runner = "@duckscript" | ||||||
|  | env = { "ZELLIJ_CONFIG_FILE" = "${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/emulation/zellij-config.sh" } | ||||||
|  | script = [ | ||||||
|  | ''' | ||||||
|  |     writefile ${ZELLIJ_CONFIG_FILE} "QEMU=${QEMU}\n" | ||||||
|  |     appendfile ${ZELLIJ_CONFIG_FILE} "QEMU_OPTS=\"${QEMU_OPTS}\"\n" | ||||||
|  |     appendfile ${ZELLIJ_CONFIG_FILE} "QEMU_RUNNER_OPTS=${QEMU_RUNNER_OPTS}\n" | ||||||
|  |     appendfile ${ZELLIJ_CONFIG_FILE} "CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY=${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}\n" | ||||||
|  |     appendfile ${ZELLIJ_CONFIG_FILE} "TARGET_DTB=${TARGET_DTB}\n" | ||||||
|  |     appendfile ${ZELLIJ_CONFIG_FILE} "KERNEL_BIN=${KERNEL_BIN}\n" | ||||||
|  | ''' | ||||||
|  | ] | ||||||
|  | install_crate = { crate_name = "zellij", binary = "zellij", test_arg = ["--help"] } | ||||||
|  | 
 | ||||||
|  | [tasks.openocd] | ||||||
|  | script = [ | ||||||
|  |     "${OPENOCD} -f interface/jlink.cfg -f ../ocd/${TARGET_BOARD}_target.cfg" | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [tasks.sdeject] | ||||||
|  | dependencies = ["sdcard"] | ||||||
|  | script = [ | ||||||
|  |     "diskutil unmount ${VOLUME}" | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [tasks.chainboot] | ||||||
|  | dependencies = ["build", "kernel-binary"] | ||||||
|  | command = "echo" | ||||||
|  | args = ["\n***===***\n", "Run the following command in your terminal:\n", "  ${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/target/debug/chainofcommand ${CHAINBOOT_SERIAL} ${CHAINBOOT_BAUD} --kernel ${KERNEL_BIN}\n", "***===***\n\n"] | ||||||
							
								
								
									
										106
									
								
								README.md
								
								
								
								
							
							
						
						
									
										106
									
								
								README.md
								
								
								
								
							|  | @ -2,7 +2,7 @@ | ||||||
| 
 | 
 | ||||||
| ## About kernel | ## About kernel | ||||||
| 
 | 
 | ||||||
| Vesper is a capability-based single-address-space exokernel, it tries to remain small and secure. To achieve this, kernel functionality is extremely limited - it provides only address space isolation and IPC, after bootup kernel does not allocate any memory itself. | Vesper is a capability-based single-address-space nanokernel, it tries to remain small and secure. To achieve this, kernel functionality is extremely limited - it provides only address space isolation and IPC, after bootup kernel does not allocate any memory itself. | ||||||
| 
 | 
 | ||||||
| Exokernel's distinctive trait is that it provides mechanisms but not policies. Vesper tries to move as many policy decisions as possible to the library OS. | Exokernel's distinctive trait is that it provides mechanisms but not policies. Vesper tries to move as many policy decisions as possible to the library OS. | ||||||
| 
 | 
 | ||||||
|  | @ -30,22 +30,116 @@ Vesper has been influenced by the kernels in L4 family, notably seL4. Fawn and N | ||||||
| 
 | 
 | ||||||
| ## Build instructions | ## Build instructions | ||||||
| 
 | 
 | ||||||
| Use rustc nightly 2018-04-01 or later because of [bugs fixed](https://github.com/rust-lang/rust/issues/48884). | Use at least rustc nightly 2020-09-30 with cargo nightly of the same or later date. It adds support for `cargo build --build-std` feature (since 2020-07-15) and support for compiler_builtins memory operations ([since 2020-09-30](https://github.com/rust-lang/rust/pull/77284)). | ||||||
|  | 
 | ||||||
|  | * Install tools: `cargo install just cargo-make`. | ||||||
|  | * Install qemu (at least version 4.1.1): `brew install qemu`. | ||||||
|  | * Optionally install OpenOCD with [RTT patches](http://openocd.zylin.com/#/c/4055/11). | ||||||
|  | * Install aarch64 gdb. | ||||||
|  | 
 | ||||||
|  | You can override invoked `qemu`, `openocd` and `gdb` by specifying full paths to them as env variables `QEMU`, `OPENOCD` and `GDB`, respectively. | ||||||
|  | 
 | ||||||
|  | You can override the name of mounted sdcard volume by specifying env variable `VOLUME` (it defaults to `/Volumes/BOOT`). | ||||||
|  | 
 | ||||||
|  | ### To build kernel and run it in QEMU emulator | ||||||
| 
 | 
 | ||||||
| ``` | ``` | ||||||
| make qemu | just qemu | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| Will run `cargo xbuild` to create kernel and run it in qemu emulator. | ### To build kernel for Raspberry Pi and copy it to the mounted SDCard | ||||||
| 
 | 
 | ||||||
| ``` | ``` | ||||||
| make device | just device | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| Will build kernel, and copy it to sdcard at /Volumes/BOOT/ | On the device boot SD card you'll need a configuration file instructing RasPi to launch in 64-bit mode. | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | # config.txt on RPi3 | ||||||
|  | arm_64bit=1 | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### To run tests (tests require QEMU) | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | just test | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### To launch JTAG connected JLink probe | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | just ocd | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### To launch GDB and load kernel binary into it | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | just gdb | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | If you launch OpenOCD or QEMU before, then gdb shall connect to it and allow you to load the kernel binary directly into memory. Type `load` in gdb to do that. | ||||||
|  | 
 | ||||||
|  | ### To see kernel symbols and their values | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | just nm | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### To see kernel disassembly | ||||||
|  | 
 | ||||||
|  | You need to have [Hopper](https://hopperapp.com) and hopperv4 cli helper installed. | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | just disasm | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### To see other available commands | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | just | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | It will list all just commands with their short descriptions. | ||||||
|  | 
 | ||||||
|  | ## Development flow | ||||||
|  | 
 | ||||||
|  | `mainline`, `develop` and `released` branches: | ||||||
|  | 
 | ||||||
|  | - `feature branches` are fluid development lines which may be discarded or merged into `develop`. Feature branches must be either merged or fast-forward merged ("landed") into develop. Squashing history during merge is not permitted - commits must be sorted and squashed as necessary before merge. | ||||||
|  | - `develop` is currenly developed changes. History is recommended to be immutable, however mutations are possible in some cases. Feature branches are merged into develop for stabilisation, then develop is merged into the mainline. `Develop` must be either merged or fast-forward merged ("landed") into `mainline`. Squashing history during merge is not permitted - commits must be sorted as necessary before merge. Avoid direct commits to develop. It is recommended to perform stabilisation fixes in a separate branch and then landing it into develop. | ||||||
|  | - `mainline` is for generally accepted changes. History is immutable, to record reversals make a revert commit with explanations why. Changes from `develop` are merged or landed into the mainline after stabilisation. | ||||||
|  | - `released` branch records points from mainline which were officially released. Mutations are not possible. Only non-fast-forward merges from mainline are acceptable. Releases are marked as annotated tags on this branch. | ||||||
| 
 | 
 | ||||||
| ## OSdev help | ## OSdev help | ||||||
| 
 | 
 | ||||||
| Based on [Raspi3 tutorials by Andre Richter](https://github.com/rust-embedded/rust-raspi3-tutorial/blob/master/05_uart0/src/uart.rs), | Based on [Raspi3 tutorials by Andre Richter](https://github.com/rust-embedded/rust-raspi3-tutorial/blob/master/05_uart0/src/uart.rs), | ||||||
| which are in turn based on [Raspi3 tutorials by bzt](https://github.com/bztsrc/raspi3-tutorial/). | which are in turn based on [Raspi3 tutorials by bzt](https://github.com/bztsrc/raspi3-tutorial/). | ||||||
| Various references from [OSDev Wiki](https://wiki.osdev.org/Raspberry_Pi_Bare_Bones) and [RaspberryPi.org manuals](https://www.raspberrypi.org/app/uploads/2012/02/BCM2835-ARM-Peripherals.pdf). | Various references from [OSDev Wiki](https://wiki.osdev.org/Raspberry_Pi_Bare_Bones) and [RaspberryPi.org manuals](https://www.raspberrypi.org/app/uploads/2012/02/BCM2835-ARM-Peripherals.pdf). | ||||||
|  | 
 | ||||||
|  | [Debug with JTAG](doc/rpi3_jtag.md). | ||||||
|  | 
 | ||||||
|  | ## Badges | ||||||
|  | 
 | ||||||
|  | [](https://sagiegurari.github.io/cargo-make) | ||||||
|  | 
 | ||||||
|  |  | ||||||
|  | 
 | ||||||
|  | [](https://deps.rs/repo/github/metta-systems/vesper) | ||||||
|  | 
 | ||||||
|  | [](https://gitpod.io/#https://github.com/metta-systems/vesper) | ||||||
|  | 
 | ||||||
|  | ### License scan | ||||||
|  | 
 | ||||||
|  | [](https://app.fossa.com/projects/git%2Bgithub.com%2Fmetta-systems%2Fvesper?ref=badge_large) | ||||||
|  | 
 | ||||||
|  | Individual files contain the following tag instead of the full license text. | ||||||
|  | 
 | ||||||
|  |     SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  | 
 | ||||||
|  | This enables machine processing of license information based on the SPDX | ||||||
|  | License Identifiers that are here available: http://spdx.org/licenses/ | ||||||
|  | 
 | ||||||
|  | ---- | ||||||
|  | 
 | ||||||
|  | For more information please re-read. | ||||||
|  |  | ||||||
|  | @ -0,0 +1,39 @@ | ||||||
|  | [package] | ||||||
|  | name = "chainboot" | ||||||
|  | version = "0.0.1" | ||||||
|  | authors = ["Berkus Decker <berkus+vesper@metta.systems>"] | ||||||
|  | description = "Chain boot loader" | ||||||
|  | license = "BlueOak-1.0.0" | ||||||
|  | categories = ["no-std", "embedded", "os"] | ||||||
|  | publish = false | ||||||
|  | edition = "2021" | ||||||
|  | 
 | ||||||
|  | [badges] | ||||||
|  | maintenance = { status = "experimental" } | ||||||
|  | 
 | ||||||
|  | [features] | ||||||
|  | default = ["asm"] | ||||||
|  | # Build for running under QEMU with semihosting, so various halt/reboot options would for example quit QEMU instead. | ||||||
|  | qemu = ["machine/qemu"] | ||||||
|  | # Build for debugging it over JTAG/SWD connection - halts on first non-startup function start. | ||||||
|  | jtag = ["machine/jtag"] | ||||||
|  | # Dummy feature, ignored in this crate. | ||||||
|  | noserial = [] | ||||||
|  | # Startup relocation code is implemented in assembly | ||||||
|  | asm = [] | ||||||
|  | # Mutually exclusive features to choose a target board | ||||||
|  | rpi3 = ["machine/rpi3"] | ||||||
|  | rpi4 = ["machine/rpi4"] | ||||||
|  | 
 | ||||||
|  | [dependencies] | ||||||
|  | machine = { path = "../../machine" } | ||||||
|  | r0 = "1.0" | ||||||
|  | cortex-a = "7.0" | ||||||
|  | tock-registers = "0.7" | ||||||
|  | ux = { version = "0.1", default-features = false } | ||||||
|  | usize_conversions = "0.2" | ||||||
|  | bit_field = "0.10" | ||||||
|  | bitflags = "1.3" | ||||||
|  | cfg-if = "1.0" | ||||||
|  | snafu = { version = "0.7", default-features = false } | ||||||
|  | seahash = "4.1" | ||||||
|  | @ -0,0 +1,52 @@ | ||||||
|  | [env] | ||||||
|  | CHAINBOOT_ELF = "${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/target/${TARGET}/release/chainboot" | ||||||
|  | CHAINBOOT_BIN = "${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/target/chainboot.bin" | ||||||
|  | 
 | ||||||
|  | [tasks.kernel-binary] | ||||||
|  | env = { "BINARY_FILE" = "${CHAINBOOT_ELF}" } | ||||||
|  | run_task = "custom-binary" | ||||||
|  | 
 | ||||||
|  | [tasks.hopper] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.zellij-nucleus] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.zellij-cb] | ||||||
|  | env = { "KERNEL_BIN" = "${CHAINBOOT_BIN}", "QEMU_OPTS" = "${QEMU_OPTS} ${QEMU_DISASM_OPTS}" } | ||||||
|  | run_task = "zellij-config" | ||||||
|  | 
 | ||||||
|  | [tasks.zellij-cb-gdb] | ||||||
|  | env = { "KERNEL_BIN" = "${CHAINBOOT_BIN}", "QEMU_OPTS" = "${QEMU_OPTS} ${QEMU_DISASM_OPTS} ${QEMU_GDB_OPTS}", "TARGET_BOARD" = "rpi3", "TARGET_DTB" = "${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/targets/bcm2710-rpi-3-b-plus.dtb" } | ||||||
|  | run_task = "zellij-config" | ||||||
|  | 
 | ||||||
|  | [tasks.qemu] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.qemu-cb] | ||||||
|  | env = { "QEMU_RUNNER_OPTS" = "${QEMU_DISASM_OPTS} -serial pty", "KERNEL_BIN" = "${CHAINBOOT_BIN}", "TARGET_DTB" = "${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/targets/bcm2710-rpi-3-b-plus.dtb" } | ||||||
|  | extend = "qemu-runner" | ||||||
|  | 
 | ||||||
|  | [tasks.gdb] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.gdb-cb] | ||||||
|  | dependencies = ["build", "kernel-binary", "gdb-config"] | ||||||
|  | env = { "RUST_GDB" = "${GDB}" } | ||||||
|  | script = [ | ||||||
|  |     "rust-gdb -x ${GDB_CONNECT_FILE} ${CHAINBOOT_ELF}" | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [tasks.sdcard] | ||||||
|  | dependencies = ["build", "kernel-binary"] | ||||||
|  | script_runner = "@duckscript" | ||||||
|  | script = [ | ||||||
|  | ''' | ||||||
|  |     kernelImage = set "chain_boot_rpi4.img" | ||||||
|  |     cp ${CHAINBOOT_BIN} ${VOLUME}/${kernelImage} | ||||||
|  |     echo "Copied chainboot to ${VOLUME}/${kernelImage}" | ||||||
|  | ''' | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [tasks.cb-eject] | ||||||
|  | dependencies = ["sdeject"] | ||||||
|  | @ -0,0 +1,6 @@ | ||||||
|  | const LINKER_SCRIPT: &str = "bin/chainboot/src/link.ld"; | ||||||
|  | 
 | ||||||
|  | fn main() { | ||||||
|  |     println!("cargo:rerun-if-changed={}", LINKER_SCRIPT); | ||||||
|  |     println!("cargo:rustc-link-arg=--script={}", LINKER_SCRIPT); | ||||||
|  | } | ||||||
|  | @ -0,0 +1,68 @@ | ||||||
|  | // Assembly counterpart to this file.
 | ||||||
|  | #[cfg(feature = "asm")] | ||||||
|  | core::arch::global_asm!(include_str!("boot.s")); | ||||||
|  | 
 | ||||||
|  | // This is quite impossible - the linker constants are resolved to fully constant offsets in asm
 | ||||||
|  | // version, but are image-relative symbols in rust, and I see no way to force it otherwise.
 | ||||||
|  | #[no_mangle] | ||||||
|  | #[link_section = ".text._start"] | ||||||
|  | #[cfg(not(feature = "asm"))] | ||||||
|  | pub unsafe extern "C" fn _start() -> ! { | ||||||
|  |     use { | ||||||
|  |         cortex_a::registers::{MPIDR_EL1, SP}, | ||||||
|  |         machine::endless_sleep, | ||||||
|  |         tock_registers::interfaces::{Readable, Writeable}, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     const CORE_0: u64 = 0; | ||||||
|  |     const CORE_MASK: u64 = 0x3; | ||||||
|  | 
 | ||||||
|  |     if CORE_0 == MPIDR_EL1.get() & CORE_MASK { | ||||||
|  |         // if not core0, infinitely wait for events
 | ||||||
|  |         endless_sleep() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // These are a problem, because they are not interpreted as constants here.
 | ||||||
|  |     // Subsequently, this code tries to read values from not-yet-existing data locations.
 | ||||||
|  |     extern "C" { | ||||||
|  |         // Boundaries of the .bss section, provided by the linker script
 | ||||||
|  |         static mut __bss_start: u64; | ||||||
|  |         static mut __bss_end_exclusive: u64; | ||||||
|  |         // Load address of the kernel binary
 | ||||||
|  |         static mut __binary_nonzero_lma: u64; | ||||||
|  |         // Address to relocate to and image size
 | ||||||
|  |         static mut __binary_nonzero_vma: u64; | ||||||
|  |         static mut __binary_nonzero_vma_end_exclusive: u64; | ||||||
|  |         // Stack top
 | ||||||
|  |         static mut __boot_core_stack_end_exclusive: u64; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Set stack pointer.
 | ||||||
|  |     SP.set(&mut __boot_core_stack_end_exclusive as *mut u64 as u64); | ||||||
|  | 
 | ||||||
|  |     // Zeroes the .bss section
 | ||||||
|  |     r0::zero_bss(&mut __bss_start, &mut __bss_end_exclusive); | ||||||
|  | 
 | ||||||
|  |     // Relocate the code
 | ||||||
|  |     core::ptr::copy_nonoverlapping( | ||||||
|  |         &mut __binary_nonzero_lma as *const u64, | ||||||
|  |         &mut __binary_nonzero_vma as *mut u64, | ||||||
|  |         (&mut __binary_nonzero_vma_end_exclusive as *mut u64 as u64 | ||||||
|  |             - &mut __binary_nonzero_vma as *mut u64 as u64) as usize, | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     _start_rust(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | //--------------------------------------------------------------------------------------------------
 | ||||||
|  | // Public Code
 | ||||||
|  | //--------------------------------------------------------------------------------------------------
 | ||||||
|  | 
 | ||||||
|  | /// The Rust entry of the `kernel` binary.
 | ||||||
|  | ///
 | ||||||
|  | /// The function is called from the assembly `_start` function, keep it to support "asm" feature.
 | ||||||
|  | #[no_mangle] | ||||||
|  | #[inline(always)] | ||||||
|  | pub unsafe fn _start_rust(max_kernel_size: u64) -> ! { | ||||||
|  |     crate::kernel_init(max_kernel_size) | ||||||
|  | } | ||||||
|  | @ -0,0 +1,93 @@ | ||||||
|  | // SPDX-License-Identifier: MIT OR Apache-2.0 | ||||||
|  | // | ||||||
|  | // Copyright (c) 2021 Andre Richter <andre.o.richter@gmail.com>
 | ||||||
|  | // Modifications | ||||||
|  | // Copyright (c) 2021- Berkus <berkus+github@metta.systems>
 | ||||||
|  | 
 | ||||||
|  | //-------------------------------------------------------------------------------------------------- | ||||||
|  | // Definitions | ||||||
|  | //-------------------------------------------------------------------------------------------------- | ||||||
|  | 
 | ||||||
|  | // Load the address of a symbol into a register, PC-relative. | ||||||
|  | // | ||||||
|  | // The symbol must lie within +/- 4 GiB of the Program Counter. | ||||||
|  | // | ||||||
|  | // # Resources | ||||||
|  | // | ||||||
|  | // - https://sourceware.org/binutils/docs-2.36/as/AArch64_002dRelocations.html | ||||||
|  | .macro ADR_REL register, symbol | ||||||
|  |     adrp	\register, \symbol | ||||||
|  |     add	\register, \register, #:lo12:\symbol | ||||||
|  | .endm | ||||||
|  | 
 | ||||||
|  | // Load the address of a symbol into a register, absolute. | ||||||
|  | // | ||||||
|  | // # Resources | ||||||
|  | // | ||||||
|  | // - https://sourceware.org/binutils/docs-2.36/as/AArch64_002dRelocations.html | ||||||
|  | .macro ADR_ABS register, symbol | ||||||
|  |     movz	\register, #:abs_g2:\symbol | ||||||
|  |     movk	\register, #:abs_g1_nc:\symbol | ||||||
|  |     movk	\register, #:abs_g0_nc:\symbol | ||||||
|  | .endm | ||||||
|  | 
 | ||||||
|  | //-------------------------------------------------------------------------------------------------- | ||||||
|  | // Public Code | ||||||
|  | //-------------------------------------------------------------------------------------------------- | ||||||
|  | .section .text._start | ||||||
|  | 
 | ||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  | // fn _start() | ||||||
|  | //------------------------------------------------------------------------------ | ||||||
|  | _start: | ||||||
|  |     // Only proceed on the boot core. Park it otherwise. | ||||||
|  |     mrs	x1, MPIDR_EL1 | ||||||
|  |     and	x1, x1, 0b11          // core id mask | ||||||
|  |     cmp	x1, 0                 // boot core id | ||||||
|  |     b.ne	.L_parking_loop | ||||||
|  | 
 | ||||||
|  |     // If execution reaches here, it is the boot core. | ||||||
|  | 
 | ||||||
|  |     // Initialize bss. | ||||||
|  |     ADR_ABS	x0, __bss_start | ||||||
|  |     ADR_ABS x1, __bss_end_exclusive | ||||||
|  | 
 | ||||||
|  | .L_bss_init_loop: | ||||||
|  |     cmp	x0, x1 | ||||||
|  |     b.eq	.L_relocate_binary | ||||||
|  |     stp	xzr, xzr, [x0], #16 | ||||||
|  |     b	.L_bss_init_loop | ||||||
|  | 
 | ||||||
|  |     // Next, relocate the binary. | ||||||
|  | .L_relocate_binary: | ||||||
|  |     ADR_REL	x0, __binary_nonzero_lma           // The address the binary got loaded to. | ||||||
|  |     ADR_ABS	x1, __binary_nonzero_vma           // The address the binary was linked to. | ||||||
|  |     ADR_ABS	x2, __binary_nonzero_vma_end_exclusive | ||||||
|  |     sub x4, x1, x0                             // Get difference between vma and lma as max size | ||||||
|  | 
 | ||||||
|  | .L_copy_loop: | ||||||
|  |     ldr	x3, [x0], #8 | ||||||
|  |     str	x3, [x1], #8 | ||||||
|  |     cmp	x1, x2 | ||||||
|  |     b.lo	.L_copy_loop | ||||||
|  | 
 | ||||||
|  |     // Prepare the jump to Rust code. | ||||||
|  |     // Set the stack pointer. | ||||||
|  |     ADR_ABS	x0, __rpi_phys_binary_load_addr | ||||||
|  |     mov	sp, x0 | ||||||
|  | 
 | ||||||
|  |     // Pass maximum kernel size as an argument to Rust init function. | ||||||
|  |     mov x0, x4 | ||||||
|  | 
 | ||||||
|  |     // Jump to the relocated Rust code. | ||||||
|  |     ADR_ABS	x1, _start_rust | ||||||
|  |     br	x1 | ||||||
|  | 
 | ||||||
|  |     // Infinitely wait for events (aka "park the core"). | ||||||
|  | .L_parking_loop: | ||||||
|  |     wfe | ||||||
|  |     b	.L_parking_loop | ||||||
|  | 
 | ||||||
|  | .size	_start, . - _start | ||||||
|  | .type	_start, function | ||||||
|  | .global	_start
 | ||||||
|  | @ -0,0 +1,98 @@ | ||||||
|  | /* SPDX-License-Identifier: MIT OR Apache-2.0 | ||||||
|  |  * | ||||||
|  |  * Copyright (c) 2018-2021 Andre Richter <andre.o.richter@gmail.com> | ||||||
|  |  * Copyright (c) 2021- Berkus <berkus+github@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  |  * Information from: | ||||||
|  |  * [Output Section Address](https://sourceware.org/binutils/docs/ld/Output-Section-Address.html) | ||||||
|  |  * [Output Section LMA](https://sourceware.org/binutils/docs/ld/Output-Section-LMA.html) | ||||||
|  |  * [Output Section Attributes](https://sourceware.org/binutils/docs/ld/Output-Section-Attributes.html#Output-Section-Attributes) | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /* The physical address at which the the kernel binary will be loaded by the Raspberry's firmware */ | ||||||
|  | __rpi_phys_binary_load_addr = 0x80000; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ENTRY(__rpi_phys_binary_load_addr) | ||||||
|  | 
 | ||||||
|  | /* Flags: | ||||||
|  |  *     4 == R | ||||||
|  |  *     5 == RX | ||||||
|  |  *     6 == RW | ||||||
|  |  * | ||||||
|  |  * Segments are marked PT_LOAD below so that the ELF file provides virtual and physical addresses. | ||||||
|  |  * It doesn't mean all of them need actually be loaded. | ||||||
|  |  */ | ||||||
|  | PHDRS | ||||||
|  | { | ||||||
|  |     segment_boot_core_stack PT_LOAD FLAGS(6); | ||||||
|  |     segment_start_code      PT_LOAD FLAGS(5); | ||||||
|  |     segment_code            PT_LOAD FLAGS(5); | ||||||
|  |     segment_data            PT_LOAD FLAGS(6); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | SECTIONS | ||||||
|  | { | ||||||
|  |     /*********************************************************************************************** | ||||||
|  |     * Boot Core Stack | ||||||
|  |     ***********************************************************************************************/ | ||||||
|  |     .boot_core_stack (NOLOAD) : | ||||||
|  |     { | ||||||
|  |                                              /*   ^             */ | ||||||
|  |                                              /*   | stack       */ | ||||||
|  |         . += __rpi_phys_binary_load_addr;    /*   | growth      */ | ||||||
|  |                                              /*   | direction   */ | ||||||
|  |         __boot_core_stack_end_exclusive = .; /*   |             */ | ||||||
|  |     } :segment_boot_core_stack | ||||||
|  | 
 | ||||||
|  |     . = __rpi_phys_binary_load_addr; | ||||||
|  | 
 | ||||||
|  |     .text : | ||||||
|  |     { | ||||||
|  |         KEEP(*(.text._start)) | ||||||
|  |         /* *(text.memcpy) -- only relevant for Rust relocator impl which is currently impossible */ | ||||||
|  |     } :segment_start_code | ||||||
|  | 
 | ||||||
|  |     /* Align to 8 bytes, b/c relocating the binary is done in u64 chunks */ | ||||||
|  |     . = ALIGN(8); | ||||||
|  | 
 | ||||||
|  |     __binary_nonzero_lma = .; | ||||||
|  | 
 | ||||||
|  |     /* Set the link address to 32 MiB */ | ||||||
|  |     /* This dictates the max size of the loadable kernel. */ | ||||||
|  |     . += 0x2000000; | ||||||
|  | 
 | ||||||
|  |     /*********************************************************************************************** | ||||||
|  |     * Code + RO Data + Global Offset Table | ||||||
|  |     ***********************************************************************************************/ | ||||||
|  |     __binary_nonzero_vma = .; | ||||||
|  |     .text : AT (ADDR(.text) + SIZEOF(.text)) | ||||||
|  |     { | ||||||
|  |         *(.text._start_rust)      /* The Rust entry point */ | ||||||
|  |         /* *(text.memcpy) -- only relevant for Rust relocator impl which is currently impossible */ | ||||||
|  |         *(.text*)                 /* Everything else */ | ||||||
|  |     } :segment_code | ||||||
|  | 
 | ||||||
|  |     .rodata : ALIGN(8) { *(.rodata*) } :segment_code | ||||||
|  |     .got    : ALIGN(8) { *(.got)     } :segment_code | ||||||
|  | 
 | ||||||
|  |     /*********************************************************************************************** | ||||||
|  |     * Data + BSS | ||||||
|  |     ***********************************************************************************************/ | ||||||
|  |     .data : { *(.data*) } :segment_data | ||||||
|  | 
 | ||||||
|  |     /* Fill up to 8 bytes, b/c relocating the binary is done in u64 chunks */ | ||||||
|  |     . = ALIGN(8); | ||||||
|  |     __binary_nonzero_vma_end_exclusive = .; | ||||||
|  | 
 | ||||||
|  |     /* Section is zeroed in pairs of u64. Align start and end to 16 bytes */ | ||||||
|  |     .bss (NOLOAD) : ALIGN(16) | ||||||
|  |     { | ||||||
|  |         __bss_start = .; | ||||||
|  |         *(.bss*); | ||||||
|  |         . = ALIGN(16); | ||||||
|  |         __bss_end_exclusive = .; | ||||||
|  |     } :segment_data | ||||||
|  | } | ||||||
|  | @ -0,0 +1,155 @@ | ||||||
|  | // Based on miniload by @andre-richter
 | ||||||
|  | #![feature(format_args_nl)] | ||||||
|  | #![feature(custom_test_frameworks)] | ||||||
|  | #![test_runner(machine::tests::test_runner)] | ||||||
|  | #![reexport_test_harness_main = "test_main"] | ||||||
|  | #![no_main] | ||||||
|  | #![no_std] | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     core::{hash::Hasher, panic::PanicInfo}, | ||||||
|  |     cortex_a::asm::barrier, | ||||||
|  |     machine::{ | ||||||
|  |         devices::SerialOps, | ||||||
|  |         platform::rpi3::{gpio::GPIO, pl011_uart::PL011Uart, BcmHost}, | ||||||
|  |         print, println, CONSOLE, | ||||||
|  |     }, | ||||||
|  |     seahash::SeaHasher, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | mod boot; | ||||||
|  | 
 | ||||||
|  | /// Early init code.
 | ||||||
|  | ///
 | ||||||
|  | /// # Safety
 | ||||||
|  | ///
 | ||||||
|  | /// - Only a single core must be active and running this function.
 | ||||||
|  | /// - The init calls in this function must appear in the correct order.
 | ||||||
|  | #[inline(always)] | ||||||
|  | unsafe fn kernel_init(max_kernel_size: u64) -> ! { | ||||||
|  |     #[cfg(feature = "jtag")] | ||||||
|  |     machine::arch::jtag::wait_debugger(); | ||||||
|  | 
 | ||||||
|  |     let gpio = GPIO::default(); | ||||||
|  |     let uart = PL011Uart::default(); | ||||||
|  |     let uart = uart.prepare(&gpio).expect("What could go wrong?"); | ||||||
|  |     CONSOLE.lock(|c| { | ||||||
|  |         // Move uart into the global CONSOLE.
 | ||||||
|  |         c.replace_with(uart.into()); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     // println! is usable from here on.
 | ||||||
|  | 
 | ||||||
|  |     // Transition from unsafe to safe.
 | ||||||
|  |     kernel_main(max_kernel_size) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // https://onlineasciitools.com/convert-text-to-ascii-art (FIGlet) with `cricket` font
 | ||||||
|  | const LOGO: &str = r#" | ||||||
|  |        __          __       __                __   
 | ||||||
|  |  .----|  |--.---.-|__.-----|  |--.-----.-----|  |_ 
 | ||||||
|  |  |  __|     |  _  |  |     |  _  |  _  |  _  |   _| | ||||||
|  |  |____|__|__|___._|__|__|__|_____|_____|_____|____| | ||||||
|  | "#;
 | ||||||
|  | 
 | ||||||
|  | fn read_u64() -> u64 { | ||||||
|  |     CONSOLE.lock(|c| { | ||||||
|  |         let mut val: u64 = u64::from(c.read_byte()); | ||||||
|  |         val |= u64::from(c.read_byte()) << 8; | ||||||
|  |         val |= u64::from(c.read_byte()) << 16; | ||||||
|  |         val |= u64::from(c.read_byte()) << 24; | ||||||
|  |         val |= u64::from(c.read_byte()) << 32; | ||||||
|  |         val |= u64::from(c.read_byte()) << 40; | ||||||
|  |         val |= u64::from(c.read_byte()) << 48; | ||||||
|  |         val |= u64::from(c.read_byte()) << 56; | ||||||
|  |         val | ||||||
|  |     }) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// The main function running after the early init.
 | ||||||
|  | #[inline(always)] | ||||||
|  | fn kernel_main(max_kernel_size: u64) -> ! { | ||||||
|  |     #[cfg(test)] | ||||||
|  |     test_main(); | ||||||
|  | 
 | ||||||
|  |     print!("{}", LOGO); | ||||||
|  |     println!("{:>51}\n", BcmHost::board_name()); | ||||||
|  |     println!("[<<] Requesting kernel image..."); | ||||||
|  | 
 | ||||||
|  |     let kernel_addr: *mut u8 = BcmHost::kernel_load_address() as *mut u8; | ||||||
|  | 
 | ||||||
|  |     loop { | ||||||
|  |         CONSOLE.lock(|c| c.flush()); | ||||||
|  | 
 | ||||||
|  |         // Discard any spurious received characters before starting with the loader protocol.
 | ||||||
|  |         CONSOLE.lock(|c| c.clear_rx()); | ||||||
|  | 
 | ||||||
|  |         // Notify `chainofcommand` to send the binary.
 | ||||||
|  |         for _ in 0..3 { | ||||||
|  |             CONSOLE.lock(|c| c.write_byte(3u8)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Read the binary's size.
 | ||||||
|  |         let size = read_u64(); | ||||||
|  | 
 | ||||||
|  |         // Check the size to fit RAM
 | ||||||
|  |         if size > max_kernel_size { | ||||||
|  |             println!("ERR Kernel image too big (over {} bytes)", max_kernel_size); | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         print!("OK"); | ||||||
|  | 
 | ||||||
|  |         // We use seahash, simple and with no_std implementation.
 | ||||||
|  |         let mut hasher = SeaHasher::new(); | ||||||
|  | 
 | ||||||
|  |         // Read the kernel byte by byte.
 | ||||||
|  |         for i in 0..size { | ||||||
|  |             let val = CONSOLE.lock(|c| c.read_byte()); | ||||||
|  |             unsafe { | ||||||
|  |                 core::ptr::write_volatile(kernel_addr.offset(i as isize), val); | ||||||
|  |             } | ||||||
|  |             let written = unsafe { core::ptr::read_volatile(kernel_addr.offset(i as isize)) }; | ||||||
|  |             hasher.write_u8(written); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Read the binary's checksum.
 | ||||||
|  |         let checksum = read_u64(); | ||||||
|  | 
 | ||||||
|  |         let valid = hasher.finish() == checksum; | ||||||
|  |         if !valid { | ||||||
|  |             println!("ERR Kernel image checksum mismatch"); | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         print!("OK"); | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     println!( | ||||||
|  |         "[<<] Loaded! Executing the payload now from {:p}\n", | ||||||
|  |         kernel_addr | ||||||
|  |     ); | ||||||
|  |     CONSOLE.lock(|c| c.flush()); | ||||||
|  | 
 | ||||||
|  |     // Use black magic to create a function pointer.
 | ||||||
|  |     let kernel: fn() -> ! = unsafe { core::mem::transmute(kernel_addr) }; | ||||||
|  | 
 | ||||||
|  |     // Force everything to complete before we jump.
 | ||||||
|  |     unsafe { barrier::isb(barrier::SY) }; | ||||||
|  | 
 | ||||||
|  |     // Jump to loaded kernel!
 | ||||||
|  |     kernel() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(not(test))] | ||||||
|  | #[panic_handler] | ||||||
|  | fn panicked(info: &PanicInfo) -> ! { | ||||||
|  |     machine::panic::handler(info) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(test)] | ||||||
|  | #[panic_handler] | ||||||
|  | fn panicked(info: &PanicInfo) -> ! { | ||||||
|  |     machine::panic::handler_for_tests(info) | ||||||
|  | } | ||||||
|  | @ -0,0 +1,25 @@ | ||||||
|  | [package] | ||||||
|  | name = "chainofcommand" | ||||||
|  | version = "0.0.1" | ||||||
|  | authors = ["Berkus Decker <berkus+vesper@metta.systems>"] | ||||||
|  | description = "Host server for chainboot" | ||||||
|  | license = "BlueOak-1.0.0" | ||||||
|  | categories = ["no-std", "embedded", "os"] | ||||||
|  | publish = false | ||||||
|  | edition = "2021" | ||||||
|  | 
 | ||||||
|  | [badges] | ||||||
|  | maintenance = { status = "experimental" } | ||||||
|  | 
 | ||||||
|  | [dependencies] | ||||||
|  | clap = "3.0" | ||||||
|  | seahash = "4.1" | ||||||
|  | anyhow = "1.0" | ||||||
|  | fehler = "1.0" | ||||||
|  | crossterm = { version = "0.23", features = ["event-stream"] } | ||||||
|  | tokio-serial = "5.4" | ||||||
|  | tokio = { version = "1.16", features = ["full"] } | ||||||
|  | futures = "0.3" | ||||||
|  | defer = "0.1" | ||||||
|  | tokio-util = { version = "0.7", features = ["codec"] } | ||||||
|  | bytes = "1.1" | ||||||
|  | @ -0,0 +1,44 @@ | ||||||
|  | [tasks.build] | ||||||
|  | command = "cargo" | ||||||
|  | args = ["build"] | ||||||
|  | 
 | ||||||
|  | [tasks.test] | ||||||
|  | command = "cargo" | ||||||
|  | args = ["test"] | ||||||
|  | 
 | ||||||
|  | [tasks.clippy] | ||||||
|  | command = "cargo" | ||||||
|  | args = ["clippy", "--", "-D", "warnings"] | ||||||
|  | 
 | ||||||
|  | [tasks.hopper] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.kernel-binary] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.zellij-nucleus] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.zellij-cb] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.zellij-cb-gdb] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.qemu] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.qemu-cb] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.sdcard] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.cb-eject] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.gdb] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.gdb-cb] | ||||||
|  | disabled = true | ||||||
|  | @ -0,0 +1,436 @@ | ||||||
|  | #![feature(trait_alias)] | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     anyhow::{anyhow, Result}, | ||||||
|  |     bytes::Bytes, | ||||||
|  |     clap::{App, AppSettings, Arg}, | ||||||
|  |     crossterm::{ | ||||||
|  |         cursor, | ||||||
|  |         event::{Event, EventStream, KeyCode, KeyEvent, KeyModifiers}, | ||||||
|  |         execute, style, terminal, | ||||||
|  |         tty::IsTty, | ||||||
|  |     }, | ||||||
|  |     defer::defer, | ||||||
|  |     futures::{future::FutureExt, StreamExt}, | ||||||
|  |     seahash::SeaHasher, | ||||||
|  |     std::{ | ||||||
|  |         fs::File, | ||||||
|  |         hash::Hasher, | ||||||
|  |         io::{BufRead, BufReader}, | ||||||
|  |         path::Path, | ||||||
|  |         time::Duration, | ||||||
|  |     }, | ||||||
|  |     tokio::{io::AsyncReadExt, sync::mpsc}, | ||||||
|  |     tokio_serial::{SerialPortBuilderExt, SerialStream}, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | trait Writable = std::io::Write + Send; | ||||||
|  | trait ThePath = AsRef<Path> + std::fmt::Display + Clone + Sync + Send + 'static; | ||||||
|  | 
 | ||||||
|  | async fn expect( | ||||||
|  |     to_console2: &mpsc::Sender<Vec<u8>>, | ||||||
|  |     from_serial: &mut mpsc::Receiver<Vec<u8>>, | ||||||
|  |     m: &str, | ||||||
|  | ) -> Result<()> { | ||||||
|  |     if let Some(buf) = from_serial.recv().await { | ||||||
|  |         if buf.len() == m.len() && String::from_utf8_lossy(buf.as_ref()) == m { | ||||||
|  |             return Ok(()); | ||||||
|  |         } | ||||||
|  |         to_console2.send(buf).await?; | ||||||
|  |         return Err(anyhow!("Failed to receive expected value")); | ||||||
|  |     } | ||||||
|  |     Err(anyhow!("Failed to receive expected value")) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async fn load_kernel<P>(to_console2: &mpsc::Sender<Vec<u8>>, kernel: P) -> Result<(File, u64)> | ||||||
|  | where | ||||||
|  |     P: ThePath, | ||||||
|  | { | ||||||
|  |     to_console2 | ||||||
|  |         .send("[>>] Loading kernel image\n".into()) | ||||||
|  |         .await?; | ||||||
|  | 
 | ||||||
|  |     let kernel_file = match std::fs::File::open(kernel.clone()) { | ||||||
|  |         Ok(file) => file, | ||||||
|  |         Err(_) => return Err(anyhow!("Couldn't open kernel file {}", kernel)), | ||||||
|  |     }; | ||||||
|  |     let kernel_size: u64 = kernel_file.metadata()?.len(); | ||||||
|  | 
 | ||||||
|  |     to_console2 | ||||||
|  |         .send(format!("[>>] .. {} ({} bytes)\n", kernel, kernel_size).into()) | ||||||
|  |         .await?; | ||||||
|  | 
 | ||||||
|  |     Ok((kernel_file, kernel_size)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async fn send_kernel<P>( | ||||||
|  |     to_console2: &mpsc::Sender<Vec<u8>>, | ||||||
|  |     to_serial: &mpsc::Sender<Vec<u8>>, | ||||||
|  |     from_serial: &mut mpsc::Receiver<Vec<u8>>, | ||||||
|  |     kernel: P, | ||||||
|  | ) -> Result<()> | ||||||
|  | where | ||||||
|  |     P: ThePath, | ||||||
|  | { | ||||||
|  |     let (kernel_file, kernel_size) = load_kernel(to_console2, kernel).await?; | ||||||
|  | 
 | ||||||
|  |     to_console2.send("[>>] Sending image size\n".into()).await?; | ||||||
|  | 
 | ||||||
|  |     to_serial.send(kernel_size.to_le_bytes().into()).await?; | ||||||
|  | 
 | ||||||
|  |     // Wait for OK response
 | ||||||
|  |     expect(to_console2, from_serial, "OK").await?; | ||||||
|  | 
 | ||||||
|  |     to_console2 | ||||||
|  |         .send("[>>] Sending kernel image\n".into()) | ||||||
|  |         .await?; | ||||||
|  | 
 | ||||||
|  |     let mut hasher = SeaHasher::new(); | ||||||
|  |     let mut reader = BufReader::with_capacity(1, kernel_file); | ||||||
|  |     loop { | ||||||
|  |         let length = { | ||||||
|  |             let buf = reader.fill_buf()?; | ||||||
|  |             to_serial.send(buf.into()).await?; | ||||||
|  |             hasher.write(buf); | ||||||
|  |             buf.len() | ||||||
|  |         }; | ||||||
|  |         if length == 0 { | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         reader.consume(length); | ||||||
|  |     } | ||||||
|  |     let hashed_value: u64 = hasher.finish(); | ||||||
|  | 
 | ||||||
|  |     to_console2 | ||||||
|  |         .send(format!("[>>] Sending image checksum {:x}\n", hashed_value).into()) | ||||||
|  |         .await?; | ||||||
|  | 
 | ||||||
|  |     to_serial.send(hashed_value.to_le_bytes().into()).await?; | ||||||
|  | 
 | ||||||
|  |     expect(to_console2, from_serial, "OK").await?; | ||||||
|  | 
 | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Async reading using Tokio: https://fasterthanli.me/articles/a-terminal-case-of-linux
 | ||||||
|  | 
 | ||||||
|  | async fn serial_loop( | ||||||
|  |     mut port: tokio_serial::SerialStream, | ||||||
|  |     to_console: mpsc::Sender<Vec<u8>>, | ||||||
|  |     mut from_console: mpsc::Receiver<Vec<u8>>, | ||||||
|  | ) -> Result<()> { | ||||||
|  |     let mut buf = [0; 256]; | ||||||
|  |     loop { | ||||||
|  |         tokio::select! { | ||||||
|  |             // _ = poll_send => {},
 | ||||||
|  | 
 | ||||||
|  |             Some(msg) = from_console.recv() => { | ||||||
|  |                 // debug!("serial write {} bytes", msg.len());
 | ||||||
|  |                 tokio::io::AsyncWriteExt::write_all(&mut port, msg.as_ref()).await?; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             res = port.read(&mut buf) => { | ||||||
|  |                 match res { | ||||||
|  |                     Ok(0) => { | ||||||
|  |                         // info!("Serial <EOF>");
 | ||||||
|  |                         return Ok(()); | ||||||
|  |                     } | ||||||
|  |                     Ok(n) => { | ||||||
|  |                         // debug!("Serial read {n} bytes.");
 | ||||||
|  |                         to_console.send(buf[0..n].to_owned()).await?; | ||||||
|  |                     } | ||||||
|  |                     Err(e) => { | ||||||
|  |             //             if e.kind() == ErrorKind::TimedOut {
 | ||||||
|  |             //                 execute!(w, style::Print("\r\nTimeout: the serial device has been unplugged!"))?;
 | ||||||
|  |             //             } else {
 | ||||||
|  |             //                 execute!(w, style::Print(format!("\r\nSerial Error: {:?}\r", e)))?;
 | ||||||
|  |             //             }
 | ||||||
|  |             //             break;
 | ||||||
|  |                         return Err(anyhow!(e)); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async fn console_loop<P>( | ||||||
|  |     to_console2: mpsc::Sender<Vec<u8>>, | ||||||
|  |     mut from_internal: mpsc::Receiver<Vec<u8>>, | ||||||
|  |     to_serial: mpsc::Sender<Vec<u8>>, | ||||||
|  |     mut from_serial: mpsc::Receiver<Vec<u8>>, | ||||||
|  |     kernel: P, | ||||||
|  | ) -> Result<()> | ||||||
|  | where | ||||||
|  |     P: ThePath, | ||||||
|  | { | ||||||
|  |     let mut w = std::io::stdout(); | ||||||
|  | 
 | ||||||
|  |     let mut breaks = 0; | ||||||
|  | 
 | ||||||
|  |     let mut event_reader = EventStream::new(); | ||||||
|  | 
 | ||||||
|  |     loop { | ||||||
|  |         tokio::select! { | ||||||
|  |             biased; | ||||||
|  | 
 | ||||||
|  |             Some(received) = from_internal.recv() => { | ||||||
|  |                 for &x in &received[..] { | ||||||
|  |                     execute!(w, style::Print(format!("{}", x as char)))?; | ||||||
|  |                 } | ||||||
|  |                 w.flush()?; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             Some(received) = from_serial.recv() => { | ||||||
|  |                 // execute!(w, cursor::MoveToNextLine(1), style::Print(format!("[>>] Received {} bytes from serial", from_serial.len())), cursor::MoveToNextLine(1))?;
 | ||||||
|  | 
 | ||||||
|  |                 for &x in &received[..] { | ||||||
|  |                     if x == 0x3 { | ||||||
|  |                         // execute!(w, cursor::MoveToNextLine(1), style::Print("[>>] Received a BREAK"), cursor::MoveToNextLine(1))?;
 | ||||||
|  |                         breaks += 1; | ||||||
|  |                         // Await for 3 consecutive \3 to start downloading
 | ||||||
|  |                         if breaks == 3 { | ||||||
|  |                             // execute!(w, cursor::MoveToNextLine(1), style::Print("[>>] Received 3 BREAKs"), cursor::MoveToNextLine(1))?;
 | ||||||
|  |                             breaks = 0; | ||||||
|  |                             send_kernel(&to_console2, &to_serial, &mut from_serial, kernel.clone()).await?; | ||||||
|  |                             to_console2.send("[>>] Send successful, pass-through\n".into()).await?; | ||||||
|  |                         } | ||||||
|  |                     } else { | ||||||
|  |                         while breaks > 0 { | ||||||
|  |                             execute!(w, style::Print(format!("{}", 3 as char)))?; | ||||||
|  |                             breaks -= 1; | ||||||
|  |                         } | ||||||
|  |                         execute!(w, style::Print(format!("{}", x as char)))?; | ||||||
|  |                         w.flush()?; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             maybe_event = event_reader.next().fuse() => { | ||||||
|  |                 match maybe_event { | ||||||
|  |                     Some(Ok(Event::Key(key_event))) => { | ||||||
|  |                         if key_event.code == KeyCode::Char('c') && key_event.modifiers == KeyModifiers::CONTROL { | ||||||
|  |                             return Ok(()); | ||||||
|  |                         } | ||||||
|  |                         if let Some(key) = handle_key_event(key_event) { | ||||||
|  |                             to_serial.send(key.to_vec()).await?; | ||||||
|  |                             // Local echo
 | ||||||
|  |                             execute!(w, style::Print(format!("{:?}", key)))?; | ||||||
|  |                             w.flush()?; | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                     Some(Ok(_)) => {}, | ||||||
|  |                     Some(Err(e)) => { | ||||||
|  |                         execute!(w, style::Print(format!("Console read error: {:?}\r", e)))?; | ||||||
|  |                         w.flush()?; | ||||||
|  |                     }, | ||||||
|  |                     None => return Err(anyhow!("woops")), | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async fn main_loop<P>(port: SerialStream, kernel: P) -> Result<()> | ||||||
|  | where | ||||||
|  |     P: ThePath, | ||||||
|  | { | ||||||
|  |     // read from serial -> to_console==>from_serial -> output to console
 | ||||||
|  |     let (to_console, from_serial) = mpsc::channel(256); | ||||||
|  |     let (to_console2, from_internal) = mpsc::channel(256); | ||||||
|  | 
 | ||||||
|  |     // read from console -> to_serial==>from_console -> output to serial
 | ||||||
|  |     let (to_serial, from_console) = mpsc::channel(256); | ||||||
|  | 
 | ||||||
|  |     tokio::spawn(serial_loop(port, to_console.clone(), from_console)); | ||||||
|  |     console_loop(to_console2, from_internal, to_serial, from_serial, kernel).await | ||||||
|  | 
 | ||||||
|  |     // TODO: framed
 | ||||||
|  | 
 | ||||||
|  |     // rx_device -> serial_reader -> app
 | ||||||
|  |     // app -> serial_writer -> serial_consumer -> (poll_send to drive) -> serial_sink -> tx_device
 | ||||||
|  |     // let (rx_device, tx_device) = split(port);
 | ||||||
|  | 
 | ||||||
|  |     // let mut serial_reader = FramedRead::new(rx_device, BytesCodec::new());
 | ||||||
|  |     // let serial_sink = FramedWrite::new(tx_device, BytesCodec::new());
 | ||||||
|  |     //
 | ||||||
|  |     // let (serial_writer, serial_consumer) = mpsc::unbounded::<Bytes>();
 | ||||||
|  |     // let mut poll_send = serial_consumer.map(Ok).forward(serial_sink);
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // From remote_serial -- https://github.com/zhp-rs/remote_serial/ (Licensed under MIT License)
 | ||||||
|  | fn handle_key_event(key_event: KeyEvent) -> Option<Bytes> { | ||||||
|  |     let mut buf = [0; 4]; | ||||||
|  | 
 | ||||||
|  |     let key_str: Option<&[u8]> = match key_event.code { | ||||||
|  |         KeyCode::Backspace => Some(b"\x08"), | ||||||
|  |         KeyCode::Enter => Some(b"\x0D"), | ||||||
|  |         KeyCode::Left => Some(b"\x1b[D"), | ||||||
|  |         KeyCode::Right => Some(b"\x1b[C"), | ||||||
|  |         KeyCode::Home => Some(b"\x1b[H"), | ||||||
|  |         KeyCode::End => Some(b"\x1b[F"), | ||||||
|  |         KeyCode::Up => Some(b"\x1b[A"), | ||||||
|  |         KeyCode::Down => Some(b"\x1b[B"), | ||||||
|  |         KeyCode::Tab => Some(b"\x09"), | ||||||
|  |         KeyCode::Delete => Some(b"\x1b[3~"), | ||||||
|  |         KeyCode::Insert => Some(b"\x1b[2~"), | ||||||
|  |         KeyCode::Esc => Some(b"\x1b"), | ||||||
|  |         KeyCode::Char(ch) => { | ||||||
|  |             if key_event.modifiers & KeyModifiers::CONTROL == KeyModifiers::CONTROL { | ||||||
|  |                 buf[0] = ch as u8; | ||||||
|  |                 if ('a'..='z').contains(&ch) || (ch == ' ') { | ||||||
|  |                     buf[0] &= 0x1f; | ||||||
|  |                     Some(&buf[0..1]) | ||||||
|  |                 } else if ('4'..='7').contains(&ch) { | ||||||
|  |                     // crossterm returns Control-4 thru 7 for \x1c thru \x1f
 | ||||||
|  |                     buf[0] = (buf[0] + 8) & 0x1f; | ||||||
|  |                     Some(&buf[0..1]) | ||||||
|  |                 } else { | ||||||
|  |                     Some(ch.encode_utf8(&mut buf).as_bytes()) | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 Some(ch.encode_utf8(&mut buf).as_bytes()) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         _ => None, | ||||||
|  |     }; | ||||||
|  |     key_str.map(Bytes::copy_from_slice) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 1. connect to given serial port, e.g. /dev/ttyUSB23234
 | ||||||
|  | // 2. Await for \3\3\3 start signal, meanwhile pass-through all traffic to console
 | ||||||
|  | // 3. send selected kernel binary with checksum to the target
 | ||||||
|  | // 4. go to 2
 | ||||||
|  | 
 | ||||||
|  | #[tokio::main] | ||||||
|  | async fn main() -> Result<()> { | ||||||
|  |     let matches = App::new("ChainOfCommand - command chainboot protocol") | ||||||
|  |         .about("Use to send freshly built kernel to chainboot-compatible boot loader") | ||||||
|  |         .setting(AppSettings::DisableVersionFlag) | ||||||
|  |         .arg( | ||||||
|  |             Arg::new("port") | ||||||
|  |                 .help("The device path to a serial port, e.g. /dev/ttyUSB0") | ||||||
|  |                 .required(true), | ||||||
|  |         ) | ||||||
|  |         .arg( | ||||||
|  |             Arg::new("baud") | ||||||
|  |                 .help("The baud rate to connect at") | ||||||
|  |                 .use_delimiter(false) | ||||||
|  |                 .required(true), // .validator(valid_baud),
 | ||||||
|  |         ) | ||||||
|  |         .arg( | ||||||
|  |             Arg::new("kernel") | ||||||
|  |                 .long("kernel") | ||||||
|  |                 .help("Path of the binary kernel image to send") | ||||||
|  |                 .takes_value(true) | ||||||
|  |                 .default_value("kernel8.img"), | ||||||
|  |         ) | ||||||
|  |         .get_matches(); | ||||||
|  |     let port_name = matches.value_of("port").unwrap(); | ||||||
|  |     let baud_rate = matches.value_of("baud").unwrap().parse::<u32>().unwrap(); | ||||||
|  |     let kernel = matches.value_of("kernel").unwrap().to_owned(); | ||||||
|  | 
 | ||||||
|  |     // Check that STDIN is a proper tty
 | ||||||
|  |     if !std::io::stdin().is_tty() { | ||||||
|  |         panic!("Must have a TTY for stdin"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Disable line buffering, local echo, etc.
 | ||||||
|  |     terminal::enable_raw_mode()?; | ||||||
|  |     defer(|| terminal::disable_raw_mode().unwrap_or(())); | ||||||
|  | 
 | ||||||
|  |     let mut serial_toggle = false; | ||||||
|  |     let mut stdout = std::io::stdout(); | ||||||
|  | 
 | ||||||
|  |     execute!(stdout, cursor::SavePosition)?; | ||||||
|  | 
 | ||||||
|  |     loop { | ||||||
|  |         execute!( | ||||||
|  |             stdout, | ||||||
|  |             cursor::RestorePosition, | ||||||
|  |             style::Print("[>>] Opening serial port       ") | ||||||
|  |         )?; | ||||||
|  | 
 | ||||||
|  |         // tokio_serial::new() creates a builder with 8N1 setup without flow control by default.
 | ||||||
|  |         let port = tokio_serial::new(port_name, baud_rate).open_native_async(); | ||||||
|  |         if let Err(e) = port { | ||||||
|  |             let cont = match e.kind { | ||||||
|  |                 tokio_serial::ErrorKind::NoDevice => true, | ||||||
|  |                 tokio_serial::ErrorKind::Io(e) | ||||||
|  |                     if e == std::io::ErrorKind::NotFound | ||||||
|  |                         || e == std::io::ErrorKind::PermissionDenied => | ||||||
|  |                 { | ||||||
|  |                     true | ||||||
|  |                 } | ||||||
|  |                 _ => false, | ||||||
|  |             }; | ||||||
|  |             if cont { | ||||||
|  |                 execute!( | ||||||
|  |                     stdout, | ||||||
|  |                     cursor::RestorePosition, | ||||||
|  |                     style::Print(format!( | ||||||
|  |                         "[>>] Waiting for serial port {}\r", | ||||||
|  |                         if serial_toggle { "# " } else { " #" } | ||||||
|  |                     )) | ||||||
|  |                 )?; | ||||||
|  |                 stdout.flush()?; | ||||||
|  |                 serial_toggle = !serial_toggle; | ||||||
|  | 
 | ||||||
|  |                 if crossterm::event::poll(Duration::from_millis(1000))? { | ||||||
|  |                     if let Event::Key(KeyEvent { code, modifiers }) = crossterm::event::read()? { | ||||||
|  |                         if code == KeyCode::Char('c') && modifiers == KeyModifiers::CONTROL { | ||||||
|  |                             return Ok(()); | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |             return Err(e.into()); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         execute!( | ||||||
|  |             stdout, | ||||||
|  |             style::Print("\n[>>] Waiting for handshake, pass-through"), | ||||||
|  |         )?; | ||||||
|  |         stdout.flush()?; | ||||||
|  | 
 | ||||||
|  |         // Run in pass-through mode by default.
 | ||||||
|  |         // Once we receive BREAK (0x3) three times, switch to kernel send mode and upload kernel,
 | ||||||
|  |         // then switch back to pass-through mode.
 | ||||||
|  | 
 | ||||||
|  |         // Input from STDIN should pass through to serial
 | ||||||
|  |         // Input from serial should pass through to STDOUT
 | ||||||
|  | 
 | ||||||
|  |         let port = port?; | ||||||
|  | 
 | ||||||
|  |         if let Err(e) = main_loop(port, kernel.clone()).await { | ||||||
|  |             execute!(stdout, style::Print(format!("\nError: {:?}\n", e)))?; | ||||||
|  |             stdout.flush()?; | ||||||
|  | 
 | ||||||
|  |             let cont = match e.downcast_ref::<std::io::Error>() { | ||||||
|  |                 Some(e) | ||||||
|  |                     if e.kind() == std::io::ErrorKind::NotFound | ||||||
|  |                         || e.kind() == std::io::ErrorKind::PermissionDenied => | ||||||
|  |                 { | ||||||
|  |                     true | ||||||
|  |                 } | ||||||
|  |                 _ => false, | ||||||
|  |             } || matches!(e.downcast_ref::<tokio_serial::Error>(), Some(e) if e.kind == tokio_serial::ErrorKind::NoDevice) | ||||||
|  |                 || matches!( | ||||||
|  |                     e.downcast_ref::<tokio::sync::mpsc::error::SendError<Vec<u8>>>(), | ||||||
|  |                     Some(_) | ||||||
|  |                 ); | ||||||
|  | 
 | ||||||
|  |             if !cont { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             // main_loop() returned Ok() we're good to finish
 | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         execute!(stdout, cursor::SavePosition)?; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
							
								
								
									
										50
									
								
								build.ninja
								
								
								
								
							
							
						
						
									
										50
									
								
								build.ninja
								
								
								
								
							|  | @ -1,50 +0,0 @@ | ||||||
| TARGET = aarch64-vesper-metta |  | ||||||
| 
 |  | ||||||
| UTILS_CONTAINER = andrerichter/raspi3-utils |  | ||||||
| DOCKER_CMD = docker run -it --rm -v $$(pwd):/work -w /work |  | ||||||
| QEMU_CMD = qemu-system-aarch64 |  | ||||||
| 
 |  | ||||||
| # -d in_asm,unimp,int -S |  | ||||||
| QEMU_OPTS = -M raspi3 -d int -serial null -serial stdio |  | ||||||
| 
 |  | ||||||
| QEMU = /usr/local/Cellar/qemu/HEAD-3365de01b5-custom/bin/qemu-system-aarch64 |  | ||||||
| 
 |  | ||||||
| rule cargo_build |  | ||||||
|   command = cargo xbuild --target=targets/$TARGET.json --release $FEATURES |  | ||||||
|   description = Building kernel |  | ||||||
| 
 |  | ||||||
| rule strip |  | ||||||
|   command = cargo objcopy -- --strip-all -O binary $in $out |  | ||||||
|   description = Converting kernel ELF to binary |  | ||||||
| 
 |  | ||||||
| rule docker_qemulate |  | ||||||
|   command = $DOCKER_CMD $UTILS_CONTAINER $QEMU_CMD $QEMU_OPTS -kernel $in |  | ||||||
|   description = Running QEMU in Docker |  | ||||||
| 
 |  | ||||||
| rule qemulate |  | ||||||
|   command = $QEMU $QEMU_OPTS -kernel $in |  | ||||||
|   description = Running QEMU |  | ||||||
| 
 |  | ||||||
| rule copy_file |  | ||||||
|   command = cp -f $in /Volumes/BOOT/ |  | ||||||
|   description = Copying binary image to sdcard |  | ||||||
| 
 |  | ||||||
| # todo this doesn't have deps so builds always... |  | ||||||
| build target/${TARGET}/release/vesper | kernel: cargo_build |  | ||||||
|   FEATURES = --features "noserial" |  | ||||||
|   description = Building device kernel |  | ||||||
| 
 |  | ||||||
| #build target/${TARGET}/release/vesper qemu_kernel: cargo_build |  | ||||||
| #  FEATURES = |  | ||||||
| #  description = Building QEMU kernel |  | ||||||
| 
 |  | ||||||
| build kernel8.img: strip target/${TARGET}/release/vesper | kernel |  | ||||||
| 
 |  | ||||||
| build qemu_kernel8.img: strip target/${TARGET}/release/vesper | qemu_kernel |  | ||||||
| 
 |  | ||||||
| build qemu: qemulate qemu_kernel8.img |  | ||||||
| 
 |  | ||||||
| build device: copy_file kernel8.img |  | ||||||
| 
 |  | ||||||
| default kernel8.img |  | ||||||
| 
 |  | ||||||
|  | @ -1,5 +0,0 @@ | ||||||
| #!/bin/sh |  | ||||||
| cargo xbuild --target=targets/aarch64-vesper-metta.json --release --features "noserial" && \ |  | ||||||
| sh .cargo/runscript.sh target/aarch64-vesper-metta/release/vesper && \ |  | ||||||
| cp target/aarch64-vesper-metta/release/vesper.bin /Volumes/boot/vesper && \ |  | ||||||
| diskutil eject /Volumes/boot/ |  | ||||||
|  | @ -1,65 +1,7 @@ | ||||||
| # Contributor Covenant Code of Conduct | This is our Code of Conduct and Ethics (the “Code”) | ||||||
| 
 | 
 | ||||||
| ## Our Pledge | It is based on our values and aspirations (be good, innovate, work together, be part of the solution). It describes what we expect from the members of the community in three main rules: | ||||||
| 
 | 
 | ||||||
| In the interest of fostering an open and welcoming environment, we as | 1. Do the right thing. Always act with honesty, integrity, and reliability. Keep moral and ethical standards high. | ||||||
| contributors and maintainers pledge to making participation in our project and | 2. Be nice. Treat people with dignity and respect, regardless of who they are and where they came from. Stay decent and courteous in all relationships. | ||||||
| our community a harassment-free experience for everyone, regardless of age, body size, disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, | 3. Play fair. Don’t cheat. Be careful to account for and balance the interests of all groups (developers, artists and designers, end users and the general public). | ||||||
| personal appearance, race, religion, or sexual identity and orientation. |  | ||||||
| 
 |  | ||||||
| ## Our Standards |  | ||||||
| 
 |  | ||||||
| Examples of behavior that contributes to creating a positive environment include: |  | ||||||
| 
 |  | ||||||
| * Using welcoming and inclusive language |  | ||||||
| * Being respectful of differing viewpoints and experiences |  | ||||||
| * Gracefully accepting constructive criticism |  | ||||||
| * Focusing on what is best for the community |  | ||||||
| * Showing empathy towards other community members |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| Examples of unacceptable behavior by participants include: |  | ||||||
| 
 |  | ||||||
| * The use of sexualized language or imagery and unwelcome sexual attention or advances |  | ||||||
| * Trolling, insulting/derogatory comments, and personal or political attacks |  | ||||||
| * Public or private harassment |  | ||||||
| * Publishing others’ private information, such as a physical or electronic address, without explicit permission |  | ||||||
| * Other conduct which could reasonably be considered inappropriate in a professional setting |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| ## Our Responsibilities |  | ||||||
| 
 |  | ||||||
| Maintainers are responsible for clarifying the standards of acceptable behavior |  | ||||||
| and are expected to take appropriate and fair corrective action in response to |  | ||||||
| any instances of unacceptable behavior. |  | ||||||
| 
 |  | ||||||
| Maintainers have the right and responsibility to remove, edit, or reject |  | ||||||
| comments, commits, code, wiki edits, issues, and other contributions that are |  | ||||||
| not aligned to this Code of Conduct, or to ban temporarily or permanently any |  | ||||||
| contributor for other behaviors that they deem inappropriate, threatening, |  | ||||||
| offensive, or harmful. |  | ||||||
| 
 |  | ||||||
| ## Scope |  | ||||||
| 
 |  | ||||||
| This Code of Conduct applies both within project spaces and in public spaces |  | ||||||
| when an individual is representing the project or its community. Examples of |  | ||||||
| representing a project or community include using an official project e-mail |  | ||||||
| address, posting via an official social media account, or acting as an appointed representative at an online or offline event. Representation of a project may be further defined and clarified by project maintainers. |  | ||||||
| 
 |  | ||||||
| ## Enforcement |  | ||||||
| 
 |  | ||||||
| Instances of abusive, harassing, or otherwise unacceptable behavior may be |  | ||||||
| reported by contacting the Technical Advisory Board (TAB) at |  | ||||||
| <tab@metta.systems>. All complaints will be reviewed and |  | ||||||
| investigated and will result in a response that is deemed necessary and |  | ||||||
| appropriate to the circumstances. The TAB is obligated to maintain |  | ||||||
| confidentiality with regard to the reporter of an incident.  Further details of |  | ||||||
| specific enforcement policies may be posted separately. |  | ||||||
| 
 |  | ||||||
| Maintainers who do not follow or enforce the Code of Conduct in good faith may |  | ||||||
| face temporary or permanent repercussions as determined by other members of the |  | ||||||
| project’s leadership. |  | ||||||
| 
 |  | ||||||
| ## Attribution |  | ||||||
| 
 |  | ||||||
| This Code of Conduct is adapted from the Contributor Covenant, version 1.4, available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html |  | ||||||
|  |  | ||||||
										
											Binary file not shown.
										
									
								
							|  | @ -1,40 +0,0 @@ | ||||||
| # Canadian-Cross-build rustc for arvm7 RPi3 to build for aarch64-unknown |  | ||||||
| 
 |  | ||||||
| Japaric's explanations of [building rustc](https://www.reddit.com/r/rust/comments/5ag60z/how_do_i_bootstrap_rust_to_crosscompile_for_a_new/) with notes on used repos, namely `rust-buildbot` and crosstool-ng examples. |  | ||||||
| 
 |  | ||||||
| Crosstool for gcc toolchain:  `git clone git@github.com:asymptotik/crosstool-arm-osx.git` -- this is shit, avoid! |  | ||||||
| 
 |  | ||||||
| `https://stackoverflow.com/questions/50955843/linking-with-arm-linux-gnueabihf-gcc-failed-when-cross-compiling-a-rust-applic` |  | ||||||
| 
 |  | ||||||
| Crosstool for llvm toolchain: https://medium.com/@zw3rk/making-a-raspbian-cross-compilation-sdk-830fe56d75ba |  | ||||||
| 
 |  | ||||||
| ``` |  | ||||||
| ./configure --prefix="/Users/berkus/Hobby/Metta/cross-tools/rpi3-cross-llvm/prebuilt" \ |  | ||||||
|             --target=arm-linux-gnueabihf \ |  | ||||||
|             --enable-gold=yes \ |  | ||||||
|             --enable-ld=yes \ |  | ||||||
|             --enable-targets=arm-linux-gnueabihf \ |  | ||||||
|             --enable-multilib \ |  | ||||||
|             --enable-interwork \ |  | ||||||
|             --disable-werror \ |  | ||||||
|             --quiet |  | ||||||
| make && make install |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| ``` |  | ||||||
| mkdir sdkroot |  | ||||||
| rsync -rzLR --safe-links \ |  | ||||||
|       pi@172.20.10.2:/usr/lib/arm-linux-gnueabihf \ |  | ||||||
|       pi@172.20.10.2:/usr/lib/gcc/arm-linux-gnueabihf \ |  | ||||||
|       pi@172.20.10.2:/usr/include \ |  | ||||||
|       pi@172.20.10.2:/lib/arm-linux-gnueabihf \ |  | ||||||
|       sysroot/ |  | ||||||
| ``` |  | ||||||
| 
 |  | ||||||
| `cp /Users/berkus/Hobby/Metta/cross-tools/rpi3-cross-llvm/sysroot/usr/lib/gcc/arm-linux-gnueabihf/6.3.0/crt* /Users/berkus/Hobby/Metta/cross-tools/rpi3-cross-llvm/prebuilt/arm-linux-gnueabihf/lib/` |  | ||||||
| 
 |  | ||||||
| `cp -r /Users/berkus/Hobby/Metta/cross-tools/rpi3-cross-llvm/sysroot/usr/lib/gcc /Users/berkus/Hobby/Metta/cross-tools/rpi3-cross-llvm/prebuilt/arm-linux-gnueabihf/lib/` |  | ||||||
| 
 |  | ||||||
| `set -x PATH /Users/berkus/Hobby/Metta/cross-tools/rpi3-cross-llvm/prebuilt/bin $PATH` |  | ||||||
| `./x.py build --host armv7-unknown-linux-gnueabihf --target aarch64-unknown-none --stage 1 src/libtest` |  | ||||||
| 
 |  | ||||||
|  | @ -1,22 +0,0 @@ | ||||||
| The Device memory type has several attributes: |  | ||||||
| 
 |  | ||||||
|     G or nG - Gathering or non-Gathering. Multiple accesses to a device can be merged into a single transaction except for operations with memory ordering semantics, for example, memory barrier instructions, load acquire/store release. |  | ||||||
| 
 |  | ||||||
|     R or nR - Reordering. |  | ||||||
| 
 |  | ||||||
|     E or nE - Early Write Acknowledge (similar to bufferable). |  | ||||||
| 
 |  | ||||||
| Only four combinations of these attributes are valid: |  | ||||||
| 
 |  | ||||||
|     Device-nGnRnE  <-- "Strongly Ordered" |  | ||||||
|     Device-nGnRE   <-- "Device Memory" |  | ||||||
|     Device-nGRE |  | ||||||
|     Device-GRE |  | ||||||
| 
 |  | ||||||
| Typically peripheral control registers must be either `Device-nGnRE`, or `Device-nGnRnE`. This prevents reordering of the transactions in the programming sequences. |  | ||||||
| 
 |  | ||||||
| `Device-nGRE` and `Device-GRE` memory types can be useful for peripherals where memory access sequence and ordering does not affect results, for example, in bitmap or display buffers in a display interface. If the bus interface of such peripheral can only accept certain transfer sizes, the peripheral must be set to `Device-nGRE`. |  | ||||||
| 
 |  | ||||||
| Device memory is shareable, and must be cached. |  | ||||||
| 
 |  | ||||||
| [source](https://developer.arm.com/products/architecture/cpu-architecture/m-profile/docs/100699/latest/memory-type-definitions-in-armv8-m-architecture) |  | ||||||
|  | @ -1,6 +1,9 @@ | ||||||
|  | # This is a config.txt file that should be on RPi SD boot partition | ||||||
|  | 
 | ||||||
| cec_osd_name=Jellyfish | cec_osd_name=Jellyfish | ||||||
| 
 | 
 | ||||||
| # Keep rainbow splash screen on boot | # Keep rainbow splash screen on boot | ||||||
|  | # Helps in debugging boot problems. | ||||||
| disable_splash=0 | disable_splash=0 | ||||||
| 
 | 
 | ||||||
| # Set jtag debug pins to alt4, uses GPIO26 for TDI | # Set jtag debug pins to alt4, uses GPIO26 for TDI | ||||||
|  |  | ||||||
|  | @ -1,46 +0,0 @@ | ||||||
| # |  | ||||||
| # Docker image naotaco/openocd:armv8 configures JTAG similar to this |  | ||||||
| # Extracted via: |  | ||||||
| # $ docker run --rm naotaco/openocd:armv8 /bin/sh -c "cat /usr/local/share/openocd/scripts/target/rpi3.cfg" |  | ||||||
| # |  | ||||||
| transport select jtag |  | ||||||
| 
 |  | ||||||
| # we need to enable srst even though we don't connect it |  | ||||||
| reset_config trst_and_srst |  | ||||||
| 
 |  | ||||||
| adapter_khz 4000 |  | ||||||
| jtag_ntrst_delay 500 |  | ||||||
| 
 |  | ||||||
| if { [info exists CHIPNAME] } { |  | ||||||
|   set _CHIPNAME $CHIPNAME |  | ||||||
| } else { |  | ||||||
|   set _CHIPNAME rpi3 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| # |  | ||||||
| # Main DAP |  | ||||||
| # |  | ||||||
| if { [info exists DAP_TAPID] } { |  | ||||||
|    set _DAP_TAPID $DAP_TAPID |  | ||||||
| } else { |  | ||||||
|    set _DAP_TAPID 0x4ba00477 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| jtag newtap $_CHIPNAME dap -irlen 4 -ircapture 0x1 -irmask 0xf -expected-id $_DAP_TAPID -enable |  | ||||||
| 
 |  | ||||||
| set _TARGETNAME $_CHIPNAME.cpu |  | ||||||
| 
 |  | ||||||
| set DBGBASE {0x80010000 0x80012000 0x80014000 0x80016000} |  | ||||||
| set CTIBASE {0x80018000 0x80019000 0x8001a000 0x8001b000} |  | ||||||
| set _cores 4 |  | ||||||
| 
 |  | ||||||
| for { set _core 0 } { $_core < $_cores } { incr _core } { |  | ||||||
| 
 |  | ||||||
|     target create $_TARGETNAME.$_core aarch64 \ |  | ||||||
|         -chain-position $_CHIPNAME.dap -coreid $_core \ |  | ||||||
|         -dbgbase [lindex $DBGBASE $_core] -ctibase [lindex $CTIBASE $_core] |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| # In contrast with SUSE config, naotaco only configures core0 events |  | ||||||
| $_TARGETNAME.0 configure -event reset-assert-post "aarch64 dbginit" |  | ||||||
| $_TARGETNAME.0 configure -event gdb-attach { halt } |  | ||||||
|  | @ -1,48 +0,0 @@ | ||||||
| # |  | ||||||
| # https://www.suse.com/c/debugging-raspberry-pi-3-with-jtag/ configures JTAG like this |  | ||||||
| # |  | ||||||
| transport select jtag |  | ||||||
| 
 |  | ||||||
| # we need to enable srst even though we don't connect it |  | ||||||
| reset_config trst_and_srst |  | ||||||
| 
 |  | ||||||
| adapter_khz 4000 |  | ||||||
| jtag_ntrst_delay 500 |  | ||||||
| 
 |  | ||||||
| if { [info exists CHIPNAME] } { |  | ||||||
|   set _CHIPNAME $CHIPNAME |  | ||||||
| } else { |  | ||||||
|   set _CHIPNAME rpi3 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| # |  | ||||||
| # Main DAP |  | ||||||
| # |  | ||||||
| if { [info exists DAP_TAPID] } { |  | ||||||
|    set _DAP_TAPID $DAP_TAPID |  | ||||||
| } else { |  | ||||||
|    set _DAP_TAPID 0x4ba00477 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| jtag newtap $_CHIPNAME tap -irlen 4 -ircapture 0x1 -irmask 0xf -expected-id $_DAP_TAPID -enable |  | ||||||
| dap create $_CHIPNAME.dap -chain-position $_CHIPNAME.tap |  | ||||||
| 
 |  | ||||||
| set _TARGETNAME $_CHIPNAME.a53 |  | ||||||
| set _CTINAME $_CHIPNAME.cti |  | ||||||
| 
 |  | ||||||
| set DBGBASE {0x80010000 0x80012000 0x80014000 0x80016000} |  | ||||||
| set CTIBASE {0x80018000 0x80019000 0x8001a000 0x8001b000} |  | ||||||
| set _cores 4 |  | ||||||
| 
 |  | ||||||
| for { set _core 0 } { $_core < $_cores } { incr _core } { |  | ||||||
| 
 |  | ||||||
|     cti create $_CTINAME.$_core -dap $_CHIPNAME.dap -ap-num 0 \ |  | ||||||
|         -ctibase [lindex $CTIBASE $_core] |  | ||||||
| 
 |  | ||||||
|     target create $_TARGETNAME.$_core aarch64 \ |  | ||||||
|         -dap $_CHIPNAME.dap -coreid $_core \ |  | ||||||
|         -dbgbase [lindex $DBGBASE $_core] -cti $_CTINAME.$_core |  | ||||||
| 
 |  | ||||||
|     $_TARGETNAME.$_core configure -event reset-assert-post "aarch64 dbginit" |  | ||||||
|     $_TARGETNAME.$_core configure -event gdb-attach { halt } |  | ||||||
| } |  | ||||||
							
								
								
									
										187
									
								
								doc/rpi3_jtag.md
								
								
								
								
							
							
						
						
									
										187
									
								
								doc/rpi3_jtag.md
								
								
								
								
							|  | @ -1,17 +1,19 @@ | ||||||
| JTAG boards: | # Connecting RPi3 JTAG | ||||||
|  | 
 | ||||||
|  | Possible JTAG boards: | ||||||
| 
 | 
 | ||||||
| * RasPi3 | * RasPi3 | ||||||
| * Segger J-Link V9 | * Segger J-Link V9 | ||||||
| * TinCanTools Flyswatter | * TinCanTools Flyswatter | ||||||
| * OpenMoko DebugBoard_v3 - [one i have](http://wiki.openmoko.org/wiki/Debug_Board_v3) | * OpenMoko DebugBoard_v3 - [this is the version have](http://wiki.openmoko.org/wiki/Debug_Board_v3) | ||||||
| 
 | 
 | ||||||
| # RPi3 to RPi3 jtag | ## RPi3 to RPi3 JTAG | ||||||
| 
 | 
 | ||||||
| Helpful RPi3 GPIO header pinouts from element14 [for Model B](https://www.element14.com/community/docs/DOC-73950/l/raspberry-pi-3-model-b-gpio-40-pin-block-pinout) and [here for Model B+](https://www.element14.com/community/docs/DOC-88824/l/raspberry-pi-3-model-b-gpio-40-pin-block-poe-header-pinout) (which is the same). | Helpful RPi3 GPIO header pinouts from element14 [for Model B](https://www.element14.com/community/docs/DOC-73950/l/raspberry-pi-3-model-b-gpio-40-pin-block-pinout) and [here for Model B+](https://www.element14.com/community/docs/DOC-88824/l/raspberry-pi-3-model-b-gpio-40-pin-block-poe-header-pinout) (they are the same). | ||||||
| 
 | 
 | ||||||
| ## Host configuration: | ### Host configuration: | ||||||
| 
 | 
 | ||||||
| These are regular GPIO functions, which we specify in OpenOCD interface configuration to enable driving JTAG interface. | These are regular GPIO functions, which we specify in OpenOCD interface configuration to enable driving JTAG interface. We should be able to choose pretty much any available pins. | ||||||
| 
 | 
 | ||||||
| ``` | ``` | ||||||
| FUNC  |  GPIO  |  PIN # | FUNC  |  GPIO  |  PIN # | ||||||
|  | @ -28,7 +30,7 @@ RPi doesn't expose SRST so we ignore it. | ||||||
| 
 | 
 | ||||||
| [Source](https://movr0.com/2016/09/02/use-raspberry-pi-23-as-a-jtagswd-adapter/) | [Source](https://movr0.com/2016/09/02/use-raspberry-pi-23-as-a-jtagswd-adapter/) | ||||||
| 
 | 
 | ||||||
| ## Target configuration: | ### Target configuration: | ||||||
| 
 | 
 | ||||||
| These are real JTAG pins of bcm2837, enabled on target RPi via config.txt options (see below). | These are real JTAG pins of bcm2837, enabled on target RPi via config.txt options (see below). | ||||||
| 
 | 
 | ||||||
|  | @ -55,13 +57,13 @@ In config.txt: | ||||||
| gpio=22-27=a4 | gpio=22-27=a4 | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| Alternatively, just specify @todo - verify this works with all alt4 pins | Alternatively, just specify this: (@todo verify this works with all alt4 pins) | ||||||
| 
 | 
 | ||||||
| ``` | ``` | ||||||
| enable_jtag_gpio=1 | enable_jtag_gpio=1 | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ## Connection between boards | ### Wire Connection between boards | ||||||
| 
 | 
 | ||||||
| ``` | ``` | ||||||
| Func | Host Pin | Wire color | Target pin | Func | Host Pin | Wire color | Target pin | ||||||
|  | @ -74,25 +76,24 @@ TRST |    26    |    red     |    15 | ||||||
| GND  |    20    |   black    |    20 | GND  |    20    |   black    |    20 | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| 
 | ### OpenOCD configuration on the host | ||||||
| ## OpenOCD configuration on the host |  | ||||||
| 
 | 
 | ||||||
| You need two files: interface file for driving the host GPIO correctly, and target file for detecting the JTAG circuitry on the target RPi. | You need two files: interface file for driving the host GPIO correctly, and target file for detecting the JTAG circuitry on the target RPi. | ||||||
| 
 | 
 | ||||||
| Interface configuration: [rpi3_interface.cfg](./rpi3_interface.cfg) | Interface configuration: [rpi3_interface.cfg](./rpi2rpi_jtag/rpi3_interface.cfg) | ||||||
| 
 | 
 | ||||||
| [Source](https://movr0.com/2016/09/02/use-raspberry-pi-23-as-a-jtagswd-adapter/), [source #2 - rpi3 speed_coeffs](https://forum.doozan.com/read.php?3,21789) | [Source](https://movr0.com/2016/09/02/use-raspberry-pi-23-as-a-jtagswd-adapter/), [source #2 - rpi3 speed_coeffs](https://forum.doozan.com/read.php?3,21789) | ||||||
| 
 | 
 | ||||||
| Target configuration: [rpi3_target.cfg](./rpi3_target.cfg) | Target configuration: [rpi3_target.cfg](./rpi2rpi_jtag/rpi3_target.cfg) | ||||||
| 
 | 
 | ||||||
| [Source #1](https://electronics.stackexchange.com/questions/249008/how-to-use-rpi-2-to-debug-rpi-model-b-via-jtag-with-openocd/419724#419724), [source #2](https://sysprogs.com/tutorials/preparing-raspberry-pi-for-jtag-debugging/), [source #3](http://openocd.org/doc/html/Reset-Configuration.html), [source #4](http://infocenter.arm.com/help/topic/com.arm.doc.faqs/ka3854.html), [source #5](https://www.raspberrypi.org/forums/viewtopic.php?p=1013802), [source #6 - proper rpi3 ocd config](https://www.suse.com/c/debugging-raspberry-pi-3-with-jtag/), [source #7 - simpler rpi3 ocd config](https://github.com/daniel-k/openocd/blob/armv8/tcl/target/rpi3.cfg), [source #8 - explanations about SRST](https://catch22.eu/baremetal/openocd_sysfs_stm32/), [source #9 - example RPi target config](https://github.com/OP-TEE/build/blob/master/rpi3/debugger/pi3.cfg), [source #10 - some JTAG debug hints on rpi](https://www.raspberrypi.org/forums/viewtopic.php?p=1013802), [source #11 - jtag vs swd and CoreSight info links](https://electronics.stackexchange.com/questions/53571/jtag-vs-swd-debugging?rq=1) | [Source #1](https://electronics.stackexchange.com/questions/249008/how-to-use-rpi-2-to-debug-rpi-model-b-via-jtag-with-openocd/419724#419724), [source #2](https://sysprogs.com/tutorials/preparing-raspberry-pi-for-jtag-debugging/), [source #3](http://openocd.org/doc/html/Reset-Configuration.html), [source #4](http://infocenter.arm.com/help/topic/com.arm.doc.faqs/ka3854.html), [source #5](https://www.raspberrypi.org/forums/viewtopic.php?p=1013802), [source #6 - proper rpi3 ocd config](https://www.suse.com/c/debugging-raspberry-pi-3-with-jtag/), [source #7 - simpler rpi3 ocd config](https://github.com/daniel-k/openocd/blob/armv8/tcl/target/rpi3.cfg), [source #8 - explanations about SRST](https://catch22.eu/baremetal/openocd_sysfs_stm32/), [source #9 - example RPi target config](https://github.com/OP-TEE/build/blob/master/rpi3/debugger/pi3.cfg), [source #10 - some JTAG debug hints on rpi](https://www.raspberrypi.org/forums/viewtopic.php?p=1013802), [source #11 - jtag vs swd and CoreSight info links](https://electronics.stackexchange.com/questions/53571/jtag-vs-swd-debugging?rq=1) | ||||||
| 
 | 
 | ||||||
| > If an SoC provides a JTAG debug interface and contains any CoreSight debug components (including any Cortex processor) you should expect to see the standard JTAG IDCODE of a single CoreSight SWJ-DP as one TAP on the JTAG chain. | > If a SoC provides a JTAG debug interface and contains any CoreSight debug components (including any Cortex processor) you should expect to see the standard JTAG IDCODE of a single CoreSight SWJ-DP as one TAP on the JTAG chain. | ||||||
| 
 | 
 | ||||||
| 
 | ### Run OpenOCD, GDB and attach to target | ||||||
| ## Run OpenOCD, GDB and attach to target |  | ||||||
| 
 | 
 | ||||||
| Need to verify if the following bug is still valid: | Need to verify if the following bug is still valid: | ||||||
|  | 
 | ||||||
| > There is a bug in OpenOCD that will prevent Raspberry PI from continuing correctly after a stop unless the initialization is done twice. Close OpenOCD with Ctrl-C and re-run it again. Now the debugging will be usable. | > There is a bug in OpenOCD that will prevent Raspberry PI from continuing correctly after a stop unless the initialization is done twice. Close OpenOCD with Ctrl-C and re-run it again. Now the debugging will be usable. | ||||||
| 
 | 
 | ||||||
| [Source](https://sysprogs.com/tutorials/preparing-raspberry-pi-for-jtag-debugging/) | [Source](https://sysprogs.com/tutorials/preparing-raspberry-pi-for-jtag-debugging/) | ||||||
|  | @ -112,51 +113,157 @@ If `stepi` command causes CPU to make one instruction step, everything is workin | ||||||
| 
 | 
 | ||||||
| [Source](https://sysprogs.com/tutorials/preparing-raspberry-pi-for-jtag-debugging/), [source #2](https://www.op-tee.org/docs/rpi3/#6-openocd-and-jtag), [source #3 - monitor reset halt](http://www.openstm32.org/forumthread823) | [Source](https://sysprogs.com/tutorials/preparing-raspberry-pi-for-jtag-debugging/), [source #2](https://www.op-tee.org/docs/rpi3/#6-openocd-and-jtag), [source #3 - monitor reset halt](http://www.openstm32.org/forumthread823) | ||||||
| 
 | 
 | ||||||
| 
 | I got RPi3-to-RPi3 JTAG working and even debugged a bit directly on the CPU, but a few things make it not an ideal experience: | ||||||
| I got RPi3-to-RPi3 JTAG working and even debugged a bit directly on the CPU, but a few things impeded my happiness: |  | ||||||
| 
 | 
 | ||||||
| * RPi is a bit too slow for bitbanging and oftentimes opening a browser window, or running some other command caused OpenOCD to spew JTAG synchronization errors. | * RPi is a bit too slow for bitbanging and oftentimes opening a browser window, or running some other command caused OpenOCD to spew JTAG synchronization errors. | ||||||
| * To properly debug my kernel from RPi I would need to compile it locally (otherwise all the paths in the debug info are wrong and GDB will not find the source files, I did not want to mess around with symlinks). | * To properly debug my kernel from RPi I would need to compile it locally (otherwise all the paths in the debug info are wrong and GDB will not find the source files, I did not want to mess around with symlinks). Compiling rust on rpi3 is _slow_. | ||||||
| 
 | 
 | ||||||
| Fortunately, at this point a Segger J-Link 9 arrived and I went to use it. | Fortunately, at this point a Segger J-Link 9 arrived and I went to use it. | ||||||
| 
 | 
 | ||||||
| # J-Link to RPi3 jtag | ## J-Link to RPi3 JTAG | ||||||
| 
 | 
 | ||||||
| https://www.segger.com/downloads/jlink/ | > https://www.segger.com/downloads/jlink/ | ||||||
| https://habr.com/ru/post/259205/ |  | ||||||
| 
 | 
 | ||||||
| JTAG pinout on segger is in UM08001_JLink.pdf distributed with the J-Link software kit, in section 17.1.1. | > https://habr.com/ru/post/259205/ | ||||||
|  | 
 | ||||||
|  | JTAG pinout on JLink is in UM08001_JLink.pdf distributed with the J-Link software kit, in section `18.1.1 Pinout for JTAG`. | ||||||
|  | 
 | ||||||
|  | Reproduced here in ASCII: | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  |        +-----------------+ | ||||||
|  | VTRef  |  1 *       * 2  | NC | ||||||
|  | nTRST  |  3 *       * 4  | GND | ||||||
|  | TDI    |  5 *       * 6  | GND | ||||||
|  | TMS    |  7 *       * 8  | GND | ||||||
|  | TCK   ||  9 *       * 10 | GND | ||||||
|  | RTCK  || 11 *       * 12 | GND | ||||||
|  | TDO    | 13 *       * 14 | * | ||||||
|  | RESET  | 15 *       * 16 | * | ||||||
|  | DBGRQ  | 17 *       * 18 | * | ||||||
|  | +5V    | 19 *       * 20 | * | ||||||
|  |        +-----------------+ | ||||||
|  | ``` | ||||||
| 
 | 
 | ||||||
| This adds VTref for target voltage detection. | This adds VTref for target voltage detection. | ||||||
|  | Additionally, with this pinout J-Link is able to power and boot up RPi3 board itself! | ||||||
| 
 | 
 | ||||||
| Pinout: | Explanation of pins from J-Link manual: | ||||||
| 
 | 
 | ||||||
| J-Link and connection to Raspi3: | <table> | ||||||
|  |     <thead> | ||||||
|  |     <tr> | ||||||
|  |         <th>Pin</th> | ||||||
|  |         <th>Signal</th> | ||||||
|  |         <th>Direction</th> | ||||||
|  |         <th>Description</th> | ||||||
|  |     </tr> | ||||||
|  |     </thead> | ||||||
|  |     <tbody> | ||||||
|  |     <tr> | ||||||
|  |         <td>1</td> | ||||||
|  |         <td>VTref</td> | ||||||
|  |         <td>Input</td> | ||||||
|  |         <td>This is the target reference voltage. It is used to check if the target has power, to create the logic-level reference for the input comparators and to control the output logic levels to the target. It is normally fed from VDD of the target board and must not have a series resistor.</td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |         <td>2</td> | ||||||
|  |         <td>NC</td> | ||||||
|  |         <td>Not connected</td> | ||||||
|  |         <td>This pin is not connected in J-Link.</td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |         <td>3</td> | ||||||
|  |         <td>nTRST</td> | ||||||
|  |         <td>Output</td> | ||||||
|  |         <td>JTAG Reset. Output from J-Link to the Reset signal of the target JTAG port. Typically connected to nTRST of the target CPU. This pin is normally pulled HIGH on the target to avoid unintentional resets when there is no connection.</td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |         <td>4, 6, 8, 10, 12</td> | ||||||
|  |         <td>GND</td> | ||||||
|  |         <td>Ground</td> | ||||||
|  |         <td>Pins connected to GND in J-Link. They should also be connected to GND in the target system.</td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |         <td>5</td> | ||||||
|  |         <td>TDI</td> | ||||||
|  |         <td>Output</td> | ||||||
|  |         <td>JTAG data input of target CPU. It is recommended that this pin is pulled to a defined state on the target board. Typically connected to TDI of the target CPU.</td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |         <td>7</td> | ||||||
|  |         <td>TMS</td> | ||||||
|  |         <td>Output</td> | ||||||
|  |         <td>JTAG mode set input of target CPU. This pin should be pulled up on the target. Typically connected to TMS of the target CPU.</td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |         <td>9</td> | ||||||
|  |         <td>TCK</td> | ||||||
|  |         <td>Output</td> | ||||||
|  |         <td>JTAG clock signal to target CPU. It is recommended that this pin is pulled to a defined state of the target board. Typically connected to TCK of the target CPU.</td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |         <td>11</td> | ||||||
|  |         <td>RTCK</td> | ||||||
|  |         <td>Input</td> | ||||||
|  |         <td>Return test clock signal from the target. Some targets must synchronize the JTAG inputs to internal clocks. To assist in meeting this requirement, you can use a returned, and re-timed, TCK to dynamically control the TCK rate. J-Link supports adaptive clocking, which waits for TCK changes to be echoed correctly before making further changes. Connect to RTCK if available, <b>otherwise to GND.</b></td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |         <td>13</td> | ||||||
|  |         <td>TDO</td> | ||||||
|  |         <td>Input</td> | ||||||
|  |         <td>JTAG data output from target CPU. Typically connected to TDO of the target CPU.</td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |         <td>15</td> | ||||||
|  |         <td>nRESET</td> | ||||||
|  |         <td>I/O</td> | ||||||
|  |         <td>Target CPU reset signal. Typically connected to the RESET pin of the target CPU, which is typically called “nRST”, “nRESET” or “RESET”. This signal is an active low signal.</td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |         <td>17</td> | ||||||
|  |         <td>DBGRQ</td> | ||||||
|  |         <td>Not connected</td> | ||||||
|  |         <td>This pin is not connected in J-Link</td> | ||||||
|  |     </tr> | ||||||
|  |     <tr> | ||||||
|  |         <td>19</td> | ||||||
|  |         <td>5V-Supply</td> | ||||||
|  |         <td>Output</td> | ||||||
|  |         <td>This pin can be used to supply power to the target hardware. Older J-Links may not be able to supply power on this pin.</td> | ||||||
|  |     </tr> | ||||||
|  |     </tbody> | ||||||
|  | </table> | ||||||
|  | 
 | ||||||
|  | ### J-Link wire connection with RPi3 | ||||||
| 
 | 
 | ||||||
| ``` | ``` | ||||||
| Func  |  J-Link Pin  | Wire color  | Target pin | Func  |  J-Link Pin  | Wire color  | Target pin | Target GPIO | Target Func | ||||||
| ------+--------------+-------------+----------- | ------+--------------+-------------+------------+-------------+------------- | ||||||
| VTref |       1      | white       |  1 | VTref |       1      | white       |  1         |             | | ||||||
| TCK   |       9      | yellow      | 22 | nTRST |       3      | red         | 15         | GPIO22      | Alt4 | ||||||
| TMS   |       7      | brown       | 13 | TDI   |       5      | green       | 37         | GPIO26      | Alt4 | ||||||
| TDI   |       5      | green       | 37 | TMS   |       7      | brown       | 13         | GPIO27      | Alt4 | ||||||
| TDO   |      13      | orange      | 18 | TCK   |       9      | yellow      | 22         | GPIO25      | Alt4 | ||||||
| nTRST |       3      | red         | 15 | RTCK  |      11      | magenta     | 16         | GPIO23      | Alt4 | ||||||
| RTCK  |      11      | magenta     | 16 | TDO   |      13      | orange      | 18         | GPIO24      | Alt4 | ||||||
| GND   |       4      | black       | 20 | GND   |       4      | black       | 20, 14     |             | | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
|  | [Useful article](https://www.suse.com/c/debugging-raspberry-pi-3-with-jtag/). | ||||||
| 
 | 
 | ||||||
| [Useful article](https://www.suse.com/c/debugging-raspberry-pi-3-with-jtag/) | ### Run with OpenOCD | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
| Rebuild openocd from git and voila, it works with  | Rebuild openocd from git and voila, it works with  | ||||||
| 
 | 
 | ||||||
| `openocd -f interface/jlink.cfg -f rpi3_jtag.cfg` | `openocd -f interface/jlink.cfg -f rpi3_jtag.cfg` | ||||||
| 
 | 
 | ||||||
|  | ### Run with probe-rs | ||||||
| 
 | 
 | ||||||
| # Andre Richter's tutorials | To be written when probe-rs starts supporting RPi3/4. | ||||||
| 
 | 
 | ||||||
| Finally, while I was messing with all this, Andre Richter has created another entry in his excellent Raspi tutorials dedicated exactly to [JTAG debugging](fixme) and he has also provided useful docker containers with configured [gdb-dashboard](https://github.com/rust-embedded/rust-raspi3-OS-tutorials/blob/JTAG/docker/raspi3-gdb/) | ## Andre Richter's tutorials | ||||||
| 
 | 
 | ||||||
| So debugging is a lot easier now - just drop specifically-built JTAG enabler binary to sdcard, connect over JTAG via openocd and gdb and go load your kernel! | Andre Richter has created an entry in his excellent RPi tutorials dedicated exactly to [JTAG debugging](https://github.com/rust-embedded/rust-raspberrypi-OS-tutorials/tree/master/09_hw_debug_JTAG). | ||||||
|  | 
 | ||||||
|  | So debugging is a lot easier now - just drop [specifically-built JTAG enabler](https://github.com/rust-embedded/rust-raspberrypi-OS-tutorials/tree/master/X1_JTAG_boot) binary to sdcard, connect over JTAG via openocd and gdb and go load your kernel! | ||||||
|  |  | ||||||
|  | @ -2,34 +2,31 @@ | ||||||
| 
 | 
 | ||||||
| ## Using cp2104 usb-to-ttl converter | ## Using cp2104 usb-to-ttl converter | ||||||
| 
 | 
 | ||||||
| Download drivers from: https://www.silabs.com/products/development-tools/software/usb-to-uart-bridge-vcp-drivers | Download drivers from [SiLabs driver page](https://www.silabs.com/products/development-tools/software/usb-to-uart-bridge-vcp-drivers). | ||||||
| 
 | 
 | ||||||
| Install `minicom`: `brew install minicom` | Install `minicom`: `brew install minicom` | ||||||
| 
 | 
 | ||||||
| Configure minicom to use `/dev/tty.SLAB_USBtoUART` port. | Configure minicom to use `/dev/tty.SLAB_USBtoUART` port. On macOS Big Sur it might be `/dev/tty.usbserial-019586E3` or similar depending on the serial number of the adapter. | ||||||
| 
 | 
 | ||||||
| Connect rpi wires to cp2014: | Connect RPi wires to cp2014: | ||||||
|  | 
 | ||||||
|  | _NB:_ Swap the TXD and RXD wires. I.e. RXD pin of CP2104 should go to TXD pin on RPi and vice versa. | ||||||
| 
 | 
 | ||||||
| ``` | ``` | ||||||
| UART0 | UART0 | ||||||
| 
 | 
 | ||||||
| FUNC  |  GPIO  |  PIN #  |  MODE  | Wire color | RPi Func  |  RPi GPIO  |  PIN #  |  MODE  | CP2104 Pin | Wire color | ||||||
| ------+--------+---------+--------+------------ | ----------+------------+---------+--------+------------+------------ | ||||||
| RXD0  | GPIO15 |   10    |  Alt0  | Brown | RXD0      | GPIO15     |   10    |  Alt0  | TXD        | Red | ||||||
| TXD0  | GPIO14 |    8    |  Alt0  | Red | TXD0      | GPIO14     |    8    |  Alt0  | RXD        | Brown | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ``` | ``` | ||||||
| MiniUart (UART1) | MiniUart (UART1) | ||||||
| 
 | 
 | ||||||
| FUNC  |  GPIO  |  PIN #  |  MODE  | Wire color | RPi Func  |  RPi GPIO  |  PIN #  |  MODE  | CP2104 Pin | Wire color | ||||||
| ------+--------+---------+--------+------------ | ----------+------------+---------+--------+------------+------------ | ||||||
| RXD1  | GPIO15 |   10    |  Alt5  | Brown | RXD1      | GPIO15     |   10    |  Alt5  | TXD        | Red | ||||||
| TXD1  | GPIO14 |    8    |  Alt5  | Red | TXD1      | GPIO14     |    8    |  Alt5  | RXD        | Brown | ||||||
| GND   | GND    |    6    |        | Green | GND       | GND        |    6    |        | GND        | Green | ||||||
| ``` | ``` | ||||||
| 
 |  | ||||||
| NB: SWAP the TXD and RXD wires. |  | ||||||
| 
 |  | ||||||
| I.e. RXD pin of CP2104 should go to TXD pin on RPi and vice versa. |  | ||||||
| 
 |  | ||||||
|  |  | ||||||
|  | @ -0,0 +1 @@ | ||||||
|  | zellij-config.sh | ||||||
|  | @ -1,14 +1,12 @@ | ||||||
| ######################################################### | ######################################################### | ||||||
| #         Welcome to the Raspi3 Emulation with          # | #         Welcome to the RasPi3 Emulation with          # | ||||||
| #               MiniUart and PL011UART                  # | #               MiniUart and PL011UART                  # | ||||||
| #                                                       # | #                                                       # | ||||||
| #                                                       # | #                                                       # | ||||||
| # 1. Use the mouse to select different panes            # | # 1. Use the mouse to select different panes            # | ||||||
| #    and interact with the UARTS.                       # | #    and interact with the UARTs.                       # | ||||||
| #                                                       # | #                                                       # | ||||||
| # 2. To quit the emulation:                             # | # 2. To quit the emulation:                             # | ||||||
| #    2.1 Select the pane in which QEMU runs (this one). # | #    2.1 Select the pane in which QEMU runs (this one). # | ||||||
| #    2.2 CTRL-C to close qemu.                          # | #    2.2 CTRL-C to close qemu.                          # | ||||||
| ######################################################### | ######################################################### | ||||||
| 
 |  | ||||||
| === QEMU === |  | ||||||
|  |  | ||||||
|  | @ -0,0 +1,39 @@ | ||||||
|  | --- | ||||||
|  | template: | ||||||
|  |   direction: Horizontal | ||||||
|  |   parts: | ||||||
|  |     - direction: Vertical | ||||||
|  |       borderless: true | ||||||
|  |       split_size: | ||||||
|  |         Fixed: 1 | ||||||
|  |       run: | ||||||
|  |         plugin: | ||||||
|  |           location: "zellij:tab-bar" | ||||||
|  |     - direction: Vertical | ||||||
|  |       body: true | ||||||
|  | 
 | ||||||
|  | tabs: | ||||||
|  |   - direction: Vertical | ||||||
|  |     parts: | ||||||
|  |       - direction: Horizontal | ||||||
|  |         borderless: true | ||||||
|  |         run: | ||||||
|  |           command: | ||||||
|  |             cmd: "bash" | ||||||
|  |             args: ["-c", "bash emulation/qemu_multi_uart.sh"] | ||||||
|  |       - direction: Horizontal | ||||||
|  |         parts: | ||||||
|  |           - direction: Vertical | ||||||
|  |             split_size: | ||||||
|  |               Percent: 30 | ||||||
|  |             run: | ||||||
|  |               command: | ||||||
|  |                 cmd: "bash" | ||||||
|  |                 args: ["-c", "clear; echo -e \"\\033]0;MiniUart\\007\"; bash /dev/ptmx FIRST=1"] | ||||||
|  |           - direction: Vertical | ||||||
|  |             split_size: | ||||||
|  |               Percent: 70 | ||||||
|  |             run: | ||||||
|  |               command: | ||||||
|  |                 cmd: "bash" | ||||||
|  |                 args: ["-c", "clear; echo -e \"\\033]0;PL011 Uart\\007\"; bash /dev/ptmx SECOND=1"] | ||||||
|  | @ -1,46 +1,35 @@ | ||||||
| #!/usr/bin/env bash | #!/bin/bash | ||||||
| # | # | ||||||
| # MIT License | # MIT License | ||||||
| # | # | ||||||
| # Copyright (c) 2018-2019 Andre Richter <andre.o.richter@gmail.com> | # Copyright (c) 2018-2019 Berkus Decker <berkus+vesper@metta.systems> | ||||||
| # | # Based on QEMU runner example by Andre Richter <andre.o.richter@gmail.com> | ||||||
| # Permission is hereby granted, free of charge, to any person obtaining a copy |  | ||||||
| # of this software and associated documentation files (the "Software"), to deal |  | ||||||
| # in the Software without restriction, including without limitation the rights |  | ||||||
| # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |  | ||||||
| # copies of the Software, and to permit persons to whom the Software is |  | ||||||
| # furnished to do so, subject to the following conditions: |  | ||||||
| # |  | ||||||
| # The above copyright notice and this permission notice shall be included in all |  | ||||||
| # copies or substantial portions of the Software. |  | ||||||
| # |  | ||||||
| # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |  | ||||||
| # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |  | ||||||
| # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |  | ||||||
| # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |  | ||||||
| # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |  | ||||||
| # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |  | ||||||
| # SOFTWARE. |  | ||||||
| # |  | ||||||
| 
 | 
 | ||||||
| QEMU=/usr/local/Cellar/qemu/HEAD-3365de01b5-custom/bin/qemu-system-aarch64 | echo -e "\033]0;QEMU\007" | ||||||
| QEMU_OPTS=-M raspi3 -d int |  | ||||||
| 
 | 
 | ||||||
| tmux new-session -d -s raspi3 & | clear | ||||||
|  | cat emulation/instructions.txt | ||||||
|  | 
 | ||||||
|  | # Generated by `cargo make zellij-config` | ||||||
|  | source emulation/zellij-config.sh | ||||||
|  | 
 | ||||||
|  | # Let the two other serials boot up | ||||||
| sleep 1 | sleep 1 | ||||||
| tmux new-window -t raspi3:1 |  | ||||||
| tmux new-window -t raspi3:2 |  | ||||||
| 
 | 
 | ||||||
| tmux send-keys -t raspi3:0 "clear; echo '=== MiniUart ==='; bash /dev/ptmx" C-m | FIRST_TTY=$(ps a | grep -e [F]IRST | awk '{ print $2 }') | ||||||
| tmux send-keys -t raspi3:1 "clear; printf '=== PL011 Uart ===\n\n';bash /dev/ptmx" C-m | SECOND_TTY=$(ps a | grep -e [S]ECOND | awk '{ print $2 }') | ||||||
| 
 | 
 | ||||||
| FIRST=$(ps aux | grep ptmx | sort | awk '{print $7}' | sed '1q;d') | FIRST_PID=$(ps a | grep -e [F]IRST | awk '{ print $1 }') | ||||||
| SECOND=$(ps aux | grep ptmx | sort | awk '{print $7}' | sed '2q;d') | SECOND_PID=$(ps a | grep -e [S]ECOND | awk '{ print $1 }') | ||||||
| 
 | 
 | ||||||
| tmux send-keys -t raspi3:2 "clear; cat emulation/instructions.txt && ${QEMU} ${QEMU_OPTS} -kernel kernel8.img -serial /dev/$SECOND -serial /dev/$FIRST && tmux kill-session" C-m | function cleanup() { | ||||||
|  |     # Drop ptmx shells | ||||||
|  |     kill $FIRST_PID | ||||||
|  |     kill $SECOND_PID | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| tmux join-pane -s raspi3:0 -t 2 | trap cleanup EXIT | ||||||
| tmux join-pane -s raspi3:1 -t 2 |  | ||||||
| 
 | 
 | ||||||
| tmux select-pane -t 1 | echo "=== Running QEMU with MiniUart ${FIRST_TTY} and PL011 Uart ${SECOND_TTY} ===" | ||||||
| tmux attach-session -t raspi3 | 
 | ||||||
|  | ${QEMU} ${QEMU_OPTS} ${QEMU_RUNNER_OPTS} -dtb ${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/targets/bcm2710-rpi-3-b-plus.dtb -kernel ${KERNEL_BIN} -serial /dev/tty${SECOND_TTY} -serial /dev/tty${FIRST_TTY} | ||||||
|  |  | ||||||
|  | @ -0,0 +1,5 @@ | ||||||
|  | QEMU= | ||||||
|  | QEMU_OPTS= | ||||||
|  | QEMU_RUNNER_OPTS= | ||||||
|  | CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY= | ||||||
|  | KERNEL_BIN= | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| #!/bin/sh |  | ||||||
| hopperv4 -e target/aarch64-vesper-metta/release/vesper.bin -R --base-address 0x80000 --entrypoint 0x80000 --file-offset 0 --aarch64 |  | ||||||
|  | @ -1,41 +1,36 @@ | ||||||
| /* | /* | ||||||
|  * MIT License |  * SPDX-License-Identifier: MIT OR BlueOak-1.0.0 | ||||||
|  * |  | ||||||
|  * Copyright (c) 2018 Andre Richter <andre.o.richter@gmail.com> |  * Copyright (c) 2018 Andre Richter <andre.o.richter@gmail.com> | ||||||
|  * |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy |  * Original code distributed under MIT, additional changes are under BlueOak-1.0.0 | ||||||
|  * of this software and associated documentation files (the "Software"), to deal |  | ||||||
|  * in the Software without restriction, including without limitation the rights |  | ||||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |  | ||||||
|  * copies of the Software, and to permit persons to whom the Software is |  | ||||||
|  * furnished to do so, subject to the following conditions: |  | ||||||
|  * |  | ||||||
|  * The above copyright notice and this permission notice shall be included in all |  | ||||||
|  * copies or substantial portions of the Software. |  | ||||||
|  * |  | ||||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |  | ||||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |  | ||||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |  | ||||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |  | ||||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |  | ||||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |  | ||||||
|  * SOFTWARE. |  | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| ENTRY(_boot_cores); | ENTRY(_boot_cores); | ||||||
| 
 | 
 | ||||||
|  | /* Symbols between __BOOT_START and __BOOT_END should be dropped after init is complete. | ||||||
|  |    Symbols between __RO_START and __RO_END are the kernel code. | ||||||
|  |    Symbols between __BSS_START and __BSS_END must be initialized to zero by r0 code in kernel. | ||||||
|  | */ | ||||||
| SECTIONS | SECTIONS | ||||||
| { | { | ||||||
|     . = 0x80000; /* AArch64 boot address is 0x80000, 4K-aligned */ |     . = 0x80000; /* AArch64 boot address is 0x80000, 4K-aligned */ | ||||||
|     __ro_start = .; |     __STACK_START = 0x80000; /* Stack grows from here towards 0x0. */ | ||||||
|  |     __BOOT_START = .; | ||||||
|     .text : |     .text : | ||||||
|     { |     { | ||||||
|         KEEP(*(.text.boot)) *(.text .text.*) |         KEEP(*(.text.boot.entry)) // Entry point must go first | ||||||
|  |         *(.text.boot) | ||||||
|  |         . = ALIGN(4096); | ||||||
|  |         *(.data.boot) | ||||||
|  |         . = ALIGN(4096); /* Here boot code ends */ | ||||||
|  |         __BOOT_END = .; // __BOOT_END must be 4KiB aligned | ||||||
|  |         __RO_START = .; | ||||||
|  |         *(.text .text.*) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     .vectors ALIGN(2048): |     .vectors ALIGN(2048): | ||||||
|     { |     { | ||||||
|         *(.vectors) |         KEEP(*(.vectors)) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     .rodata ALIGN(4): |     .rodata ALIGN(4): | ||||||
|  | @ -44,20 +39,24 @@ SECTIONS | ||||||
|         FILL(0x00) |         FILL(0x00) | ||||||
|     } |     } | ||||||
|     . = ALIGN(4096); /* Fill up to 4KiB */ |     . = ALIGN(4096); /* Fill up to 4KiB */ | ||||||
|     __ro_end = .; |     __RO_END = .; /* __RO_END must be 4KiB aligned */ | ||||||
|  |     __DATA_START = .; /* __DATA_START must be 4KiB aligned */ | ||||||
| 
 | 
 | ||||||
|     .data : |     .data : /* @todo align data to 4K -- it's already aligned up to __RO_END marker now */ | ||||||
|     { |     { | ||||||
|         *(.data .data.*) |         *(.data .data.*) | ||||||
|         FILL(0x00) |         FILL(0x00) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     .bss ALIGN(8): |     /* @todo could insert .data.boot here with proper alignment */ | ||||||
|  | 
 | ||||||
|  |     .bss ALIGN(8) (NOLOAD): | ||||||
|     { |     { | ||||||
|         __bss_start = .; |         __BSS_START = .; | ||||||
|         *(.bss .bss.*) |         *(.bss .bss.*) | ||||||
|         *(COMMON) |         *(COMMON) | ||||||
|         __bss_end = .; |         . = ALIGN(4096); /* Align up to 4KiB */ | ||||||
|  |         __BSS_END = .; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /DISCARD/ : { *(.comment) *(.gnu*) *(.note*) *(.eh_frame*) } |     /DISCARD/ : { *(.comment) *(.gnu*) *(.note*) *(.eh_frame*) } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,40 @@ | ||||||
|  | [package] | ||||||
|  | name = "machine" | ||||||
|  | version = "0.0.1" | ||||||
|  | authors = ["Berkus Decker <berkus+vesper@metta.systems>"] | ||||||
|  | description = "Vesper nanokernel shared code library, useful also for the chainboot loader." | ||||||
|  | documentation = "https://docs.metta.systems/vesper" | ||||||
|  | homepage = "https://github.com/metta-systems/vesper" | ||||||
|  | repository = "https://github.com/metta-systems/vesper" | ||||||
|  | readme = "README.md" | ||||||
|  | license = "BlueOak-1.0.0" | ||||||
|  | categories = ["no-std", "embedded", "os"] | ||||||
|  | publish = false | ||||||
|  | edition = "2021" | ||||||
|  | 
 | ||||||
|  | [badges] | ||||||
|  | maintenance = { status = "experimental" } | ||||||
|  | 
 | ||||||
|  | [features] | ||||||
|  | default = [] | ||||||
|  | noserial = [] | ||||||
|  | # Enable JTAG debugging of kernel - enable jtag helpers and | ||||||
|  | # block waiting for JTAG probe attach at the start of kernel main. | ||||||
|  | jtag = [] | ||||||
|  | # Build for running under QEMU with semihosting, so various halt/reboot options would for example quit QEMU instead. | ||||||
|  | qemu = ["rpi3"] | ||||||
|  | # Mutually exclusive features to choose a target board | ||||||
|  | rpi3 = [] | ||||||
|  | rpi4 = [] | ||||||
|  | 
 | ||||||
|  | [dependencies] | ||||||
|  | r0 = "1.0" | ||||||
|  | qemu-exit = "3.0" | ||||||
|  | cortex-a = "7.0" | ||||||
|  | tock-registers = "0.7" | ||||||
|  | ux = { version = "0.1", default-features = false } | ||||||
|  | usize_conversions = "0.2" | ||||||
|  | bit_field = "0.10" | ||||||
|  | bitflags = "1.3" | ||||||
|  | cfg-if = "1.0" | ||||||
|  | snafu = { version = "0.7", default-features = false } | ||||||
|  | @ -3,3 +3,7 @@ | ||||||
| This directory contains code specific to a certain architecture. | This directory contains code specific to a certain architecture. | ||||||
| 
 | 
 | ||||||
| Implementations of arch-specific kernel calls are also placed here. | Implementations of arch-specific kernel calls are also placed here. | ||||||
|  | 
 | ||||||
|  | ---- | ||||||
|  | 
 | ||||||
|  | For more information please re-read. | ||||||
|  | @ -0,0 +1,223 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * | ||||||
|  |  * Based on ideas from Jorge Aparicio, Andre Richter, Phil Oppenheimer, Sergio Benitez. | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | //! Low-level boot of the Raspberry's processor
 | ||||||
|  | //! <http://infocenter.arm.com/help/topic/com.arm.doc.dai0527a/DAI0527A_baremetal_boot_code_for_ARMv8_A_processors.pdf>
 | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     crate::endless_sleep, | ||||||
|  |     cortex_a::{asm, registers::*}, | ||||||
|  |     tock_registers::interfaces::{Readable, Writeable}, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // Stack placed before first executable instruction
 | ||||||
|  | const STACK_START: u64 = 0x0008_0000; // Keep in sync with linker script
 | ||||||
|  | 
 | ||||||
|  | /// Type check the user-supplied entry function.
 | ||||||
|  | #[macro_export] | ||||||
|  | macro_rules! entry { | ||||||
|  |     ($path:path) => { | ||||||
|  |         /// # Safety
 | ||||||
|  |         /// Only type-checks!
 | ||||||
|  |         #[export_name = "main"] | ||||||
|  |         #[inline(always)] | ||||||
|  |         pub unsafe fn __main() -> ! { | ||||||
|  |             // type check the given path
 | ||||||
|  |             let f: fn() -> ! = $path; | ||||||
|  | 
 | ||||||
|  |             f() | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Reset function.
 | ||||||
|  | ///
 | ||||||
|  | /// Initializes the bss section before calling into the user's `main()`.
 | ||||||
|  | ///
 | ||||||
|  | /// # Safety
 | ||||||
|  | ///
 | ||||||
|  | /// Totally unsafe! We're in the hardware land.
 | ||||||
|  | #[link_section = ".text.boot"] | ||||||
|  | unsafe fn reset() -> ! { | ||||||
|  |     extern "C" { | ||||||
|  |         // Boundaries of the .bss section, provided by the linker script
 | ||||||
|  |         // The type, `u64`, indicates that the memory is 8-byte aligned
 | ||||||
|  |         static mut __BSS_START: u64; | ||||||
|  |         static mut __BSS_END: u64; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Zeroes the .bss section
 | ||||||
|  |     r0::zero_bss(&mut __BSS_START, &mut __BSS_END); | ||||||
|  | 
 | ||||||
|  |     extern "Rust" { | ||||||
|  |         fn main() -> !; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     main() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // [ARMv6 unaligned data access restrictions](https://developer.arm.com/documentation/ddi0333/h/unaligned-and-mixed-endian-data-access-support/unaligned-access-support/armv6-unaligned-data-access-restrictions?lang=en)
 | ||||||
|  | // dictates that compatibility bit U in CP15 must be set to 1 to allow Unaligned accesses while MMU is off.
 | ||||||
|  | // (In addition to SCTLR_EL1.A being 0)
 | ||||||
|  | // See also [CP15 C1 docs](https://developer.arm.com/documentation/ddi0290/g/system-control-coprocessor/system-control-processor-registers/c1--control-register).
 | ||||||
|  | // #[link_section = ".text.boot"]
 | ||||||
|  | // #[inline]
 | ||||||
|  | // fn enable_armv6_unaligned_access() {
 | ||||||
|  | //     unsafe {
 | ||||||
|  | //         core::arch::asm!(
 | ||||||
|  | //             "mrc p15, 0, {u}, c1, c0, 0",
 | ||||||
|  | //             "or {u}, {u}, {CR_U}",
 | ||||||
|  | //             "mcr p15, 0, {u}, c1, c0, 0",
 | ||||||
|  | //             u = out(reg) _,
 | ||||||
|  | //             CR_U = const 1 << 22
 | ||||||
|  | //         );
 | ||||||
|  | //     }
 | ||||||
|  | // }
 | ||||||
|  | 
 | ||||||
|  | #[link_section = ".text.boot"] | ||||||
|  | #[inline] | ||||||
|  | fn shared_setup_and_enter_pre() { | ||||||
|  |     // Enable timer counter registers for EL1
 | ||||||
|  |     CNTHCTL_EL2.write(CNTHCTL_EL2::EL1PCEN::SET + CNTHCTL_EL2::EL1PCTEN::SET); | ||||||
|  | 
 | ||||||
|  |     // No virtual offset for reading the counters
 | ||||||
|  |     CNTVOFF_EL2.set(0); | ||||||
|  | 
 | ||||||
|  |     // Set System Control Register (EL1)
 | ||||||
|  |     // Make memory non-cacheable and disable MMU mapping.
 | ||||||
|  |     // Disable alignment checks, because Rust fmt module uses a little optimization
 | ||||||
|  |     // that happily reads and writes half-words (ldrh/strh) from/to unaligned addresses.
 | ||||||
|  |     SCTLR_EL1.write( | ||||||
|  |         SCTLR_EL1::I::NonCacheable | ||||||
|  |             + SCTLR_EL1::C::NonCacheable | ||||||
|  |             + SCTLR_EL1::M::Disable | ||||||
|  |             + SCTLR_EL1::A::Disable | ||||||
|  |             + SCTLR_EL1::SA::Disable | ||||||
|  |             + SCTLR_EL1::SA0::Disable, | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     // enable_armv6_unaligned_access();
 | ||||||
|  | 
 | ||||||
|  |     // Set Hypervisor Configuration Register (EL2)
 | ||||||
|  |     // Set EL1 execution state to AArch64
 | ||||||
|  |     // @todo Explain the SWIO bit (SWIO hardwired on Pi3)
 | ||||||
|  |     HCR_EL2.write(HCR_EL2::RW::EL1IsAarch64 + HCR_EL2::SWIO::SET); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[link_section = ".text.boot"] | ||||||
|  | #[inline] | ||||||
|  | fn shared_setup_and_enter_post() -> ! { | ||||||
|  |     // Set up SP_EL1 (stack pointer), which will be used by EL1 once
 | ||||||
|  |     // we "return" to it.
 | ||||||
|  |     SP_EL1.set(STACK_START); | ||||||
|  | 
 | ||||||
|  |     // Use `eret` to "return" to EL1. This will result in execution of
 | ||||||
|  |     // `reset()` in EL1.
 | ||||||
|  |     asm::eret() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Real hardware boot-up sequence.
 | ||||||
|  | ///
 | ||||||
|  | /// Prepare and execute transition from EL2 to EL1.
 | ||||||
|  | #[link_section = ".text.boot"] | ||||||
|  | #[inline] | ||||||
|  | fn setup_and_enter_el1_from_el2() -> ! { | ||||||
|  |     // Set Saved Program Status Register (EL2)
 | ||||||
|  |     // Set up a simulated exception return.
 | ||||||
|  |     //
 | ||||||
|  |     // Fake a saved program status, where all interrupts were
 | ||||||
|  |     // masked and SP_EL1 was used as a stack pointer.
 | ||||||
|  |     SPSR_EL2.write( | ||||||
|  |         SPSR_EL2::D::Masked | ||||||
|  |             + SPSR_EL2::A::Masked | ||||||
|  |             + SPSR_EL2::I::Masked | ||||||
|  |             + SPSR_EL2::F::Masked | ||||||
|  |             + SPSR_EL2::M::EL1h, // Use SP_EL1
 | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     // Make the Exception Link Register (EL2) point to reset().
 | ||||||
|  |     ELR_EL2.set(reset as *const () as u64); | ||||||
|  | 
 | ||||||
|  |     shared_setup_and_enter_post() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// QEMU boot-up sequence.
 | ||||||
|  | ///
 | ||||||
|  | /// Processors enter EL3 after reset.
 | ||||||
|  | /// ref: http://infocenter.arm.com/help/topic/com.arm.doc.dai0527a/DAI0527A_baremetal_boot_code_for_ARMv8_A_processors.pdf
 | ||||||
|  | /// section: 5.5.1
 | ||||||
|  | /// However, GPU init code must be switching it down to EL2.
 | ||||||
|  | /// QEMU can't emulate Raspberry Pi properly (no VC boot code), so it starts in EL3.
 | ||||||
|  | ///
 | ||||||
|  | /// Prepare and execute transition from EL3 to EL1.
 | ||||||
|  | /// (from https://github.com/s-matyukevich/raspberry-pi-os/blob/master/docs/lesson02/rpi-os.md)
 | ||||||
|  | #[cfg(qemu)] | ||||||
|  | #[link_section = ".text.boot"] | ||||||
|  | #[inline] | ||||||
|  | fn setup_and_enter_el1_from_el3() -> ! { | ||||||
|  |     // Set Secure Configuration Register (EL3)
 | ||||||
|  |     SCR_EL3.write(SCR_EL3::RW::NextELIsAarch64 + SCR_EL3::NS::NonSecure); | ||||||
|  | 
 | ||||||
|  |     // Set Saved Program Status Register (EL3)
 | ||||||
|  |     // Set up a simulated exception return.
 | ||||||
|  |     //
 | ||||||
|  |     // Fake a saved program status, where all interrupts were
 | ||||||
|  |     // masked and SP_EL1 was used as a stack pointer.
 | ||||||
|  |     SPSR_EL3.write( | ||||||
|  |         SPSR_EL3::D::Masked | ||||||
|  |             + SPSR_EL3::A::Masked | ||||||
|  |             + SPSR_EL3::I::Masked | ||||||
|  |             + SPSR_EL3::F::Masked | ||||||
|  |             + SPSR_EL3::M::EL1h, // Use SP_EL1
 | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     // Make the Exception Link Register (EL3) point to reset().
 | ||||||
|  |     ELR_EL3.set(reset as *const () as u64); | ||||||
|  | 
 | ||||||
|  |     shared_setup_and_enter_post() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Entrypoint of the processor.
 | ||||||
|  | ///
 | ||||||
|  | /// Parks all cores except core0 and checks if we started in EL2/EL3. If
 | ||||||
|  | /// so, proceeds with setting up EL1.
 | ||||||
|  | ///
 | ||||||
|  | /// This is invoked from the linker script, does arch-specific init
 | ||||||
|  | /// and passes control to the kernel boot function reset().
 | ||||||
|  | ///
 | ||||||
|  | /// Dissection of various RPi core boot stubs is available
 | ||||||
|  | /// [here](https://leiradel.github.io/2019/01/20/Raspberry-Pi-Stubs.html).
 | ||||||
|  | ///
 | ||||||
|  | #[no_mangle] | ||||||
|  | #[link_section = ".text.boot.entry"] | ||||||
|  | pub unsafe extern "C" fn _boot_cores() -> ! { | ||||||
|  |     const CORE_0: u64 = 0; | ||||||
|  |     const CORE_MASK: u64 = 0x3; | ||||||
|  |     // Can't match values with dots in match, so use intermediate consts.
 | ||||||
|  |     #[cfg(qemu)] | ||||||
|  |     const EL3: u64 = CurrentEL::EL::EL3.value; | ||||||
|  |     const EL2: u64 = CurrentEL::EL::EL2.value; | ||||||
|  |     const EL1: u64 = CurrentEL::EL::EL1.value; | ||||||
|  | 
 | ||||||
|  |     // Set stack pointer. Used in case we started in EL1.
 | ||||||
|  |     SP.set(STACK_START); | ||||||
|  | 
 | ||||||
|  |     shared_setup_and_enter_pre(); | ||||||
|  | 
 | ||||||
|  |     if CORE_0 == MPIDR_EL1.get() & CORE_MASK { | ||||||
|  |         match CurrentEL.get() { | ||||||
|  |             #[cfg(qemu)] | ||||||
|  |             EL3 => setup_and_enter_el1_from_el3(), | ||||||
|  |             EL2 => setup_and_enter_el1_from_el2(), | ||||||
|  |             EL1 => reset(), | ||||||
|  |             _ => endless_sleep(), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // if not core0 or not EL3/EL2/EL1, infinitely wait for events
 | ||||||
|  |     endless_sleep() | ||||||
|  | } | ||||||
|  | @ -0,0 +1,20 @@ | ||||||
|  | //! JTAG helper functions.
 | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     core::ptr::{read_volatile, write_volatile}, | ||||||
|  |     cortex_a::asm, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #[no_mangle] | ||||||
|  | static mut WAIT_FLAG: bool = true; | ||||||
|  | 
 | ||||||
|  | /// Wait for debugger to attach.
 | ||||||
|  | /// Then in gdb issue `> set var *(&WAIT_FLAG) = 0`
 | ||||||
|  | /// from inside this function's frame to continue running.
 | ||||||
|  | pub fn wait_debugger() { | ||||||
|  |     while unsafe { read_volatile(&WAIT_FLAG) } { | ||||||
|  |         asm::nop(); | ||||||
|  |     } | ||||||
|  |     // Reset the flag so that next jtag::wait_debugger() would block again.
 | ||||||
|  |     unsafe { write_volatile(&mut WAIT_FLAG, true) } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,11 @@ | ||||||
|  | # Memory Configuration | ||||||
|  | 
 | ||||||
|  | The types VirtAddr and PhysAddr are representing the addresses before and after the mapping in the MMU. | ||||||
|  | 
 | ||||||
|  | Page table types must represent pages of differing sizes. | ||||||
|  | For every entry in the MMU page table we should be able to receive a proper page type - e.g. Invalid, further page table, or | ||||||
|  | a specific-size page. | ||||||
|  | 
 | ||||||
|  | ---- | ||||||
|  | 
 | ||||||
|  | For more information please re-read. | ||||||
|  | @ -0,0 +1,7 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #[allow(dead_code)] | ||||||
|  | pub type ASID = u16; | ||||||
|  | @ -0,0 +1,10 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | mod asid; | ||||||
|  | mod phys_addr; | ||||||
|  | mod virt_addr; | ||||||
|  | 
 | ||||||
|  | pub use {asid::*, phys_addr::*, virt_addr::*}; | ||||||
|  | @ -0,0 +1,252 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     crate::mm::{align_down, align_up}, | ||||||
|  |     bit_field::BitField, | ||||||
|  |     core::{ | ||||||
|  |         convert::{From, Into}, | ||||||
|  |         fmt, | ||||||
|  |         ops::{Add, AddAssign, Shl, Shr, Sub, SubAssign}, | ||||||
|  |     }, | ||||||
|  |     usize_conversions::FromUsize, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /// A 64-bit physical memory address.
 | ||||||
|  | ///
 | ||||||
|  | /// This is a wrapper type around an `u64`, so it is always 8 bytes, even when compiled
 | ||||||
|  | /// on non 64-bit systems. The `UsizeConversions` trait can be used for performing conversions
 | ||||||
|  | /// between `u64` and `usize`.
 | ||||||
|  | ///
 | ||||||
|  | /// On `aarch64`, only the 52 lower bits of a physical address can be used. The top 12 bits need
 | ||||||
|  | /// to be zero. This type guarantees that it always represents a valid physical address.
 | ||||||
|  | #[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Default)] | ||||||
|  | #[repr(transparent)] | ||||||
|  | pub struct PhysAddr(u64); | ||||||
|  | 
 | ||||||
|  | /// A passed `u64` was not a valid physical address.
 | ||||||
|  | ///
 | ||||||
|  | /// This means that bits 52 to 64 were not all null.
 | ||||||
|  | #[derive(Debug, PartialEq, Eq, PartialOrd, Ord)] | ||||||
|  | pub struct PhysAddrNotValid(u64); | ||||||
|  | 
 | ||||||
|  | impl PhysAddr { | ||||||
|  |     /// Creates a new physical address.
 | ||||||
|  |     ///
 | ||||||
|  |     /// Panics if any bits in the bit position 52 to 64 is set.
 | ||||||
|  |     pub fn new(addr: u64) -> PhysAddr { | ||||||
|  |         assert_eq!( | ||||||
|  |             addr.get_bits(52..64), | ||||||
|  |             0, | ||||||
|  |             "physical addresses must not have any set bits in positions 52 to 64" | ||||||
|  |         ); | ||||||
|  |         PhysAddr(addr) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Tries to create a new physical address.
 | ||||||
|  |     ///
 | ||||||
|  |     /// Fails if any bits in the bit positions 52 to 64 are set.
 | ||||||
|  |     pub fn try_new(addr: u64) -> Result<PhysAddr, PhysAddrNotValid> { | ||||||
|  |         match addr.get_bits(52..64) { | ||||||
|  |             0 => Ok(PhysAddr(addr)), // address is valid
 | ||||||
|  |             _ => Err(PhysAddrNotValid(addr)), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Creates a physical address that points to `0`.
 | ||||||
|  |     pub const fn zero() -> PhysAddr { | ||||||
|  |         PhysAddr(0) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Converts the address to an `u64`.
 | ||||||
|  |     pub fn as_u64(self) -> u64 { | ||||||
|  |         self.0 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Convenience method for checking if a physical address is null.
 | ||||||
|  |     pub fn is_null(&self) -> bool { | ||||||
|  |         self.0 == 0 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Aligns the physical address upwards to the given alignment.
 | ||||||
|  |     ///
 | ||||||
|  |     /// See the `align_up` function for more information.
 | ||||||
|  |     #[must_use] | ||||||
|  |     pub fn aligned_up<U>(self, align: U) -> Self | ||||||
|  |     where | ||||||
|  |         U: Into<u64>, | ||||||
|  |     { | ||||||
|  |         PhysAddr(align_up(self.0, align.into())) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Aligns the physical address downwards to the given alignment.
 | ||||||
|  |     ///
 | ||||||
|  |     /// See the `align_down` function for more information.
 | ||||||
|  |     #[must_use] | ||||||
|  |     pub fn aligned_down<U>(self, align: U) -> Self | ||||||
|  |     where | ||||||
|  |         U: Into<u64>, | ||||||
|  |     { | ||||||
|  |         PhysAddr(align_down(self.0, align.into())) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Checks whether the physical address has the demanded alignment.
 | ||||||
|  |     pub fn is_aligned<U>(self, align: U) -> bool | ||||||
|  |     where | ||||||
|  |         U: Into<u64>, | ||||||
|  |     { | ||||||
|  |         self.aligned_down(align) == self | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::Debug for PhysAddr { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||||
|  |         write!(f, "PhysAddr({:#x})", self.0) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::Binary for PhysAddr { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||||
|  |         self.0.fmt(f) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::LowerHex for PhysAddr { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||||
|  |         self.0.fmt(f) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::Octal for PhysAddr { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||||
|  |         self.0.fmt(f) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::UpperHex for PhysAddr { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||||
|  |         self.0.fmt(f) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<u64> for PhysAddr { | ||||||
|  |     fn from(value: u64) -> Self { | ||||||
|  |         PhysAddr::new(value) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<PhysAddr> for u64 { | ||||||
|  |     fn from(value: PhysAddr) -> Self { | ||||||
|  |         value.as_u64() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<PhysAddr> for u128 { | ||||||
|  |     fn from(value: PhysAddr) -> Self { | ||||||
|  |         value.as_u64() as u128 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Add<u64> for PhysAddr { | ||||||
|  |     type Output = Self; | ||||||
|  |     fn add(self, rhs: u64) -> Self::Output { | ||||||
|  |         PhysAddr::new(self.0 + rhs) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl AddAssign<u64> for PhysAddr { | ||||||
|  |     fn add_assign(&mut self, rhs: u64) { | ||||||
|  |         *self = *self + rhs; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Add<usize> for PhysAddr | ||||||
|  | where | ||||||
|  |     u64: FromUsize, | ||||||
|  | { | ||||||
|  |     type Output = Self; | ||||||
|  |     fn add(self, rhs: usize) -> Self::Output { | ||||||
|  |         self + u64::from_usize(rhs) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl AddAssign<usize> for PhysAddr | ||||||
|  | where | ||||||
|  |     u64: FromUsize, | ||||||
|  | { | ||||||
|  |     fn add_assign(&mut self, rhs: usize) { | ||||||
|  |         self.add_assign(u64::from_usize(rhs)) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Sub<u64> for PhysAddr { | ||||||
|  |     type Output = Self; | ||||||
|  |     fn sub(self, rhs: u64) -> Self::Output { | ||||||
|  |         PhysAddr::new(self.0.checked_sub(rhs).unwrap()) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl SubAssign<u64> for PhysAddr { | ||||||
|  |     fn sub_assign(&mut self, rhs: u64) { | ||||||
|  |         *self = *self - rhs; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Sub<usize> for PhysAddr | ||||||
|  | where | ||||||
|  |     u64: FromUsize, | ||||||
|  | { | ||||||
|  |     type Output = Self; | ||||||
|  |     fn sub(self, rhs: usize) -> Self::Output { | ||||||
|  |         self - u64::from_usize(rhs) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl SubAssign<usize> for PhysAddr | ||||||
|  | where | ||||||
|  |     u64: FromUsize, | ||||||
|  | { | ||||||
|  |     fn sub_assign(&mut self, rhs: usize) { | ||||||
|  |         self.sub_assign(u64::from_usize(rhs)) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Sub<PhysAddr> for PhysAddr { | ||||||
|  |     type Output = u64; | ||||||
|  |     fn sub(self, rhs: PhysAddr) -> Self::Output { | ||||||
|  |         self.as_u64().checked_sub(rhs.as_u64()).unwrap() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Shr<usize> for PhysAddr { | ||||||
|  |     type Output = PhysAddr; | ||||||
|  | 
 | ||||||
|  |     fn shr(self, shift: usize) -> Self::Output { | ||||||
|  |         PhysAddr::new(self.0 >> shift) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Shl<usize> for PhysAddr { | ||||||
|  |     type Output = PhysAddr; | ||||||
|  | 
 | ||||||
|  |     fn shl(self, shift: usize) -> Self::Output { | ||||||
|  |         PhysAddr::new(self.0 << shift) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(test)] | ||||||
|  | mod tests { | ||||||
|  |     use super::*; | ||||||
|  | 
 | ||||||
|  |     #[test_case] | ||||||
|  |     pub fn test_invalid_phys_addr() { | ||||||
|  |         let result = PhysAddr::try_new(0xfafa_0123_3210_3210); | ||||||
|  |         if let Err(e) = result { | ||||||
|  |             assert_eq!(e, PhysAddrNotValid(0xfafa_0123_3210_3210)); | ||||||
|  |         } else { | ||||||
|  |             assert!(false) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,295 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     crate::mm::{align_down, align_up}, | ||||||
|  |     bit_field::BitField, | ||||||
|  |     core::{ | ||||||
|  |         convert::{From, Into, TryInto}, | ||||||
|  |         fmt, | ||||||
|  |         ops::{Add, AddAssign, Rem, RemAssign, Sub, SubAssign}, | ||||||
|  |     }, | ||||||
|  |     usize_conversions::{usize_from, FromUsize}, | ||||||
|  |     ux::*, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /// A canonical 64-bit virtual memory address.
 | ||||||
|  | ///
 | ||||||
|  | /// This is a wrapper type around an `u64`, so it is always 8 bytes, even when compiled
 | ||||||
|  | /// on non 64-bit systems. The `UsizeConversions` trait can be used for performing conversions
 | ||||||
|  | /// between `u64` and `usize`.
 | ||||||
|  | ///
 | ||||||
|  | /// On `x86_64`, only the 48 lower bits of a virtual address can be used. The top 16 bits need
 | ||||||
|  | /// to be copies of bit 47, i.e. the most significant bit. Addresses that fulfil this criterium
 | ||||||
|  | /// are called “canonical”. This type guarantees that it always represents a canonical address.
 | ||||||
|  | #[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Default)] | ||||||
|  | #[repr(transparent)] | ||||||
|  | pub struct VirtAddr(u64); | ||||||
|  | 
 | ||||||
|  | /// A passed `u64` was not a valid virtual address.
 | ||||||
|  | ///
 | ||||||
|  | /// This means that bits 48 to 64 are not
 | ||||||
|  | /// a valid sign extension and are not null either. So automatic sign extension would have
 | ||||||
|  | /// overwritten possibly meaningful bits. This likely indicates a bug, for example an invalid
 | ||||||
|  | /// address calculation.
 | ||||||
|  | #[derive(Debug)] | ||||||
|  | pub struct VirtAddrNotValid(u64); | ||||||
|  | 
 | ||||||
|  | impl VirtAddr { | ||||||
|  |     /// Creates a new canonical virtual address.
 | ||||||
|  |     ///
 | ||||||
|  |     /// This function performs sign extension of bit 47 to make the address canonical. Panics
 | ||||||
|  |     /// if the bits in the range 48 to 64 contain data (i.e. are not null and not a sign extension).
 | ||||||
|  |     ///
 | ||||||
|  |     /// @todo Support ASID byte in top bits of the address.
 | ||||||
|  |     pub fn new(addr: u64) -> VirtAddr { | ||||||
|  |         Self::try_new(addr).expect( | ||||||
|  |             "address passed to VirtAddr::new must not contain any data \ | ||||||
|  |              in bits 48 to 64",
 | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Tries to create a new canonical virtual address.
 | ||||||
|  |     ///
 | ||||||
|  |     /// This function tries to performs sign extension of bit 47 to make the address canonical.
 | ||||||
|  |     /// It succeeds if bits 48 to 64 are either a correct sign extension (i.e. copies of bit 47)
 | ||||||
|  |     /// or all null. Else, an error is returned.
 | ||||||
|  |     pub fn try_new(addr: u64) -> Result<VirtAddr, VirtAddrNotValid> { | ||||||
|  |         match addr.get_bits(47..64) { | ||||||
|  |             0 | 0x1ffff => Ok(VirtAddr(addr)),      // address is canonical
 | ||||||
|  |             1 => Ok(VirtAddr::new_unchecked(addr)), // address needs sign extension
 | ||||||
|  |             _ => Err(VirtAddrNotValid(addr)), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Creates a new canonical virtual address without checks.
 | ||||||
|  |     ///
 | ||||||
|  |     /// This function performs sign extension of bit 47 to make the address canonical, so
 | ||||||
|  |     /// bits 48 to 64 are overwritten. If you want to check that these bits contain no data,
 | ||||||
|  |     /// use `new` or `try_new`.
 | ||||||
|  |     pub fn new_unchecked(mut addr: u64) -> VirtAddr { | ||||||
|  |         if addr.get_bit(47) { | ||||||
|  |             addr.set_bits(48..64, 0xffff); | ||||||
|  |         } else { | ||||||
|  |             addr.set_bits(48..64, 0); | ||||||
|  |         } | ||||||
|  |         VirtAddr(addr) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Creates a virtual address that points to `0`.
 | ||||||
|  |     pub const fn zero() -> VirtAddr { | ||||||
|  |         VirtAddr(0) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Converts the address to an `u64`.
 | ||||||
|  |     pub fn as_u64(self) -> u64 { | ||||||
|  |         self.0 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Creates a virtual address from the given pointer
 | ||||||
|  |     pub fn from_ptr<T>(ptr: *const T) -> Self { | ||||||
|  |         Self::new(u64::from_usize(ptr as usize)) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Converts the address to a raw pointer.
 | ||||||
|  |     #[cfg(target_pointer_width = "64")] | ||||||
|  |     pub fn as_ptr<T>(self) -> *const T { | ||||||
|  |         usize_from(self.as_u64()) as *const T | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Converts the address to a mutable raw pointer.
 | ||||||
|  |     #[cfg(target_pointer_width = "64")] | ||||||
|  |     pub fn as_mut_ptr<T>(self) -> *mut T { | ||||||
|  |         self.as_ptr::<T>() as *mut T | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Aligns the virtual address upwards to the given alignment.
 | ||||||
|  |     ///
 | ||||||
|  |     /// See the `align_up` free function for more information.
 | ||||||
|  |     #[must_use] | ||||||
|  |     pub fn aligned_up<U>(self, align: U) -> Self | ||||||
|  |     where | ||||||
|  |         U: Into<u64>, | ||||||
|  |     { | ||||||
|  |         VirtAddr(align_up(self.0, align.into())) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Aligns the virtual address downwards to the given alignment.
 | ||||||
|  |     ///
 | ||||||
|  |     /// See the `align_down` free function for more information.
 | ||||||
|  |     #[must_use] | ||||||
|  |     pub fn aligned_down<U>(self, align: U) -> Self | ||||||
|  |     where | ||||||
|  |         U: Into<u64>, | ||||||
|  |     { | ||||||
|  |         VirtAddr(align_down(self.0, align.into())) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Checks whether the virtual address has the demanded alignment.
 | ||||||
|  |     pub fn is_aligned<U>(self, align: U) -> bool | ||||||
|  |     where | ||||||
|  |         U: Into<u64>, | ||||||
|  |     { | ||||||
|  |         self.aligned_down(align) == self | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Returns the 12-bit page offset of this virtual address.
 | ||||||
|  |     pub fn page_offset(&self) -> u12 { | ||||||
|  |         u12::new((self.0 & 0xfff).try_into().unwrap()) | ||||||
|  |     } | ||||||
|  |     // ^ @todo this only works for 4KiB pages
 | ||||||
|  | 
 | ||||||
|  |     /// Returns the 9-bit level 3 page table index.
 | ||||||
|  |     pub fn l3_index(&self) -> u9 { | ||||||
|  |         u9::new(((self.0 >> 12) & 0o777).try_into().unwrap()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Returns the 9-bit level 2 page table index.
 | ||||||
|  |     pub fn l2_index(&self) -> u9 { | ||||||
|  |         u9::new(((self.0 >> 12 >> 9) & 0o777).try_into().unwrap()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Returns the 9-bit level 1 page table index.
 | ||||||
|  |     pub fn l1_index(&self) -> u9 { | ||||||
|  |         u9::new(((self.0 >> 12 >> 9 >> 9) & 0o777).try_into().unwrap()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Returns the 9-bit level 0 page table index.
 | ||||||
|  |     pub fn l0_index(&self) -> u9 { | ||||||
|  |         u9::new(((self.0 >> 12 >> 9 >> 9 >> 9) & 0o777).try_into().unwrap()) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::Debug for VirtAddr { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||||
|  |         write!(f, "VirtAddr({:#x})", self.0) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::Binary for VirtAddr { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||||
|  |         self.0.fmt(f) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::LowerHex for VirtAddr { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||||
|  |         self.0.fmt(f) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::Octal for VirtAddr { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||||
|  |         self.0.fmt(f) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::UpperHex for VirtAddr { | ||||||
|  |     fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { | ||||||
|  |         self.0.fmt(f) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<u64> for VirtAddr { | ||||||
|  |     fn from(value: u64) -> Self { | ||||||
|  |         VirtAddr::new(value) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<VirtAddr> for u64 { | ||||||
|  |     fn from(value: VirtAddr) -> Self { | ||||||
|  |         value.as_u64() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Add<u64> for VirtAddr { | ||||||
|  |     type Output = Self; | ||||||
|  |     fn add(self, rhs: u64) -> Self::Output { | ||||||
|  |         VirtAddr::new(self.0 + rhs) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl AddAssign<u64> for VirtAddr { | ||||||
|  |     fn add_assign(&mut self, rhs: u64) { | ||||||
|  |         *self = *self + rhs; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Add<usize> for VirtAddr | ||||||
|  | where | ||||||
|  |     u64: FromUsize, | ||||||
|  | { | ||||||
|  |     type Output = Self; | ||||||
|  |     fn add(self, rhs: usize) -> Self::Output { | ||||||
|  |         self + u64::from_usize(rhs) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl AddAssign<usize> for VirtAddr | ||||||
|  | where | ||||||
|  |     u64: FromUsize, | ||||||
|  | { | ||||||
|  |     fn add_assign(&mut self, rhs: usize) { | ||||||
|  |         self.add_assign(u64::from_usize(rhs)) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Sub<u64> for VirtAddr { | ||||||
|  |     type Output = Self; | ||||||
|  |     fn sub(self, rhs: u64) -> Self::Output { | ||||||
|  |         VirtAddr::new(self.0.checked_sub(rhs).unwrap()) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl SubAssign<u64> for VirtAddr { | ||||||
|  |     fn sub_assign(&mut self, rhs: u64) { | ||||||
|  |         *self = *self - rhs; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Sub<usize> for VirtAddr | ||||||
|  | where | ||||||
|  |     u64: FromUsize, | ||||||
|  | { | ||||||
|  |     type Output = Self; | ||||||
|  |     fn sub(self, rhs: usize) -> Self::Output { | ||||||
|  |         self - u64::from_usize(rhs) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl SubAssign<usize> for VirtAddr | ||||||
|  | where | ||||||
|  |     u64: FromUsize, | ||||||
|  | { | ||||||
|  |     fn sub_assign(&mut self, rhs: usize) { | ||||||
|  |         self.sub_assign(u64::from_usize(rhs)) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Sub<VirtAddr> for VirtAddr { | ||||||
|  |     type Output = u64; | ||||||
|  |     fn sub(self, rhs: VirtAddr) -> Self::Output { | ||||||
|  |         self.as_u64().checked_sub(rhs.as_u64()).unwrap() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Rem<usize> for VirtAddr | ||||||
|  | where | ||||||
|  |     u64: FromUsize, | ||||||
|  | { | ||||||
|  |     type Output = u64; | ||||||
|  |     fn rem(self, rhs: usize) -> Self::Output { | ||||||
|  |         self.0 % u64::from_usize(rhs) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl RemAssign<usize> for VirtAddr | ||||||
|  | where | ||||||
|  |     u64: FromUsize, | ||||||
|  | { | ||||||
|  |     fn rem_assign(&mut self, rhs: usize) { | ||||||
|  |         *self = VirtAddr::new(self.0 % u64::from_usize(rhs)); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,713 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: MIT OR BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) 2018-2019 Andre Richter <andre.o.richter@gmail.com> | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  * Original code distributed under MIT, additional changes are under BlueOak-1.0.0 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | //! MMU initialisation.
 | ||||||
|  | //!
 | ||||||
|  | //! Paging is mostly based on [previous version](https://os.phil-opp.com/page-tables/) of
 | ||||||
|  | //! Phil Opp's [paging guide](https://os.phil-opp.com/paging-implementation/) and
 | ||||||
|  | //! [ARMv8 ARM memory addressing](https://static.docs.arm.com/100940/0100/armv8_a_address%20translation_100940_0100_en.pdf).
 | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     crate::{ | ||||||
|  |         arch::aarch64::memory::{get_virt_addr_properties, AttributeFields}, | ||||||
|  |         println, | ||||||
|  |     }, | ||||||
|  |     core::{ | ||||||
|  |         marker::PhantomData, | ||||||
|  |         ops::{Index, IndexMut}, | ||||||
|  |     }, | ||||||
|  |     cortex_a::{ | ||||||
|  |         asm::barrier, | ||||||
|  |         registers::{ID_AA64MMFR0_EL1, SCTLR_EL1, TCR_EL1, TTBR0_EL1}, | ||||||
|  |     }, | ||||||
|  |     tock_registers::{ | ||||||
|  |         fields::FieldValue, | ||||||
|  |         interfaces::{ReadWriteable, Readable, Writeable}, | ||||||
|  |         register_bitfields, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | mod mair { | ||||||
|  |     use {cortex_a::registers::MAIR_EL1, tock_registers::interfaces::Writeable}; | ||||||
|  | 
 | ||||||
|  |     /// Setup function for the MAIR_EL1 register.
 | ||||||
|  |     pub fn set_up() { | ||||||
|  |         // Define the three memory types that we will map. Normal DRAM, Uncached and device.
 | ||||||
|  |         MAIR_EL1.write( | ||||||
|  |             // Attribute 2 -- Device Memory
 | ||||||
|  |             MAIR_EL1::Attr2_Device::nonGathering_nonReordering_EarlyWriteAck | ||||||
|  |                 // Attribute 1 -- Non Cacheable DRAM
 | ||||||
|  |                 + MAIR_EL1::Attr1_Normal_Outer::NonCacheable | ||||||
|  |                 + MAIR_EL1::Attr1_Normal_Inner::NonCacheable | ||||||
|  |                 // Attribute 0 -- Regular Cacheable
 | ||||||
|  |                 + MAIR_EL1::Attr0_Normal_Outer::WriteBack_NonTransient_ReadWriteAlloc | ||||||
|  |                 + MAIR_EL1::Attr0_Normal_Inner::WriteBack_NonTransient_ReadWriteAlloc, | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Three descriptive consts for indexing into the correct MAIR_EL1 attributes.
 | ||||||
|  |     pub mod attr { | ||||||
|  |         pub const NORMAL: u64 = 0; | ||||||
|  |         pub const NORMAL_NON_CACHEABLE: u64 = 1; | ||||||
|  |         pub const DEVICE_NGNRE: u64 = 2; | ||||||
|  |         // DEVICE_GRE
 | ||||||
|  |         // DEVICE_NGNRNE
 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Parse the ID_AA64MMFR0_EL1 register for runtime information about supported MMU features.
 | ||||||
|  | /// Print the current state of TCR register.
 | ||||||
|  | pub fn print_features() { | ||||||
|  |     // use crate::cortex_a::regs::RegisterReadWrite;
 | ||||||
|  |     let sctlr = SCTLR_EL1.extract(); | ||||||
|  | 
 | ||||||
|  |     if let Some(SCTLR_EL1::M::Value::Enable) = sctlr.read_as_enum(SCTLR_EL1::M) { | ||||||
|  |         println!("[i] MMU currently enabled"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if let Some(SCTLR_EL1::I::Value::Cacheable) = sctlr.read_as_enum(SCTLR_EL1::I) { | ||||||
|  |         println!("[i] MMU I-cache enabled"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if let Some(SCTLR_EL1::C::Value::Cacheable) = sctlr.read_as_enum(SCTLR_EL1::C) { | ||||||
|  |         println!("[i] MMU D-cache enabled"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let mmfr = ID_AA64MMFR0_EL1.extract(); | ||||||
|  | 
 | ||||||
|  |     if let Some(ID_AA64MMFR0_EL1::TGran4::Value::Supported) = | ||||||
|  |         mmfr.read_as_enum(ID_AA64MMFR0_EL1::TGran4) | ||||||
|  |     { | ||||||
|  |         println!("[i] MMU: 4 KiB granule supported!"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if let Some(ID_AA64MMFR0_EL1::TGran16::Value::Supported) = | ||||||
|  |         mmfr.read_as_enum(ID_AA64MMFR0_EL1::TGran16) | ||||||
|  |     { | ||||||
|  |         println!("[i] MMU: 16 KiB granule supported!"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if let Some(ID_AA64MMFR0_EL1::TGran64::Value::Supported) = | ||||||
|  |         mmfr.read_as_enum(ID_AA64MMFR0_EL1::TGran64) | ||||||
|  |     { | ||||||
|  |         println!("[i] MMU: 64 KiB granule supported!"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     match mmfr.read_as_enum(ID_AA64MMFR0_EL1::ASIDBits) { | ||||||
|  |         Some(ID_AA64MMFR0_EL1::ASIDBits::Value::Bits_16) => { | ||||||
|  |             println!("[i] MMU: 16 bit ASIDs supported!") | ||||||
|  |         } | ||||||
|  |         Some(ID_AA64MMFR0_EL1::ASIDBits::Value::Bits_8) => { | ||||||
|  |             println!("[i] MMU: 8 bit ASIDs supported!") | ||||||
|  |         } | ||||||
|  |         _ => println!("[i] MMU: Invalid ASID bits specified!"), | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     match mmfr.read_as_enum(ID_AA64MMFR0_EL1::PARange) { | ||||||
|  |         Some(ID_AA64MMFR0_EL1::PARange::Value::Bits_32) => { | ||||||
|  |             println!("[i] MMU: Up to 32 Bit physical address range supported!") | ||||||
|  |         } | ||||||
|  |         Some(ID_AA64MMFR0_EL1::PARange::Value::Bits_36) => { | ||||||
|  |             println!("[i] MMU: Up to 36 Bit physical address range supported!") | ||||||
|  |         } | ||||||
|  |         Some(ID_AA64MMFR0_EL1::PARange::Value::Bits_40) => { | ||||||
|  |             println!("[i] MMU: Up to 40 Bit physical address range supported!") | ||||||
|  |         } | ||||||
|  |         Some(ID_AA64MMFR0_EL1::PARange::Value::Bits_42) => { | ||||||
|  |             println!("[i] MMU: Up to 42 Bit physical address range supported!") | ||||||
|  |         } | ||||||
|  |         Some(ID_AA64MMFR0_EL1::PARange::Value::Bits_44) => { | ||||||
|  |             println!("[i] MMU: Up to 44 Bit physical address range supported!") | ||||||
|  |         } | ||||||
|  |         Some(ID_AA64MMFR0_EL1::PARange::Value::Bits_48) => { | ||||||
|  |             println!("[i] MMU: Up to 48 Bit physical address range supported!") | ||||||
|  |         } | ||||||
|  |         Some(ID_AA64MMFR0_EL1::PARange::Value::Bits_52) => { | ||||||
|  |             println!("[i] MMU: Up to 52 Bit physical address range supported!") | ||||||
|  |         } | ||||||
|  |         _ => println!("[i] MMU: Invalid PARange specified!"), | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let tcr = TCR_EL1.extract(); | ||||||
|  | 
 | ||||||
|  |     match tcr.read_as_enum(TCR_EL1::IPS) { | ||||||
|  |         Some(TCR_EL1::IPS::Value::Bits_32) => { | ||||||
|  |             println!("[i] MMU: 32 Bit intermediate physical address size supported!") | ||||||
|  |         } | ||||||
|  |         Some(TCR_EL1::IPS::Value::Bits_36) => { | ||||||
|  |             println!("[i] MMU: 36 Bit intermediate physical address size supported!") | ||||||
|  |         } | ||||||
|  |         Some(TCR_EL1::IPS::Value::Bits_40) => { | ||||||
|  |             println!("[i] MMU: 40 Bit intermediate physical address size supported!") | ||||||
|  |         } | ||||||
|  |         Some(TCR_EL1::IPS::Value::Bits_42) => { | ||||||
|  |             println!("[i] MMU: 42 Bit intermediate physical address size supported!") | ||||||
|  |         } | ||||||
|  |         Some(TCR_EL1::IPS::Value::Bits_44) => { | ||||||
|  |             println!("[i] MMU: 44 Bit intermediate physical address size supported!") | ||||||
|  |         } | ||||||
|  |         Some(TCR_EL1::IPS::Value::Bits_48) => { | ||||||
|  |             println!("[i] MMU: 48 Bit intermediate physical address size supported!") | ||||||
|  |         } | ||||||
|  |         Some(TCR_EL1::IPS::Value::Bits_52) => { | ||||||
|  |             println!("[i] MMU: 52 Bit intermediate physical address size supported!") | ||||||
|  |         } | ||||||
|  |         _ => println!("[i] MMU: Invalid IPS specified!"), | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     match tcr.read_as_enum(TCR_EL1::TG0) { | ||||||
|  |         Some(TCR_EL1::TG0::Value::KiB_4) => println!("[i] MMU: TTBR0 4 KiB granule active!"), | ||||||
|  |         Some(TCR_EL1::TG0::Value::KiB_16) => println!("[i] MMU: TTBR0 16 KiB granule active!"), | ||||||
|  |         Some(TCR_EL1::TG0::Value::KiB_64) => println!("[i] MMU: TTBR0 64 KiB granule active!"), | ||||||
|  |         _ => println!("[i] MMU: Invalid TTBR0 granule size specified!"), | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let t0sz = tcr.read(TCR_EL1::T0SZ); | ||||||
|  |     println!("[i] MMU: T0sz = 64-{} = {} bits", t0sz, 64 - t0sz); | ||||||
|  | 
 | ||||||
|  |     match tcr.read_as_enum(TCR_EL1::TG1) { | ||||||
|  |         Some(TCR_EL1::TG1::Value::KiB_4) => println!("[i] MMU: TTBR1 4 KiB granule active!"), | ||||||
|  |         Some(TCR_EL1::TG1::Value::KiB_16) => println!("[i] MMU: TTBR1 16 KiB granule active!"), | ||||||
|  |         Some(TCR_EL1::TG1::Value::KiB_64) => println!("[i] MMU: TTBR1 64 KiB granule active!"), | ||||||
|  |         _ => println!("[i] MMU: Invalid TTBR1 granule size specified!"), | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let t1sz = tcr.read(TCR_EL1::T1SZ); | ||||||
|  |     println!("[i] MMU: T1sz = 64-{} = {} bits", t1sz, 64 - t1sz); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | register_bitfields! { | ||||||
|  |     u64, | ||||||
|  |     // AArch64 Reference Manual page 2150, D5-2445
 | ||||||
|  |     STAGE1_DESCRIPTOR [ | ||||||
|  |         // In table descriptors
 | ||||||
|  | 
 | ||||||
|  |         NSTable_EL3   OFFSET(63) NUMBITS(1) [], | ||||||
|  | 
 | ||||||
|  |         /// Access Permissions for subsequent tables
 | ||||||
|  |         APTable  OFFSET(61) NUMBITS(2) [ | ||||||
|  |             RW_EL1 = 0b00, | ||||||
|  |             RW_EL1_EL0 = 0b01, | ||||||
|  |             RO_EL1 = 0b10, | ||||||
|  |             RO_EL1_EL0 = 0b11 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         // User execute-never for subsequent tables
 | ||||||
|  |         UXNTable OFFSET(60) NUMBITS(1) [ | ||||||
|  |             Execute = 0, | ||||||
|  |             NeverExecute = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// Privileged execute-never for subsequent tables
 | ||||||
|  |         PXNTable OFFSET(59) NUMBITS(1) [ | ||||||
|  |             Execute = 0, | ||||||
|  |             NeverExecute = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         // In block descriptors
 | ||||||
|  | 
 | ||||||
|  |         // OS-specific data
 | ||||||
|  |         OSData      OFFSET(55) NUMBITS(4) [], | ||||||
|  | 
 | ||||||
|  |         // User execute-never
 | ||||||
|  |         UXN      OFFSET(54) NUMBITS(1) [ | ||||||
|  |             Execute = 0, | ||||||
|  |             NeverExecute = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// Privileged execute-never
 | ||||||
|  |         PXN      OFFSET(53) NUMBITS(1) [ | ||||||
|  |             Execute = 0, | ||||||
|  |             NeverExecute = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         // @fixme ?? where is this described
 | ||||||
|  |         CONTIGUOUS OFFSET(52) NUMBITS(1) [ | ||||||
|  |             False = 0, | ||||||
|  |             True = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         // @fixme ?? where is this described
 | ||||||
|  |         DIRTY OFFSET(51) NUMBITS(1) [ | ||||||
|  |             False = 0, | ||||||
|  |             True = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// Various address fields, depending on use case
 | ||||||
|  |         LVL2_OUTPUT_ADDR_4KiB    OFFSET(21) NUMBITS(27) [], // [47:21]
 | ||||||
|  |         NEXT_LVL_TABLE_ADDR_4KiB OFFSET(12) NUMBITS(36) [], // [47:12]
 | ||||||
|  | 
 | ||||||
|  |         // @fixme ?? where is this described
 | ||||||
|  |         NON_GLOBAL OFFSET(11) NUMBITS(1) [ | ||||||
|  |             False = 0, | ||||||
|  |             True = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// Access flag
 | ||||||
|  |         AF       OFFSET(10) NUMBITS(1) [ | ||||||
|  |             False = 0, | ||||||
|  |             True = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// Shareability field
 | ||||||
|  |         SH       OFFSET(8) NUMBITS(2) [ | ||||||
|  |             OuterShareable = 0b10, | ||||||
|  |             InnerShareable = 0b11 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// Access Permissions
 | ||||||
|  |         AP       OFFSET(6) NUMBITS(2) [ | ||||||
|  |             RW_EL1 = 0b00, | ||||||
|  |             RW_EL1_EL0 = 0b01, | ||||||
|  |             RO_EL1 = 0b10, | ||||||
|  |             RO_EL1_EL0 = 0b11 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         NS_EL3   OFFSET(5) NUMBITS(1) [], | ||||||
|  | 
 | ||||||
|  |         /// Memory attributes index into the MAIR_EL1 register
 | ||||||
|  |         AttrIndx OFFSET(2) NUMBITS(3) [], | ||||||
|  | 
 | ||||||
|  |         TYPE     OFFSET(1) NUMBITS(1) [ | ||||||
|  |             Block = 0, | ||||||
|  |             Table = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         VALID    OFFSET(0) NUMBITS(1) [ | ||||||
|  |             False = 0, | ||||||
|  |             True = 1 | ||||||
|  |         ] | ||||||
|  |     ] | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// A function that maps the generic memory range attributes to HW-specific
 | ||||||
|  | /// attributes of the MMU.
 | ||||||
|  | fn into_mmu_attributes( | ||||||
|  |     attribute_fields: AttributeFields, | ||||||
|  | ) -> FieldValue<u64, STAGE1_DESCRIPTOR::Register> { | ||||||
|  |     use super::{AccessPermissions, MemAttributes}; | ||||||
|  | 
 | ||||||
|  |     // Memory attributes
 | ||||||
|  |     let mut desc = match attribute_fields.mem_attributes { | ||||||
|  |         MemAttributes::CacheableDRAM => { | ||||||
|  |             STAGE1_DESCRIPTOR::SH::InnerShareable | ||||||
|  |                 + STAGE1_DESCRIPTOR::AttrIndx.val(mair::attr::NORMAL) | ||||||
|  |         } | ||||||
|  |         MemAttributes::NonCacheableDRAM => { | ||||||
|  |             STAGE1_DESCRIPTOR::SH::InnerShareable | ||||||
|  |                 + STAGE1_DESCRIPTOR::AttrIndx.val(mair::attr::NORMAL_NON_CACHEABLE) | ||||||
|  |         } | ||||||
|  |         MemAttributes::Device => { | ||||||
|  |             STAGE1_DESCRIPTOR::SH::OuterShareable | ||||||
|  |                 + STAGE1_DESCRIPTOR::AttrIndx.val(mair::attr::DEVICE_NGNRE) | ||||||
|  |         } | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     // Access Permissions
 | ||||||
|  |     desc += match attribute_fields.acc_perms { | ||||||
|  |         AccessPermissions::ReadOnly => STAGE1_DESCRIPTOR::AP::RO_EL1, | ||||||
|  |         AccessPermissions::ReadWrite => STAGE1_DESCRIPTOR::AP::RW_EL1, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     // Execute Never
 | ||||||
|  |     desc += if attribute_fields.execute_never { | ||||||
|  |         STAGE1_DESCRIPTOR::PXN::NeverExecute | ||||||
|  |     } else { | ||||||
|  |         STAGE1_DESCRIPTOR::PXN::Execute | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     desc | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  |  *  With 4k page granule, a virtual address is split into 4 lookup parts | ||||||
|  |  *  spanning 9 bits each: | ||||||
|  |  * | ||||||
|  |  *    _______________________________________________ | ||||||
|  |  *   |       |       |       |       |       |       | | ||||||
|  |  *   | signx |  Lv0  |  Lv1  |  Lv2  |  Lv3  |  off  | | ||||||
|  |  *   |_______|_______|_______|_______|_______|_______| | ||||||
|  |  *     63-48   47-39   38-30   29-21   20-12   11-00 | ||||||
|  |  * | ||||||
|  |  *             mask        page size | ||||||
|  |  * | ||||||
|  |  *    Lv0: FF8000000000       -- | ||||||
|  |  *    Lv1:   7FC0000000       1G | ||||||
|  |  *    Lv2:     3FE00000       2M | ||||||
|  |  *    Lv3:       1FF000       4K | ||||||
|  |  *    off:          FFF | ||||||
|  |  * | ||||||
|  |  * RPi3 supports 64K and 4K granules, also 40-bit physical addresses. | ||||||
|  |  * It also can address only 1G physical memory, so these 40-bit phys addresses are a fake. | ||||||
|  |  * | ||||||
|  |  * 48-bit virtual address space; different mappings in VBAR0 (EL0) and VBAR1 (EL1+). | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /// Number of entries in a 4KiB mmu table.
 | ||||||
|  | pub const NUM_ENTRIES_4KIB: u64 = 512; | ||||||
|  | 
 | ||||||
|  | /// Trait for abstracting over the possible page sizes, 4KiB, 16KiB, 2MiB, 1GiB.
 | ||||||
|  | pub trait PageSize: Copy + Eq + PartialOrd + Ord { | ||||||
|  |     /// The page size in bytes.
 | ||||||
|  |     const SIZE: u64; | ||||||
|  | 
 | ||||||
|  |     /// A string representation of the page size for debug output.
 | ||||||
|  |     const SIZE_AS_DEBUG_STR: &'static str; | ||||||
|  | 
 | ||||||
|  |     /// The page shift in bits.
 | ||||||
|  |     const SHIFT: usize; | ||||||
|  | 
 | ||||||
|  |     /// The page mask in bits.
 | ||||||
|  |     const MASK: u64; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// This trait is implemented for 4KiB, 16KiB, and 2MiB pages, but not for 1GiB pages.
 | ||||||
|  | pub trait NotGiantPageSize: PageSize {} // @todo doesn't have to be pub??
 | ||||||
|  | 
 | ||||||
|  | /// A standard 4KiB page.
 | ||||||
|  | #[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)] | ||||||
|  | pub enum Size4KiB {} | ||||||
|  | 
 | ||||||
|  | impl PageSize for Size4KiB { | ||||||
|  |     const SIZE: u64 = 4096; | ||||||
|  |     const SIZE_AS_DEBUG_STR: &'static str = "4KiB"; | ||||||
|  |     const SHIFT: usize = 12; | ||||||
|  |     const MASK: u64 = 0xfff; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl NotGiantPageSize for Size4KiB {} | ||||||
|  | 
 | ||||||
|  | /// A “huge” 2MiB page.
 | ||||||
|  | #[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)] | ||||||
|  | pub enum Size2MiB {} | ||||||
|  | 
 | ||||||
|  | impl PageSize for Size2MiB { | ||||||
|  |     const SIZE: u64 = Size4KiB::SIZE * NUM_ENTRIES_4KIB; | ||||||
|  |     const SIZE_AS_DEBUG_STR: &'static str = "2MiB"; | ||||||
|  |     const SHIFT: usize = 21; | ||||||
|  |     const MASK: u64 = 0x1fffff; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl NotGiantPageSize for Size2MiB {} | ||||||
|  | 
 | ||||||
|  | type EntryFlags = tock_registers::fields::FieldValue<u64, STAGE1_DESCRIPTOR::Register>; | ||||||
|  | // type EntryRegister = register::LocalRegisterCopy<u64, STAGE1_DESCRIPTOR::Register>;
 | ||||||
|  | 
 | ||||||
|  | /// L0 table -- only pointers to L1 tables
 | ||||||
|  | pub enum PageGlobalDirectory {} | ||||||
|  | /// L1 tables -- pointers to L2 tables or giant 1GiB pages
 | ||||||
|  | pub enum PageUpperDirectory {} | ||||||
|  | /// L2 tables -- pointers to L3 tables or huge 2MiB pages
 | ||||||
|  | pub enum PageDirectory {} | ||||||
|  | /// L3 tables -- only pointers to 4/16KiB pages
 | ||||||
|  | pub enum PageTable {} | ||||||
|  | 
 | ||||||
|  | /// Shared trait for specific table levels.
 | ||||||
|  | pub trait TableLevel {} | ||||||
|  | 
 | ||||||
|  | /// Shared trait for hierarchical table levels.
 | ||||||
|  | ///
 | ||||||
|  | /// Specifies what is the next level of page table hierarchy.
 | ||||||
|  | pub trait HierarchicalLevel: TableLevel { | ||||||
|  |     /// Level of the next translation table below this one.
 | ||||||
|  |     type NextLevel: TableLevel; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl TableLevel for PageGlobalDirectory {} | ||||||
|  | impl TableLevel for PageUpperDirectory {} | ||||||
|  | impl TableLevel for PageDirectory {} | ||||||
|  | impl TableLevel for PageTable {} | ||||||
|  | 
 | ||||||
|  | impl HierarchicalLevel for PageGlobalDirectory { | ||||||
|  |     type NextLevel = PageUpperDirectory; | ||||||
|  | } | ||||||
|  | impl HierarchicalLevel for PageUpperDirectory { | ||||||
|  |     type NextLevel = PageDirectory; | ||||||
|  | } | ||||||
|  | impl HierarchicalLevel for PageDirectory { | ||||||
|  |     type NextLevel = PageTable; | ||||||
|  | } | ||||||
|  | // PageTables do not have next level, therefore they are not HierarchicalLevel
 | ||||||
|  | 
 | ||||||
|  | /// MMU address translation table.
 | ||||||
|  | /// Contains just u64 internally, provides enum interface on top
 | ||||||
|  | #[repr(C)] | ||||||
|  | #[repr(align(4096))] | ||||||
|  | pub struct Table<L: TableLevel> { | ||||||
|  |     entries: [u64; NUM_ENTRIES_4KIB as usize], | ||||||
|  |     level: PhantomData<L>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Implementation code shared for all levels of page tables
 | ||||||
|  | impl<L> Table<L> | ||||||
|  | where | ||||||
|  |     L: TableLevel, | ||||||
|  | { | ||||||
|  |     /// Zero out entire table.
 | ||||||
|  |     pub fn zero(&mut self) { | ||||||
|  |         for entry in self.entries.iter_mut() { | ||||||
|  |             *entry = 0; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<L> Index<usize> for Table<L> | ||||||
|  | where | ||||||
|  |     L: TableLevel, | ||||||
|  | { | ||||||
|  |     type Output = u64; | ||||||
|  | 
 | ||||||
|  |     fn index(&self, index: usize) -> &u64 { | ||||||
|  |         &self.entries[index] | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<L> IndexMut<usize> for Table<L> | ||||||
|  | where | ||||||
|  |     L: TableLevel, | ||||||
|  | { | ||||||
|  |     fn index_mut(&mut self, index: usize) -> &mut u64 { | ||||||
|  |         &mut self.entries[index] | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Type-safe enum wrapper covering Table<L>'s 64-bit entries.
 | ||||||
|  | #[derive(Clone)] | ||||||
|  | // #[repr(transparent)]
 | ||||||
|  | enum PageTableEntry { | ||||||
|  |     /// Empty page table entry.
 | ||||||
|  |     Invalid, | ||||||
|  |     /// Table descriptor is a L0, L1 or L2 table pointing to another table.
 | ||||||
|  |     /// L0 tables can only point to L1 tables.
 | ||||||
|  |     /// A descriptor pointing to the next page table.
 | ||||||
|  |     TableDescriptor(EntryFlags), | ||||||
|  |     /// A Level2 block descriptor with 2 MiB aperture.
 | ||||||
|  |     ///
 | ||||||
|  |     /// The output points to physical memory.
 | ||||||
|  |     Lvl2BlockDescriptor(EntryFlags), | ||||||
|  |     /// A page PageTableEntry::descriptor with 4 KiB aperture.
 | ||||||
|  |     ///
 | ||||||
|  |     /// The output points to physical memory.
 | ||||||
|  |     PageDescriptor(EntryFlags), | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// A descriptor pointing to the next page table. (within PageTableEntry enum)
 | ||||||
|  | // struct TableDescriptor(register::FieldValue<u64, STAGE1_DESCRIPTOR::Register>);
 | ||||||
|  | 
 | ||||||
|  | impl PageTableEntry { | ||||||
|  |     fn new_table_descriptor(next_lvl_table_addr: usize) -> Result<PageTableEntry, &'static str> { | ||||||
|  |         if next_lvl_table_addr % Size4KiB::SIZE as usize != 0 { | ||||||
|  |             // @todo SIZE must be usize
 | ||||||
|  |             return Err("TableDescriptor: Address is not 4 KiB aligned."); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         let shifted = next_lvl_table_addr >> Size4KiB::SHIFT; | ||||||
|  | 
 | ||||||
|  |         Ok(PageTableEntry::TableDescriptor( | ||||||
|  |             STAGE1_DESCRIPTOR::VALID::True | ||||||
|  |                 + STAGE1_DESCRIPTOR::TYPE::Table | ||||||
|  |                 + STAGE1_DESCRIPTOR::NEXT_LVL_TABLE_ADDR_4KiB.val(shifted as u64), | ||||||
|  |         )) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// A Level2 block descriptor with 2 MiB aperture.
 | ||||||
|  | ///
 | ||||||
|  | /// The output points to physical memory.
 | ||||||
|  | // struct Lvl2BlockDescriptor(register::FieldValue<u64, STAGE1_DESCRIPTOR::Register>);
 | ||||||
|  | 
 | ||||||
|  | impl PageTableEntry { | ||||||
|  |     fn new_lvl2_block_descriptor( | ||||||
|  |         output_addr: usize, | ||||||
|  |         attribute_fields: AttributeFields, | ||||||
|  |     ) -> Result<PageTableEntry, &'static str> { | ||||||
|  |         if output_addr % Size2MiB::SIZE as usize != 0 { | ||||||
|  |             return Err("BlockDescriptor: Address is not 2 MiB aligned."); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         let shifted = output_addr >> Size2MiB::SHIFT; | ||||||
|  | 
 | ||||||
|  |         Ok(PageTableEntry::Lvl2BlockDescriptor( | ||||||
|  |             STAGE1_DESCRIPTOR::VALID::True | ||||||
|  |                 + STAGE1_DESCRIPTOR::AF::True | ||||||
|  |                 + into_mmu_attributes(attribute_fields) | ||||||
|  |                 + STAGE1_DESCRIPTOR::TYPE::Block | ||||||
|  |                 + STAGE1_DESCRIPTOR::LVL2_OUTPUT_ADDR_4KiB.val(shifted as u64), | ||||||
|  |         )) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// A page descriptor with 4 KiB aperture.
 | ||||||
|  | ///
 | ||||||
|  | /// The output points to physical memory.
 | ||||||
|  | 
 | ||||||
|  | impl PageTableEntry { | ||||||
|  |     fn new_page_descriptor( | ||||||
|  |         output_addr: usize, | ||||||
|  |         attribute_fields: AttributeFields, | ||||||
|  |     ) -> Result<PageTableEntry, &'static str> { | ||||||
|  |         if output_addr % Size4KiB::SIZE as usize != 0 { | ||||||
|  |             return Err("PageDescriptor: Address is not 4 KiB aligned."); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         let shifted = output_addr >> Size4KiB::SHIFT; | ||||||
|  | 
 | ||||||
|  |         Ok(PageTableEntry::PageDescriptor( | ||||||
|  |             STAGE1_DESCRIPTOR::VALID::True | ||||||
|  |                 + STAGE1_DESCRIPTOR::AF::True | ||||||
|  |                 + into_mmu_attributes(attribute_fields) | ||||||
|  |                 + STAGE1_DESCRIPTOR::TYPE::Table | ||||||
|  |                 + STAGE1_DESCRIPTOR::NEXT_LVL_TABLE_ADDR_4KiB.val(shifted as u64), | ||||||
|  |         )) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<u64> for PageTableEntry { | ||||||
|  |     fn from(_val: u64) -> PageTableEntry { | ||||||
|  |         // xxx0 -> Invalid
 | ||||||
|  |         // xx11 -> TableDescriptor on L0, L1 and L2
 | ||||||
|  |         // xx10 -> Block Entry L1 and L2
 | ||||||
|  |         // xx11 -> PageDescriptor L3
 | ||||||
|  |         PageTableEntry::Invalid | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<PageTableEntry> for u64 { | ||||||
|  |     fn from(val: PageTableEntry) -> u64 { | ||||||
|  |         match val { | ||||||
|  |             PageTableEntry::Invalid => 0, | ||||||
|  |             PageTableEntry::TableDescriptor(x) | ||||||
|  |             | PageTableEntry::Lvl2BlockDescriptor(x) | ||||||
|  |             | PageTableEntry::PageDescriptor(x) => x.value, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static mut LVL2_TABLE: Table<PageDirectory> = Table::<PageDirectory> { | ||||||
|  |     entries: [0; NUM_ENTRIES_4KIB as usize], | ||||||
|  |     level: PhantomData, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static mut LVL3_TABLE: Table<PageTable> = Table::<PageTable> { | ||||||
|  |     entries: [0; NUM_ENTRIES_4KIB as usize], | ||||||
|  |     level: PhantomData, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | trait BaseAddr { | ||||||
|  |     fn base_addr_u64(&self) -> u64; | ||||||
|  |     fn base_addr_usize(&self) -> usize; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl BaseAddr for [u64; 512] { | ||||||
|  |     fn base_addr_u64(&self) -> u64 { | ||||||
|  |         self as *const u64 as u64 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn base_addr_usize(&self) -> usize { | ||||||
|  |         self as *const u64 as usize | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Set up identity mapped page tables for the first 1 gigabyte of address space.
 | ||||||
|  | /// default: 880 MB ARM ram, 128MB VC
 | ||||||
|  | ///
 | ||||||
|  | /// # Safety
 | ||||||
|  | ///
 | ||||||
|  | /// Completely unsafe, we're in the hardware land! Incorrectly initialised tables will just
 | ||||||
|  | /// restart the CPU.
 | ||||||
|  | pub unsafe fn init() -> Result<(), &'static str> { | ||||||
|  |     // Prepare the memory attribute indirection register.
 | ||||||
|  |     mair::set_up(); | ||||||
|  | 
 | ||||||
|  |     // Point the first 2 MiB of virtual addresses to the follow-up LVL3
 | ||||||
|  |     // page-table.
 | ||||||
|  |     LVL2_TABLE.entries[0] = | ||||||
|  |         PageTableEntry::new_table_descriptor(LVL3_TABLE.entries.base_addr_usize())?.into(); | ||||||
|  | 
 | ||||||
|  |     // Fill the rest of the LVL2 (2 MiB) entries as block descriptors.
 | ||||||
|  |     //
 | ||||||
|  |     // Notice the skip(1) which makes the iteration start at the second 2 MiB
 | ||||||
|  |     // block (0x20_0000).
 | ||||||
|  |     for (block_descriptor_nr, entry) in LVL2_TABLE.entries.iter_mut().enumerate().skip(1) { | ||||||
|  |         let virt_addr = block_descriptor_nr << Size2MiB::SHIFT; | ||||||
|  | 
 | ||||||
|  |         let (output_addr, attribute_fields) = match get_virt_addr_properties(virt_addr) { | ||||||
|  |             Err(s) => return Err(s), | ||||||
|  |             Ok((a, b)) => (a, b), | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         let block_desc = | ||||||
|  |             match PageTableEntry::new_lvl2_block_descriptor(output_addr, attribute_fields) { | ||||||
|  |                 Err(s) => return Err(s), | ||||||
|  |                 Ok(desc) => desc, | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |         *entry = block_desc.into(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Finally, fill the single LVL3 table (4 KiB granule).
 | ||||||
|  |     for (page_descriptor_nr, entry) in LVL3_TABLE.entries.iter_mut().enumerate() { | ||||||
|  |         let virt_addr = page_descriptor_nr << Size4KiB::SHIFT; | ||||||
|  | 
 | ||||||
|  |         let (output_addr, attribute_fields) = match get_virt_addr_properties(virt_addr) { | ||||||
|  |             Err(s) => return Err(s), | ||||||
|  |             Ok((a, b)) => (a, b), | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         let page_desc = match PageTableEntry::new_page_descriptor(output_addr, attribute_fields) { | ||||||
|  |             Err(s) => return Err(s), | ||||||
|  |             Ok(desc) => desc, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         *entry = page_desc.into(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Point to the LVL2 table base address in TTBR0.
 | ||||||
|  |     TTBR0_EL1.set_baddr(LVL2_TABLE.entries.base_addr_u64()); // User (lo-)space addresses
 | ||||||
|  | 
 | ||||||
|  |     // TTBR1_EL1.set_baddr(LVL2_TABLE.entries.base_addr_u64()); // Kernel (hi-)space addresses
 | ||||||
|  | 
 | ||||||
|  |     // Configure various settings of stage 1 of the EL1 translation regime.
 | ||||||
|  |     let ips = ID_AA64MMFR0_EL1.read(ID_AA64MMFR0_EL1::PARange); | ||||||
|  |     TCR_EL1.write( | ||||||
|  |         TCR_EL1::TBI0::Ignored // @todo TBI1 also set to Ignored??
 | ||||||
|  |             + TCR_EL1::IPS.val(ips) // Intermediate Physical Address Size
 | ||||||
|  |             // ttbr0 user memory addresses
 | ||||||
|  |             + TCR_EL1::TG0::KiB_4 // 4 KiB granule
 | ||||||
|  |             + TCR_EL1::SH0::Inner | ||||||
|  |             + TCR_EL1::ORGN0::WriteBack_ReadAlloc_WriteAlloc_Cacheable | ||||||
|  |             + TCR_EL1::IRGN0::WriteBack_ReadAlloc_WriteAlloc_Cacheable | ||||||
|  |             + TCR_EL1::EPD0::EnableTTBR0Walks | ||||||
|  |             + TCR_EL1::T0SZ.val(34) // ARMv8ARM Table D5-11 minimum TxSZ for starting table level 2
 | ||||||
|  |             // ttbr1 kernel memory addresses
 | ||||||
|  |             + TCR_EL1::TG1::KiB_4 // 4 KiB granule
 | ||||||
|  |             + TCR_EL1::SH1::Inner | ||||||
|  |             + TCR_EL1::ORGN1::WriteBack_ReadAlloc_WriteAlloc_Cacheable | ||||||
|  |             + TCR_EL1::IRGN1::WriteBack_ReadAlloc_WriteAlloc_Cacheable | ||||||
|  |             + TCR_EL1::EPD1::EnableTTBR1Walks | ||||||
|  |             + TCR_EL1::T1SZ.val(34), // ARMv8ARM Table D5-11 minimum TxSZ for starting table level 2
 | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     // Switch the MMU on.
 | ||||||
|  |     //
 | ||||||
|  |     // First, force all previous changes to be seen before the MMU is enabled.
 | ||||||
|  |     barrier::isb(barrier::SY); | ||||||
|  | 
 | ||||||
|  |     // use cortex_a::regs::RegisterReadWrite;
 | ||||||
|  |     // Enable the MMU and turn on data and instruction caching.
 | ||||||
|  |     SCTLR_EL1.modify(SCTLR_EL1::M::Enable + SCTLR_EL1::C::Cacheable + SCTLR_EL1::I::Cacheable); | ||||||
|  | 
 | ||||||
|  |     // Force MMU init to complete before next instruction
 | ||||||
|  |     /* | ||||||
|  |      * Invalidate the local I-cache so that any instructions fetched | ||||||
|  |      * speculatively from the PoC are discarded, since they may have | ||||||
|  |      * been dynamically patched at the PoU. | ||||||
|  |      */ | ||||||
|  |     barrier::isb(barrier::SY); | ||||||
|  | 
 | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  | @ -1,129 +1,110 @@ | ||||||
| // mod arch::aarch64::memory
 | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
| 
 | 
 | ||||||
| // mod area_frame_allocator;
 | //! Memory management functions for aarch64.
 | ||||||
| // mod paging;
 |  | ||||||
| 
 | 
 | ||||||
| // pub use self::area_frame_allocator::AreaFrameAllocator;
 | use { | ||||||
|  |     crate::println, | ||||||
|  |     core::{fmt, ops::RangeInclusive}, | ||||||
|  | }; | ||||||
| 
 | 
 | ||||||
| mod bump_allocator; | mod addr; | ||||||
| pub use bump_allocator::BumpAllocator; | pub mod mmu; | ||||||
| 
 | 
 | ||||||
| pub type PhysicalAddress = usize; | pub use addr::{PhysAddr, VirtAddr}; | ||||||
| pub type VirtualAddress = usize; |  | ||||||
| 
 | 
 | ||||||
| // use self::paging::PAGE_SIZE;
 | // aarch64 granules and page sizes howto:
 | ||||||
|  | // https://stackoverflow.com/questions/34269185/simultaneous-existence-of-different-sized-pages-on-aarch64
 | ||||||
|  | 
 | ||||||
|  | /// Default page size used by the kernel.
 | ||||||
| pub const PAGE_SIZE: usize = 4096; | pub const PAGE_SIZE: usize = 4096; | ||||||
| 
 | 
 | ||||||
| /**
 |  | ||||||
|  * Frame is an addressable unit of the physical address space. |  | ||||||
|  */ |  | ||||||
| #[derive(Debug, PartialEq, Eq, PartialOrd, Ord)] |  | ||||||
| pub struct Frame { |  | ||||||
|     number: usize, |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl Frame { |  | ||||||
|     fn containing_address(address: usize) -> Frame { |  | ||||||
|         Frame { |  | ||||||
|             number: address / PAGE_SIZE, |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn start_address(&self) -> PhysicalAddress { |  | ||||||
|         self.number * PAGE_SIZE |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| pub trait FrameAllocator { |  | ||||||
|     fn allocate_frame(&mut self) -> Option<Frame>; |  | ||||||
|     fn deallocate_frame(&mut self, frame: Frame); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // --------------------------------------------
 |  | ||||||
| 
 |  | ||||||
| /* |  | ||||||
|  * MIT License |  | ||||||
|  * |  | ||||||
|  * Copyright (c) 2019 Andre Richter <andre.o.richter@gmail.com> |  | ||||||
|  * |  | ||||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy |  | ||||||
|  * of this software and associated documentation files (the "Software"), to deal |  | ||||||
|  * in the Software without restriction, including without limitation the rights |  | ||||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |  | ||||||
|  * copies of the Software, and to permit persons to whom the Software is |  | ||||||
|  * furnished to do so, subject to the following conditions: |  | ||||||
|  * |  | ||||||
|  * The above copyright notice and this permission notice shall be included in all |  | ||||||
|  * copies or substantial portions of the Software. |  | ||||||
|  * |  | ||||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |  | ||||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |  | ||||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |  | ||||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |  | ||||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |  | ||||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |  | ||||||
|  * SOFTWARE. |  | ||||||
|  */ |  | ||||||
| 
 |  | ||||||
| use crate::println; |  | ||||||
| use core::fmt; |  | ||||||
| use core::ops::RangeInclusive; |  | ||||||
| 
 |  | ||||||
| /// System memory map.
 | /// System memory map.
 | ||||||
|  | /// This is a fixed memory map for RasPi3,
 | ||||||
|  | /// @todo we need to infer the memory map from the provided DTB.
 | ||||||
| #[rustfmt::skip] | #[rustfmt::skip] | ||||||
| pub mod map { | pub mod map { | ||||||
|  |     /// Beginning of memory.
 | ||||||
|     pub const START:                   usize =             0x0000_0000; |     pub const START:                   usize =             0x0000_0000; | ||||||
|  |     /// End of memory.
 | ||||||
|     pub const END:                     usize =             0x3FFF_FFFF; |     pub const END:                     usize =             0x3FFF_FFFF; | ||||||
| 
 | 
 | ||||||
|     pub mod physical { |     /// Physical RAM addresses.
 | ||||||
|  |     pub mod phys { | ||||||
|  |         /// Base address of video (VC) memory.
 | ||||||
|         pub const VIDEOMEM_BASE:       usize =             0x3e00_0000; |         pub const VIDEOMEM_BASE:       usize =             0x3e00_0000; | ||||||
|  |         /// Base address of MMIO register range.
 | ||||||
|         pub const MMIO_BASE:           usize =             0x3F00_0000; |         pub const MMIO_BASE:           usize =             0x3F00_0000; | ||||||
|  |         /// Base address of ARM<->VC mailbox area.
 | ||||||
|         pub const VIDEOCORE_MBOX_BASE: usize = MMIO_BASE + 0x0000_B880; |         pub const VIDEOCORE_MBOX_BASE: usize = MMIO_BASE + 0x0000_B880; | ||||||
|  |         /// Base address of GPIO registers.
 | ||||||
|         pub const GPIO_BASE:           usize = MMIO_BASE + 0x0020_0000; |         pub const GPIO_BASE:           usize = MMIO_BASE + 0x0020_0000; | ||||||
|  |         /// Base address of regular UART.
 | ||||||
|         pub const PL011_UART_BASE:     usize = MMIO_BASE + 0x0020_1000; |         pub const PL011_UART_BASE:     usize = MMIO_BASE + 0x0020_1000; | ||||||
|  |         /// Base address of MiniUART.
 | ||||||
|         pub const MINI_UART_BASE:      usize = MMIO_BASE + 0x0021_5000; |         pub const MINI_UART_BASE:      usize = MMIO_BASE + 0x0021_5000; | ||||||
|  |         /// End of MMIO memory.
 | ||||||
|         pub const MMIO_END:            usize =             super::END; |         pub const MMIO_END:            usize =             super::END; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /// Virtual (mapped) addresses.
 | ||||||
|     pub mod virt { |     pub mod virt { | ||||||
|  |         /// Start (top) of kernel stack.
 | ||||||
|         pub const KERN_STACK_START:    usize =             super::START; |         pub const KERN_STACK_START:    usize =             super::START; | ||||||
|  |         /// End (bottom) of kernel stack. SP starts at KERN_STACK_END + 1.
 | ||||||
|         pub const KERN_STACK_END:      usize =             0x0007_FFFF; |         pub const KERN_STACK_END:      usize =             0x0007_FFFF; | ||||||
| 
 | 
 | ||||||
|         // The second 2 MiB block.
 |         /// Location of DMA-able memory region (in the second 2 MiB block).
 | ||||||
|         pub const DMA_HEAP_START:      usize =             0x0020_0000; |         pub const DMA_HEAP_START:      usize =             0x0020_0000; | ||||||
|  |         /// End of DMA-able memory region.
 | ||||||
|         pub const DMA_HEAP_END:        usize =             0x005F_FFFF; |         pub const DMA_HEAP_END:        usize =             0x005F_FFFF; | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Types used for compiling the virtual memory layout of the kernel using
 | /// Types used for compiling the virtual memory layout of the kernel using address ranges.
 | ||||||
| /// address ranges.
 |  | ||||||
| pub mod kernel_mem_range { | pub mod kernel_mem_range { | ||||||
|     use core::ops::RangeInclusive; |     use core::ops::RangeInclusive; | ||||||
| 
 | 
 | ||||||
|  |     /// Memory region attributes.
 | ||||||
|     #[derive(Copy, Clone)] |     #[derive(Copy, Clone)] | ||||||
|     pub enum MemAttributes { |     pub enum MemAttributes { | ||||||
|  |         /// Regular memory
 | ||||||
|         CacheableDRAM, |         CacheableDRAM, | ||||||
|  |         /// Memory without caching
 | ||||||
|         NonCacheableDRAM, |         NonCacheableDRAM, | ||||||
|  |         /// Device memory
 | ||||||
|         Device, |         Device, | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /// Memory region access permissions.
 | ||||||
|     #[derive(Copy, Clone)] |     #[derive(Copy, Clone)] | ||||||
|     pub enum AccessPermissions { |     pub enum AccessPermissions { | ||||||
|  |         /// Read-only access
 | ||||||
|         ReadOnly, |         ReadOnly, | ||||||
|  |         /// Read-write access
 | ||||||
|         ReadWrite, |         ReadWrite, | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /// Memory region translation.
 | ||||||
|     #[allow(dead_code)] |     #[allow(dead_code)] | ||||||
|     #[derive(Copy, Clone)] |     #[derive(Copy, Clone)] | ||||||
|     pub enum Translation { |     pub enum Translation { | ||||||
|  |         /// One-to-one address mapping
 | ||||||
|         Identity, |         Identity, | ||||||
|  |         /// Mapping with a specified offset
 | ||||||
|         Offset(usize), |         Offset(usize), | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /// Summary structure of memory region properties.
 | ||||||
|     #[derive(Copy, Clone)] |     #[derive(Copy, Clone)] | ||||||
|     pub struct AttributeFields { |     pub struct AttributeFields { | ||||||
|  |         /// Attributes
 | ||||||
|         pub mem_attributes: MemAttributes, |         pub mem_attributes: MemAttributes, | ||||||
|  |         /// Permissions
 | ||||||
|         pub acc_perms: AccessPermissions, |         pub acc_perms: AccessPermissions, | ||||||
|  |         /// Disable executable code in this region
 | ||||||
|         pub execute_never: bool, |         pub execute_never: bool, | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -137,10 +118,17 @@ pub mod kernel_mem_range { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /// Memory region descriptor.
 | ||||||
|  |     ///
 | ||||||
|  |     /// Used to construct iterable kernel memory ranges.
 | ||||||
|     pub struct Descriptor { |     pub struct Descriptor { | ||||||
|  |         /// Name of the region
 | ||||||
|         pub name: &'static str, |         pub name: &'static str, | ||||||
|  |         /// Virtual memory range
 | ||||||
|         pub virtual_range: fn() -> RangeInclusive<usize>, |         pub virtual_range: fn() -> RangeInclusive<usize>, | ||||||
|  |         /// Mapping translation
 | ||||||
|         pub translation: Translation, |         pub translation: Translation, | ||||||
|  |         /// Attributes
 | ||||||
|         pub attribute_fields: AttributeFields, |         pub attribute_fields: AttributeFields, | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -152,7 +140,7 @@ pub use kernel_mem_range::*; | ||||||
| ///
 | ///
 | ||||||
| /// Contains only special ranges, aka anything that is _not_ normal cacheable
 | /// Contains only special ranges, aka anything that is _not_ normal cacheable
 | ||||||
| /// DRAM.
 | /// DRAM.
 | ||||||
| static KERNEL_VIRTUAL_LAYOUT: [Descriptor; 5] = [ | static KERNEL_VIRTUAL_LAYOUT: [Descriptor; 6] = [ | ||||||
|     Descriptor { |     Descriptor { | ||||||
|         name: "Kernel stack", |         name: "Kernel stack", | ||||||
|         virtual_range: || { |         virtual_range: || { | ||||||
|  | @ -166,28 +154,61 @@ static KERNEL_VIRTUAL_LAYOUT: [Descriptor; 5] = [ | ||||||
|         }, |         }, | ||||||
|     }, |     }, | ||||||
|     Descriptor { |     Descriptor { | ||||||
|         name: "Kernel code and RO data", |         name: "Boot code and data", | ||||||
|         virtual_range: || { |         virtual_range: || { | ||||||
|             // Using the linker script, we ensure that the RO area is consecutive and 4
 |             // Using the linker script, we ensure that the boot area is consecutive and 4
 | ||||||
|             // KiB aligned, and we export the boundaries via symbols:
 |             // KiB aligned, and we export the boundaries via symbols:
 | ||||||
|             //
 |             //
 | ||||||
|             // [__ro_start, __ro_end)
 |             // [__BOOT_START, __BOOT_END)
 | ||||||
|             extern "C" { |             extern "C" { | ||||||
|                 // The inclusive start of the read-only area, aka the address of the
 |                 // The inclusive start of the boot area, aka the address of the
 | ||||||
|                 // first byte of the area.
 |                 // first byte of the area.
 | ||||||
|                 static __ro_start: u64; |                 static __BOOT_START: u64; | ||||||
| 
 | 
 | ||||||
|                 // The exclusive end of the read-only area, aka the address of
 |                 // The exclusive end of the boot area, aka the address of
 | ||||||
|                 // the first byte _after_ the RO area.
 |                 // the first byte _after_ the RO area.
 | ||||||
|                 static __ro_end: u64; |                 static __BOOT_END: u64; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             unsafe { |             unsafe { | ||||||
|                 // Notice the subtraction to turn the exclusive end into an
 |                 // Notice the subtraction to turn the exclusive end into an
 | ||||||
|                 // inclusive end
 |                 // inclusive end
 | ||||||
|                 RangeInclusive::new( |                 RangeInclusive::new( | ||||||
|                     &__ro_start as *const _ as usize, |                     &__BOOT_START as *const _ as usize, | ||||||
|                     &__ro_end as *const _ as usize - 1, |                     &__BOOT_END as *const _ as usize - 1, | ||||||
|  |                 ) | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |         translation: Translation::Identity, | ||||||
|  |         attribute_fields: AttributeFields { | ||||||
|  |             mem_attributes: MemAttributes::CacheableDRAM, | ||||||
|  |             acc_perms: AccessPermissions::ReadOnly, | ||||||
|  |             execute_never: false, | ||||||
|  |         }, | ||||||
|  |     }, | ||||||
|  |     Descriptor { | ||||||
|  |         name: "Kernel code and RO data", | ||||||
|  |         virtual_range: || { | ||||||
|  |             // Using the linker script, we ensure that the RO area is consecutive and 4
 | ||||||
|  |             // KiB aligned, and we export the boundaries via symbols:
 | ||||||
|  |             //
 | ||||||
|  |             // [__RO_START, __RO_END)
 | ||||||
|  |             extern "C" { | ||||||
|  |                 // The inclusive start of the read-only area, aka the address of the
 | ||||||
|  |                 // first byte of the area.
 | ||||||
|  |                 static __RO_START: u64; | ||||||
|  | 
 | ||||||
|  |                 // The exclusive end of the read-only area, aka the address of
 | ||||||
|  |                 // the first byte _after_ the RO area.
 | ||||||
|  |                 static __RO_END: u64; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             unsafe { | ||||||
|  |                 // Notice the subtraction to turn the exclusive end into an
 | ||||||
|  |                 // inclusive end
 | ||||||
|  |                 RangeInclusive::new( | ||||||
|  |                     &__RO_START as *const _ as usize, | ||||||
|  |                     &__RO_END as *const _ as usize - 1, | ||||||
|                 ) |                 ) | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|  | @ -202,14 +223,14 @@ static KERNEL_VIRTUAL_LAYOUT: [Descriptor; 5] = [ | ||||||
|         name: "Kernel data and BSS", |         name: "Kernel data and BSS", | ||||||
|         virtual_range: || { |         virtual_range: || { | ||||||
|             extern "C" { |             extern "C" { | ||||||
|                 static __ro_end: u64; |                 static __DATA_START: u64; | ||||||
|                 static __bss_end: u64; |                 static __BSS_END: u64; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             unsafe { |             unsafe { | ||||||
|                 RangeInclusive::new( |                 RangeInclusive::new( | ||||||
|                     &__ro_end as *const _ as usize, |                     &__DATA_START as *const _ as usize, | ||||||
|                     &__bss_end as *const _ as usize - 1, |                     &__BSS_END as *const _ as usize - 1, | ||||||
|                 ) |                 ) | ||||||
|             } |             } | ||||||
|         }, |         }, | ||||||
|  | @ -232,9 +253,7 @@ static KERNEL_VIRTUAL_LAYOUT: [Descriptor; 5] = [ | ||||||
|     }, |     }, | ||||||
|     Descriptor { |     Descriptor { | ||||||
|         name: "Device MMIO", |         name: "Device MMIO", | ||||||
|         virtual_range: || { |         virtual_range: || RangeInclusive::new(map::phys::VIDEOMEM_BASE, map::phys::MMIO_END), | ||||||
|             RangeInclusive::new(map::physical::VIDEOMEM_BASE, map::physical::MMIO_END) |  | ||||||
|         }, |  | ||||||
|         translation: Translation::Identity, |         translation: Translation::Identity, | ||||||
|         attribute_fields: AttributeFields { |         attribute_fields: AttributeFields { | ||||||
|             mem_attributes: MemAttributes::Device, |             mem_attributes: MemAttributes::Device, | ||||||
|  | @ -326,10 +345,3 @@ pub fn print_layout() { | ||||||
|         println!("{}", i); |         println!("{}", i); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 |  | ||||||
| /// Calculate the next possible aligned address without sanity checking the
 |  | ||||||
| /// input parameters.
 |  | ||||||
| #[inline] |  | ||||||
| fn aligned_addr_unchecked(addr: usize, alignment: usize) -> usize { |  | ||||||
|     (addr + (alignment - 1)) & !(alignment - 1) |  | ||||||
| } |  | ||||||
|  | @ -0,0 +1,52 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | //! Implementation of aarch64 kernel functions.
 | ||||||
|  | 
 | ||||||
|  | use cortex_a::asm; | ||||||
|  | 
 | ||||||
|  | mod boot; | ||||||
|  | #[cfg(feature = "jtag")] | ||||||
|  | pub mod jtag; | ||||||
|  | pub mod memory; | ||||||
|  | pub mod traps; | ||||||
|  | 
 | ||||||
|  | /// Loop forever in sleep mode.
 | ||||||
|  | #[inline] | ||||||
|  | pub fn endless_sleep() -> ! { | ||||||
|  |     loop { | ||||||
|  |         asm::wfe(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Loop for a given number of `nop` instructions.
 | ||||||
|  | #[inline] | ||||||
|  | pub fn loop_delay(rounds: u32) { | ||||||
|  |     for _ in 0..rounds { | ||||||
|  |         asm::nop(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Loop until a passed function returns `true`.
 | ||||||
|  | #[inline] | ||||||
|  | pub fn loop_until<F: Fn() -> bool>(f: F) { | ||||||
|  |     loop { | ||||||
|  |         if f() { | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         asm::nop(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Loop while a passed function returns `true`.
 | ||||||
|  | #[inline] | ||||||
|  | pub fn loop_while<F: Fn() -> bool>(f: F) { | ||||||
|  |     loop { | ||||||
|  |         if !f() { | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         asm::nop(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,379 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | //! Interrupt handling
 | ||||||
|  | //!
 | ||||||
|  | //! The base address is given by VBAR_ELn and each entry has a defined offset from this
 | ||||||
|  | //! base address. Each table has 16 entries, with each entry being 128 bytes (32 instructions)
 | ||||||
|  | //! in size. The table effectively consists of 4 sets of 4 entries.
 | ||||||
|  | //!
 | ||||||
|  | //! Minimal implementation to help catch MMU traps.
 | ||||||
|  | //! Reads ESR_ELx to understand why trap was taken.
 | ||||||
|  | //!
 | ||||||
|  | //! VBAR_EL1, VBAR_EL2, VBAR_EL3
 | ||||||
|  | //!
 | ||||||
|  | //! CurrentEL with SP0: +0x0
 | ||||||
|  | //!
 | ||||||
|  | //! * Synchronous
 | ||||||
|  | //! * IRQ/vIRQ
 | ||||||
|  | //! * FIQ
 | ||||||
|  | //! * SError/vSError
 | ||||||
|  | //!
 | ||||||
|  | //! CurrentEL with SPx: +0x200
 | ||||||
|  | //!
 | ||||||
|  | //! * Synchronous
 | ||||||
|  | //! * IRQ/vIRQ
 | ||||||
|  | //! * FIQ
 | ||||||
|  | //! * SError/vSError
 | ||||||
|  | //!
 | ||||||
|  | //! Lower EL using AArch64: +0x400
 | ||||||
|  | //!
 | ||||||
|  | //! * Synchronous
 | ||||||
|  | //! * IRQ/vIRQ
 | ||||||
|  | //! * FIQ
 | ||||||
|  | //! * SError/vSError
 | ||||||
|  | //!
 | ||||||
|  | //! Lower EL using AArch32: +0x600
 | ||||||
|  | //!
 | ||||||
|  | //! * Synchronous
 | ||||||
|  | //! * IRQ/vIRQ
 | ||||||
|  | //! * FIQ
 | ||||||
|  | //! * SError/vSError
 | ||||||
|  | //!
 | ||||||
|  | //! When the processor takes an exception to AArch64 execution state,
 | ||||||
|  | //! all of the PSTATE interrupt masks is set automatically. This means
 | ||||||
|  | //! that further exceptions are disabled. If software is to support
 | ||||||
|  | //! nested exceptions, for example, to allow a higher priority interrupt
 | ||||||
|  | //! to interrupt the handling of a lower priority source, then software needs
 | ||||||
|  | //! to explicitly re-enable interrupts
 | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     crate::{arch::endless_sleep, println}, | ||||||
|  |     cortex_a::{ | ||||||
|  |         asm::barrier, | ||||||
|  |         registers::{ESR_EL1, FAR_EL1, VBAR_EL1}, | ||||||
|  |     }, | ||||||
|  |     snafu::Snafu, | ||||||
|  |     tock_registers::{ | ||||||
|  |         interfaces::{Readable, Writeable}, | ||||||
|  |         register_bitfields, LocalRegisterCopy, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | core::arch::global_asm!(include_str!("vectors.S")); | ||||||
|  | 
 | ||||||
|  | /// Errors possibly returned from the traps module.
 | ||||||
|  | #[derive(Debug, Snafu)] | ||||||
|  | pub enum Error { | ||||||
|  |     /// IVT address is unaligned.
 | ||||||
|  |     #[snafu(display("Unaligned base address for interrupt vector table"))] | ||||||
|  |     Unaligned, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Configure base address of interrupt vectors table.
 | ||||||
|  | /// Checks that address is properly 2KiB aligned.
 | ||||||
|  | ///
 | ||||||
|  | /// # Safety
 | ||||||
|  | ///
 | ||||||
|  | /// Totally unsafe in the land of the hardware.
 | ||||||
|  | pub unsafe fn set_vbar_el1_checked(vec_base_addr: u64) -> Result<(), Error> { | ||||||
|  |     if vec_base_addr.trailing_zeros() < 11 { | ||||||
|  |         return Err(Error::Unaligned); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     VBAR_EL1.set(vec_base_addr); | ||||||
|  | 
 | ||||||
|  |     // Force VBAR update to complete before next instruction.
 | ||||||
|  |     barrier::isb(barrier::SY); | ||||||
|  | 
 | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// A blob of general-purpose registers.
 | ||||||
|  | #[repr(C)] | ||||||
|  | pub struct GPR { | ||||||
|  |     x: [u64; 31], | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Saved exception context.
 | ||||||
|  | #[repr(C)] | ||||||
|  | pub struct ExceptionContext { | ||||||
|  |     // General Purpose Registers
 | ||||||
|  |     gpr: GPR, | ||||||
|  |     spsr_el1: u64, | ||||||
|  |     elr_el1: u64, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// The default exception, invoked for every exception type unless the handler
 | ||||||
|  | /// is overridden.
 | ||||||
|  | /// Default pointer is configured in the linker script.
 | ||||||
|  | ///
 | ||||||
|  | /// # Safety
 | ||||||
|  | ///
 | ||||||
|  | /// Totally unsafe in the land of the hardware.
 | ||||||
|  | #[no_mangle] | ||||||
|  | unsafe extern "C" fn default_exception_handler() -> ! { | ||||||
|  |     println!("Unexpected exception. Halting CPU."); | ||||||
|  | 
 | ||||||
|  |     endless_sleep() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // To implement an exception handler, override it by defining the respective
 | ||||||
|  | // function below.
 | ||||||
|  | // Don't forget the #[no_mangle] attribute.
 | ||||||
|  | //
 | ||||||
|  | /// # Safety
 | ||||||
|  | ///
 | ||||||
|  | /// Totally unsafe in the land of the hardware.
 | ||||||
|  | #[no_mangle] | ||||||
|  | unsafe extern "C" fn current_el0_synchronous(e: &mut ExceptionContext) { | ||||||
|  |     println!("[!] USER synchronous exception happened."); | ||||||
|  |     synchronous_common(e) | ||||||
|  | } | ||||||
|  | // unsafe extern "C" fn current_el0_irq(e: &mut ExceptionContext);
 | ||||||
|  | // unsafe extern "C" fn current_el0_serror(e: &mut ExceptionContext);
 | ||||||
|  | 
 | ||||||
|  | /// # Safety
 | ||||||
|  | ///
 | ||||||
|  | /// Totally unsafe in the land of the hardware.
 | ||||||
|  | #[no_mangle] | ||||||
|  | unsafe extern "C" fn current_elx_synchronous(e: &mut ExceptionContext) { | ||||||
|  |     println!("[!] KERNEL synchronous exception happened."); | ||||||
|  |     synchronous_common(e); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // unsafe extern "C" fn current_elx_irq(e: &mut ExceptionContext);
 | ||||||
|  | /// # Safety
 | ||||||
|  | ///
 | ||||||
|  | /// Totally unsafe in the land of the hardware.
 | ||||||
|  | #[no_mangle] | ||||||
|  | unsafe extern "C" fn current_elx_serror(e: &mut ExceptionContext) { | ||||||
|  |     println!("[!] KERNEL serror exception happened."); | ||||||
|  |     synchronous_common(e); | ||||||
|  |     endless_sleep() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn cause_to_string(cause: u64) -> &'static str { | ||||||
|  |     if cause == ESR_EL1::EC::DataAbortCurrentEL.read(ESR_EL1::EC) { | ||||||
|  |         "Data Alignment Check" | ||||||
|  |     } else { | ||||||
|  |         "Unknown" | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | register_bitfields! { | ||||||
|  |     u64, | ||||||
|  |     /// ISS structure for Data Abort exceptions
 | ||||||
|  |     ISS_DA [ | ||||||
|  |         /// Instruction Syndrome Valid. Indicates whether the syndrome information in ISS[23:14] is valid.
 | ||||||
|  |         /// (This includes SAS, SSE, SRT, SF, and AR)
 | ||||||
|  |         ISV   OFFSET(24) NUMBITS(1) [], | ||||||
|  |         SAS   OFFSET(22) NUMBITS(2) [ | ||||||
|  |             Byte = 0b00, | ||||||
|  |             Halfword = 0b01, | ||||||
|  |             Word = 0b10, | ||||||
|  |             DoubleWord = 0b11 | ||||||
|  |         ], | ||||||
|  |         SSE   OFFSET(21) NUMBITS(1) [], | ||||||
|  |         SRT   OFFSET(16) NUMBITS(5) [], | ||||||
|  |         SF    OFFSET(15) NUMBITS(1) [], | ||||||
|  |         AR    OFFSET(14) NUMBITS(1) [], | ||||||
|  |         VNCR  OFFSET(13) NUMBITS(1) [], | ||||||
|  |         SET   OFFSET(11) NUMBITS(2) [ | ||||||
|  |             UER = 0b00, // Recoverable state
 | ||||||
|  |             UC = 0b10, // Uncontainable
 | ||||||
|  |             UEO = 0b11 // Restartable state
 | ||||||
|  |         ], | ||||||
|  |         FNV   OFFSET(10) NUMBITS(1) [], | ||||||
|  |         EA    OFFSET(9)  NUMBITS(1) [], | ||||||
|  |         CM    OFFSET(8)  NUMBITS(1) [], | ||||||
|  |         S1PTW OFFSET(7)  NUMBITS(1) [], | ||||||
|  |         WNR   OFFSET(6)  NUMBITS(1) [], | ||||||
|  |         DFSC  OFFSET(0)  NUMBITS(6) [ | ||||||
|  |             /// Address size fault, level 0 of translation or translation table base register.
 | ||||||
|  |             AddressSizeTL0 = 0b000000, | ||||||
|  |             /// Address size fault, level 1.
 | ||||||
|  |             AddressSizeTL1 = 0b000001, | ||||||
|  |             ///Address size fault, level 2.
 | ||||||
|  |             AddressSizeTL2 = 0b000010, | ||||||
|  |             /// Address size fault, level 3.
 | ||||||
|  |             AddressSizeTL3 = 0b000011, | ||||||
|  |             /// Translation fault, level 0.
 | ||||||
|  |             TranslationFaultTL0 = 0b000100, | ||||||
|  |             /// Translation fault, level 1.
 | ||||||
|  |             TranslationFaultTL1 = 0b000101, | ||||||
|  |             /// Translation fault, level 2.
 | ||||||
|  |             TranslationFaultTL2 = 0b000110, | ||||||
|  |             /// Translation fault, level 3.
 | ||||||
|  |             TranslationFaultTL3 = 0b000111, | ||||||
|  |             /// Access flag fault, level 1.
 | ||||||
|  |             AccessFaultTL1 = 0b001001, | ||||||
|  |             /// Access flag fault, level 2.
 | ||||||
|  |             AccessFaultTL2 = 0b001010, | ||||||
|  |             /// Access flag fault, level 3.
 | ||||||
|  |             AccessFaultTL3 = 0b001011, | ||||||
|  |             /// Permission fault, level 1.
 | ||||||
|  |             PermissionFaultTL1 = 0b001101, | ||||||
|  |             /// Permission fault, level 2.
 | ||||||
|  |             PermissionFaultTL2 = 0b001110, | ||||||
|  |             /// Permission fault, level 3.
 | ||||||
|  |             PermissionFaultTL3 = 0b001111, | ||||||
|  |             /// Synchronous External abort, not on translation table walk or hardware update of translation table.
 | ||||||
|  |             SyncExternalAbort = 0b010000, | ||||||
|  |             /// Synchronous Tag Check Fault.
 | ||||||
|  |             /// (When FEAT_MTE is implemented)
 | ||||||
|  |             SyncTagCheckFault = 0b010001, | ||||||
|  |             /// Synchronous External abort on translation table walk or hardware update of translation table, level 0.
 | ||||||
|  |             SyncAbortOnTranslationTL0 = 0b010100, | ||||||
|  |             /// Synchronous External abort on translation table walk or hardware update of translation table, level 1.
 | ||||||
|  |             SyncAbortOnTranslationTL1 = 0b010101, | ||||||
|  |             /// Synchronous External abort on translation table walk or hardware update of translation table, level 2.
 | ||||||
|  |             SyncAbortOnTranslationTL2 = 0b010110, | ||||||
|  |             /// Synchronous External abort on translation table walk or hardware update of translation table, level 3.
 | ||||||
|  |             SyncAbortOnTranslationTL3 = 0b010111, | ||||||
|  |             /// Synchronous parity or ECC error on memory access, not on translation table walk.
 | ||||||
|  |             /// (When FEAT_RAS is not implemented)
 | ||||||
|  |             SyncParityError = 0b011000, | ||||||
|  |             /// Synchronous parity or ECC error on memory access on translation table walk or hardware update of translation table, level 0.
 | ||||||
|  |             /// (When FEAT_RAS is not implemented)
 | ||||||
|  |             SyncParityErrorOnTranslationTL0 = 0b011100, | ||||||
|  |             /// Synchronous parity or ECC error on memory access on translation table walk or hardware update of translation table, level 1.
 | ||||||
|  |             /// (When FEAT_RAS is not implemented)
 | ||||||
|  |             SyncParityErrorOnTranslationTL1 = 0b011101, | ||||||
|  |             /// Synchronous parity or ECC error on memory access on translation table walk or hardware update of translation table, level 2.
 | ||||||
|  |             /// (When FEAT_RAS is not implemented)
 | ||||||
|  |             SyncParityErrorOnTranslationTL2 = 0b011110, | ||||||
|  |             /// Synchronous parity or ECC error on memory access on translation table walk or hardware update of translation table, level 3.
 | ||||||
|  |             /// (When FEAT_RAS is not implemented)
 | ||||||
|  |             SyncParityErrorOnTranslationTL3 = 0b011111, | ||||||
|  |             /// Alignment fault.
 | ||||||
|  |             AlignmentFault = 0b100001, | ||||||
|  |             /// TLB conflict abort.
 | ||||||
|  |             TlbConflictAbort = 0b110000, | ||||||
|  |             /// Unsupported atomic hardware update fault.
 | ||||||
|  |             /// (When FEAT_HAFDBS is implemented)
 | ||||||
|  |             UnsupportedAtomicUpdate = 0b110001, | ||||||
|  |             /// IMPLEMENTATION DEFINED fault (Lockdown).
 | ||||||
|  |             Lockdown = 0b110100, | ||||||
|  |             /// IMPLEMENTATION DEFINED fault (Unsupported Exclusive or Atomic access).
 | ||||||
|  |             UnsupportedAccess = 0b110101 | ||||||
|  |         ] | ||||||
|  |     ] | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type IssForDataAbort = LocalRegisterCopy<u64, ISS_DA::Register>; | ||||||
|  | 
 | ||||||
|  | fn iss_dfsc_to_string(iss: IssForDataAbort) -> &'static str { | ||||||
|  |     match iss.read_as_enum(ISS_DA::DFSC) { | ||||||
|  |         Some(ISS_DA::DFSC::Value::AddressSizeTL0) => "Address size fault, level 0 of translation or translation table base register", | ||||||
|  |         Some(ISS_DA::DFSC::Value::AddressSizeTL1) => "Address size fault, level 1", | ||||||
|  |         Some(ISS_DA::DFSC::Value::AddressSizeTL2) => "Address size fault, level 2", | ||||||
|  |         Some(ISS_DA::DFSC::Value::AddressSizeTL3) => "Address size fault, level 3", | ||||||
|  |         Some(ISS_DA::DFSC::Value::TranslationFaultTL0) => "Translation fault, level 0", | ||||||
|  |         Some(ISS_DA::DFSC::Value::TranslationFaultTL1) => "Translation fault, level 1", | ||||||
|  |         Some(ISS_DA::DFSC::Value::TranslationFaultTL2) => "Translation fault, level 2", | ||||||
|  |         Some(ISS_DA::DFSC::Value::TranslationFaultTL3) => "Translation fault, level 3", | ||||||
|  |         Some(ISS_DA::DFSC::Value::AccessFaultTL1) => "Access flag fault, level 1", | ||||||
|  |         Some(ISS_DA::DFSC::Value::AccessFaultTL2) => "Access flag fault, level 2", | ||||||
|  |         Some(ISS_DA::DFSC::Value::AccessFaultTL3) => "Access flag fault, level 3", | ||||||
|  |         Some(ISS_DA::DFSC::Value::PermissionFaultTL1) => "Permission fault, level 1", | ||||||
|  |         Some(ISS_DA::DFSC::Value::PermissionFaultTL2) => "Permission fault, level 2", | ||||||
|  |         Some(ISS_DA::DFSC::Value::PermissionFaultTL3) => "Permission fault, level 3", | ||||||
|  |         Some(ISS_DA::DFSC::Value::SyncExternalAbort) => "Synchronous External abort, not on translation table walk or hardware update of translation table", | ||||||
|  |         Some(ISS_DA::DFSC::Value::SyncTagCheckFault) => "Synchronous Tag Check Fault", | ||||||
|  |         Some(ISS_DA::DFSC::Value::SyncAbortOnTranslationTL0) => "Synchronous External abort on translation table walk or hardware update of translation table, level 0", | ||||||
|  |         Some(ISS_DA::DFSC::Value::SyncAbortOnTranslationTL1) => "Synchronous External abort on translation table walk or hardware update of translation table, level 1", | ||||||
|  |         Some(ISS_DA::DFSC::Value::SyncAbortOnTranslationTL2) => "Synchronous External abort on translation table walk or hardware update of translation table, level 2", | ||||||
|  |         Some(ISS_DA::DFSC::Value::SyncAbortOnTranslationTL3) => "Synchronous External abort on translation table walk or hardware update of translation table, level 3", | ||||||
|  |         Some(ISS_DA::DFSC::Value::SyncParityError) => "Synchronous parity or ECC error on memory access, not on translation table walk", | ||||||
|  |         Some(ISS_DA::DFSC::Value::SyncParityErrorOnTranslationTL0) => "Synchronous parity or ECC error on memory access on translation table walk or hardware update of translation table, level 0", | ||||||
|  |         Some(ISS_DA::DFSC::Value::SyncParityErrorOnTranslationTL1) => "Synchronous parity or ECC error on memory access on translation table walk or hardware update of translation table, level 1", | ||||||
|  |         Some(ISS_DA::DFSC::Value::SyncParityErrorOnTranslationTL2) => "Synchronous parity or ECC error on memory access on translation table walk or hardware update of translation table, level 2", | ||||||
|  |         Some(ISS_DA::DFSC::Value::SyncParityErrorOnTranslationTL3) => "Synchronous parity or ECC error on memory access on translation table walk or hardware update of translation table, level 3", | ||||||
|  |         Some(ISS_DA::DFSC::Value::AlignmentFault) => "Alignment fault", | ||||||
|  |         Some(ISS_DA::DFSC::Value::TlbConflictAbort) => "TLB conflict abort", | ||||||
|  |         Some(ISS_DA::DFSC::Value::UnsupportedAtomicUpdate) => "Unsupported atomic hardware update fault", | ||||||
|  |         Some(ISS_DA::DFSC::Value::Lockdown) => "Lockdown (IMPLEMENTATION DEFINED fault)", | ||||||
|  |         Some(ISS_DA::DFSC::Value::UnsupportedAccess) => "Unsupported Exclusive or Atomic access (IMPLEMENTATION DEFINED fault)", | ||||||
|  |         _ => "Unknown", | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // unsafe extern "C" fn lower_aarch64_synchronous(e: &mut ExceptionContext);
 | ||||||
|  | // unsafe extern "C" fn lower_aarch64_irq(e: &mut ExceptionContext);
 | ||||||
|  | // unsafe extern "C" fn lower_aarch64_serror(e: &mut ExceptionContext);
 | ||||||
|  | 
 | ||||||
|  | // unsafe extern "C" fn lower_aarch32_synchronous(e: &mut ExceptionContext);
 | ||||||
|  | // unsafe extern "C" fn lower_aarch32_irq(e: &mut ExceptionContext);
 | ||||||
|  | // unsafe extern "C" fn lower_aarch32_serror(e: &mut ExceptionContext);
 | ||||||
|  | 
 | ||||||
|  | /// Helper function to 1) display current exception, 2) skip the offending asm instruction.
 | ||||||
|  | /// Not for production use!
 | ||||||
|  | fn synchronous_common(e: &mut ExceptionContext) { | ||||||
|  |     println!("      ESR_EL1: {:#010x} (syndrome)", ESR_EL1.get()); | ||||||
|  |     let cause = ESR_EL1.read(ESR_EL1::EC); | ||||||
|  |     println!( | ||||||
|  |         "           EC: {:#06b} (cause) -- {}", | ||||||
|  |         cause, | ||||||
|  |         cause_to_string(cause) | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     // Print more details about Data Alignment Check
 | ||||||
|  |     if cause == ESR_EL1::EC::DataAbortCurrentEL.read(ESR_EL1::EC) { | ||||||
|  |         let iss = ESR_EL1.read(ESR_EL1::ISS); | ||||||
|  |         let iss = IssForDataAbort::new(iss); | ||||||
|  |         if iss.is_set(ISS_DA::ISV) { | ||||||
|  |             println!( | ||||||
|  |                 "               Access size: {} bytes ({}signed) to {}{}", | ||||||
|  |                 2u64.pow(iss.read(ISS_DA::SAS) as u32), | ||||||
|  |                 if iss.is_set(ISS_DA::SSE) { "" } else { "un" }, | ||||||
|  |                 if iss.is_set(ISS_DA::SF) { "x" } else { "r" }, | ||||||
|  |                 iss.read(ISS_DA::SRT) | ||||||
|  |             ); | ||||||
|  |             println!( | ||||||
|  |                 "               Acq/Rel semantics: {}present", | ||||||
|  |                 if iss.is_set(ISS_DA::AR) { "" } else { "not " } | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |         // data abort specific encoding
 | ||||||
|  |         println!( | ||||||
|  |             "               {} address {:#016x} ({}valid)", | ||||||
|  |             if iss.is_set(ISS_DA::WNR) { | ||||||
|  |                 "Writing to" | ||||||
|  |             } else { | ||||||
|  |                 "Reading from" | ||||||
|  |             }, | ||||||
|  |             FAR_EL1.get(), | ||||||
|  |             if iss.is_set(ISS_DA::FNV) { "not " } else { "" } | ||||||
|  |         ); | ||||||
|  |         println!("               Specific fault: {}", iss_dfsc_to_string(iss)); | ||||||
|  |     } else { | ||||||
|  |         println!("      FAR_EL1: {:#016x} (location)", FAR_EL1.get()); | ||||||
|  |         println!("        Stack: {:#016x}", e.spsr_el1); | ||||||
|  |     } | ||||||
|  |     println!("      ELR_EL1: {:#010x}", e.elr_el1); | ||||||
|  | 
 | ||||||
|  |     println!("      x00: 0000000000000000    x01: {:016x}", e.gpr.x[0]); | ||||||
|  | 
 | ||||||
|  |     for index in 0..15 { | ||||||
|  |         println!( | ||||||
|  |             "      x{:02}: {:016x}    x{:02}: {:016x}", | ||||||
|  |             index * 2 + 2, | ||||||
|  |             e.gpr.x[index * 2 + 1], | ||||||
|  |             index * 2 + 3, | ||||||
|  |             e.gpr.x[index * 2 + 2] | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     println!( | ||||||
|  |         "      Incrementing ELR_EL1 by 4 to continue with the first \ | ||||||
|  |          instruction after the exception!" | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     e.elr_el1 += 4; | ||||||
|  | 
 | ||||||
|  |     println!("      ELR_EL1 modified: {:#010x}", e.elr_el1); | ||||||
|  |     println!("      Returning from exception...\n"); | ||||||
|  | } | ||||||
|  | @ -1,26 +1,9 @@ | ||||||
| // | /* | ||||||
| //  MIT License |  * SPDX-License-Identifier: MIT OR BlueOak-1.0.0 | ||||||
| // |  * Copyright (c) 2018-2019 Andre Richter <andre.o.richter@gmail.com>
 | ||||||
| //  Copyright (c) 2018-2019 Andre Richter <andre.o.richter@gmail.com>
 |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems>
 | ||||||
| // |  * Original code distributed under MIT, additional changes are under BlueOak-1.0.0 | ||||||
| //  Permission is hereby granted, free of charge, to any person obtaining a copy |  */ | ||||||
| //  of this software and associated documentation files (the "Software"), to deal |  | ||||||
| //  in the Software without restriction, including without limitation the rights |  | ||||||
| //  to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |  | ||||||
| //  copies of the Software, and to permit persons to whom the Software is |  | ||||||
| //  furnished to do so, subject to the following conditions: |  | ||||||
| // |  | ||||||
| //  The above copyright notice and this permission notice shall be included in all |  | ||||||
| //  copies or substantial portions of the Software. |  | ||||||
| // |  | ||||||
| //  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |  | ||||||
| //  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |  | ||||||
| //  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |  | ||||||
| //  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |  | ||||||
| //  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |  | ||||||
| //  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |  | ||||||
| //  SOFTWARE. |  | ||||||
| // |  | ||||||
| 
 | 
 | ||||||
| .macro SAVE_CONTEXT_CALL_HANDLER_AND_RESTORE handler | .macro SAVE_CONTEXT_CALL_HANDLER_AND_RESTORE handler | ||||||
| .balign 0x80
 | .balign 0x80
 | ||||||
|  | @ -0,0 +1,10 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #[cfg(target_arch = "aarch64")] | ||||||
|  | #[macro_use] | ||||||
|  | pub mod aarch64; | ||||||
|  | #[cfg(target_arch = "aarch64")] | ||||||
|  | pub use self::aarch64::*; | ||||||
|  | @ -0,0 +1,177 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #![allow(dead_code)] | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     crate::{devices::SerialOps, platform}, | ||||||
|  |     core::fmt, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /// A trait that must be implemented by devices that are candidates for the
 | ||||||
|  | /// global console.
 | ||||||
|  | #[allow(unused_variables)] | ||||||
|  | pub trait ConsoleOps: SerialOps { | ||||||
|  |     /// Send a character
 | ||||||
|  |     fn write_char(&self, c: char); | ||||||
|  |     /// Display a string
 | ||||||
|  |     fn write_string(&self, string: &str); | ||||||
|  |     /// Receive a character
 | ||||||
|  |     fn read_char(&self) -> char; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// A dummy console that just ignores its inputs.
 | ||||||
|  | pub struct NullConsole; | ||||||
|  | 
 | ||||||
|  | impl Drop for NullConsole { | ||||||
|  |     fn drop(&mut self) {} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl ConsoleOps for NullConsole { | ||||||
|  |     fn write_char(&self, _c: char) {} | ||||||
|  | 
 | ||||||
|  |     fn write_string(&self, _string: &str) {} | ||||||
|  | 
 | ||||||
|  |     fn read_char(&self) -> char { | ||||||
|  |         ' ' | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl SerialOps for NullConsole { | ||||||
|  |     fn read_byte(&self) -> u8 { | ||||||
|  |         0 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn write_byte(&self, _byte: u8) {} | ||||||
|  | 
 | ||||||
|  |     fn flush(&self) {} | ||||||
|  | 
 | ||||||
|  |     fn clear_rx(&self) {} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Possible outputs which the console can store.
 | ||||||
|  | pub enum Output { | ||||||
|  |     None(NullConsole), | ||||||
|  |     MiniUart(platform::rpi3::mini_uart::PreparedMiniUart), | ||||||
|  |     Uart(platform::rpi3::pl011_uart::PreparedPL011Uart), | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Generate boilerplate for converting into one of Output enum values
 | ||||||
|  | macro output_from($name:ty, $optname:ident) { | ||||||
|  |     impl From<$name> for Output { | ||||||
|  |         fn from(instance: $name) -> Self { | ||||||
|  |             Output::$optname(instance) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | output_from!(NullConsole, None); | ||||||
|  | output_from!(platform::rpi3::mini_uart::PreparedMiniUart, MiniUart); | ||||||
|  | output_from!(platform::rpi3::pl011_uart::PreparedPL011Uart, Uart); | ||||||
|  | 
 | ||||||
|  | pub struct Console { | ||||||
|  |     output: Output, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Default for Console { | ||||||
|  |     fn default() -> Self { | ||||||
|  |         Console { | ||||||
|  |             output: (NullConsole {}).into(), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Console { | ||||||
|  |     pub const fn new() -> Console { | ||||||
|  |         Console { | ||||||
|  |             output: Output::None(NullConsole {}), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn current_ptr(&self) -> &dyn ConsoleOps { | ||||||
|  |         match &self.output { | ||||||
|  |             Output::None(i) => i, | ||||||
|  |             Output::MiniUart(i) => i, | ||||||
|  |             Output::Uart(i) => i, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Overwrite the current output. The old output will go out of scope and
 | ||||||
|  |     /// it's Drop function will be called.
 | ||||||
|  |     pub fn replace_with(&mut self, x: Output) { | ||||||
|  |         self.current_ptr().flush(); | ||||||
|  | 
 | ||||||
|  |         self.output = x; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// A command prompt.
 | ||||||
|  |     pub fn command_prompt<'a>(&self, buf: &'a mut [u8]) -> &'a [u8] { | ||||||
|  |         self.write_string("\n$> "); | ||||||
|  | 
 | ||||||
|  |         let mut i = 0; | ||||||
|  |         let mut input; | ||||||
|  |         loop { | ||||||
|  |             input = self.read_char(); | ||||||
|  | 
 | ||||||
|  |             if input == '\n' { | ||||||
|  |                 self.write_char('\n'); // do \r\n output
 | ||||||
|  |                 return &buf[..i]; | ||||||
|  |             } else { | ||||||
|  |                 if i < buf.len() { | ||||||
|  |                     buf[i] = input as u8; | ||||||
|  |                     i += 1; | ||||||
|  |                 } else { | ||||||
|  |                     return &buf[..i]; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 self.write_char(input); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Drop for Console { | ||||||
|  |     fn drop(&mut self) {} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Dispatch the respective function to the currently stored output device.
 | ||||||
|  | impl ConsoleOps for Console { | ||||||
|  |     fn write_char(&self, c: char) { | ||||||
|  |         self.current_ptr().write_char(c); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn write_string(&self, string: &str) { | ||||||
|  |         self.current_ptr().write_string(string); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn read_char(&self) -> char { | ||||||
|  |         self.current_ptr().read_char() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl SerialOps for Console { | ||||||
|  |     fn read_byte(&self) -> u8 { | ||||||
|  |         self.current_ptr().read_byte() | ||||||
|  |     } | ||||||
|  |     fn write_byte(&self, byte: u8) { | ||||||
|  |         self.current_ptr().write_byte(byte) | ||||||
|  |     } | ||||||
|  |     fn flush(&self) { | ||||||
|  |         self.current_ptr().flush() | ||||||
|  |     } | ||||||
|  |     fn clear_rx(&self) { | ||||||
|  |         self.current_ptr().clear_rx() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Implementing this trait enables usage of the format_args! macros, which in
 | ||||||
|  | /// turn are used to implement the kernel's print! and println! macros.
 | ||||||
|  | ///
 | ||||||
|  | /// See src/macros.rs.
 | ||||||
|  | impl fmt::Write for Console { | ||||||
|  |     fn write_str(&mut self, s: &str) -> fmt::Result { | ||||||
|  |         self.current_ptr().write_string(s); | ||||||
|  |         Ok(()) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,10 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  */ | ||||||
|  | pub mod console; | ||||||
|  | pub mod serial; | ||||||
|  | 
 | ||||||
|  | pub use { | ||||||
|  |     console::{Console, ConsoleOps}, | ||||||
|  |     serial::SerialOps, | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,12 @@ | ||||||
|  | pub trait SerialOps { | ||||||
|  |     /// Read one byte from serial without translation.
 | ||||||
|  |     fn read_byte(&self) -> u8; | ||||||
|  |     /// Write one byte to serial without translation.
 | ||||||
|  |     fn write_byte(&self, byte: u8); | ||||||
|  |     /// Wait until the TX FIFO is empty, aka all characters have been put on the
 | ||||||
|  |     /// line.
 | ||||||
|  |     fn flush(&self); | ||||||
|  |     /// Consume input until RX FIFO is empty, aka all pending characters have been
 | ||||||
|  |     /// consumed.
 | ||||||
|  |     fn clear_rx(&self); | ||||||
|  | } | ||||||
|  | @ -0,0 +1,57 @@ | ||||||
|  | #![no_std] | ||||||
|  | #![no_main] | ||||||
|  | #![feature(decl_macro)] | ||||||
|  | #![feature(allocator_api)] | ||||||
|  | #![feature(format_args_nl)] | ||||||
|  | #![feature(const_fn_fn_ptr_basics)] | ||||||
|  | #![feature(nonnull_slice_from_raw_parts)] | ||||||
|  | #![feature(custom_test_frameworks)] | ||||||
|  | #![test_runner(crate::tests::test_runner)] | ||||||
|  | #![reexport_test_harness_main = "test_main"] | ||||||
|  | #![allow(clippy::upper_case_acronyms)] | ||||||
|  | #![allow(clippy::enum_variant_names)] | ||||||
|  | #![allow(clippy::nonstandard_macro_braces)] // https://github.com/shepmaster/snafu/issues/296
 | ||||||
|  | #![allow(missing_docs)] // Temp: switch to deny
 | ||||||
|  | #![deny(warnings)] | ||||||
|  | 
 | ||||||
|  | #[cfg(not(target_arch = "aarch64"))] | ||||||
|  | use architecture_not_supported_sorry; | ||||||
|  | 
 | ||||||
|  | /// Architecture-specific code.
 | ||||||
|  | #[macro_use] | ||||||
|  | pub mod arch; | ||||||
|  | 
 | ||||||
|  | pub use arch::*; | ||||||
|  | 
 | ||||||
|  | pub mod devices; | ||||||
|  | pub mod macros; | ||||||
|  | mod mm; | ||||||
|  | pub mod panic; | ||||||
|  | pub mod platform; | ||||||
|  | pub mod qemu; | ||||||
|  | mod sync; | ||||||
|  | pub mod tests; | ||||||
|  | pub mod write_to; | ||||||
|  | 
 | ||||||
|  | /// The global console. Output of the kernel print! and println! macros goes here.
 | ||||||
|  | pub static CONSOLE: sync::NullLock<devices::Console> = sync::NullLock::new(devices::Console::new()); | ||||||
|  | 
 | ||||||
|  | /// The global allocator for DMA-able memory. That is, memory which is tagged
 | ||||||
|  | /// non-cacheable in the page tables.
 | ||||||
|  | #[allow(dead_code)] | ||||||
|  | static DMA_ALLOCATOR: sync::NullLock<mm::BumpAllocator> = | ||||||
|  |     sync::NullLock::new(mm::BumpAllocator::new( | ||||||
|  |         // @todo Init this after we loaded boot memory map
 | ||||||
|  |         memory::map::virt::DMA_HEAP_START as usize, | ||||||
|  |         memory::map::virt::DMA_HEAP_END as usize, | ||||||
|  |         "Global DMA Allocator", | ||||||
|  |         // Try the following arguments instead to see all mailbox operations
 | ||||||
|  |         // fail. It will cause the allocator to use memory that are marked
 | ||||||
|  |         // cacheable and therefore not DMA-safe. The answer from the VideoCore
 | ||||||
|  |         // won't be received by the CPU because it reads an old cached value
 | ||||||
|  |         // that resembles an error case instead.
 | ||||||
|  | 
 | ||||||
|  |         // 0x00600000 as usize,
 | ||||||
|  |         // 0x007FFFFF as usize,
 | ||||||
|  |         // "Global Non-DMA Allocator",
 | ||||||
|  |     )); | ||||||
|  | @ -0,0 +1,41 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /// Macro similar to [std](https://doc.rust-lang.org/src/std/macros.rs.html)
 | ||||||
|  | /// but for writing into kernel-specific output (UART or QEMU console).
 | ||||||
|  | #[macro_export] | ||||||
|  | macro_rules! print { | ||||||
|  |     ($($arg:tt)*) => ($crate::macros::_print(format_args!($($arg)*))); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Macro similar to [std](https://doc.rust-lang.org/src/std/macros.rs.html)
 | ||||||
|  | /// but for writing into kernel-specific output (UART or QEMU console).
 | ||||||
|  | #[macro_export] | ||||||
|  | macro_rules! println { | ||||||
|  |     () => (print!("\n")); | ||||||
|  |     ($($arg:tt)*) => ({ | ||||||
|  |         $crate::macros::_print(format_args_nl!($($arg)*)); | ||||||
|  |     }) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[doc(hidden)] | ||||||
|  | #[cfg(not(any(test, qemu)))] | ||||||
|  | pub fn _print(args: core::fmt::Arguments) { | ||||||
|  |     use core::fmt::Write; | ||||||
|  | 
 | ||||||
|  |     crate::CONSOLE.lock(|c| { | ||||||
|  |         c.write_fmt(args).unwrap(); | ||||||
|  |     }) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// qemu-based tests use semihosting write0 syscall.
 | ||||||
|  | #[doc(hidden)] | ||||||
|  | #[cfg(any(test, qemu))] // qemu feature not enabled here?? we pass --features=qemu to cargo test
 | ||||||
|  | pub fn _print(args: core::fmt::Arguments) { | ||||||
|  |     use crate::{qemu, write_to}; | ||||||
|  | 
 | ||||||
|  |     let mut buf = [0u8; 2048]; // Increase this buffer size to allow dumping larger panic texts.
 | ||||||
|  |     qemu::semihosting::sys_write0_call(write_to::c_show(&mut buf, args).unwrap()); | ||||||
|  | } | ||||||
|  | @ -0,0 +1,93 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | // @todo Use alloc-fmt crate for logging in allocators
 | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     crate::println, | ||||||
|  |     core::{ | ||||||
|  |         alloc::{AllocError, Allocator, Layout}, | ||||||
|  |         cell::Cell, | ||||||
|  |         ptr::NonNull, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | pub struct BumpAllocator { | ||||||
|  |     next: Cell<usize>, | ||||||
|  |     pool_end: usize, | ||||||
|  |     name: &'static str, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | unsafe impl Allocator for BumpAllocator { | ||||||
|  |     /// Allocate a memory block from the pool.
 | ||||||
|  |     fn allocate(&self, layout: Layout) -> Result<NonNull<[u8]>, AllocError> { | ||||||
|  |         let start = crate::mm::aligned_addr_unchecked(self.next.get(), layout.align()); | ||||||
|  |         let end = start + layout.size(); | ||||||
|  | 
 | ||||||
|  |         println!( | ||||||
|  |             "[i] {}:\n    Allocating Start {:#010x} End {:#010x}", | ||||||
|  |             self.name, start, end | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |         if end > self.pool_end { | ||||||
|  |             return Err(AllocError); | ||||||
|  |         } | ||||||
|  |         self.next.set(end); | ||||||
|  | 
 | ||||||
|  |         println!( | ||||||
|  |             "[i] {}:\n    Allocated Addr {:#010x} Size {:#x}", | ||||||
|  |             self.name, | ||||||
|  |             start, | ||||||
|  |             layout.size() | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |         Ok(NonNull::slice_from_raw_parts( | ||||||
|  |             unsafe { NonNull::new_unchecked(start as *mut u8) }, | ||||||
|  |             layout.size(), | ||||||
|  |         )) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// A bump allocator doesn't care about releasing memory.
 | ||||||
|  |     unsafe fn deallocate(&self, _ptr: NonNull<u8>, _layout: Layout) {} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl BumpAllocator { | ||||||
|  |     /// Create a named bump allocator between start and end addresses.
 | ||||||
|  |     #[allow(dead_code)] | ||||||
|  |     pub const fn new(pool_start: usize, pool_end: usize, name: &'static str) -> Self { | ||||||
|  |         Self { | ||||||
|  |             next: Cell::new(pool_start), | ||||||
|  |             pool_end, | ||||||
|  |             name, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(test)] | ||||||
|  | mod tests { | ||||||
|  |     use super::*; | ||||||
|  | 
 | ||||||
|  |     // Validate allocator allocates from the provided address range
 | ||||||
|  |     // Validate allocation fails when range is exhausted
 | ||||||
|  |     #[test_case] | ||||||
|  |     fn test_allocates_within_init_range() { | ||||||
|  |         let allocator = BumpAllocator::new(256, 512, "Test allocator 1"); | ||||||
|  |         let result1 = allocator.allocate(unsafe { Layout::from_size_align_unchecked(128, 1) }); | ||||||
|  |         assert!(result1.is_ok()); | ||||||
|  |         let result2 = allocator.allocate(unsafe { Layout::from_size_align_unchecked(128, 32) }); | ||||||
|  |         println!("{:?}", result2); | ||||||
|  |         assert!(result2.is_ok()); | ||||||
|  |         let result3 = allocator.allocate(unsafe { Layout::from_size_align_unchecked(1, 1) }); | ||||||
|  |         assert!(result3.is_err()); | ||||||
|  |     } | ||||||
|  |     // Creating with end <= start sshould fail
 | ||||||
|  |     // @todo return Result<> from new?
 | ||||||
|  |     #[test_case] | ||||||
|  |     fn test_bad_allocator() { | ||||||
|  |         let bad_allocator = BumpAllocator::new(512, 256, "Test allocator 2"); | ||||||
|  |         let result1 = bad_allocator.allocate(unsafe { Layout::from_size_align_unchecked(1, 1) }); | ||||||
|  |         assert!(result1.is_err()); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,59 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | pub mod bump_allocator; | ||||||
|  | pub use bump_allocator::BumpAllocator; | ||||||
|  | 
 | ||||||
|  | /// Align address downwards.
 | ||||||
|  | ///
 | ||||||
|  | /// Returns the greatest x with alignment `align` so that x <= addr.
 | ||||||
|  | /// The alignment must be a power of 2.
 | ||||||
|  | pub fn align_down(addr: u64, align: u64) -> u64 { | ||||||
|  |     assert!(align.is_power_of_two(), "`align` must be a power of two"); | ||||||
|  |     addr & !(align - 1) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Align address upwards.
 | ||||||
|  | ///
 | ||||||
|  | /// Returns the smallest x with alignment `align` so that x >= addr.
 | ||||||
|  | /// The alignment must be a power of 2.
 | ||||||
|  | pub fn align_up(addr: u64, align: u64) -> u64 { | ||||||
|  |     assert!(align.is_power_of_two(), "`align` must be a power of two"); | ||||||
|  |     let align_mask = align - 1; | ||||||
|  |     if addr & align_mask == 0 { | ||||||
|  |         addr // already aligned
 | ||||||
|  |     } else { | ||||||
|  |         (addr | align_mask) + 1 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Calculate the next possible aligned address without sanity checking the
 | ||||||
|  | /// input parameters.
 | ||||||
|  | #[inline] | ||||||
|  | fn aligned_addr_unchecked(addr: usize, alignment: usize) -> usize { | ||||||
|  |     (addr + (alignment - 1)) & !(alignment - 1) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(test)] | ||||||
|  | mod tests { | ||||||
|  |     use super::*; | ||||||
|  | 
 | ||||||
|  |     #[test_case] | ||||||
|  |     pub fn test_align_up() { | ||||||
|  |         // align 1
 | ||||||
|  |         assert_eq!(align_up(0, 1), 0); | ||||||
|  |         assert_eq!(align_up(1234, 1), 1234); | ||||||
|  |         assert_eq!(align_up(0xffff_ffff_ffff_ffff, 1), 0xffff_ffff_ffff_ffff); | ||||||
|  |         // align 2
 | ||||||
|  |         assert_eq!(align_up(0, 2), 0); | ||||||
|  |         assert_eq!(align_up(1233, 2), 1234); | ||||||
|  |         assert_eq!(align_up(0xffff_ffff_ffff_fffe, 2), 0xffff_ffff_ffff_fffe); | ||||||
|  |         // address 0
 | ||||||
|  |         assert_eq!(align_up(0, 128), 0); | ||||||
|  |         assert_eq!(align_up(0, 1), 0); | ||||||
|  |         assert_eq!(align_up(0, 2), 0); | ||||||
|  |         assert_eq!(align_up(0, 0x8000_0000_0000_0000), 0); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,12 @@ | ||||||
|  | pub fn handler(info: &core::panic::PanicInfo) -> ! { | ||||||
|  |     // @todo This may fail to print if the panic message is too long for local print buffer.
 | ||||||
|  |     crate::println!("{}", info); | ||||||
|  |     crate::endless_sleep() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub fn handler_for_tests(info: &core::panic::PanicInfo) -> ! { | ||||||
|  |     crate::println!("\n[failed]\n"); | ||||||
|  |     // @todo This may fail to print if the panic message is too long for local print buffer.
 | ||||||
|  |     crate::println!("\nError: {}\n", info); | ||||||
|  |     crate::qemu::semihosting::exit_failure() | ||||||
|  | } | ||||||
|  | @ -1,3 +1,7 @@ | ||||||
| # Board Support Packages | # Board Support Packages | ||||||
| 
 | 
 | ||||||
| This directory contains support for specific Boards like RaspberryPi3 etc. | This directory contains support for specific Boards like RaspberryPi3 etc. | ||||||
|  | 
 | ||||||
|  | ---- | ||||||
|  | 
 | ||||||
|  | For more information please re-read. | ||||||
|  | @ -0,0 +1,52 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | use core::{marker::PhantomData, ops}; | ||||||
|  | 
 | ||||||
|  | //--------------------------------------------------------------------------------------------------
 | ||||||
|  | // Public Definitions
 | ||||||
|  | //--------------------------------------------------------------------------------------------------
 | ||||||
|  | 
 | ||||||
|  | pub mod rpi3; | ||||||
|  | 
 | ||||||
|  | pub struct MMIODerefWrapper<T> { | ||||||
|  |     base_addr: usize, | ||||||
|  |     phantom: PhantomData<fn() -> T>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | //--------------------------------------------------------------------------------------------------
 | ||||||
|  | // Public Code
 | ||||||
|  | //--------------------------------------------------------------------------------------------------
 | ||||||
|  | 
 | ||||||
|  | impl<T> MMIODerefWrapper<T> { | ||||||
|  |     /// Create an instance.
 | ||||||
|  |     ///
 | ||||||
|  |     /// # Safety
 | ||||||
|  |     ///
 | ||||||
|  |     /// Unsafe, duh!
 | ||||||
|  |     pub const unsafe fn new(start_addr: usize) -> Self { | ||||||
|  |         Self { | ||||||
|  |             base_addr: start_addr, | ||||||
|  |             phantom: PhantomData, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Deref to RegisterBlock
 | ||||||
|  | ///
 | ||||||
|  | /// Allows writing
 | ||||||
|  | /// ```
 | ||||||
|  | /// self.GPPUD.read()
 | ||||||
|  | /// ```
 | ||||||
|  | /// instead of something along the lines of
 | ||||||
|  | /// ```
 | ||||||
|  | /// unsafe { (*GPIO::ptr()).GPPUD.read() }
 | ||||||
|  | /// ```
 | ||||||
|  | impl<T> ops::Deref for MMIODerefWrapper<T> { | ||||||
|  |     type Target = T; | ||||||
|  | 
 | ||||||
|  |     fn deref(&self) -> &Self::Target { | ||||||
|  |         unsafe { &*(self.base_addr as *const _) } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,3 +1,9 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | use snafu::Snafu; | ||||||
|  | 
 | ||||||
| /* Character cells are 8x8 */ | /* Character cells are 8x8 */ | ||||||
| pub const CHARSIZE_X: u32 = 8; | pub const CHARSIZE_X: u32 = 8; | ||||||
| pub const CHARSIZE_Y: u32 = 8; | pub const CHARSIZE_Y: u32 = 8; | ||||||
|  | @ -10,27 +16,29 @@ pub struct Size2d { | ||||||
| pub struct Color(pub u32); | pub struct Color(pub u32); | ||||||
| 
 | 
 | ||||||
| impl Color { | impl Color { | ||||||
|     pub fn rgb(r: u8, g: u8, b: u8) -> Color { |     pub const fn rgb(r: u8, g: u8, b: u8) -> Color { | ||||||
|         Color(u32::from(b) << 16 | u32::from(g) << 8 | u32::from(r)) |         // @todo use u32::from(g) when it's declared const
 | ||||||
|  |         // (see https://doc.rust-lang.org/src/core/convert/num.rs.html#49-54)
 | ||||||
|  |         Color((b as u32) << 16 | (g as u32) << 8 | (r as u32)) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn black() -> Color { |     pub const fn black() -> Color { | ||||||
|         Color::rgb(0, 0, 0) |         Color::rgb(0, 0, 0) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn white() -> Color { |     pub const fn white() -> Color { | ||||||
|         Color::rgb(255, 255, 255) |         Color::rgb(255, 255, 255) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn red() -> Color { |     pub const fn red() -> Color { | ||||||
|         Color::rgb(255, 0, 0) |         Color::rgb(255, 0, 0) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn green() -> Color { |     pub const fn green() -> Color { | ||||||
|         Color::rgb(0, 255, 0) |         Color::rgb(0, 255, 0) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn blue() -> Color { |     pub const fn blue() -> Color { | ||||||
|         Color::rgb(0, 0, 255) |         Color::rgb(0, 0, 255) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -53,6 +61,13 @@ pub struct Display { | ||||||
|     order: PixelOrder, |     order: PixelOrder, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | #[derive(Debug, Snafu)] | ||||||
|  | pub enum DrawError { | ||||||
|  |     #[snafu(display("Character not in font"))] | ||||||
|  |     CharacterNotInFont, | ||||||
|  | } | ||||||
|  | type Result<T, E = DrawError> = ::core::result::Result<T, E>; | ||||||
|  | 
 | ||||||
| // https://github.com/david-griffith/rust-bitmap/blob/master/src/lib.rs
 | // https://github.com/david-griffith/rust-bitmap/blob/master/src/lib.rs
 | ||||||
| #[rustfmt::skip] | #[rustfmt::skip] | ||||||
| static CHAR_ARRAY: [u64; 95] = [ | static CHAR_ARRAY: [u64; 95] = [ | ||||||
|  | @ -92,6 +107,7 @@ static CHAR_ARRAY: [u64; 95] = [ | ||||||
| ]; | ]; | ||||||
| 
 | 
 | ||||||
| impl Display { | impl Display { | ||||||
|  |     #[allow(clippy::too_many_arguments)] // Sorry, Clips, this api stays for now, ugly as it is.
 | ||||||
|     pub fn new( |     pub fn new( | ||||||
|         base: u32, |         base: u32, | ||||||
|         size: u32, |         size: u32, | ||||||
|  | @ -137,9 +153,9 @@ impl Display { | ||||||
|     /// Set a pixel value on display at given coordinates.
 |     /// Set a pixel value on display at given coordinates.
 | ||||||
|     #[inline] |     #[inline] | ||||||
|     pub fn putpixel(&mut self, x: u32, y: u32, color: u32) { |     pub fn putpixel(&mut self, x: u32, y: u32, color: u32) { | ||||||
|         self.write_pixel_component(x, y, 0, color & 0xff); |         self.write_pixel_component(x, y, 0, color & 0xff); // R
 | ||||||
|         self.write_pixel_component(x, y, 1, (color >> 8) & 0xff); |         self.write_pixel_component(x, y, 1, (color >> 8) & 0xff); // G
 | ||||||
|         self.write_pixel_component(x, y, 2, (color >> 16) & 0xff); |         self.write_pixel_component(x, y, 2, (color >> 16) & 0xff); // B
 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn rect(&mut self, x1: u32, y1: u32, x2: u32, y2: u32, color: Color) { |     pub fn rect(&mut self, x1: u32, y1: u32, x2: u32, y2: u32, color: Color) { | ||||||
|  | @ -154,14 +170,14 @@ impl Display { | ||||||
|         self.rect(0, 0, self.width, self.height, color) |         self.rect(0, 0, self.width, self.height, color) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     pub fn draw_text(&mut self, x: u32, y: u32, text: &str, color: Color) { |     pub fn draw_text(&mut self, x: u32, y: u32, text: &str, color: Color) -> Result<()> { | ||||||
|         for i in 0..8 { |         for i in 0..8 { | ||||||
|             // Take an 8 bit slice from each array value.
 |             // Take an 8 bit slice from each array value.
 | ||||||
|             for (char_off, my_char) in text.as_bytes().iter().enumerate() { |             for (char_off, my_char) in text.as_bytes().iter().enumerate() { | ||||||
|                 let off = (char_off * 8) as u32; |                 let off = (char_off * 8) as u32; | ||||||
| 
 | 
 | ||||||
|                 if (*my_char as isize - 0x20 > 95) || (*my_char as isize - 0x20 < 0) { |                 if (*my_char as isize - 0x20 > 95) || (*my_char as isize - 0x20 < 0) { | ||||||
|                     return; // Err("Character not in font.");
 |                     return Err(DrawError::CharacterNotInFont); | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 let mut myval = CHAR_ARRAY[*my_char as usize - 0x20]; |                 let mut myval = CHAR_ARRAY[*my_char as usize - 0x20]; | ||||||
|  | @ -179,5 +195,6 @@ impl Display { | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |         Ok(()) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -0,0 +1,70 @@ | ||||||
|  | use super::mailbox::{self, LocalMailboxStorage, Mailbox, MailboxError, MailboxOps}; | ||||||
|  | 
 | ||||||
|  | /// FrameBuffer channel supported structure - use with mailbox::channel::FrameBuffer
 | ||||||
|  | /// Must have the same alignment as the mailbox buffers.
 | ||||||
|  | type FrameBufferData = LocalMailboxStorage<10>; | ||||||
|  | 
 | ||||||
|  | mod index { | ||||||
|  |     pub const WIDTH: usize = 0; | ||||||
|  |     pub const HEIGHT: usize = 1; | ||||||
|  |     pub const VIRTUAL_WIDTH: usize = 2; | ||||||
|  |     pub const VIRTUAL_HEIGHT: usize = 3; | ||||||
|  |     pub const PITCH: usize = 4; | ||||||
|  |     pub const DEPTH: usize = 5; | ||||||
|  |     pub const X_OFFSET: usize = 6; | ||||||
|  |     pub const Y_OFFSET: usize = 7; | ||||||
|  |     pub const POINTER: usize = 8; // FIXME: could be 4096 for the alignment restriction.
 | ||||||
|  |     pub const SIZE: usize = 9; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub struct FrameBuffer { | ||||||
|  |     mailbox: Mailbox<10, FrameBufferData>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl core::fmt::Debug for FrameBufferData { | ||||||
|  |     fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result { | ||||||
|  |         write!( | ||||||
|  |             f, | ||||||
|  |             "\n\n\n#### FrameBuffer({}x{}, {}x{}, d{}, --{}--, +{}x{}, {}@{:x})\n\n\n", | ||||||
|  |             self.storage[index::WIDTH], | ||||||
|  |             self.storage[index::HEIGHT], | ||||||
|  |             self.storage[index::VIRTUAL_WIDTH], | ||||||
|  |             self.storage[index::VIRTUAL_HEIGHT], | ||||||
|  |             self.storage[index::HEIGHT], | ||||||
|  |             self.storage[index::PITCH], | ||||||
|  |             self.storage[index::X_OFFSET], | ||||||
|  |             self.storage[index::Y_OFFSET], | ||||||
|  |             self.storage[index::SIZE], | ||||||
|  |             self.storage[index::POINTER], | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl FrameBuffer { | ||||||
|  |     pub fn new( | ||||||
|  |         base_addr: usize, | ||||||
|  |         width: u32, | ||||||
|  |         height: u32, | ||||||
|  |         depth: u32, | ||||||
|  |     ) -> Result<FrameBuffer, MailboxError> { | ||||||
|  |         let mut fb = FrameBuffer { | ||||||
|  |             mailbox: unsafe { Mailbox::<10, FrameBufferData>::new(base_addr)? }, | ||||||
|  |         }; | ||||||
|  |         fb.mailbox.buffer.storage[index::WIDTH] = width; | ||||||
|  |         fb.mailbox.buffer.storage[index::VIRTUAL_WIDTH] = width; | ||||||
|  |         fb.mailbox.buffer.storage[index::HEIGHT] = height; | ||||||
|  |         fb.mailbox.buffer.storage[index::VIRTUAL_HEIGHT] = height; | ||||||
|  |         fb.mailbox.buffer.storage[index::DEPTH] = depth; | ||||||
|  |         Ok(fb) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl MailboxOps for FrameBuffer { | ||||||
|  |     fn write(&self, _channel: u32) -> mailbox::Result<()> { | ||||||
|  |         self.mailbox.do_write(mailbox::channel::FrameBuffer) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn read(&self, _channel: u32) -> mailbox::Result<()> { | ||||||
|  |         unsafe { self.mailbox.do_read(mailbox::channel::FrameBuffer, 0) } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,362 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: MIT OR BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) 2018-2019 Andre Richter <andre.o.richter@gmail.com> | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  * Original code distributed under MIT, additional changes are under BlueOak-1.0.0 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     super::BcmHost, | ||||||
|  |     crate::platform::MMIODerefWrapper, | ||||||
|  |     core::marker::PhantomData, | ||||||
|  |     tock_registers::{ | ||||||
|  |         fields::FieldValue, | ||||||
|  |         interfaces::{ReadWriteable, Readable, Writeable}, | ||||||
|  |         register_structs, | ||||||
|  |         registers::{ReadOnly, ReadWrite, WriteOnly}, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // Descriptions taken from
 | ||||||
|  | // https://github.com/raspberrypi/documentation/files/1888662/BCM2837-ARM-Peripherals.-.Revised.-.V2-1.pdf
 | ||||||
|  | 
 | ||||||
|  | /// Generates `pub enums` with no variants for each `ident` passed in.
 | ||||||
|  | macro states($($name:ident),*) { | ||||||
|  | $(pub enum $name {  })* | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Possible states for a GPIO pin.
 | ||||||
|  | states! { | ||||||
|  |     Uninitialized, Input, Output, Alt | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | register_structs! { | ||||||
|  |     /// The offsets for each register.
 | ||||||
|  |     /// From <https://wiki.osdev.org/Raspberry_Pi_Bare_Bones> and
 | ||||||
|  |     /// <https://github.com/raspberrypi/documentation/files/1888662/BCM2837-ARM-Peripherals.-.Revised.-.V2-1.pdf>
 | ||||||
|  |     #[allow(non_snake_case)] | ||||||
|  |     RegisterBlock { | ||||||
|  |         (0x00 => pub FSEL: [ReadWrite<u32>; 6]), // function select
 | ||||||
|  |         (0x18 => __reserved_1), | ||||||
|  |         (0x1c => pub SET: [WriteOnly<u32>; 2]), // set output pin
 | ||||||
|  |         (0x24 => __reserved_2), | ||||||
|  |         (0x28 => pub CLR: [WriteOnly<u32>; 2]), // clear output pin
 | ||||||
|  |         (0x30 => __reserved_3), | ||||||
|  |         (0x34 => pub LEV: [ReadOnly<u32>; 2]), // get input pin level
 | ||||||
|  |         (0x3c => __reserved_4), | ||||||
|  |         (0x40 => pub EDS: [ReadWrite<u32>; 2]), | ||||||
|  |         (0x48 => __reserved_5), | ||||||
|  |         (0x4c => pub REN: [ReadWrite<u32>; 2]), | ||||||
|  |         (0x54 => __reserved_6), | ||||||
|  |         (0x58 => pub FEN: [ReadWrite<u32>; 2]), | ||||||
|  |         (0x60 => __reserved_7), | ||||||
|  |         (0x64 => pub HEN: [ReadWrite<u32>; 2]), | ||||||
|  |         (0x6c => __reserved_8), | ||||||
|  |         (0x70 => pub LEN: [ReadWrite<u32>; 2]), | ||||||
|  |         (0x78 => __reserved_9), | ||||||
|  |         (0x7c => pub AREN: [ReadWrite<u32>; 2]), | ||||||
|  |         (0x84 => __reserved_10), | ||||||
|  |         (0x88 => pub AFEN: [ReadWrite<u32>; 2]), | ||||||
|  |         (0x90 => __reserved_11), | ||||||
|  |         #[cfg(feature = "rpi3")] | ||||||
|  |         (0x94 => pub PUD: ReadWrite<u32>), // pull up down
 | ||||||
|  |         #[cfg(feature = "rpi3")] | ||||||
|  |         (0x98 => pub PUDCLK: [ReadWrite<u32>; 2]), | ||||||
|  |         #[cfg(feature = "rpi3")] | ||||||
|  |         (0xa0 => __reserved_12), | ||||||
|  |         #[cfg(feature = "rpi4")] | ||||||
|  |         (0xe4 => PullUpDownControl: [ReadWrite<u32>; 4]), | ||||||
|  |         (0xf4 => @END), | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Hide RegisterBlock from public api.
 | ||||||
|  | type Registers = MMIODerefWrapper<RegisterBlock>; | ||||||
|  | 
 | ||||||
|  | /// Public interface to the GPIO MMIO area
 | ||||||
|  | pub struct GPIO { | ||||||
|  |     registers: Registers, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub const GPIO_START: usize = 0x20_0000; | ||||||
|  | 
 | ||||||
|  | impl Default for GPIO { | ||||||
|  |     fn default() -> GPIO { | ||||||
|  |         // Default RPi3 GPIO base address
 | ||||||
|  |         const GPIO_BASE: usize = BcmHost::get_peripheral_address() + GPIO_START; | ||||||
|  |         unsafe { GPIO::new(GPIO_BASE) } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl GPIO { | ||||||
|  |     /// # Safety
 | ||||||
|  |     ///
 | ||||||
|  |     /// Unsafe, duh!
 | ||||||
|  |     pub const unsafe fn new(base_addr: usize) -> GPIO { | ||||||
|  |         GPIO { | ||||||
|  |             registers: Registers::new(base_addr), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     pub fn get_pin(&self, pin: usize) -> Pin<Uninitialized> { | ||||||
|  |         unsafe { Pin::new(pin, self.registers.base_addr) } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[cfg(feature = "rpi3")] | ||||||
|  |     pub fn power_off(&self) { | ||||||
|  |         use crate::arch::loop_delay; | ||||||
|  | 
 | ||||||
|  |         // power off gpio pins (but not VCC pins)
 | ||||||
|  |         for bank in 0..5 { | ||||||
|  |             self.registers.FSEL[bank].set(0); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         self.registers.PUD.set(0); | ||||||
|  | 
 | ||||||
|  |         loop_delay(2000); | ||||||
|  | 
 | ||||||
|  |         self.registers.PUDCLK[0].set(0xffff_ffff); | ||||||
|  |         self.registers.PUDCLK[1].set(0xffff_ffff); | ||||||
|  | 
 | ||||||
|  |         loop_delay(2000); | ||||||
|  | 
 | ||||||
|  |         // flush GPIO setup
 | ||||||
|  |         self.registers.PUDCLK[0].set(0); | ||||||
|  |         self.registers.PUDCLK[1].set(0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[cfg(feature = "rpi4")] | ||||||
|  |     pub fn power_off(&self) { | ||||||
|  |         todo!() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// An alternative GPIO function.
 | ||||||
|  | #[repr(u8)] | ||||||
|  | pub enum Function { | ||||||
|  |     Input = 0b000, | ||||||
|  |     Output = 0b001, | ||||||
|  |     Alt0 = 0b100, | ||||||
|  |     Alt1 = 0b101, | ||||||
|  |     Alt2 = 0b110, | ||||||
|  |     Alt3 = 0b111, | ||||||
|  |     Alt4 = 0b011, | ||||||
|  |     Alt5 = 0b010, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl ::core::convert::From<Function> for u32 { | ||||||
|  |     fn from(f: Function) -> Self { | ||||||
|  |         f as u32 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Pull up/down resistor setup.
 | ||||||
|  | #[repr(u8)] | ||||||
|  | #[derive(PartialEq)] | ||||||
|  | pub enum PullUpDown { | ||||||
|  |     None = 0b00, | ||||||
|  |     Up = 0b01, | ||||||
|  |     Down = 0b10, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl ::core::convert::From<PullUpDown> for u32 { | ||||||
|  |     fn from(p: PullUpDown) -> Self { | ||||||
|  |         p as u32 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// A GPIO pin in state `State`.
 | ||||||
|  | ///
 | ||||||
|  | /// The `State` generic always corresponds to an un-instantiable type that is
 | ||||||
|  | /// used solely to mark and track the state of a given GPIO pin. A `Pin`
 | ||||||
|  | /// structure starts in the `Uninitialized` state and must be transitioned into
 | ||||||
|  | /// one of `Input`, `Output`, or `Alt` via the `into_input`, `into_output`, and
 | ||||||
|  | /// `into_alt` methods before it can be used.
 | ||||||
|  | pub struct Pin<State> { | ||||||
|  |     pin: usize, | ||||||
|  |     registers: Registers, | ||||||
|  |     _state: PhantomData<State>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<State> Pin<State> { | ||||||
|  |     /// Transitions `self` to state `NewState`, consuming `self` and returning a new
 | ||||||
|  |     /// `Pin` instance in state `NewState`. This method should _never_ be exposed to
 | ||||||
|  |     /// the public!
 | ||||||
|  |     #[inline(always)] | ||||||
|  |     fn transition<NewState>(self) -> Pin<NewState> { | ||||||
|  |         Pin { | ||||||
|  |             pin: self.pin, | ||||||
|  |             registers: self.registers, | ||||||
|  |             _state: PhantomData, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[cfg(feature = "rpi3")] | ||||||
|  |     pub fn set_pull_up_down(&self, pull: PullUpDown) { | ||||||
|  |         use crate::arch::loop_delay; | ||||||
|  | 
 | ||||||
|  |         let bank = self.pin / 32; | ||||||
|  |         let off = self.pin % 32; | ||||||
|  | 
 | ||||||
|  |         self.registers.PUD.set(0); | ||||||
|  | 
 | ||||||
|  |         loop_delay(2000); | ||||||
|  | 
 | ||||||
|  |         self.registers.PUDCLK[bank].modify(FieldValue::<u32, ()>::new( | ||||||
|  |             0b1, | ||||||
|  |             off, | ||||||
|  |             if pull == PullUpDown::Up { 1 } else { 0 }, | ||||||
|  |         )); | ||||||
|  | 
 | ||||||
|  |         loop_delay(2000); | ||||||
|  | 
 | ||||||
|  |         self.registers.PUD.set(0); | ||||||
|  |         self.registers.PUDCLK[bank].set(0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[cfg(feature = "rpi4")] | ||||||
|  |     pub fn set_pull_up_down(&self, pull: PullUpDown) { | ||||||
|  |         let bank = self.pin / 16; | ||||||
|  |         let off = self.pin % 16; | ||||||
|  |         self.registers.PullUpDownControl[bank].modify(FieldValue::<u32, ()>::new( | ||||||
|  |             0b11, | ||||||
|  |             off * 2, | ||||||
|  |             pull.into(), | ||||||
|  |         )); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Pin<Uninitialized> { | ||||||
|  |     /// Returns a new GPIO `Pin` structure for pin number `pin`.
 | ||||||
|  |     ///
 | ||||||
|  |     /// # Panics
 | ||||||
|  |     ///
 | ||||||
|  |     /// Panics if `pin` > `53`.
 | ||||||
|  |     unsafe fn new(pin: usize, base_addr: usize) -> Pin<Uninitialized> { | ||||||
|  |         if pin > 53 { | ||||||
|  |             panic!("gpio::Pin::new(): pin {} exceeds maximum of 53", pin); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         Pin { | ||||||
|  |             registers: Registers::new(base_addr), | ||||||
|  |             pin, | ||||||
|  |             _state: PhantomData, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Enables the alternative function `function` for `self`. Consumes self
 | ||||||
|  |     /// and returns a `Pin` structure in the `Alt` state.
 | ||||||
|  |     pub fn into_alt(self, function: Function) -> Pin<Alt> { | ||||||
|  |         let bank = self.pin / 10; | ||||||
|  |         let off = self.pin % 10; | ||||||
|  |         self.registers.FSEL[bank].modify(FieldValue::<u32, ()>::new( | ||||||
|  |             0b111, | ||||||
|  |             off * 3, | ||||||
|  |             function.into(), | ||||||
|  |         )); | ||||||
|  |         self.transition() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Sets this pin to be an _output_ pin. Consumes self and returns a `Pin`
 | ||||||
|  |     /// structure in the `Output` state.
 | ||||||
|  |     pub fn into_output(self) -> Pin<Output> { | ||||||
|  |         self.into_alt(Function::Output).transition() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Sets this pin to be an _input_ pin. Consumes self and returns a `Pin`
 | ||||||
|  |     /// structure in the `Input` state.
 | ||||||
|  |     pub fn into_input(self) -> Pin<Input> { | ||||||
|  |         self.into_alt(Function::Input).transition() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Pin<Output> { | ||||||
|  |     /// Sets (turns on) this pin.
 | ||||||
|  |     pub fn set(&mut self) { | ||||||
|  |         // Guarantees: pin number is between [0; 53] by construction.
 | ||||||
|  |         let bank = self.pin / 32; | ||||||
|  |         let shift = self.pin % 32; | ||||||
|  |         self.registers.SET[bank].set(1 << shift); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Clears (turns off) this pin.
 | ||||||
|  |     pub fn clear(&mut self) { | ||||||
|  |         // Guarantees: pin number is between [0; 53] by construction.
 | ||||||
|  |         let bank = self.pin / 32; | ||||||
|  |         let shift = self.pin % 32; | ||||||
|  |         self.registers.CLR[bank].set(1 << shift); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub type Level = bool; | ||||||
|  | 
 | ||||||
|  | impl Pin<Input> { | ||||||
|  |     /// Reads the pin's value. Returns `true` if the level is high and `false`
 | ||||||
|  |     /// if the level is low.
 | ||||||
|  |     pub fn level(&self) -> Level { | ||||||
|  |         // Guarantees: pin number is between [0; 53] by construction.
 | ||||||
|  |         let bank = self.pin / 32; | ||||||
|  |         let off = self.pin % 32; | ||||||
|  |         self.registers.LEV[bank].matches_all(FieldValue::<u32, ()>::new(1, off, 1)) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(test)] | ||||||
|  | mod tests { | ||||||
|  |     use super::*; | ||||||
|  | 
 | ||||||
|  |     #[test_case] | ||||||
|  |     fn test_pin_transitions() { | ||||||
|  |         let mut reg = [0u32; 40]; | ||||||
|  |         let gpio = unsafe { GPIO::new(&mut reg as *mut _ as usize) }; | ||||||
|  | 
 | ||||||
|  |         let _out = gpio.get_pin(1).into_output(); | ||||||
|  |         assert_eq!(reg[0], 0b001_000); | ||||||
|  |         let _inp = gpio.get_pin(12).into_input(); | ||||||
|  |         assert_eq!(reg[1], 0b000_000_000); | ||||||
|  |         let _alt = gpio.get_pin(35).into_alt(Function::Alt1); | ||||||
|  |         assert_eq!(reg[3], 0b101_000_000_000_000_000); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[test_case] | ||||||
|  |     fn test_pin_outputs() { | ||||||
|  |         let mut reg = [0u32; 40]; | ||||||
|  |         let gpio = unsafe { GPIO::new(&mut reg as *mut _ as usize) }; | ||||||
|  | 
 | ||||||
|  |         let pin = gpio.get_pin(1); | ||||||
|  |         let mut out = pin.into_output(); | ||||||
|  |         out.set(); | ||||||
|  |         assert_eq!(reg[7], 0b10); // SET pin 1 = 1 << 1
 | ||||||
|  |         out.clear(); | ||||||
|  |         assert_eq!(reg[10], 0b10); // CLR pin 1 = 1 << 1
 | ||||||
|  | 
 | ||||||
|  |         let pin = gpio.get_pin(35); | ||||||
|  |         let mut out = pin.into_output(); | ||||||
|  |         out.set(); | ||||||
|  |         assert_eq!(reg[8], 0b1000); // SET pin 35 = 1 << (35 - 32)
 | ||||||
|  |         out.clear(); | ||||||
|  |         assert_eq!(reg[11], 0b1000); // CLR pin 35 = 1 << (35 - 32)
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[test_case] | ||||||
|  |     fn test_pin_inputs() { | ||||||
|  |         let mut reg = [0u32; 40]; | ||||||
|  |         let gpio = unsafe { GPIO::new(&mut reg as *mut _ as usize) }; | ||||||
|  | 
 | ||||||
|  |         let pin = gpio.get_pin(1); | ||||||
|  |         let inp = pin.into_input(); | ||||||
|  |         assert_eq!(inp.level(), false); | ||||||
|  | 
 | ||||||
|  |         reg[13] = 0b10; | ||||||
|  | 
 | ||||||
|  |         assert_eq!(inp.level(), true); | ||||||
|  | 
 | ||||||
|  |         let pin = gpio.get_pin(35); | ||||||
|  |         let inp = pin.into_input(); | ||||||
|  |         assert_eq!(inp.level(), false); | ||||||
|  | 
 | ||||||
|  |         reg[14] = 0b1000; | ||||||
|  | 
 | ||||||
|  |         assert_eq!(inp.level(), true); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,630 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  * | ||||||
|  |  * Based on https://github.com/rust-embedded/rust-raspi3-tutorial/blob/master/04_mailboxes/src/mbox.rs
 | ||||||
|  |  * by Andre Richter of Tock OS. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | //! Broadcom mailbox interface between the VideoCore and the ARM Core.
 | ||||||
|  | //!
 | ||||||
|  | 
 | ||||||
|  | #![allow(dead_code)] | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     super::BcmHost, | ||||||
|  |     crate::{platform::MMIODerefWrapper, println}, | ||||||
|  |     core::{ | ||||||
|  |         result::Result as CoreResult, | ||||||
|  |         sync::atomic::{compiler_fence, Ordering}, | ||||||
|  |     }, | ||||||
|  |     cortex_a::asm::barrier, | ||||||
|  |     snafu::Snafu, | ||||||
|  |     tock_registers::{ | ||||||
|  |         interfaces::{Readable, Writeable}, | ||||||
|  |         register_bitfields, register_structs, | ||||||
|  |         registers::{ReadOnly, WriteOnly}, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /// Public interface to the mailbox.
 | ||||||
|  | /// The address for the buffer needs to be 16-byte aligned
 | ||||||
|  | /// so that the VideoCore can handle it properly.
 | ||||||
|  | /// The reason is that lowest 4 bits of the address will contain the channel number.
 | ||||||
|  | pub struct Mailbox<const N_SLOTS: usize, Storage = LocalMailboxStorage<N_SLOTS>> { | ||||||
|  |     registers: Registers, | ||||||
|  |     pub buffer: Storage, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Mailbox that is ready to be called.
 | ||||||
|  | /// This prevents invalid use of the mailbox until it is fully prepared.
 | ||||||
|  | pub struct PreparedMailbox<const N_SLOTS: usize, Storage = LocalMailboxStorage<N_SLOTS>>( | ||||||
|  |     Mailbox<N_SLOTS, Storage>, | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | const MAILBOX_ALIGNMENT: usize = 16; | ||||||
|  | const MAILBOX_ITEMS_COUNT: usize = 36; | ||||||
|  | 
 | ||||||
|  | /// We've identity mapped the MMIO register region on kernel start.
 | ||||||
|  | const MAILBOX_BASE: usize = BcmHost::get_peripheral_address() + 0xb880; | ||||||
|  | /// Lowest 4-bits are channel ID.
 | ||||||
|  | const CHANNEL_MASK: u32 = 0xf; | ||||||
|  | 
 | ||||||
|  | // Mailbox Peek  Read/Write  Status  Sender  Config
 | ||||||
|  | //    0    0x10  0x00        0x18    0x14    0x1c
 | ||||||
|  | //    1    0x30  0x20        0x38    0x34    0x3c
 | ||||||
|  | //
 | ||||||
|  | // Only mailbox 0's status can trigger interrupts on the ARM, so Mailbox 0 is
 | ||||||
|  | // always for communication from VC to ARM and Mailbox 1 is for ARM to VC.
 | ||||||
|  | //
 | ||||||
|  | // The ARM should never write Mailbox 0 or read Mailbox 1.
 | ||||||
|  | //
 | ||||||
|  | // There are 32 mailboxes on the ARM, which could be used for in-processor or inter-processor comms,
 | ||||||
|  | // TODO: allow using all of them.
 | ||||||
|  | 
 | ||||||
|  | register_bitfields! { | ||||||
|  |     u32, | ||||||
|  | 
 | ||||||
|  |     STATUS [ | ||||||
|  |         /* Bit 31 set in status register if the write mailbox is full */ | ||||||
|  |         FULL  OFFSET(31) NUMBITS(1) [], | ||||||
|  |         /* Bit 30 set in status register if the read mailbox is empty */ | ||||||
|  |         EMPTY OFFSET(30) NUMBITS(1) [] | ||||||
|  |     ] | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | register_structs! { | ||||||
|  |     #[allow(non_snake_case)] | ||||||
|  |     pub RegisterBlock { | ||||||
|  |         (0x00 => READ: ReadOnly<u32>), // This is Mailbox0 read for ARM, can't write
 | ||||||
|  |         (0x04 => __reserved_1), | ||||||
|  |         (0x18 => STATUS: ReadOnly<u32, STATUS::Register>), | ||||||
|  |         (0x1c => __reserved_2), | ||||||
|  |         (0x20 => WRITE: WriteOnly<u32>), // This is Mailbox1 write for ARM, can't read
 | ||||||
|  |         (0x24 => @END), | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Hide RegisterBlock from public api.
 | ||||||
|  | type Registers = MMIODerefWrapper<RegisterBlock>; | ||||||
|  | 
 | ||||||
|  | #[derive(Snafu, Debug)] | ||||||
|  | pub enum MailboxError { | ||||||
|  |     #[snafu(display("ResponseError"))] | ||||||
|  |     Response, | ||||||
|  |     #[snafu(display("UnknownError"))] | ||||||
|  |     Unknown, | ||||||
|  |     #[snafu(display("Timeout"))] | ||||||
|  |     Timeout, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub type Result<T> = CoreResult<T, MailboxError>; | ||||||
|  | 
 | ||||||
|  | /// Typical operations with a mailbox.
 | ||||||
|  | pub trait MailboxOps { | ||||||
|  |     fn write(&self, channel: u32) -> Result<()>; | ||||||
|  |     fn read(&self, channel: u32) -> Result<()>; | ||||||
|  |     fn call(&self, channel: u32) -> Result<()> { | ||||||
|  |         self.write(channel)?; | ||||||
|  |         self.read(channel) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub trait MailboxStorage { | ||||||
|  |     fn new() -> Self; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub trait MailboxStorageRef { | ||||||
|  |     fn as_ref(&self) -> &[u32]; | ||||||
|  |     fn as_mut(&mut self) -> &mut [u32]; | ||||||
|  |     fn as_ptr(&self) -> *const u32; | ||||||
|  |     fn value_at(&self, index: usize) -> u32; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // TODO: allow from 2 to 36 slots (2 because you need at least an End tag)
 | ||||||
|  | #[repr(align(16))] // MAILBOX_ALIGNMENT
 | ||||||
|  | pub struct LocalMailboxStorage<const N_SLOTS: usize> { | ||||||
|  |     pub storage: [u32; N_SLOTS], | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<const N_SLOTS: usize> MailboxStorage for LocalMailboxStorage<N_SLOTS> { | ||||||
|  |     fn new() -> Self { | ||||||
|  |         Self { | ||||||
|  |             storage: [0u32; N_SLOTS], | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<const N_SLOTS: usize> MailboxStorageRef for LocalMailboxStorage<N_SLOTS> { | ||||||
|  |     fn as_ref(&self) -> &[u32] { | ||||||
|  |         &self.storage | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn as_mut(&mut self) -> &mut [u32] { | ||||||
|  |         &mut self.storage | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn as_ptr(&self) -> *const u32 { | ||||||
|  |         self.storage.as_ptr() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // @todo Probably need a ResultMailbox for accessing data after call()?
 | ||||||
|  |     fn value_at(&self, index: usize) -> u32 { | ||||||
|  |         self.storage[index] | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  |  * Source https://elinux.org/RPi_Framebuffer
 | ||||||
|  |  * Source for channels 8 and 9: https://github.com/raspberrypi/firmware/wiki/Mailboxes
 | ||||||
|  |  */ | ||||||
|  | #[allow(non_upper_case_globals)] | ||||||
|  | pub mod channel { | ||||||
|  |     pub const Power: u32 = 0; | ||||||
|  |     pub const FrameBuffer: u32 = 1; | ||||||
|  |     pub const VirtualUart: u32 = 2; | ||||||
|  |     pub const VChiq: u32 = 3; | ||||||
|  |     pub const Leds: u32 = 4; | ||||||
|  |     pub const Buttons: u32 = 5; | ||||||
|  |     pub const TouchScreen: u32 = 6; | ||||||
|  |     // Count = 7,
 | ||||||
|  |     pub const PropertyTagsArmToVc: u32 = 8; | ||||||
|  |     pub const PropertyTagsVcToArm: u32 = 9; | ||||||
|  |     /// Channel number is ignored. Use for implementations of MailboxOps that use hardcoded
 | ||||||
|  |     /// channel number.
 | ||||||
|  |     pub const Ignored: u32 = !0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Single code indicating request
 | ||||||
|  | pub const REQUEST: u32 = 0; | ||||||
|  | 
 | ||||||
|  | // Possible responses
 | ||||||
|  | pub mod response { | ||||||
|  |     pub const SUCCESS: u32 = 0x8000_0000; | ||||||
|  |     pub const ERROR: u32 = 0x8000_0001; // error parsing request buffer (partial response)
 | ||||||
|  |     /** When responding, the VC sets this bit in val_len to indicate a response. */ | ||||||
|  |     /** Each tag with this bit set will contain VC response data. */ | ||||||
|  |     pub const VAL_LEN_FLAG: u32 = 0x8000_0000; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[allow(non_upper_case_globals)] | ||||||
|  | pub mod tag { | ||||||
|  |     pub const GetBoardRev: u32 = 0x0001_0002; | ||||||
|  |     pub const GetMacAddress: u32 = 0x0001_0003; | ||||||
|  |     pub const GetBoardSerial: u32 = 0x0001_0004; | ||||||
|  |     pub const GetArmMemory: u32 = 0x0001_0005; | ||||||
|  |     pub const GetPowerState: u32 = 0x0002_0001; | ||||||
|  |     pub const SetPowerState: u32 = 0x0002_8001; | ||||||
|  |     pub const GetClockRate: u32 = 0x0003_0002; | ||||||
|  |     pub const SetClockRate: u32 = 0x0003_8002; | ||||||
|  |     // GPU
 | ||||||
|  |     pub const AllocateMemory: u32 = 0x0003_000c; //< Allocate contiguous memory buffer
 | ||||||
|  |     pub const LockMemory: u32 = 0x0003_000d; | ||||||
|  |     pub const UnlockMemory: u32 = 0x0003_000e; | ||||||
|  |     pub const ReleaseMemory: u32 = 0x003_000f; | ||||||
|  |     pub const ExecuteCode: u32 = 0x0003_0010; | ||||||
|  |     pub const GetDispmanxResourceMemHandle: u32 = 0x0003_0014; | ||||||
|  |     pub const GetEdidBlock: u32 = 0x0003_0020; | ||||||
|  |     // FB
 | ||||||
|  |     pub const AllocateBuffer: u32 = 0x0004_0001; //< Allocate framebuffer
 | ||||||
|  |     pub const ReleaseBuffer: u32 = 0x0004_8001; | ||||||
|  |     pub const BlankScreen: u32 = 0x0004_0002; | ||||||
|  |     /* Physical means output signal */ | ||||||
|  |     pub const GetPhysicalWH: u32 = 0x0004_0003; | ||||||
|  |     pub const TestPhysicalWH: u32 = 0x0004_4003; | ||||||
|  |     pub const SetPhysicalWH: u32 = 0x0004_8003; | ||||||
|  |     /* Virtual means display buffer */ | ||||||
|  |     pub const GetVirtualWH: u32 = 0x0004_0004; | ||||||
|  |     pub const TestVirtualWH: u32 = 0x0004_4004; | ||||||
|  |     pub const SetVirtualWH: u32 = 0x0004_8004; | ||||||
|  |     pub const GetDepth: u32 = 0x0004_0005; | ||||||
|  |     pub const TestDepth: u32 = 0x0004_4005; | ||||||
|  |     pub const SetDepth: u32 = 0x0004_8005; | ||||||
|  |     pub const GetPixelOrder: u32 = 0x0004_0006; | ||||||
|  |     pub const TestPixelOrder: u32 = 0x0004_4006; | ||||||
|  |     pub const SetPixelOrder: u32 = 0x0004_8006; | ||||||
|  |     pub const GetAlphaMode: u32 = 0x0004_0007; | ||||||
|  |     pub const TestAlphaMode: u32 = 0x0004_4007; | ||||||
|  |     pub const SetAlphaMode: u32 = 0x0004_8007; | ||||||
|  |     pub const GetPitch: u32 = 0x0004_0008; | ||||||
|  |     /* Offset of display window within buffer */ | ||||||
|  |     pub const GetVirtualOffset: u32 = 0x0004_0009; | ||||||
|  |     pub const TestVirtualOffset: u32 = 0x0004_4009; | ||||||
|  |     pub const SetVirtualOffset: u32 = 0x0004_8009; | ||||||
|  |     pub const GetOverscan: u32 = 0x0004_000a; | ||||||
|  |     pub const TestOverscan: u32 = 0x0004_400a; | ||||||
|  |     pub const SetOverscan: u32 = 0x0004_800a; | ||||||
|  |     pub const GetPalette: u32 = 0x0004_000b; | ||||||
|  |     pub const TestPalette: u32 = 0x0004_400b; | ||||||
|  |     pub const SetPalette: u32 = 0x0004_800b; | ||||||
|  |     pub const SetCursorInfo: u32 = 0x0000_8010; | ||||||
|  |     pub const SetCursorState: u32 = 0x0000_8011; | ||||||
|  |     pub const GetGpioState: u32 = 0x0003_0041; | ||||||
|  |     pub const SetGpioState: u32 = 0x0003_8041; | ||||||
|  |     pub const End: u32 = 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub mod power { | ||||||
|  |     pub const SDHCI: u32 = 0; | ||||||
|  |     pub const UART0: u32 = 1; | ||||||
|  |     pub const UART1: u32 = 2; | ||||||
|  |     pub const USB_HCD: u32 = 3; | ||||||
|  |     pub const I2C0: u32 = 4; | ||||||
|  |     pub const I2C1: u32 = 5; | ||||||
|  |     pub const I2C2: u32 = 6; | ||||||
|  |     pub const SPI: u32 = 7; | ||||||
|  |     pub const CCP2TX: u32 = 8; | ||||||
|  | 
 | ||||||
|  |     pub mod response { | ||||||
|  |         pub const ON: u32 = 1; | ||||||
|  |         pub const NO_DEV: u32 = 2; /* Device doesn't exist */ | ||||||
|  |     } | ||||||
|  |     pub mod request { | ||||||
|  |         pub const ON: u32 = 1; | ||||||
|  |         pub const WAIT: u32 = 2; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub mod clock { | ||||||
|  |     pub const EMMC: u32 = 1; | ||||||
|  |     pub const UART: u32 = 2; | ||||||
|  |     pub const ARM: u32 = 3; | ||||||
|  |     pub const CORE: u32 = 4; | ||||||
|  |     pub const V3D: u32 = 5; | ||||||
|  |     pub const H264: u32 = 6; | ||||||
|  |     pub const ISP: u32 = 7; | ||||||
|  |     pub const SDRAM: u32 = 8; | ||||||
|  |     pub const PIXEL: u32 = 9; | ||||||
|  |     pub const PWM: u32 = 10; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub mod alpha_mode { | ||||||
|  |     pub const OPAQUE_0: u32 = 0; // 255 is transparent
 | ||||||
|  |     pub const TRANSPARENT_0: u32 = 1; // 255 is opaque
 | ||||||
|  |     pub const IGNORED: u32 = 2; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<const N_SLOTS: usize> core::fmt::Debug for Mailbox<N_SLOTS> { | ||||||
|  |     fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result { | ||||||
|  |         let count = self.buffer.as_ref()[0] / 4; | ||||||
|  |         assert_eq!(self.buffer.as_ref()[0], count * 4); | ||||||
|  |         assert!(count <= 36); | ||||||
|  |         for i in 0usize..count as usize { | ||||||
|  |             writeln!(f, "[{:02}] {:08x}", i, self.buffer.value_at(i))?; | ||||||
|  |         } | ||||||
|  |         Ok(()) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<const N_SLOTS: usize> core::fmt::Debug for PreparedMailbox<N_SLOTS> { | ||||||
|  |     fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result { | ||||||
|  |         self.0.fmt(f) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<const N_SLOTS: usize> Default for Mailbox<N_SLOTS> { | ||||||
|  |     fn default() -> Self { | ||||||
|  |         unsafe { Self::new(MAILBOX_BASE) }.expect("Couldn't allocate a default mailbox") | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<const N_SLOTS: usize, Storage: MailboxStorage + MailboxStorageRef> Mailbox<N_SLOTS, Storage> { | ||||||
|  |     /// Create a new mailbox locally in an aligned stack space.
 | ||||||
|  |     /// # Safety
 | ||||||
|  |     /// Caller is responsible for picking the correct MMIO register base address.
 | ||||||
|  |     pub unsafe fn new(base_addr: usize) -> Result<Mailbox<N_SLOTS, Storage>> { | ||||||
|  |         Ok(Mailbox { | ||||||
|  |             registers: Registers::new(base_addr), | ||||||
|  |             buffer: Storage::new(), | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Specific mailbox functions
 | ||||||
|  | 
 | ||||||
|  |     /// Start mailbox request.
 | ||||||
|  |     ///
 | ||||||
|  |     /// @returns index of the next available slot.
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn request(&mut self) -> usize { | ||||||
|  |         self.buffer.as_mut()[1] = REQUEST; | ||||||
|  |         2 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Mark mailbox payload as completed.
 | ||||||
|  |     /// Consumes the Mailbox and returns a Preparedmailbox that can be called.
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn end(mut self, index: usize) -> PreparedMailbox<N_SLOTS, Storage> { | ||||||
|  |         // @todo return Result
 | ||||||
|  |         self.buffer.as_mut()[index] = tag::End; | ||||||
|  |         self.buffer.as_mut()[0] = (index as u32 + 1) * 4; | ||||||
|  |         PreparedMailbox(self) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ///
 | ||||||
|  |     /// @returns index of the next available slot.
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn set_physical_wh(&mut self, index: usize, width: u32, height: u32) -> usize { | ||||||
|  |         let buf = self.buffer.as_mut(); | ||||||
|  |         buf[index] = tag::SetPhysicalWH; | ||||||
|  |         buf[index + 1] = 8; // Buffer size   // val buf size
 | ||||||
|  |         buf[index + 2] = 8; // Request size  // val size
 | ||||||
|  |         buf[index + 3] = width; // Space for horizontal resolution
 | ||||||
|  |         buf[index + 4] = height; // Space for vertical resolution
 | ||||||
|  |         index + 5 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ///
 | ||||||
|  |     /// @returns index of the next available slot.
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn set_virtual_wh(&mut self, index: usize, width: u32, height: u32) -> usize { | ||||||
|  |         let buf = self.buffer.as_mut(); | ||||||
|  |         buf[index] = tag::SetVirtualWH; | ||||||
|  |         buf[index + 1] = 8; // Buffer size   // val buf size
 | ||||||
|  |         buf[index + 2] = 8; // Request size  // val size
 | ||||||
|  |         buf[index + 3] = width; // Space for horizontal resolution
 | ||||||
|  |         buf[index + 4] = height; // Space for vertical resolution
 | ||||||
|  |         index + 5 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ///
 | ||||||
|  |     /// @returns index of the next available slot.
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn set_depth(&mut self, index: usize, depth: u32) -> usize { | ||||||
|  |         let buf = self.buffer.as_mut(); | ||||||
|  |         buf[index] = tag::SetDepth; | ||||||
|  |         buf[index + 1] = 4; // Buffer size   // val buf size
 | ||||||
|  |         buf[index + 2] = 4; // Request size  // val size
 | ||||||
|  |         buf[index + 3] = depth; // bpp
 | ||||||
|  |         index + 4 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ///
 | ||||||
|  |     /// @returns index of the next available slot.
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn allocate_buffer_aligned(&mut self, index: usize, alignment: u32) -> usize { | ||||||
|  |         let buf = self.buffer.as_mut(); | ||||||
|  |         buf[index] = tag::AllocateBuffer; | ||||||
|  |         buf[index + 1] = 8; // Buffer size   // val buf size
 | ||||||
|  |         buf[index + 2] = 4; // Request size  // val size
 | ||||||
|  |         buf[index + 3] = alignment; // Alignment = 16 -- fb_ptr will be here
 | ||||||
|  |         buf[index + 4] = 0; // Space for response -- fb_size will be here
 | ||||||
|  |         index + 5 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ///
 | ||||||
|  |     /// @returns index of the next available slot.
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn set_led_on(&mut self, index: usize, enable: bool) -> usize { | ||||||
|  |         let buf = self.buffer.as_mut(); | ||||||
|  |         buf[index] = tag::SetGpioState; | ||||||
|  |         buf[index + 1] = 8; // Buffer size   // val buf size
 | ||||||
|  |         buf[index + 2] = 0; // Response size  // val size
 | ||||||
|  |         buf[index + 3] = 130; // Pin Number
 | ||||||
|  |         buf[index + 4] = if enable { 1 } else { 0 }; | ||||||
|  |         index + 5 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn set_clock_rate(&mut self, index: usize, channel: u32, rate: u32) -> usize { | ||||||
|  |         let buf = self.buffer.as_mut(); | ||||||
|  |         buf[index] = tag::SetClockRate; | ||||||
|  |         buf[index + 1] = 12; // Buffer size   // val buf size
 | ||||||
|  |         buf[index + 2] = 8; // Response size  // val size
 | ||||||
|  |         buf[index + 3] = channel; // mailbox::clock::*
 | ||||||
|  |         buf[index + 4] = rate; | ||||||
|  |         buf[index + 5] = 0; // skip turbo setting
 | ||||||
|  |         index + 6 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// NB: Do not intermix Get/Set and Test tags in one request!
 | ||||||
|  |     /// See <https://github.com/raspberrypi/firmware/wiki/Mailbox-property-interface>
 | ||||||
|  |     /// * It is not valid to mix Test tags with Get/Set tags in the same operation
 | ||||||
|  |     ///   and no tags will be returned.
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn set_pixel_order(&mut self, index: usize, order: u32) -> usize { | ||||||
|  |         let buf = self.buffer.as_mut(); | ||||||
|  |         buf[index] = tag::SetPixelOrder; | ||||||
|  |         buf[index + 1] = 4; // Buffer size   // val buf size
 | ||||||
|  |         buf[index + 2] = 4; // Response size  // val size
 | ||||||
|  |         buf[index + 3] = order; | ||||||
|  |         index + 4 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// NB: Do not intermix Get/Set and Test tags in one request!
 | ||||||
|  |     /// See <https://github.com/raspberrypi/firmware/wiki/Mailbox-property-interface>
 | ||||||
|  |     /// * It is not valid to mix Test tags with Get/Set tags in the same operation
 | ||||||
|  |     ///   and no tags will be returned.
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn test_pixel_order(&mut self, index: usize, order: u32) -> usize { | ||||||
|  |         let buf = self.buffer.as_mut(); | ||||||
|  |         buf[index] = tag::TestPixelOrder; | ||||||
|  |         buf[index + 1] = 4; // Buffer size   // val buf size
 | ||||||
|  |         buf[index + 2] = 4; // Response size  // val size
 | ||||||
|  |         buf[index + 3] = order; | ||||||
|  |         index + 4 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn set_alpha_mode(&mut self, index: usize, mode: u32) -> usize { | ||||||
|  |         let buf = self.buffer.as_mut(); | ||||||
|  |         buf[index] = tag::SetAlphaMode; | ||||||
|  |         buf[index + 1] = 4; // Buffer size   // val buf size
 | ||||||
|  |         buf[index + 2] = 4; // Response size  // val size
 | ||||||
|  |         buf[index + 3] = mode; | ||||||
|  |         index + 4 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn get_pitch(&mut self, index: usize) -> usize { | ||||||
|  |         let buf = self.buffer.as_mut(); | ||||||
|  |         buf[index] = tag::GetPitch; | ||||||
|  |         buf[index + 1] = 4; // Buffer size   // val buf size
 | ||||||
|  |         buf[index + 2] = 4; // Response size  // val size
 | ||||||
|  |         buf[index + 3] = 0; // Result placeholder
 | ||||||
|  |         index + 4 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[inline] | ||||||
|  |     pub fn set_device_power(&mut self, index: usize, device_id: u32, power_flags: u32) -> usize { | ||||||
|  |         let buf = self.buffer.as_mut(); | ||||||
|  |         buf[index] = tag::SetPowerState; | ||||||
|  |         buf[index + 1] = 8; // Buffer size   // val buf size
 | ||||||
|  |         buf[index + 2] = 8; // Response size  // val size
 | ||||||
|  |         buf[index + 3] = device_id; | ||||||
|  |         buf[index + 4] = power_flags; // bit 0: off, bit 1: no wait
 | ||||||
|  |         index + 5 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Actual work functions
 | ||||||
|  | 
 | ||||||
|  |     /// <https://github.com/raspberrypi/firmware/wiki/Accessing-mailboxes> says:
 | ||||||
|  |     /// **With the exception of the property tags mailbox channel,**
 | ||||||
|  |     /// when passing memory addresses as the data part of a mailbox message,
 | ||||||
|  |     /// the addresses should be **bus addresses as seen from the VC.**
 | ||||||
|  |     pub fn do_write(&self, channel: u32) -> Result<()> { | ||||||
|  |         let buf_ptr = self.buffer.as_ptr() as *const u32 as u32; | ||||||
|  |         let buf_ptr = if channel != channel::PropertyTagsArmToVc { | ||||||
|  |             BcmHost::phys2bus(buf_ptr as usize) as u32 | ||||||
|  |         } else { | ||||||
|  |             buf_ptr | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         let mut count: u32 = 0; | ||||||
|  | 
 | ||||||
|  |         println!("Mailbox::write {:#08x}/{:#x}", buf_ptr, channel); | ||||||
|  | 
 | ||||||
|  |         // Insert a compiler fence that ensures that all stores to the
 | ||||||
|  |         // mailbox buffer are finished before the GPU is signaled (which is
 | ||||||
|  |         // done by a store operation as well).
 | ||||||
|  |         compiler_fence(Ordering::Release); | ||||||
|  | 
 | ||||||
|  |         while self.registers.STATUS.is_set(STATUS::FULL) { | ||||||
|  |             count += 1; | ||||||
|  |             if count > (1 << 25) { | ||||||
|  |                 return Err(MailboxError::Timeout); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         unsafe { | ||||||
|  |             barrier::dmb(barrier::SY); | ||||||
|  |         } | ||||||
|  |         self.registers | ||||||
|  |             .WRITE | ||||||
|  |             .set((buf_ptr & !CHANNEL_MASK) | (channel & CHANNEL_MASK)); | ||||||
|  |         Ok(()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Perform the mailbox read.
 | ||||||
|  |     ///
 | ||||||
|  |     /// # Safety
 | ||||||
|  |     ///
 | ||||||
|  |     /// Buffer will be mutated by the hardware before read operation is completed.
 | ||||||
|  |     pub unsafe fn do_read(&self, channel: u32, expected: u32) -> Result<()> { | ||||||
|  |         loop { | ||||||
|  |             let mut count: u32 = 0; | ||||||
|  |             while self.registers.STATUS.is_set(STATUS::EMPTY) { | ||||||
|  |                 count += 1; | ||||||
|  |                 if count > (1 << 25) { | ||||||
|  |                     println!("Timed out waiting for mailbox response"); | ||||||
|  |                     return Err(MailboxError::Timeout); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             /* Read the data
 | ||||||
|  |              * Data memory barriers as we've switched peripheral | ||||||
|  |              */ | ||||||
|  |             barrier::dmb(barrier::SY); | ||||||
|  |             let data: u32 = self.registers.READ.get(); | ||||||
|  |             barrier::dmb(barrier::SY); | ||||||
|  | 
 | ||||||
|  |             println!( | ||||||
|  |                 "Received mailbox response {:#08x}, expecting {:#08x}", | ||||||
|  |                 data, expected | ||||||
|  |             ); | ||||||
|  | 
 | ||||||
|  |             // is it a response to our message?
 | ||||||
|  |             if ((data & CHANNEL_MASK) == channel) && ((data & !CHANNEL_MASK) == expected) { | ||||||
|  |                 // is it a valid successful response?
 | ||||||
|  |                 return match self.buffer.value_at(1) { | ||||||
|  |                     response::SUCCESS => { | ||||||
|  |                         println!("\n######\nMailbox::returning SUCCESS"); | ||||||
|  |                         Ok(()) | ||||||
|  |                     } | ||||||
|  |                     response::ERROR => { | ||||||
|  |                         println!("\n######\nMailbox::returning ResponseError"); | ||||||
|  |                         Err(MailboxError::Response) | ||||||
|  |                     } | ||||||
|  |                     _ => { | ||||||
|  |                         println!("\n######\nMailbox::returning UnknownError"); | ||||||
|  |                         println!("{:x}\n######", self.buffer.value_at(1)); | ||||||
|  |                         Err(MailboxError::Unknown) | ||||||
|  |                     } | ||||||
|  |                 }; | ||||||
|  |             } else { | ||||||
|  |                 // ignore invalid responses and loop again.
 | ||||||
|  |                 // will return Timeout above if no matching response is received.
 | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<const N_SLOTS: usize, Storage: MailboxStorage + MailboxStorageRef> MailboxOps | ||||||
|  |     for PreparedMailbox<N_SLOTS, Storage> | ||||||
|  | { | ||||||
|  |     fn write(&self, channel: u32) -> Result<()> { | ||||||
|  |         self.0.do_write(channel) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // @todo read() should probably consume PreparedMailbox completely - because request is overwritten with response
 | ||||||
|  |     fn read(&self, channel: u32) -> Result<()> { | ||||||
|  |         unsafe { self.0.do_read(channel, self.0.buffer.as_ptr() as u32) } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<const N_SLOTS: usize, Storage: MailboxStorage + MailboxStorageRef> MailboxStorageRef | ||||||
|  |     for PreparedMailbox<N_SLOTS, Storage> | ||||||
|  | { | ||||||
|  |     fn as_ref(&self) -> &[u32] { | ||||||
|  |         self.0.buffer.as_ref() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn as_mut(&mut self) -> &mut [u32] { | ||||||
|  |         self.0.buffer.as_mut() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn as_ptr(&self) -> *const u32 { | ||||||
|  |         self.0.buffer.as_ptr() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // @todo Probably need a ResultMailbox for accessing data after call()?
 | ||||||
|  |     fn value_at(&self, index: usize) -> u32 { | ||||||
|  |         self.0.buffer.value_at(index) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(test)] | ||||||
|  | mod tests { | ||||||
|  |     use super::*; | ||||||
|  | 
 | ||||||
|  |     // Validate the buffer is filled correctly
 | ||||||
|  |     // Validate the buffer is properly terminated when call()ed -- this invariant must be maintained
 | ||||||
|  |     // by the end() fn.
 | ||||||
|  |     #[test_case] | ||||||
|  |     fn test_prepare_mailbox() { | ||||||
|  |         let mut mailbox = Mailbox::default(); | ||||||
|  |         let index = mailbox.request(); | ||||||
|  |         let index = mailbox.set_led_on(index, true); | ||||||
|  |         let mailbox = mailbox.end(index); | ||||||
|  |         // Instead of calling just check the filled buffer format:
 | ||||||
|  |         assert_eq!( | ||||||
|  |             unsafe { mailbox.0.buffer.as_ref()[0] } as usize, | ||||||
|  |             (index + 1) * 4 | ||||||
|  |         ); | ||||||
|  |         assert_eq!(unsafe { mailbox.0.buffer.as_ref()[1] }, REQUEST); | ||||||
|  |         assert_eq!(unsafe { mailbox.0.buffer.as_ref()[2] }, tag::SetGpioState); | ||||||
|  |         assert_eq!(unsafe { mailbox.0.buffer.as_ref()[3] }, 8); | ||||||
|  |         assert_eq!(unsafe { mailbox.0.buffer.as_ref()[4] }, 0); | ||||||
|  |         assert_eq!(unsafe { mailbox.0.buffer.as_ref()[5] }, 130); | ||||||
|  |         assert_eq!(unsafe { mailbox.0.buffer.as_ref()[6] }, 1); | ||||||
|  |         assert_eq!(unsafe { mailbox.0.buffer.as_ref()[7] }, tag::End); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,328 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: MIT OR BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) 2018-2019 Andre Richter <andre.o.richter@gmail.com> | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  * Original code distributed under MIT, additional changes are under BlueOak-1.0.0 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #[cfg(not(feature = "noserial"))] | ||||||
|  | use tock_registers::interfaces::{Readable, Writeable}; | ||||||
|  | use { | ||||||
|  |     super::{gpio, BcmHost}, | ||||||
|  |     crate::{ | ||||||
|  |         devices::{ConsoleOps, SerialOps}, | ||||||
|  |         platform::MMIODerefWrapper, | ||||||
|  |     }, | ||||||
|  |     cfg_if::cfg_if, | ||||||
|  |     core::{convert::From, fmt}, | ||||||
|  |     tock_registers::{ | ||||||
|  |         interfaces::ReadWriteable, | ||||||
|  |         register_bitfields, register_structs, | ||||||
|  |         registers::{ReadOnly, ReadWrite, WriteOnly}, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // Auxiliary mini UART registers
 | ||||||
|  | //
 | ||||||
|  | // Descriptions taken from
 | ||||||
|  | // https://github.com/raspberrypi/documentation/files/1888662/BCM2837-ARM-Peripherals.-.Revised.-.V2-1.pdf
 | ||||||
|  | register_bitfields! { | ||||||
|  |     u32, | ||||||
|  | 
 | ||||||
|  |     /// Auxiliary enables
 | ||||||
|  |     AUX_ENABLES [ | ||||||
|  |         /// If set the mini UART is enabled. The UART will immediately
 | ||||||
|  |         /// start receiving data, especially if the UART1_RX line is
 | ||||||
|  |         /// low.
 | ||||||
|  |         /// If clear the mini UART is disabled. That also disables any
 | ||||||
|  |         /// mini UART register access
 | ||||||
|  |         MINI_UART_ENABLE OFFSET(0) NUMBITS(1) [] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// Mini Uart Interrupt Identify
 | ||||||
|  |     AUX_MU_IIR [ | ||||||
|  |         /// Writing with bit 1 set will clear the receive FIFO
 | ||||||
|  |         /// Writing with bit 2 set will clear the transmit FIFO
 | ||||||
|  |         FIFO_CLEAR OFFSET(1) NUMBITS(2) [ | ||||||
|  |             Rx = 0b01, | ||||||
|  |             Tx = 0b10, | ||||||
|  |             All = 0b11 | ||||||
|  |         ] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// Mini Uart Line Control
 | ||||||
|  |     AUX_MU_LCR [ | ||||||
|  |         /// Mode the UART works in
 | ||||||
|  |         DATA_SIZE OFFSET(0) NUMBITS(2) [ | ||||||
|  |             SevenBit = 0b00, | ||||||
|  |             EightBit = 0b11 | ||||||
|  |         ] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// Mini Uart Line Status
 | ||||||
|  |     AUX_MU_LSR [ | ||||||
|  |         /// This bit is set if the transmit FIFO is empty and the transmitter is
 | ||||||
|  |         /// idle. (Finished shifting out the last bit).
 | ||||||
|  |         TX_IDLE    OFFSET(6) NUMBITS(1) [], | ||||||
|  | 
 | ||||||
|  |         /// This bit is set if the transmit FIFO can accept at least
 | ||||||
|  |         /// one byte.
 | ||||||
|  |         TX_EMPTY   OFFSET(5) NUMBITS(1) [], | ||||||
|  | 
 | ||||||
|  |         /// This bit is set if the receive FIFO holds at least 1
 | ||||||
|  |         /// symbol.
 | ||||||
|  |         DATA_READY OFFSET(0) NUMBITS(1) [] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// Mini Uart Extra Control
 | ||||||
|  |     AUX_MU_CNTL [ | ||||||
|  |         /// If this bit is set the mini UART transmitter is enabled.
 | ||||||
|  |         /// If this bit is clear the mini UART transmitter is disabled.
 | ||||||
|  |         TX_EN OFFSET(1) NUMBITS(1) [ | ||||||
|  |             Disabled = 0, | ||||||
|  |             Enabled = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// If this bit is set the mini UART receiver is enabled.
 | ||||||
|  |         /// If this bit is clear the mini UART receiver is disabled.
 | ||||||
|  |         RX_EN OFFSET(0) NUMBITS(1) [ | ||||||
|  |             Disabled = 0, | ||||||
|  |             Enabled = 1 | ||||||
|  |         ] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// Mini Uart Status
 | ||||||
|  |     AUX_MU_STAT [ | ||||||
|  |         TX_DONE OFFSET(9) NUMBITS(1) [ | ||||||
|  |             No = 0, | ||||||
|  |             Yes = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// This bit is set if the transmit FIFO can accept at least
 | ||||||
|  |         /// one byte.
 | ||||||
|  |         SPACE_AVAILABLE OFFSET(1) NUMBITS(1) [ | ||||||
|  |             No = 0, | ||||||
|  |             Yes = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// This bit is set if the receive FIFO holds at least 1
 | ||||||
|  |         /// symbol.
 | ||||||
|  |         SYMBOL_AVAILABLE OFFSET(0) NUMBITS(1) [ | ||||||
|  |             No = 0, | ||||||
|  |             Yes = 1 | ||||||
|  |         ] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// Mini Uart Baud rate
 | ||||||
|  |     AUX_MU_BAUD [ | ||||||
|  |         /// Mini UART baud rate counter
 | ||||||
|  |         RATE OFFSET(0) NUMBITS(16) [] | ||||||
|  |     ] | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | register_structs! { | ||||||
|  |     #[allow(non_snake_case)] | ||||||
|  |     RegisterBlock { | ||||||
|  |         // 0x00 - AUX_IRQ?
 | ||||||
|  |         (0x00 => __reserved_1), | ||||||
|  |         (0x04 => AUX_ENABLES: ReadWrite<u32, AUX_ENABLES::Register>), | ||||||
|  |         (0x08 => __reserved_2), | ||||||
|  |         (0x40 => AUX_MU_IO: ReadWrite<u32>),//Mini Uart I/O Data
 | ||||||
|  |         (0x44 => AUX_MU_IER: WriteOnly<u32>),//Mini Uart Interrupt Enable
 | ||||||
|  |         (0x48 => AUX_MU_IIR: WriteOnly<u32, AUX_MU_IIR::Register>), | ||||||
|  |         (0x4c => AUX_MU_LCR: WriteOnly<u32, AUX_MU_LCR::Register>), | ||||||
|  |         (0x50 => AUX_MU_MCR: WriteOnly<u32>), | ||||||
|  |         (0x54 => AUX_MU_LSR: ReadOnly<u32, AUX_MU_LSR::Register>), | ||||||
|  |         // 0x58 - AUX_MU_MSR
 | ||||||
|  |         // 0x5c - AUX_MU_SCRATCH
 | ||||||
|  |         (0x58 => __reserved_3), | ||||||
|  |         (0x60 => AUX_MU_CNTL: WriteOnly<u32, AUX_MU_CNTL::Register>), | ||||||
|  |         (0x64 => AUX_MU_STAT: ReadOnly<u32, AUX_MU_STAT::Register>), | ||||||
|  |         (0x68 => AUX_MU_BAUD: WriteOnly<u32, AUX_MU_BAUD::Register>), | ||||||
|  |         (0x6c => @END), | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type Registers = MMIODerefWrapper<RegisterBlock>; | ||||||
|  | 
 | ||||||
|  | pub struct MiniUart { | ||||||
|  |     registers: Registers, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub struct PreparedMiniUart(MiniUart); | ||||||
|  | 
 | ||||||
|  | /// Divisor values for common baud rates
 | ||||||
|  | pub enum Rate { | ||||||
|  |     Baud115200 = 270, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl From<Rate> for u32 { | ||||||
|  |     fn from(r: Rate) -> Self { | ||||||
|  |         r as u32 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // [temporary] Used in mmu.rs to set up local paging
 | ||||||
|  | pub const UART1_START: usize = 0x21_5000; | ||||||
|  | 
 | ||||||
|  | impl Default for MiniUart { | ||||||
|  |     fn default() -> Self { | ||||||
|  |         const UART1_BASE: usize = BcmHost::get_peripheral_address() + UART1_START; | ||||||
|  |         unsafe { MiniUart::new(UART1_BASE) } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl MiniUart { | ||||||
|  |     /// # Safety
 | ||||||
|  |     ///
 | ||||||
|  |     /// Unsafe, duh!
 | ||||||
|  |     pub const unsafe fn new(base_addr: usize) -> MiniUart { | ||||||
|  |         MiniUart { | ||||||
|  |             registers: Registers::new(base_addr), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl MiniUart { | ||||||
|  |     cfg_if! { | ||||||
|  |         if #[cfg(not(feature = "noserial"))] { | ||||||
|  |             /// Set baud rate and characteristics (115200 8N1) and map to GPIO
 | ||||||
|  |             pub fn prepare(self, gpio: &gpio::GPIO) -> PreparedMiniUart { | ||||||
|  |                 // GPIO pins should be set up first before enabling the UART
 | ||||||
|  | 
 | ||||||
|  |                 // Pin 14
 | ||||||
|  |                 const MINI_UART_TXD: gpio::Function = gpio::Function::Alt5; | ||||||
|  |                 // Pin 15
 | ||||||
|  |                 const MINI_UART_RXD: gpio::Function = gpio::Function::Alt5; | ||||||
|  | 
 | ||||||
|  |                 // map UART1 to GPIO pins
 | ||||||
|  |                 gpio.get_pin(14).into_alt(MINI_UART_TXD).set_pull_up_down(gpio::PullUpDown::Up); | ||||||
|  |                 gpio.get_pin(15).into_alt(MINI_UART_RXD).set_pull_up_down(gpio::PullUpDown::Up); | ||||||
|  | 
 | ||||||
|  |                 // initialize UART
 | ||||||
|  |                 self.registers.AUX_ENABLES.modify(AUX_ENABLES::MINI_UART_ENABLE::SET); | ||||||
|  |                 self.registers.AUX_MU_IER.set(0); | ||||||
|  |                 self.registers.AUX_MU_CNTL.set(0); | ||||||
|  |                 self.registers.AUX_MU_LCR.write(AUX_MU_LCR::DATA_SIZE::EightBit); | ||||||
|  |                 self.registers.AUX_MU_MCR.set(0); | ||||||
|  |                 self.registers.AUX_MU_IER.set(0); | ||||||
|  |                 self.registers.AUX_MU_BAUD | ||||||
|  |                     .write(AUX_MU_BAUD::RATE.val(Rate::Baud115200.into())); | ||||||
|  | 
 | ||||||
|  |                 // Clear FIFOs before using the device
 | ||||||
|  |                 self.registers.AUX_MU_IIR.write(AUX_MU_IIR::FIFO_CLEAR::All); | ||||||
|  | 
 | ||||||
|  |                 self.registers.AUX_MU_CNTL | ||||||
|  |                     .write(AUX_MU_CNTL::RX_EN::Enabled + AUX_MU_CNTL::TX_EN::Enabled); | ||||||
|  | 
 | ||||||
|  |                 PreparedMiniUart(self) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             pub fn prepare(self, _gpio: &gpio::GPIO) -> PreparedMiniUart { | ||||||
|  |                 PreparedMiniUart(self) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Drop for PreparedMiniUart { | ||||||
|  |     fn drop(&mut self) { | ||||||
|  |         self.0 | ||||||
|  |             .registers | ||||||
|  |             .AUX_ENABLES | ||||||
|  |             .modify(AUX_ENABLES::MINI_UART_ENABLE::CLEAR); | ||||||
|  |         // @todo disable gpio.PUD ?
 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl SerialOps for PreparedMiniUart { | ||||||
|  |     cfg_if! { | ||||||
|  |         if #[cfg(not(feature = "noserial"))] { | ||||||
|  |             /// Receive a byte without console translation
 | ||||||
|  |             fn read_byte(&self) -> u8 { | ||||||
|  |                 // wait until something is in the buffer
 | ||||||
|  |                 crate::arch::loop_until(|| self.0.registers.AUX_MU_STAT.is_set(AUX_MU_STAT::SYMBOL_AVAILABLE)); | ||||||
|  | 
 | ||||||
|  |                 // read it and return
 | ||||||
|  |                 self.0.registers.AUX_MU_IO.get() as u8 | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             fn write_byte(&self, b: u8) { | ||||||
|  |                 // wait until we can send
 | ||||||
|  |                 crate::arch::loop_until(|| self.0.registers.AUX_MU_STAT.is_set(AUX_MU_STAT::SPACE_AVAILABLE)); | ||||||
|  | 
 | ||||||
|  |                 // write the character to the buffer
 | ||||||
|  |                 self.0.registers.AUX_MU_IO.set(b as u32); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             /// Wait until the TX FIFO is empty, aka all characters have been put on the
 | ||||||
|  |             /// line.
 | ||||||
|  |             fn flush(&self) { | ||||||
|  |                 crate::arch::loop_until(|| self.0.registers.AUX_MU_STAT.is_set(AUX_MU_STAT::TX_DONE)); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             /// Consume input until RX FIFO is empty, aka all pending characters have been
 | ||||||
|  |             /// consumed.
 | ||||||
|  |             fn clear_rx(&self) { | ||||||
|  |                 crate::arch::loop_while(|| { | ||||||
|  |                     let pending = self.0.registers.AUX_MU_STAT.is_set(AUX_MU_STAT::SYMBOL_AVAILABLE); | ||||||
|  |                     if pending { self.read_byte(); } | ||||||
|  |                     pending | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             fn read_byte(&self) -> u8 { 0 } | ||||||
|  |             fn write_byte(&self, _byte: u8) {} | ||||||
|  |             fn flush(&self) {} | ||||||
|  |             fn clear_rx(&self) {} | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl ConsoleOps for PreparedMiniUart { | ||||||
|  |     cfg_if! { | ||||||
|  |         if #[cfg(not(feature = "noserial"))] { | ||||||
|  |             /// Send a character
 | ||||||
|  |             fn write_char(&self, c: char) { | ||||||
|  |                 self.write_byte(c as u8); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             /// Display a string
 | ||||||
|  |             fn write_string(&self, string: &str) { | ||||||
|  |                 for c in string.chars() { | ||||||
|  |                     // convert newline to carriage return + newline
 | ||||||
|  |                     if c == '\n' { | ||||||
|  |                         self.write_char('\r') | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     self.write_char(c); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             /// Receive a character
 | ||||||
|  |             fn read_char(&self) -> char { | ||||||
|  |                 let mut ret = self.read_byte() as char; | ||||||
|  | 
 | ||||||
|  |                 // convert carriage return to newline -- this doesn't work well for reading binaries...
 | ||||||
|  |                 if ret == '\r' { | ||||||
|  |                     ret = '\n' | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 ret | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             fn write_char(&self, _c: char) {} | ||||||
|  |             fn write_string(&self, _string: &str) {} | ||||||
|  |             fn read_char(&self) -> char { | ||||||
|  |                 '\n' | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl fmt::Write for PreparedMiniUart { | ||||||
|  |     fn write_str(&mut self, s: &str) -> fmt::Result { | ||||||
|  |         self.write_string(s); | ||||||
|  |         Ok(()) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,109 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #![allow(dead_code)] | ||||||
|  | 
 | ||||||
|  | pub mod display; | ||||||
|  | pub mod fb; | ||||||
|  | pub mod gpio; | ||||||
|  | pub mod mailbox; | ||||||
|  | pub mod mini_uart; | ||||||
|  | pub mod pl011_uart; | ||||||
|  | pub mod power; | ||||||
|  | pub mod vc; | ||||||
|  | 
 | ||||||
|  | /// See BCM2835-ARM-Peripherals.pdf
 | ||||||
|  | /// See <https://www.raspberrypi.org/forums/viewtopic.php?t=186090> for more details.
 | ||||||
|  | 
 | ||||||
|  | pub struct BcmHost; | ||||||
|  | 
 | ||||||
|  | // Per <https://www.raspberrypi.com/documentation/computers/raspberry-pi.html#peripheral-addresses>:
 | ||||||
|  | //
 | ||||||
|  | // SoC     Peripheral Address	Peripheral Size	SDRAM Address	Source
 | ||||||
|  | // BCM2835 0x20000000           0x01000000      0x40000000      <https://github.com/raspberrypi/linux/blob/7f465f823c2ecbade5877b8bbcb2093a8060cb0e/arch/arm/boot/dts/bcm2835.dtsi#L21>
 | ||||||
|  | // BCM2836 0x3f000000           0x01000000      0xc0000000      <https://github.com/raspberrypi/linux/blob/7f465f823c2ecbade5877b8bbcb2093a8060cb0e/arch/arm/boot/dts/bcm2836.dtsi#L10>
 | ||||||
|  | // BCM2837 0x3f000000           0x01000000      0xc0000000      <https://github.com/raspberrypi/linux/blob/7f465f823c2ecbade5877b8bbcb2093a8060cb0e/arch/arm/boot/dts/bcm2837.dtsi#L9>
 | ||||||
|  | // BCM2711 0xfe000000           0x01800000      0xc0000000      <https://github.com/raspberrypi/linux/blob/7f465f823c2ecbade5877b8bbcb2093a8060cb0e/arch/arm/boot/dts/bcm2711.dtsi#L41>
 | ||||||
|  | 
 | ||||||
|  | // <https://www.raspberrypi.com/documentation/computers/processors.html>
 | ||||||
|  | // The BCM2835 is the Broadcom chip used in the Raspberry Pi Model A, B, B+, the Compute Module, and the Raspberry Pi Zero.
 | ||||||
|  | // The BCM2836 is used in the Raspberry Pi 2 Model B.
 | ||||||
|  | // The BCM2837 is used in the Raspberry Pi 3, and in later models of the Raspberry Pi 2.
 | ||||||
|  | // The BCM2837B0 is used in the Raspberry Pi 3B+ and 3A+.
 | ||||||
|  | // The BCM2711 is used in the Raspberry Pi 4 Model B.
 | ||||||
|  | // RP3A0 (BCM2710A1 — which is the die packaged inside the BCM2837 chip - Raspberry Pi 3) used in Raspberry Pi Zero 2 W
 | ||||||
|  | 
 | ||||||
|  | // Machine   Board  Chip
 | ||||||
|  | // raspi1    raspi  bcm2835
 | ||||||
|  | // raspi1    raspi  bcm2835
 | ||||||
|  | // raspi3b+  raspi  bcm2837
 | ||||||
|  | // raspi4    raspi  bcm2711
 | ||||||
|  | 
 | ||||||
|  | impl BcmHost { | ||||||
|  |     /// At which address to load the kernel binary.
 | ||||||
|  |     pub const fn kernel_load_address() -> u64 { | ||||||
|  |         0x8_0000 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// As per <https://www.raspberrypi.org/forums/viewtopic.php?p=1170522#p1170522>
 | ||||||
|  |     ///
 | ||||||
|  |     pub fn bus2phys(bus: usize) -> usize { | ||||||
|  |         bus & !0xc000_0000 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     pub fn phys2bus(phys: usize) -> usize { | ||||||
|  |         phys | 0xc000_0000 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // RasPi3B+
 | ||||||
|  | #[cfg(feature = "rpi3")] | ||||||
|  | impl BcmHost { | ||||||
|  |     /// Name of the hardware device this BcmHost is compiled for.
 | ||||||
|  |     pub const fn board_name() -> &'static str { | ||||||
|  |         "Raspberry Pi 3+" | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// This returns the ARM-side physical address where peripherals are mapped.
 | ||||||
|  |     ///
 | ||||||
|  |     pub const fn get_peripheral_address() -> usize { | ||||||
|  |         0x3f00_0000 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// This returns the size of the peripherals' space.
 | ||||||
|  |     pub const fn get_peripheral_size() -> usize { | ||||||
|  |         0x0100_0000 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// This returns the bus address of the SDRAM.
 | ||||||
|  |     pub const fn get_sdram_address() -> usize { | ||||||
|  |         0xc000_0000 // uncached
 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // RasPi4
 | ||||||
|  | #[cfg(feature = "rpi4")] | ||||||
|  | impl BcmHost { | ||||||
|  |     /// Name of the hardware device this BcmHost is compiled for.
 | ||||||
|  |     pub const fn board_name() -> &'static str { | ||||||
|  |         "Raspberry Pi 4+" | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// This returns the ARM-side physical address where peripherals are mapped.
 | ||||||
|  |     ///
 | ||||||
|  |     pub const fn get_peripheral_address() -> usize { | ||||||
|  |         0xfe00_0000 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// This returns the size of the peripherals' space.
 | ||||||
|  |     pub const fn get_peripheral_size() -> usize { | ||||||
|  |         0x0180_0000 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// This returns the bus address of the SDRAM.
 | ||||||
|  |     pub const fn get_sdram_address() -> usize { | ||||||
|  |         0xc000_0000 // uncached
 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,448 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: MIT OR BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) 2018-2019 Andre Richter <andre.o.richter@gmail.com> | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  * Original code distributed under MIT, additional changes are under BlueOak-1.0.0 | ||||||
|  |  * | ||||||
|  |  * http://infocenter.arm.com/help/topic/com.arm.doc.ddi0183g/DDI0183G_uart_pl011_r1p5_trm.pdf
 | ||||||
|  |  * https://docs.rs/embedded-serial/0.5.0/embedded_serial/
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     super::{ | ||||||
|  |         gpio, | ||||||
|  |         mailbox::{self, Mailbox, MailboxOps}, | ||||||
|  |         BcmHost, | ||||||
|  |     }, | ||||||
|  |     crate::{ | ||||||
|  |         arch::loop_while, | ||||||
|  |         devices::{ConsoleOps, SerialOps}, | ||||||
|  |         platform::MMIODerefWrapper, | ||||||
|  |     }, | ||||||
|  |     snafu::Snafu, | ||||||
|  |     tock_registers::{ | ||||||
|  |         interfaces::{ReadWriteable, Readable, Writeable}, | ||||||
|  |         register_bitfields, register_structs, | ||||||
|  |         registers::{ReadOnly, ReadWrite, WriteOnly}, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // PL011 UART registers.
 | ||||||
|  | //
 | ||||||
|  | // Descriptions taken from
 | ||||||
|  | // https://github.com/raspberrypi/documentation/files/1888662/BCM2837-ARM-Peripherals.-.Revised.-.V2-1.pdf
 | ||||||
|  | register_bitfields! { | ||||||
|  |     u32, | ||||||
|  | 
 | ||||||
|  |     /// Flag Register
 | ||||||
|  |     FR [ | ||||||
|  |         /// Transmit FIFO empty. The meaning of this bit depends on the
 | ||||||
|  |         /// state of the FEN bit in the Line Control Register, If the
 | ||||||
|  |         /// FIFO is disabled, this bit is set when the transmit holding
 | ||||||
|  |         /// register is empty. If the FIFO is enabled, the TXFE bit is
 | ||||||
|  |         /// set when the transmit FIFO is empty. This bit does not indicate
 | ||||||
|  |         /// if there is data in the transmit shift register.
 | ||||||
|  |         TXFE OFFSET(7) NUMBITS(1) [], | ||||||
|  | 
 | ||||||
|  |         /// Receive FIFO full. The meaning of this bit depends on the
 | ||||||
|  |         /// state of the FEN bit in the LCRH Register. If the FIFO is
 | ||||||
|  |         /// disabled, this bit is set when the receive holding register
 | ||||||
|  |         /// is full. If the FIFO is enabled, the RXFF bit is set when
 | ||||||
|  |         /// the receive FIFO is full.
 | ||||||
|  |         RXFF OFFSET(6) NUMBITS(1) [], | ||||||
|  | 
 | ||||||
|  |         /// Transmit FIFO full. The meaning of this bit depends on the
 | ||||||
|  |         /// state of the FEN bit in the LCRH Register. If the
 | ||||||
|  |         /// FIFO is disabled, this bit is set when the transmit
 | ||||||
|  |         /// holding register is full. If the FIFO is enabled, the TXFF
 | ||||||
|  |         /// bit is set when the transmit FIFO is full.
 | ||||||
|  |         TXFF OFFSET(5) NUMBITS(1) [], | ||||||
|  | 
 | ||||||
|  |         /// Receive FIFO empty. The meaning of this bit depends on the
 | ||||||
|  |         /// state of the FEN bit in the LCRH Register. If the
 | ||||||
|  |         /// FIFO is disabled, this bit is set when the receive holding
 | ||||||
|  |         /// register is empty. If the FIFO is enabled, the RXFE bit is
 | ||||||
|  |         /// set when the receive FIFO is empty.
 | ||||||
|  |         RXFE OFFSET(4) NUMBITS(1) [], | ||||||
|  | 
 | ||||||
|  |         /// UART busy. If this bit is set to 1, the UART is busy
 | ||||||
|  |         /// transmitting data. This bit remains set until the complete
 | ||||||
|  |         /// byte, including all the stop bits, has been sent from the
 | ||||||
|  |         /// shift register. This bit is set as soon as the transmit FIFO
 | ||||||
|  |         /// becomes non-empty, regardless of whether the UART is enabled or not.
 | ||||||
|  |         BUSY OFFSET(3) NUMBITS(1) [] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// Integer Baud rate divisor
 | ||||||
|  |     IBRD [ | ||||||
|  |         /// Integer Baud rate divisor
 | ||||||
|  |         IBRD OFFSET(0) NUMBITS(16) [] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// Fractional Baud rate divisor
 | ||||||
|  |     FBRD [ | ||||||
|  |         /// Fractional Baud rate divisor
 | ||||||
|  |         FBRD OFFSET(0) NUMBITS(6) [] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// Line Control register
 | ||||||
|  |     LCRH [ | ||||||
|  |         Parity OFFSET(1) NUMBITS(1) [ | ||||||
|  |             Disabled = 0, | ||||||
|  |             Enabled = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// Use 2 stop bits
 | ||||||
|  |         Stop2 OFFSET(3) NUMBITS(1) [ | ||||||
|  |             Disabled = 0, | ||||||
|  |             Enabled = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         Fifo OFFSET(4) NUMBITS(1) [ | ||||||
|  |             Disabled = 0, | ||||||
|  |             Enabled = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// Word length. These bits indicate the number of data bits
 | ||||||
|  |         /// transmitted or received in a frame.
 | ||||||
|  |         WordLength OFFSET(5) NUMBITS(2) [ | ||||||
|  |             FiveBit = 0b00, | ||||||
|  |             SixBit = 0b01, | ||||||
|  |             SevenBit = 0b10, | ||||||
|  |             EightBit = 0b11 | ||||||
|  |         ] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// Control Register
 | ||||||
|  |     CR [ | ||||||
|  |         /// Receive enable. If this bit is set to 1, the receive
 | ||||||
|  |         /// section of the UART is enabled. Data reception occurs for
 | ||||||
|  |         /// UART signals. When the UART is disabled in the middle of
 | ||||||
|  |         /// reception, it completes the current character before
 | ||||||
|  |         /// stopping.
 | ||||||
|  |         RXE    OFFSET(9) NUMBITS(1) [ | ||||||
|  |             Disabled = 0, | ||||||
|  |             Enabled = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// Transmit enable. If this bit is set to 1, the transmit
 | ||||||
|  |         /// section of the UART is enabled. Data transmission occurs
 | ||||||
|  |         /// for UART signals. When the UART is disabled in the middle
 | ||||||
|  |         /// of transmission, it completes the current character before
 | ||||||
|  |         /// stopping.
 | ||||||
|  |         TXE    OFFSET(8) NUMBITS(1) [ | ||||||
|  |             Disabled = 0, | ||||||
|  |             Enabled = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         /// UART enable
 | ||||||
|  |         UARTEN OFFSET(0) NUMBITS(1) [ | ||||||
|  |             /// If the UART is disabled in the middle of transmission
 | ||||||
|  |             /// or reception, it completes the current character
 | ||||||
|  |             /// before stopping.
 | ||||||
|  |             Disabled = 0, | ||||||
|  |             Enabled = 1 | ||||||
|  |         ] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// Interupt Clear Register
 | ||||||
|  |     ICR [ | ||||||
|  |         /// Meta field for all pending interrupts
 | ||||||
|  |         ALL OFFSET(0) NUMBITS(11) [] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// Interupt Mask Set/Clear Register
 | ||||||
|  |     IMSC [ | ||||||
|  |         /// Meta field for all interrupts
 | ||||||
|  |         ALL OFFSET(0) NUMBITS(11) [] | ||||||
|  |     ], | ||||||
|  | 
 | ||||||
|  |     /// DMA Control Register
 | ||||||
|  |     DMACR [ | ||||||
|  |         // RX DMA enabled
 | ||||||
|  |         RXDMAE OFFSET(0) NUMBITS(1) [ | ||||||
|  |             Disabled = 0, | ||||||
|  |             Enabled = 1 | ||||||
|  |         ], | ||||||
|  | 
 | ||||||
|  |         // TX DMA enabled
 | ||||||
|  |         TXDMAE OFFSET(0) NUMBITS(1) [ | ||||||
|  |             Disabled = 0, | ||||||
|  |             Enabled = 1 | ||||||
|  |         ], | ||||||
|  |     ] | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // https://developer.arm.com/documentation/ddi0183/g/programmers-model/summary-of-registers?lang=en
 | ||||||
|  | register_structs! { | ||||||
|  |     #[allow(non_snake_case)] | ||||||
|  |     RegisterBlock { | ||||||
|  |         (0x00 => Data: ReadWrite<u32>), // DR
 | ||||||
|  |         (0x04 => Status: ReadWrite<u32>), // RSR/ECR
 | ||||||
|  |         (0x08 => __reserved_1), | ||||||
|  |         (0x18 => Flag: ReadOnly<u32, FR::Register>), | ||||||
|  |         (0x1c => __reserved_2), | ||||||
|  |         (0x24 => IntegerBaudRate: ReadWrite<u32, IBRD::Register>), | ||||||
|  |         (0x28 => FractionalBaudRate: ReadWrite<u32, FBRD::Register>), | ||||||
|  |         (0x2c => LineControl: ReadWrite<u32, LCRH::Register>), | ||||||
|  |         (0x30 => Control: ReadWrite<u32, CR::Register>), | ||||||
|  |         (0x34 => InterruptFifoLevelSelect: ReadWrite<u32>), | ||||||
|  |         (0x38 => InterruptMaskSetClear: ReadWrite<u32, IMSC::Register>), | ||||||
|  |         (0x3c => RawInterruptStatus: ReadOnly<u32>), | ||||||
|  |         (0x40 => MaskedInterruptStatus: ReadOnly<u32>), | ||||||
|  |         (0x44 => InterruptClear: WriteOnly<u32, ICR::Register>), | ||||||
|  |         (0x48 => DmaControl: ReadWrite<u32, DMACR::Register>), | ||||||
|  |         (0x4c => __reserved_3), | ||||||
|  |         (0x1000 => @END), | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[derive(Debug, Snafu)] | ||||||
|  | pub enum PL011UartError { | ||||||
|  |     #[snafu(display("PL011 UART setup failed in mailbox operation"))] | ||||||
|  |     MailboxError, | ||||||
|  |     #[snafu(display(
 | ||||||
|  |         "PL011 UART setup failed due to integer baud rate divisor out of range ({})", | ||||||
|  |         ibrd | ||||||
|  |     ))] | ||||||
|  |     InvalidIntegerDivisor { ibrd: u32 }, | ||||||
|  |     #[snafu(display(
 | ||||||
|  |         "PL011 UART setup failed due to fractional baud rate divisor out of range ({})", | ||||||
|  |         fbrd | ||||||
|  |     ))] | ||||||
|  |     InvalidFractionalDivisor { fbrd: u32 }, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub type Result<T> = ::core::result::Result<T, PL011UartError>; | ||||||
|  | 
 | ||||||
|  | type Registers = MMIODerefWrapper<RegisterBlock>; | ||||||
|  | 
 | ||||||
|  | pub struct PL011Uart { | ||||||
|  |     registers: Registers, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub struct PreparedPL011Uart(PL011Uart); | ||||||
|  | 
 | ||||||
|  | pub struct RateDivisors { | ||||||
|  |     integer_baud_rate_divisor: u32, | ||||||
|  |     fractional_baud_rate_divisor: u32, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl RateDivisors { | ||||||
|  |     // Set integer & fractional part of baud rate.
 | ||||||
|  |     // Integer = clock/(16 * Baud)
 | ||||||
|  |     // e.g. 3000000 / (16 * 115200) = 1.627 = ~1.
 | ||||||
|  |     // Fraction = (Fractional part * 64) + 0.5
 | ||||||
|  |     // e.g. (.627 * 64) + 0.5 = 40.6 = ~40.
 | ||||||
|  |     //
 | ||||||
|  |     // Use integer-only calculation based on [this page](https://krinkinmu.github.io/2020/11/29/PL011.html)
 | ||||||
|  |     // Calculate 64 * clock / (16 * rate) = 4 * clock / rate, then extract 6 lowest bits for fractional part
 | ||||||
|  |     // and the next 16 bits for integer part.
 | ||||||
|  |     pub fn from_clock_and_rate(clock: u64, baud_rate: u32) -> Result<RateDivisors> { | ||||||
|  |         let value = 4 * clock / baud_rate as u64; | ||||||
|  |         let i = ((value >> 6) & 0xffff) as u32; | ||||||
|  |         let f = (value & 0x3f) as u32; | ||||||
|  |         // TODO: check for integer overflow, i.e. any bits set above the 0x3fffff mask.
 | ||||||
|  |         // FIXME: can't happen due to calculation above
 | ||||||
|  |         if i > 65535 { | ||||||
|  |             return Err(PL011UartError::InvalidIntegerDivisor { ibrd: i }); | ||||||
|  |         } | ||||||
|  |         // FIXME: can't happen due to calculation above
 | ||||||
|  |         if f > 63 { | ||||||
|  |             return Err(PL011UartError::InvalidFractionalDivisor { fbrd: f }); | ||||||
|  |         } | ||||||
|  |         Ok(RateDivisors { | ||||||
|  |             integer_baud_rate_divisor: i, | ||||||
|  |             fractional_baud_rate_divisor: f, | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub const UART0_START: usize = 0x20_1000; | ||||||
|  | 
 | ||||||
|  | impl Default for PL011Uart { | ||||||
|  |     fn default() -> Self { | ||||||
|  |         const UART0_BASE: usize = BcmHost::get_peripheral_address() + UART0_START; | ||||||
|  |         unsafe { PL011Uart::new(UART0_BASE) } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl PL011Uart { | ||||||
|  |     /// # Safety
 | ||||||
|  |     ///
 | ||||||
|  |     /// Unsafe, duh!
 | ||||||
|  |     pub const unsafe fn new(base_addr: usize) -> PL011Uart { | ||||||
|  |         PL011Uart { | ||||||
|  |             registers: Registers::new(base_addr), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Set baud rate and characteristics (115200 8N1) and map to GPIO
 | ||||||
|  |     pub fn prepare(self, gpio: &gpio::GPIO) -> Result<PreparedPL011Uart> { | ||||||
|  |         // Turn off UART
 | ||||||
|  |         self.registers.Control.set(0); | ||||||
|  | 
 | ||||||
|  |         // Wait for any ongoing transmissions to complete
 | ||||||
|  |         self.flush_internal(); | ||||||
|  | 
 | ||||||
|  |         // Flush TX FIFO
 | ||||||
|  |         self.registers.LineControl.modify(LCRH::Fifo::Disabled); | ||||||
|  | 
 | ||||||
|  |         // set up clock for consistent divisor values
 | ||||||
|  |         const CLOCK: u32 = 4_000_000; // 4Mhz
 | ||||||
|  |         const BAUD_RATE: u32 = 115_200; | ||||||
|  | 
 | ||||||
|  |         let mut mailbox = Mailbox::<9>::default(); | ||||||
|  |         let index = mailbox.request(); | ||||||
|  |         let index = mailbox.set_clock_rate(index, mailbox::clock::UART, CLOCK); | ||||||
|  |         let mailbox = mailbox.end(index); | ||||||
|  | 
 | ||||||
|  |         if mailbox.call(mailbox::channel::PropertyTagsArmToVc).is_err() { | ||||||
|  |             return Err(PL011UartError::MailboxError); // Abort if UART clocks couldn't be set
 | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         // Pin 14
 | ||||||
|  |         const UART_TXD: gpio::Function = gpio::Function::Alt0; | ||||||
|  |         // Pin 15
 | ||||||
|  |         const UART_RXD: gpio::Function = gpio::Function::Alt0; | ||||||
|  | 
 | ||||||
|  |         // Map UART0 to GPIO pins and enable pull-ups
 | ||||||
|  |         gpio.get_pin(14) | ||||||
|  |             .into_alt(UART_TXD) | ||||||
|  |             .set_pull_up_down(gpio::PullUpDown::Up); | ||||||
|  |         gpio.get_pin(15) | ||||||
|  |             .into_alt(UART_RXD) | ||||||
|  |             .set_pull_up_down(gpio::PullUpDown::Up); | ||||||
|  | 
 | ||||||
|  |         // Clear pending interrupts
 | ||||||
|  |         self.registers.InterruptClear.write(ICR::ALL::SET); | ||||||
|  | 
 | ||||||
|  |         // From the PL011 Technical Reference Manual:
 | ||||||
|  |         //
 | ||||||
|  |         // The LCR_H, IBRD, and FBRD registers form the single 30-bit wide LCR Register that is
 | ||||||
|  |         // updated on a single write strobe generated by a LCR_H write. So, to internally update the
 | ||||||
|  |         // contents of IBRD or FBRD, a LCR_H write must always be performed at the end.
 | ||||||
|  |         //
 | ||||||
|  |         // Set the baud rate divisors, 8N1 and FIFO enabled.
 | ||||||
|  |         let divisors = RateDivisors::from_clock_and_rate(CLOCK.into(), BAUD_RATE)?; | ||||||
|  |         self.registers | ||||||
|  |             .IntegerBaudRate | ||||||
|  |             .write(IBRD::IBRD.val(divisors.integer_baud_rate_divisor & 0xffff)); | ||||||
|  |         self.registers | ||||||
|  |             .FractionalBaudRate | ||||||
|  |             .write(FBRD::FBRD.val(divisors.fractional_baud_rate_divisor & 0b11_1111)); | ||||||
|  |         self.registers.LineControl.write( | ||||||
|  |             LCRH::WordLength::EightBit | ||||||
|  |                 + LCRH::Fifo::Enabled | ||||||
|  |                 + LCRH::Parity::Disabled | ||||||
|  |                 + LCRH::Stop2::Disabled, | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |         // Mask all interrupts by setting corresponding bits to 1
 | ||||||
|  |         self.registers.InterruptMaskSetClear.write(IMSC::ALL::SET); | ||||||
|  | 
 | ||||||
|  |         // Disable DMA
 | ||||||
|  |         self.registers | ||||||
|  |             .DmaControl | ||||||
|  |             .write(DMACR::RXDMAE::Disabled + DMACR::TXDMAE::Disabled); | ||||||
|  | 
 | ||||||
|  |         // Turn on UART
 | ||||||
|  |         self.registers | ||||||
|  |             .Control | ||||||
|  |             .write(CR::UARTEN::Enabled + CR::TXE::Enabled + CR::RXE::Enabled); | ||||||
|  | 
 | ||||||
|  |         Ok(PreparedPL011Uart(self)) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn flush_internal(&self) { | ||||||
|  |         loop_while(|| self.registers.Flag.is_set(FR::BUSY)); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Drop for PreparedPL011Uart { | ||||||
|  |     fn drop(&mut self) { | ||||||
|  |         self.0.registers.Control.set(0); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl SerialOps for PreparedPL011Uart { | ||||||
|  |     fn read_byte(&self) -> u8 { | ||||||
|  |         // wait until something is in the buffer
 | ||||||
|  |         loop_while(|| self.0.registers.Flag.is_set(FR::RXFE)); | ||||||
|  | 
 | ||||||
|  |         // read it and return
 | ||||||
|  |         self.0.registers.Data.get() as u8 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn write_byte(&self, b: u8) { | ||||||
|  |         // wait until we can send
 | ||||||
|  |         loop_while(|| self.0.registers.Flag.is_set(FR::TXFF)); | ||||||
|  | 
 | ||||||
|  |         // write the character to the buffer
 | ||||||
|  |         self.0.registers.Data.set(b as u32); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Wait until the TX FIFO is empty, aka all characters have been put on the
 | ||||||
|  |     /// line.
 | ||||||
|  |     fn flush(&self) { | ||||||
|  |         self.0.flush_internal(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Consume input until RX FIFO is empty, aka all pending characters have been
 | ||||||
|  |     /// consumed.
 | ||||||
|  |     fn clear_rx(&self) { | ||||||
|  |         loop_while(|| { | ||||||
|  |             let pending = !self.0.registers.Flag.is_set(FR::RXFE); | ||||||
|  |             if pending { | ||||||
|  |                 self.read_byte(); | ||||||
|  |             } | ||||||
|  |             pending | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl ConsoleOps for PreparedPL011Uart { | ||||||
|  |     /// Send a character
 | ||||||
|  |     fn write_char(&self, c: char) { | ||||||
|  |         self.write_byte(c as u8) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Display a string
 | ||||||
|  |     fn write_string(&self, string: &str) { | ||||||
|  |         for c in string.chars() { | ||||||
|  |             // convert newline to carriage return + newline
 | ||||||
|  |             if c == '\n' { | ||||||
|  |                 self.write_char('\r') | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             self.write_char(c); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Receive a character
 | ||||||
|  |     fn read_char(&self) -> char { | ||||||
|  |         let mut ret = self.read_byte() as char; | ||||||
|  | 
 | ||||||
|  |         // convert carriage return to newline
 | ||||||
|  |         if ret == '\r' { | ||||||
|  |             ret = '\n' | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         ret | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(test)] | ||||||
|  | mod tests { | ||||||
|  |     use super::*; | ||||||
|  | 
 | ||||||
|  |     #[test_case] | ||||||
|  |     fn test_divisors() { | ||||||
|  |         const CLOCK: u64 = 3_000_000; | ||||||
|  |         const BAUD_RATE: u32 = 115_200; | ||||||
|  | 
 | ||||||
|  |         let divisors = RateDivisors::from_clock_and_rate(CLOCK, BAUD_RATE); | ||||||
|  |         assert_eq!(divisors.integer_baud_rate_divisor, 1); | ||||||
|  |         assert_eq!(divisors.fractional_baud_rate_divisor, 40); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,121 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: MIT OR BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) 2018-2019 Andre Richter <andre.o.richter@gmail.com> | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  * Original code distributed under MIT, additional changes are under BlueOak-1.0.0 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | use { | ||||||
|  |     super::{ | ||||||
|  |         gpio, | ||||||
|  |         mailbox::{channel, Mailbox, MailboxOps}, | ||||||
|  |         BcmHost, | ||||||
|  |     }, | ||||||
|  |     crate::platform::MMIODerefWrapper, | ||||||
|  |     snafu::Snafu, | ||||||
|  |     tock_registers::{ | ||||||
|  |         interfaces::{Readable, Writeable}, | ||||||
|  |         register_structs, | ||||||
|  |         registers::ReadWrite, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | register_structs! { | ||||||
|  |     #[allow(non_snake_case)] | ||||||
|  |     RegisterBlock { | ||||||
|  |         (0x00 => __reserved_1), | ||||||
|  |         (0x1c => PM_RSTC: ReadWrite<u32>), | ||||||
|  |         (0x20 => PM_RSTS: ReadWrite<u32>), | ||||||
|  |         (0x24 => PM_WDOG: ReadWrite<u32>), | ||||||
|  |         (0x28 => @END), | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const PM_PASSWORD: u32 = 0x5a00_0000; | ||||||
|  | const PM_RSTC_WRCFG_CLR: u32 = 0xffff_ffcf; | ||||||
|  | const PM_RSTC_WRCFG_FULL_RESET: u32 = 0x0000_0020; | ||||||
|  | 
 | ||||||
|  | // The Raspberry Pi firmware uses the RSTS register to know which
 | ||||||
|  | // partition to boot from. The partition value is spread into bits 0, 2,
 | ||||||
|  | // 4, 6, 8, 10. Partition 63 is a special partition used by the
 | ||||||
|  | // firmware to indicate halt.
 | ||||||
|  | const PM_RSTS_RASPBERRYPI_HALT: u32 = 0x555; | ||||||
|  | 
 | ||||||
|  | const POWER_STATE_OFF: u32 = 0; | ||||||
|  | const POWER_STATE_ON: u32 = 1; | ||||||
|  | const POWER_STATE_DO_NOT_WAIT: u32 = 0; | ||||||
|  | const POWER_STATE_WAIT: u32 = 2; | ||||||
|  | 
 | ||||||
|  | #[derive(Debug, Snafu)] | ||||||
|  | pub enum PowerError { | ||||||
|  |     #[snafu(display("Power setup failed in mailbox operation"))] | ||||||
|  |     MailboxError, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub type Result<T> = ::core::result::Result<T, PowerError>; | ||||||
|  | 
 | ||||||
|  | type Registers = MMIODerefWrapper<RegisterBlock>; | ||||||
|  | 
 | ||||||
|  | const POWER_START: usize = 0x0010_0000; | ||||||
|  | 
 | ||||||
|  | /// Public interface to the Power subsystem
 | ||||||
|  | pub struct Power { | ||||||
|  |     registers: Registers, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Default for Power { | ||||||
|  |     fn default() -> Power { | ||||||
|  |         const POWER_BASE: usize = BcmHost::get_peripheral_address() + POWER_START; | ||||||
|  |         unsafe { Power::new(POWER_BASE) } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Power { | ||||||
|  |     /// # Safety
 | ||||||
|  |     ///
 | ||||||
|  |     /// Unsafe, duh!
 | ||||||
|  |     pub const unsafe fn new(base_addr: usize) -> Power { | ||||||
|  |         Power { | ||||||
|  |             registers: Registers::new(base_addr), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Shutdown the board
 | ||||||
|  |     pub fn off(&self, gpio: &gpio::GPIO) -> Result<()> { | ||||||
|  |         // power off devices one by one
 | ||||||
|  |         for dev_id in 0..16 { | ||||||
|  |             let mut mbox = Mailbox::<8>::default(); | ||||||
|  |             let index = mbox.request(); | ||||||
|  |             let index = | ||||||
|  |                 mbox.set_device_power(index, dev_id, POWER_STATE_OFF | POWER_STATE_DO_NOT_WAIT); | ||||||
|  |             let mbox = mbox.end(index); | ||||||
|  | 
 | ||||||
|  |             mbox.call(channel::PropertyTagsArmToVc) | ||||||
|  |                 .map_err(|_| PowerError::MailboxError)?; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         gpio.power_off(); | ||||||
|  | 
 | ||||||
|  |         // We set the watchdog hard reset bit here to distinguish this
 | ||||||
|  |         // reset from the normal (full) reset. bootcode.bin will not
 | ||||||
|  |         // reboot after a hard reset.
 | ||||||
|  |         let mut val = self.registers.PM_RSTS.get(); | ||||||
|  |         val |= PM_PASSWORD | PM_RSTS_RASPBERRYPI_HALT; | ||||||
|  |         self.registers.PM_RSTS.set(val); | ||||||
|  | 
 | ||||||
|  |         // Continue with normal reset mechanism
 | ||||||
|  |         self.reset(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Reboot
 | ||||||
|  |     pub fn reset(&self) -> ! { | ||||||
|  |         // use a timeout of 10 ticks (~150us)
 | ||||||
|  |         self.registers.PM_WDOG.set(PM_PASSWORD | 10); | ||||||
|  |         let mut val = self.registers.PM_RSTC.get(); | ||||||
|  |         val &= PM_RSTC_WRCFG_CLR; | ||||||
|  |         val |= PM_PASSWORD | PM_RSTC_WRCFG_FULL_RESET; | ||||||
|  |         self.registers.PM_RSTC.set(val); | ||||||
|  | 
 | ||||||
|  |         crate::endless_sleep() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,128 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | use { | ||||||
|  |     super::{ | ||||||
|  |         display::{Display, PixelOrder, CHARSIZE_X, CHARSIZE_Y}, | ||||||
|  |         mailbox::{self, channel, response::VAL_LEN_FLAG, Mailbox, MailboxOps}, | ||||||
|  |         BcmHost, | ||||||
|  |     }, | ||||||
|  |     crate::{platform::rpi3::mailbox::MailboxStorageRef, println}, | ||||||
|  |     core::convert::TryInto, | ||||||
|  |     snafu::Snafu, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | pub struct VC; | ||||||
|  | 
 | ||||||
|  | #[derive(Debug, Snafu)] | ||||||
|  | pub enum VcError { | ||||||
|  |     #[snafu(display("VC setup failed in mailbox operation"))] | ||||||
|  |     MailboxError, | ||||||
|  |     #[snafu(display("VC setup failed due to bad mailbox response {:x}", response))] | ||||||
|  |     MailboxResponseError { response: u32 }, | ||||||
|  |     #[snafu(display("Unknown pixel order received in mailbox response"))] | ||||||
|  |     InvalidPixelOrder, | ||||||
|  | } | ||||||
|  | type Result<T, E = VcError> = ::core::result::Result<T, E>; | ||||||
|  | 
 | ||||||
|  | impl VC { | ||||||
|  |     // Use framebuffer mailbox interface to initialize
 | ||||||
|  |     // https://www.raspberrypi.org/forums/viewtopic.php?f=72&t=185116
 | ||||||
|  |     pub fn init_fb(w: u32, h: u32, depth: u32) -> Result<Display> { | ||||||
|  |         /* | ||||||
|  |          *  * All tags in the request are processed in one operation. | ||||||
|  |          *  * It is not valid to mix Test tags with Get/Set tags | ||||||
|  |          *    in the same operation and no tags will be returned. | ||||||
|  |          *  * Get tags will be processed after all Set tags. | ||||||
|  |          *  * If an allocate buffer tag is omitted when setting parameters, | ||||||
|  |          *    then no change occurs unless it can be accommodated without changing | ||||||
|  |          *    the buffer base or size. | ||||||
|  |          *  * When an allocate buffer response is returned, the old buffer area | ||||||
|  |          *    (if the base or size has changed) is implicitly freed. | ||||||
|  |          */ | ||||||
|  | 
 | ||||||
|  |         let mut mbox = Mailbox::<36>::default(); | ||||||
|  |         let index = mbox.request(); | ||||||
|  |         let index = mbox.set_physical_wh(index, w, h); | ||||||
|  |         let index = mbox.set_virtual_wh(index, w, h); | ||||||
|  |         let index = mbox.set_depth(index, depth); | ||||||
|  |         let index = mbox.allocate_buffer_aligned(index, 16); | ||||||
|  |         let mbox = mbox.end(index); | ||||||
|  | 
 | ||||||
|  |         mbox.call(channel::PropertyTagsArmToVc).map_err(|e| { | ||||||
|  |             println!("Mailbox call returned error {}", e); | ||||||
|  |             println!("Mailbox contents: {:?}", mbox); | ||||||
|  |             VcError::MailboxError | ||||||
|  |         })?; | ||||||
|  | 
 | ||||||
|  |         if (mbox.value_at(18) & VAL_LEN_FLAG) == 0 { | ||||||
|  |             return Err(VcError::MailboxResponseError { | ||||||
|  |                 response: mbox.value_at(18), | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         let fb_ptr = BcmHost::bus2phys(mbox.value_at(19).try_into().unwrap()); | ||||||
|  |         let fb_size = mbox.value_at(20); | ||||||
|  | 
 | ||||||
|  |         // SetPixelOrder doesn't work in QEMU, however TestPixelOrder does.
 | ||||||
|  |         // Apparently, QEMU doesn't care about intermixing Get/Set and Test tags either.
 | ||||||
|  |         let mut mbox = Mailbox::<36>::default(); | ||||||
|  |         let index = mbox.request(); | ||||||
|  |         #[cfg(qemu)] | ||||||
|  |         let index = mbox.test_pixel_order(index, 1); | ||||||
|  |         #[cfg(not(qemu))] | ||||||
|  |         let index = mbox.set_pixel_order(index, 1); | ||||||
|  |         let index = mbox.set_alpha_mode(index, mailbox::alpha_mode::IGNORED); | ||||||
|  |         let index = mbox.get_pitch(index); | ||||||
|  |         let mbox = mbox.end(index); | ||||||
|  | 
 | ||||||
|  |         // let index = mbox.test_pixel_order(index, 1);
 | ||||||
|  | 
 | ||||||
|  |         mbox.call(channel::PropertyTagsArmToVc) | ||||||
|  |             .map_err(|_| VcError::MailboxError)?; | ||||||
|  | 
 | ||||||
|  |         if (mbox.value_at(4) & VAL_LEN_FLAG) == 0 { | ||||||
|  |             return Err(VcError::MailboxResponseError { | ||||||
|  |                 response: mbox.value_at(4), | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |         if (mbox.value_at(12) & VAL_LEN_FLAG) == 0 { | ||||||
|  |             return Err(VcError::MailboxResponseError { | ||||||
|  |                 response: mbox.value_at(12), | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         let order = match mbox.value_at(5) { | ||||||
|  |             0 => PixelOrder::BGR, | ||||||
|  |             1 => PixelOrder::RGB, | ||||||
|  |             _ => return Err(VcError::InvalidPixelOrder), | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         let pitch = mbox.value_at(13); | ||||||
|  | 
 | ||||||
|  |         /* Need to set up max_x/max_y before using Display::write */ | ||||||
|  |         let max_x = w / CHARSIZE_X; | ||||||
|  |         let max_y = h / CHARSIZE_Y; | ||||||
|  | 
 | ||||||
|  |         let x_offset = 0; | ||||||
|  |         let y_offset = 0; | ||||||
|  | 
 | ||||||
|  |         println!( | ||||||
|  |             "[i] VC init: {}x{}, {}x{}, d{}, --{}--, +{}x{}, {}@{:x}", | ||||||
|  |             w, h, w, h, depth, pitch, x_offset, y_offset, fb_size, fb_ptr | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |         Ok(Display::new( | ||||||
|  |             fb_ptr.try_into().unwrap(), | ||||||
|  |             fb_size, | ||||||
|  |             depth, | ||||||
|  |             pitch, | ||||||
|  |             max_x, | ||||||
|  |             max_y, | ||||||
|  |             w, | ||||||
|  |             h, | ||||||
|  |             order, | ||||||
|  |         )) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,35 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | pub mod semihosting { | ||||||
|  |     pub fn exit_success() -> ! { | ||||||
|  |         use qemu_exit::QEMUExit; | ||||||
|  | 
 | ||||||
|  |         #[cfg(target_arch = "aarch64")] | ||||||
|  |         let qemu_exit_handle = qemu_exit::AArch64::new(); | ||||||
|  | 
 | ||||||
|  |         qemu_exit_handle.exit_success() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     pub fn exit_failure() -> ! { | ||||||
|  |         use qemu_exit::QEMUExit; | ||||||
|  | 
 | ||||||
|  |         #[cfg(target_arch = "aarch64")] | ||||||
|  |         let qemu_exit_handle = qemu_exit::AArch64::new(); | ||||||
|  | 
 | ||||||
|  |         qemu_exit_handle.exit_failure() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     pub fn sys_write0_call(text: &str) { | ||||||
|  |         // SAFETY: text must be \0-terminated!
 | ||||||
|  |         let cmd = 0x04; | ||||||
|  |         unsafe { | ||||||
|  |             core::arch::asm!( | ||||||
|  |                 "hlt #0xF000" | ||||||
|  |                 , in("w0") cmd | ||||||
|  |                 , in("x1") text.as_ptr() as u64 | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,44 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: MIT OR BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) 2019 Andre Richter <andre.o.richter@gmail.com> | ||||||
|  |  * Original code distributed under MIT, additional changes are under BlueOak-1.0.0 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | use core::cell::UnsafeCell; | ||||||
|  | 
 | ||||||
|  | pub struct NullLock<T> { | ||||||
|  |     data: UnsafeCell<T>, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Since we are instantiating this struct as a static variable, which could
 | ||||||
|  | /// potentially be shared between different threads, we need to tell the compiler
 | ||||||
|  | /// that sharing of this struct is safe by marking it with the Sync trait.
 | ||||||
|  | ///
 | ||||||
|  | /// At this point in time, we can do so without worrying, because the kernel
 | ||||||
|  | /// anyways runs on a single core and interrupts are disabled. In short, multiple
 | ||||||
|  | /// threads don't exist yet in our code.
 | ||||||
|  | ///
 | ||||||
|  | /// Literature:
 | ||||||
|  | /// * <https://doc.rust-lang.org/beta/nomicon/send-and-sync.html>
 | ||||||
|  | /// * <https://doc.rust-lang.org/book/ch16-04-extensible-concurrency-sync-and-send.html>
 | ||||||
|  | unsafe impl<T> Sync for NullLock<T> {} | ||||||
|  | 
 | ||||||
|  | impl<T> NullLock<T> { | ||||||
|  |     pub const fn new(data: T) -> NullLock<T> { | ||||||
|  |         NullLock { | ||||||
|  |             data: UnsafeCell::new(data), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<T> NullLock<T> { | ||||||
|  |     pub fn lock<F, R>(&self, f: F) -> R | ||||||
|  |     where | ||||||
|  |         F: FnOnce(&mut T) -> R, | ||||||
|  |     { | ||||||
|  |         // In a real lock, there would be code around this line that ensures
 | ||||||
|  |         // that this mutable reference will ever only be given out to one thread
 | ||||||
|  |         // at a time.
 | ||||||
|  |         f(unsafe { &mut *self.data.get() }) | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,32 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | //============================================================================
 | ||||||
|  | // Testing environment
 | ||||||
|  | //============================================================================
 | ||||||
|  | use crate::{print, println, qemu}; | ||||||
|  | 
 | ||||||
|  | pub trait TestFn { | ||||||
|  |     fn run(&self); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<T> TestFn for T | ||||||
|  | where | ||||||
|  |     T: Fn(), | ||||||
|  | { | ||||||
|  |     fn run(&self) { | ||||||
|  |         print!("*TEST* {}...\t", core::any::type_name::<T>()); | ||||||
|  |         self(); | ||||||
|  |         println!("[ok]\n"); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub fn test_runner(tests: &[&dyn TestFn]) { | ||||||
|  |     println!("Running {} tests", tests.len()); | ||||||
|  |     for test in tests { | ||||||
|  |         test.run(); | ||||||
|  |     } | ||||||
|  |     println!("\n[success]\n"); | ||||||
|  |     qemu::semihosting::exit_success(); | ||||||
|  | } | ||||||
|  | @ -0,0 +1,105 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /// No-alloc write!() implementation from https://stackoverflow.com/a/50201632/145434
 | ||||||
|  | /// Requires you to allocate a buffer somewhere manually.
 | ||||||
|  | // @todo Try to use arrayvec::ArrayString here instead?
 | ||||||
|  | use core::{cmp::min, fmt}; | ||||||
|  | 
 | ||||||
|  | pub struct WriteTo<'a> { | ||||||
|  |     buffer: &'a mut [u8], | ||||||
|  |     // on write error (i.e. not enough space in buffer) this grows beyond
 | ||||||
|  |     // `buffer.len()`.
 | ||||||
|  |     used: usize, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<'a> WriteTo<'a> { | ||||||
|  |     #[allow(unused)] | ||||||
|  |     pub fn new(buffer: &'a mut [u8]) -> Self { | ||||||
|  |         WriteTo { buffer, used: 0 } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[allow(unused)] | ||||||
|  |     pub fn into_str(self) -> Option<&'a str> { | ||||||
|  |         if self.used <= self.buffer.len() { | ||||||
|  |             // only successful concats of str - must be a valid str.
 | ||||||
|  |             Some(unsafe { core::str::from_utf8_unchecked(&self.buffer[..self.used]) }) | ||||||
|  |         } else { | ||||||
|  |             None | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[allow(unused)] | ||||||
|  |     pub fn into_cstr(self) -> Option<&'a str> { | ||||||
|  |         if self.used < self.buffer.len() { | ||||||
|  |             self.buffer[self.used] = 0; // Terminate the string
 | ||||||
|  |             Some(unsafe { core::str::from_utf8_unchecked(&self.buffer[..=self.used]) }) | ||||||
|  |         } else { | ||||||
|  |             None | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<'a> fmt::Write for WriteTo<'a> { | ||||||
|  |     fn write_str(&mut self, s: &str) -> fmt::Result { | ||||||
|  |         if self.used > self.buffer.len() { | ||||||
|  |             return Err(fmt::Error); | ||||||
|  |         } | ||||||
|  |         let remaining_buf = &mut self.buffer[self.used..]; | ||||||
|  |         let raw_s = s.as_bytes(); | ||||||
|  |         let write_num = min(raw_s.len(), remaining_buf.len()); | ||||||
|  |         remaining_buf[..write_num].copy_from_slice(&raw_s[..write_num]); | ||||||
|  |         self.used += raw_s.len(); | ||||||
|  |         if write_num < raw_s.len() { | ||||||
|  |             Err(fmt::Error) | ||||||
|  |         } else { | ||||||
|  |             Ok(()) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[allow(unused)] | ||||||
|  | pub fn show<'a>(buffer: &'a mut [u8], args: fmt::Arguments) -> Result<&'a str, fmt::Error> { | ||||||
|  |     let mut w = WriteTo::new(buffer); | ||||||
|  |     fmt::write(&mut w, args)?; | ||||||
|  |     w.into_str().ok_or(fmt::Error) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Return a zero-terminated str
 | ||||||
|  | #[allow(unused)] | ||||||
|  | pub fn c_show<'a>(buffer: &'a mut [u8], args: fmt::Arguments) -> Result<&'a str, fmt::Error> { | ||||||
|  |     let mut w = WriteTo::new(buffer); | ||||||
|  |     fmt::write(&mut w, args)?; | ||||||
|  |     w.into_cstr().ok_or(fmt::Error) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(test)] | ||||||
|  | mod tests { | ||||||
|  |     use super::*; | ||||||
|  | 
 | ||||||
|  |     #[test_case] | ||||||
|  |     pub fn write_to_works() { | ||||||
|  |         let mut buf = [0u8; 64]; | ||||||
|  |         let s: &str = show( | ||||||
|  |             &mut buf, | ||||||
|  |             format_args!("write some stuff {:?}: {}", "foo", 42), | ||||||
|  |         ) | ||||||
|  |         .unwrap(); | ||||||
|  |         assert_eq!(s, "write some stuff \"foo\": 42"); | ||||||
|  |         assert_eq!(s.as_ptr(), buf.as_ptr()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[test_case] | ||||||
|  |     pub fn zero_terminated_write_to_works() { | ||||||
|  |         let mut buf = [0u8; 64]; | ||||||
|  |         let s: &str = c_show( | ||||||
|  |             &mut buf, | ||||||
|  |             format_args!("write some stuff {:?}: {}", "foo", 42), | ||||||
|  |         ) | ||||||
|  |         .unwrap(); | ||||||
|  |         assert_eq!(s, "write some stuff \"foo\": 42\0"); | ||||||
|  |         assert_eq!(s.as_ptr(), buf.as_ptr()); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,11 +0,0 @@ | ||||||
| #!/bin/sh |  | ||||||
| # Generate gdb-connect script with given RTT block address, to avoid typing it in manually |  | ||||||
| 
 |  | ||||||
| # Hint from this answer https://superuser.com/a/747905/355774 |  | ||||||
| [ $# -ge 1 ] && ADDR="$1" || ADDR=$(cat) |  | ||||||
| 
 |  | ||||||
| cat <<EOF > gdb-connect |  | ||||||
| target remote :3333 |  | ||||||
| monitor rttserver start 19021 0 |  | ||||||
| monitor rtt setup 0x$ADDR 24 "SEGGER RTT" |  | ||||||
| EOF |  | ||||||
|  | @ -0,0 +1,39 @@ | ||||||
|  | [package] | ||||||
|  | name = "nucleus" | ||||||
|  | version = "0.0.1" | ||||||
|  | authors = ["Berkus Decker <berkus+vesper@metta.systems>"] | ||||||
|  | description = "Vesper nanokernel binary" | ||||||
|  | documentation = "https://docs.metta.systems/vesper" | ||||||
|  | homepage = "https://github.com/metta-systems/vesper" | ||||||
|  | repository = "https://github.com/metta-systems/vesper" | ||||||
|  | readme = "README.md" | ||||||
|  | license = "BlueOak-1.0.0" | ||||||
|  | categories = ["no-std", "embedded", "os"] | ||||||
|  | publish = false | ||||||
|  | edition = "2021" | ||||||
|  | 
 | ||||||
|  | [badges] | ||||||
|  | maintenance = { status = "experimental" } | ||||||
|  | 
 | ||||||
|  | [features] | ||||||
|  | noserial = ["machine/noserial"] | ||||||
|  | # Enable JTAG debugging of kernel - enable jtag helpers and | ||||||
|  | # block waiting for JTAG probe attach at the start of kernel main. | ||||||
|  | jtag = ["machine/jtag"] | ||||||
|  | # Build for running under QEMU with semihosting, so various halt/reboot options would for example quit QEMU instead. | ||||||
|  | qemu = ["machine/qemu"] | ||||||
|  | # Mutually exclusive features to choose a target board | ||||||
|  | rpi3 = ["machine/rpi3"] | ||||||
|  | rpi4 = ["machine/rpi4"] | ||||||
|  | 
 | ||||||
|  | [dependencies] | ||||||
|  | machine = { path = "../machine" } | ||||||
|  | r0 = "1.0" | ||||||
|  | cortex-a = "7.0" | ||||||
|  | tock-registers = "0.7" | ||||||
|  | ux = { version = "0.1", default-features = false } | ||||||
|  | usize_conversions = "0.2" | ||||||
|  | bit_field = "0.10" | ||||||
|  | bitflags = "1.3" | ||||||
|  | cfg-if = "1.0" | ||||||
|  | snafu = { version = "0.7", default-features = false } | ||||||
|  | @ -0,0 +1,84 @@ | ||||||
|  | #
 | ||||||
|  | # SPDX-License-Identifier: BlueOak-1.0.0
 | ||||||
|  | #
 | ||||||
|  | # Copyright (c) Berkus Decker <berkus+vesper@metta.systems>
 | ||||||
|  | #
 | ||||||
|  | [tasks.kernel-binary] | ||||||
|  | env = { "BINARY_FILE" = "${KERNEL_ELF}" } | ||||||
|  | run_task = "custom-binary" | ||||||
|  | 
 | ||||||
|  | [tasks.qemu] | ||||||
|  | env = { "QEMU_RUNNER_OPTS" = "${QEMU_SERIAL_OPTS}", "TARGET_DTB" = "${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/targets/bcm2710-rpi-3-b-plus.dtb" } | ||||||
|  | extend = "qemu-runner" | ||||||
|  | 
 | ||||||
|  | [tasks.qemu-cb] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.qemu-gdb] | ||||||
|  | env = { "QEMU_RUNNER_OPTS" = "${QEMU_SERIAL_OPTS} ${QEMU_GDB_OPTS}", "TARGET_DTB" = "${CARGO_MAKE_WORKSPACE_WORKING_DIRECTORY}/targets/bcm2710-rpi-3-b-plus.dtb" } | ||||||
|  | extend = "qemu-runner" | ||||||
|  | 
 | ||||||
|  | [tasks.zellij-nucleus] | ||||||
|  | env = { "KERNEL_BIN" = "${KERNEL_BIN}" } | ||||||
|  | run_task = "zellij-config" | ||||||
|  | 
 | ||||||
|  | [tasks.zellij-cb] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.zellij-cb-gdb] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.gdb-config] | ||||||
|  | script_runner = "@duckscript" | ||||||
|  | script = [ | ||||||
|  | ''' | ||||||
|  |     writefile ${GDB_CONNECT_FILE} "target extended-remote :5555\n" | ||||||
|  |     appendfile ${GDB_CONNECT_FILE} "break 0x80000\n" | ||||||
|  |     appendfile ${GDB_CONNECT_FILE} "continue\n" | ||||||
|  | ''' | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | # Problem: this eats STDIN! need to run in subshell or sth
 | ||||||
|  | [tasks.gdb] | ||||||
|  | dependencies = ["build", "kernel-binary", "gdb-config"] | ||||||
|  | env = { "RUST_GDB" = "${GDB}" } | ||||||
|  | script = [ | ||||||
|  |     "rust-gdb -x ${GDB_CONNECT_FILE} ${KERNEL_ELF}" | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [tasks.gdb-cb] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.install-nm] | ||||||
|  | install_crate = { crate_name = "cargo-binutils", binary = "rust-nm", test_arg = ["--help"] } | ||||||
|  | 
 | ||||||
|  | [tasks.install-rustfilt] | ||||||
|  | install_crate = { crate_name = "rustfilt", binary = "rustfilt", test_arg = ["--help"] } | ||||||
|  | 
 | ||||||
|  | [tasks.nm] | ||||||
|  | dependencies = ["build", "kernel-binary", "install-nm", "install-rustfilt"] | ||||||
|  | script = [ | ||||||
|  |     "${NM} -- ${KERNEL_ELF} | sort -k 1 | rustfilt" | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [tasks.sdcard] | ||||||
|  | dependencies = ["build", "kernel-binary"] | ||||||
|  | script_runner = "@duckscript" | ||||||
|  | script = [ | ||||||
|  | ''' | ||||||
|  |     kernelImage = set "kernel8.img" | ||||||
|  |     cp ${KERNEL_BIN} ${VOLUME}/${kernelImage} | ||||||
|  |     echo "Copied nucleus to ${VOLUME}/${kernelImage}" | ||||||
|  | ''' | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [tasks.cb-eject] | ||||||
|  | disabled = true | ||||||
|  | 
 | ||||||
|  | [tasks.hopper] | ||||||
|  | dependencies = ["build", "kernel-binary"] | ||||||
|  | # The cmd line below causes a bug in hopper, see https://www.dropbox.com/s/zyw5mfx0bepcjb1/hopperv4-RAW-bug.mov?dl=0
 | ||||||
|  | #"hopperv4 --loader RAW --base-address 0x80000 --entrypoint 0x80000 --file-offset 0 --plugin arm --cpu aarch64 --variant generic --contains-code true --executable ${KERNEL_BIN}"
 | ||||||
|  | script = [ | ||||||
|  |     "hopperv4 --loader ELF --executable ${KERNEL_ELF}" | ||||||
|  | ] | ||||||
|  | @ -0,0 +1,6 @@ | ||||||
|  | const LINKER_SCRIPT: &str = "linker/aarch64.ld"; | ||||||
|  | 
 | ||||||
|  | fn main() { | ||||||
|  |     println!("cargo:rerun-if-changed={}", LINKER_SCRIPT); | ||||||
|  |     println!("cargo:rustc-link-arg=--script={}", LINKER_SCRIPT); | ||||||
|  | } | ||||||
|  | @ -0,0 +1,259 @@ | ||||||
|  | /* | ||||||
|  |  * SPDX-License-Identifier: BlueOak-1.0.0 | ||||||
|  |  * Copyright (c) Berkus Decker <berkus+vesper@metta.systems> | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | //! Vesper single-address-space nanokernel.
 | ||||||
|  | //!
 | ||||||
|  | //! This crate implements the kernel binary proper.
 | ||||||
|  | 
 | ||||||
|  | #![no_std] | ||||||
|  | #![no_main] | ||||||
|  | #![feature(ptr_internals)] | ||||||
|  | #![feature(format_args_nl)] | ||||||
|  | #![feature(custom_test_frameworks)] | ||||||
|  | #![test_runner(machine::tests::test_runner)] | ||||||
|  | #![reexport_test_harness_main = "test_main"] | ||||||
|  | #![deny(missing_docs)] | ||||||
|  | #![deny(warnings)] | ||||||
|  | 
 | ||||||
|  | #[cfg(not(test))] | ||||||
|  | use core::panic::PanicInfo; | ||||||
|  | #[allow(unused_imports)] | ||||||
|  | use machine::devices::SerialOps; | ||||||
|  | use { | ||||||
|  |     cfg_if::cfg_if, | ||||||
|  |     machine::{ | ||||||
|  |         arch, entry, memory, | ||||||
|  |         platform::rpi3::{ | ||||||
|  |             display::{Color, DrawError}, | ||||||
|  |             mailbox::{channel, Mailbox, MailboxOps}, | ||||||
|  |             vc::VC, | ||||||
|  |         }, | ||||||
|  |         println, CONSOLE, | ||||||
|  |     }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | entry!(kmain); | ||||||
|  | 
 | ||||||
|  | #[cfg(not(test))] | ||||||
|  | #[panic_handler] | ||||||
|  | fn panicked(info: &PanicInfo) -> ! { | ||||||
|  |     machine::panic::handler(info) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn print_mmu_state_and_features() { | ||||||
|  |     memory::mmu::print_features(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn init_mmu() { | ||||||
|  |     unsafe { | ||||||
|  |         memory::mmu::init().unwrap(); | ||||||
|  |     } | ||||||
|  |     println!("[!] MMU initialised"); | ||||||
|  |     print_mmu_state_and_features(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn init_exception_traps() { | ||||||
|  |     extern "C" { | ||||||
|  |         static __exception_vectors_start: u64; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     unsafe { | ||||||
|  |         let exception_vectors_start: u64 = &__exception_vectors_start as *const _ as u64; | ||||||
|  | 
 | ||||||
|  |         arch::traps::set_vbar_el1_checked(exception_vectors_start) | ||||||
|  |             .expect("Vector table properly aligned!"); | ||||||
|  |     } | ||||||
|  |     println!("[!] Exception traps set up"); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(not(feature = "noserial"))] | ||||||
|  | fn init_uart_serial() { | ||||||
|  |     use machine::platform::rpi3::{gpio::GPIO, mini_uart::MiniUart, pl011_uart::PL011Uart}; | ||||||
|  | 
 | ||||||
|  |     let gpio = GPIO::default(); | ||||||
|  |     let uart = MiniUart::default(); | ||||||
|  |     let uart = uart.prepare(&gpio); | ||||||
|  |     CONSOLE.lock(|c| { | ||||||
|  |         // Move uart into the global CONSOLE.
 | ||||||
|  |         c.replace_with(uart.into()); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     println!("[0] MiniUART is live!"); | ||||||
|  | 
 | ||||||
|  |     // Then immediately switch to PL011 (just as an example)
 | ||||||
|  | 
 | ||||||
|  |     let uart = PL011Uart::default(); | ||||||
|  | 
 | ||||||
|  |     // uart.init() will reconfigure the GPIO, which causes a race against
 | ||||||
|  |     // the MiniUart that is still putting out characters on the physical
 | ||||||
|  |     // line that are already buffered in its TX FIFO.
 | ||||||
|  |     //
 | ||||||
|  |     // To ensure the CPU doesn't rewire the GPIO before the MiniUart has put
 | ||||||
|  |     // its last character, explicitly flush it before rewiring.
 | ||||||
|  |     //
 | ||||||
|  |     // If you switch to an output that happens to not use the same pair of
 | ||||||
|  |     // physical wires (e.g. the Framebuffer), you don't need to do this,
 | ||||||
|  |     // because flush() is anyways called implicitly by replace_with(). This
 | ||||||
|  |     // is just a special case.
 | ||||||
|  |     CONSOLE.lock(|c| c.flush()); | ||||||
|  | 
 | ||||||
|  |     match uart.prepare(&gpio) { | ||||||
|  |         Ok(uart) => { | ||||||
|  |             CONSOLE.lock(|c| { | ||||||
|  |                 // Move uart into the global CONSOLE.
 | ||||||
|  |                 c.replace_with(uart.into()); | ||||||
|  |             }); | ||||||
|  |             println!("[0] UART0 is live!"); | ||||||
|  |         } | ||||||
|  |         Err(_) => println!("[0] Error switching to PL011 UART, continue with MiniUART"), | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Kernel entry point.
 | ||||||
|  | /// `arch` crate is responsible for calling it.
 | ||||||
|  | #[inline] | ||||||
|  | pub fn kmain() -> ! { | ||||||
|  |     #[cfg(feature = "jtag")] | ||||||
|  |     machine::arch::jtag::wait_debugger(); | ||||||
|  | 
 | ||||||
|  |     init_mmu(); | ||||||
|  |     init_exception_traps(); | ||||||
|  | 
 | ||||||
|  |     #[cfg(not(feature = "noserial"))] | ||||||
|  |     init_uart_serial(); | ||||||
|  | 
 | ||||||
|  |     #[cfg(test)] | ||||||
|  |     test_main(); | ||||||
|  | 
 | ||||||
|  |     command_prompt(); | ||||||
|  | 
 | ||||||
|  |     reboot() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | //------------------------------------------------------------
 | ||||||
|  | // Start a command prompt
 | ||||||
|  | //------------------------------------------------------------
 | ||||||
|  | fn command_prompt() { | ||||||
|  |     'cmd_loop: loop { | ||||||
|  |         let mut buf = [0u8; 64]; | ||||||
|  | 
 | ||||||
|  |         match CONSOLE.lock(|c| c.command_prompt(&mut buf)) { | ||||||
|  |             b"mmu" => init_mmu(), | ||||||
|  |             b"feats" => print_mmu_state_and_features(), | ||||||
|  |             #[cfg(not(feature = "noserial"))] | ||||||
|  |             b"uart" => init_uart_serial(), | ||||||
|  |             b"disp" => check_display_init(), | ||||||
|  |             b"trap" => check_data_abort_trap(), | ||||||
|  |             b"map" => machine::arch::memory::print_layout(), | ||||||
|  |             b"led on" => set_led(true), | ||||||
|  |             b"led off" => set_led(false), | ||||||
|  |             b"help" => print_help(), | ||||||
|  |             b"end" => break 'cmd_loop, | ||||||
|  |             x => println!("[!] Unknown command {:?}, try 'help'", x), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn print_help() { | ||||||
|  |     println!("Supported console commands:"); | ||||||
|  |     println!("  mmu  - initialize MMU"); | ||||||
|  |     println!("  feats - print MMU state and supported features"); | ||||||
|  |     #[cfg(not(feature = "noserial"))] | ||||||
|  |     println!("  uart - try to reinitialize UART serial"); | ||||||
|  |     println!("  disp - try to init VC framebuffer and draw some text"); | ||||||
|  |     println!("  trap - trigger and recover from a data abort exception"); | ||||||
|  |     println!("  map  - show kernel memory layout"); | ||||||
|  |     println!("  led [on|off]  - change RPi LED status"); | ||||||
|  |     println!("  end  - leave console and reset board"); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn set_led(enable: bool) { | ||||||
|  |     let mut mbox = Mailbox::<8>::default(); | ||||||
|  |     let index = mbox.request(); | ||||||
|  |     let index = mbox.set_led_on(index, enable); | ||||||
|  |     let mbox = mbox.end(index); | ||||||
|  | 
 | ||||||
|  |     mbox.call(channel::PropertyTagsArmToVc) | ||||||
|  |         .map_err(|e| { | ||||||
|  |             println!("Mailbox call returned error {}", e); | ||||||
|  |             println!("Mailbox contents: {:?}", mbox); | ||||||
|  |         }) | ||||||
|  |         .ok(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn reboot() -> ! { | ||||||
|  |     cfg_if! { | ||||||
|  |         if #[cfg(feature = "qemu")] { | ||||||
|  |             println!("Bye, shutting down QEMU"); | ||||||
|  |             machine::qemu::semihosting::exit_success() | ||||||
|  |         } else { | ||||||
|  |             use machine::platform::rpi3::power::Power; | ||||||
|  | 
 | ||||||
|  |             println!("Bye, going to reset now"); | ||||||
|  |             Power::default().reset() | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn check_display_init() { | ||||||
|  |     display_graphics() | ||||||
|  |         .map_err(|e| { | ||||||
|  |             println!("Error in display: {}", e); | ||||||
|  |         }) | ||||||
|  |         .ok(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn display_graphics() -> Result<(), DrawError> { | ||||||
|  |     if let Ok(mut display) = VC::init_fb(800, 600, 32) { | ||||||
|  |         println!("Display created"); | ||||||
|  | 
 | ||||||
|  |         display.clear(Color::black()); | ||||||
|  |         println!("Display cleared"); | ||||||
|  | 
 | ||||||
|  |         display.rect(10, 10, 250, 250, Color::rgb(32, 96, 64)); | ||||||
|  |         display.draw_text(50, 50, "Hello there!", Color::rgb(128, 192, 255))?; | ||||||
|  | 
 | ||||||
|  |         let mut buf = [0u8; 64]; | ||||||
|  |         let s = machine::write_to::show(&mut buf, format_args!("Display width {}", display.width)); | ||||||
|  | 
 | ||||||
|  |         if s.is_err() { | ||||||
|  |             display.draw_text(50, 150, "Error displaying", Color::red())? | ||||||
|  |         } else { | ||||||
|  |             display.draw_text(50, 150, s.unwrap(), Color::white())? | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         display.draw_text(150, 50, "RED", Color::red())?; | ||||||
|  |         display.draw_text(160, 60, "GREEN", Color::green())?; | ||||||
|  |         display.draw_text(170, 70, "BLUE", Color::blue())?; | ||||||
|  |     } | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn check_data_abort_trap() { | ||||||
|  |     // Cause an exception by accessing a virtual address for which no
 | ||||||
|  |     // address translations have been set up.
 | ||||||
|  |     //
 | ||||||
|  |     // This line of code accesses the address 3 GiB, but page tables are
 | ||||||
|  |     // only set up for the range [0..1) GiB.
 | ||||||
|  |     let big_addr: u64 = 3 * 1024 * 1024 * 1024; | ||||||
|  |     unsafe { core::ptr::read_volatile(big_addr as *mut u64) }; | ||||||
|  | 
 | ||||||
|  |     println!("[i] Whoa! We recovered from an exception."); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg(test)] | ||||||
|  | mod main_tests { | ||||||
|  |     use {super::*, core::panic::PanicInfo}; | ||||||
|  | 
 | ||||||
|  |     #[panic_handler] | ||||||
|  |     fn panicked(info: &PanicInfo) -> ! { | ||||||
|  |         machine::panic::handler_for_tests(info) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[test_case] | ||||||
|  |     fn test_data_abort_trap() { | ||||||
|  |         check_data_abort_trap() | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -1,32 +1,34 @@ | ||||||
| # Broadcom 2837 on Raspberry Pi 3 as JTAG target | # Broadcom bcm2837 on Raspberry Pi 3 as JTAG target | ||||||
| # From https://www.suse.com/c/debugging-raspberry-pi-3-with-jtag/ | # From https://www.suse.com/c/debugging-raspberry-pi-3-with-jtag/ | ||||||
| 
 | 
 | ||||||
| telnet_port 4444 | echo "Booting JTAG for Raspberry Pi 3" | ||||||
| gdb_port 5555 |  | ||||||
| 
 |  | ||||||
| transport select jtag |  | ||||||
| 
 |  | ||||||
| #reset_config trst_only |  | ||||||
| 
 |  | ||||||
| #adapter_khz 500 |  | ||||||
| #adapter_nsrst_delay 100 |  | ||||||
| #jtag_ntrst_delay 100 |  | ||||||
| 
 | 
 | ||||||
| if { [info exists CHIPNAME] } { | if { [info exists CHIPNAME] } { | ||||||
|     set _CHIPNAME $CHIPNAME |     set _CHIPNAME $CHIPNAME | ||||||
| } else { | } else { | ||||||
|     set _CHIPNAME rpi3 |     set _CHIPNAME bcm2837 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| # |  | ||||||
| # Main DAP |  | ||||||
| # |  | ||||||
| if { [info exists DAP_TAPID] } { | if { [info exists DAP_TAPID] } { | ||||||
|     set _DAP_TAPID $DAP_TAPID |     set _DAP_TAPID $DAP_TAPID | ||||||
| } else { | } else { | ||||||
|     set _DAP_TAPID 0x4ba00477 |     set _DAP_TAPID 0x4ba00477 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | adapter speed 4000 | ||||||
|  | 
 | ||||||
|  | transport select jtag | ||||||
|  | # we need to enable srst even though we don't connect it | ||||||
|  | reset_config trst_and_srst | ||||||
|  | 
 | ||||||
|  | jtag_ntrst_delay 500 | ||||||
|  | 
 | ||||||
|  | telnet_port 4444 | ||||||
|  | gdb_port 5555 | ||||||
|  | 
 | ||||||
|  | # | ||||||
|  | # Main DAP | ||||||
|  | # | ||||||
| jtag newtap $_CHIPNAME tap -irlen 4 -ircapture 0x1 -irmask 0xf -expected-id $_DAP_TAPID -enable | jtag newtap $_CHIPNAME tap -irlen 4 -ircapture 0x1 -irmask 0xf -expected-id $_DAP_TAPID -enable | ||||||
| dap create $_CHIPNAME.dap -chain-position $_CHIPNAME.tap | dap create $_CHIPNAME.dap -chain-position $_CHIPNAME.tap | ||||||
| 
 | 
 | ||||||
|  | @ -35,17 +37,28 @@ set _CTINAME $_CHIPNAME.cti | ||||||
| 
 | 
 | ||||||
| set DBGBASE {0x80010000 0x80012000 0x80014000 0x80016000} | set DBGBASE {0x80010000 0x80012000 0x80014000 0x80016000} | ||||||
| set CTIBASE {0x80018000 0x80019000 0x8001a000 0x8001b000} | set CTIBASE {0x80018000 0x80019000 0x8001a000 0x8001b000} | ||||||
|  | 
 | ||||||
| set _cores 4 | set _cores 4 | ||||||
|  | set _smp_command "" | ||||||
| 
 | 
 | ||||||
| for { set _core 0 } { $_core < $_cores } { incr _core } { | for { set _core 0 } { $_core < $_cores } { incr _core } { | ||||||
| 
 | 
 | ||||||
|     cti create $_CTINAME.$_core -dap $_CHIPNAME.dap -ap-num 0 \ |     cti create $_CTINAME.$_core -dap $_CHIPNAME.dap -ap-num 0 \ | ||||||
|         -ctibase [lindex $CTIBASE $_core] |         -baseaddr [lindex $CTIBASE $_core] | ||||||
| 
 | 
 | ||||||
|     target create $_TARGETNAME.$_core aarch64 \ |     target create $_TARGETNAME.$_core aarch64 \ | ||||||
|         -dap $_CHIPNAME.dap -coreid $_core \ |         -dap $_CHIPNAME.dap -coreid $_core \ | ||||||
|         -dbgbase [lindex $DBGBASE $_core] -cti $_CTINAME.$_core |         -dbgbase [lindex $DBGBASE $_core] -cti $_CTINAME.$_core | ||||||
| 
 | 
 | ||||||
|  |     if {$_core != 0} { | ||||||
|  |         set _smp_command "$_smp_command ${_TARGETNAME}.${_core}" | ||||||
|  |     } else { | ||||||
|  |         set _smp_command "target smp ${_TARGETNAME}.${_core}" | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     $_TARGETNAME.$_core configure -event reset-assert-post "aarch64 dbginit" |     $_TARGETNAME.$_core configure -event reset-assert-post "aarch64 dbginit" | ||||||
|     $_TARGETNAME.$_core configure -event gdb-attach { halt } |     $_TARGETNAME.$_core configure -event gdb-attach { halt } | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | eval $_smp_command | ||||||
|  | targets $_TARGETNAME.0 | ||||||
|  | @ -0,0 +1,64 @@ | ||||||
|  | # Broadcom bcm2711 on Raspberry Pi 4 as JTAG target | ||||||
|  | # From https://gist.github.com/tnishinaga/46a3380e1f47f5e892bbb74e55b3cf3e | ||||||
|  | # See also https://xihan94.gitbook.io/raspberry-pi/raspberry-pi-4-bringup | ||||||
|  | 
 | ||||||
|  | echo "Booting JTAG for Raspberry Pi 4" | ||||||
|  | 
 | ||||||
|  | if { [info exists CHIPNAME] } { | ||||||
|  |     set _CHIPNAME $CHIPNAME | ||||||
|  | } else { | ||||||
|  |     set _CHIPNAME bcm2711 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | if { [info exists DAP_TAPID] } { | ||||||
|  |     set _DAP_TAPID $DAP_TAPID | ||||||
|  | } else { | ||||||
|  |     set _DAP_TAPID 0x4ba00477 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | adapter speed 4000 | ||||||
|  | 
 | ||||||
|  | transport select jtag | ||||||
|  | # we need to enable srst even though we don't connect it | ||||||
|  | reset_config trst_and_srst | ||||||
|  | 
 | ||||||
|  | jtag_ntrst_delay 500 | ||||||
|  | 
 | ||||||
|  | telnet_port 4444 | ||||||
|  | gdb_port 5555 | ||||||
|  | 
 | ||||||
|  | # | ||||||
|  | # Main DAP | ||||||
|  | # | ||||||
|  | jtag newtap $_CHIPNAME tap -irlen 4 -expected-id $_DAP_TAPID | ||||||
|  | dap create $_CHIPNAME.dap -chain-position $_CHIPNAME.tap | ||||||
|  | 
 | ||||||
|  | set _TARGETNAME $_CHIPNAME.a72 | ||||||
|  | set _CTINAME $_CHIPNAME.cti | ||||||
|  | 
 | ||||||
|  | set DBGBASE {0x80410000 0x80510000 0x80610000 0x80710000} | ||||||
|  | set CTIBASE {0x80420000 0x80520000 0x80620000 0x80720000} | ||||||
|  | 
 | ||||||
|  | set _cores 4 | ||||||
|  | set _smp_command "" | ||||||
|  | 
 | ||||||
|  | for { set _core 0 } { $_core < $_cores } { incr _core } { | ||||||
|  |     cti create $_CTINAME.$_core -dap $_CHIPNAME.dap -ap-num 0 \ | ||||||
|  |         -baseaddr [lindex $CTIBASE $_core] | ||||||
|  | 
 | ||||||
|  |     target create ${_TARGETNAME}.${_core} aarch64 \ | ||||||
|  |         -dap ${_CHIPNAME}.dap -coreid $_core \ | ||||||
|  |         -dbgbase [lindex $DBGBASE $_core] -cti ${_CTINAME}.${_core} | ||||||
|  | 
 | ||||||
|  |     if {$_core != 0} { | ||||||
|  |         set _smp_command "$_smp_command ${_TARGETNAME}.${_core}" | ||||||
|  |     } else { | ||||||
|  |         set _smp_command "target smp ${_TARGETNAME}.${_core}" | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     $_TARGETNAME.$_core configure -event reset-assert-post "aarch64 dbginit" | ||||||
|  |     $_TARGETNAME.$_core configure -event gdb-attach { halt } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | eval $_smp_command | ||||||
|  | targets $_TARGETNAME.0 | ||||||
							
								
								
									
										3
									
								
								qemu.sh
								
								
								
								
							
							
						
						
									
										3
									
								
								qemu.sh
								
								
								
								
							|  | @ -1,3 +0,0 @@ | ||||||
| #!/bin/sh |  | ||||||
| cargo xbuild --target=targets/aarch64-vesper-metta.json --release && \ |  | ||||||
| sh .cargo/runscript.sh target/aarch64-vesper-metta/release/vesper |  | ||||||
|  | @ -1 +1,4 @@ | ||||||
| nightly | [toolchain] | ||||||
|  | channel = "nightly" | ||||||
|  | components = ["clippy", "llvm-tools-preview", "rust-src", "rustfmt"] | ||||||
|  | targets = ["aarch64-unknown-none-softfloat"] | ||||||
|  |  | ||||||
|  | @ -0,0 +1 @@ | ||||||
|  | imports_granularity = "One" | ||||||
|  | @ -1,150 +0,0 @@ | ||||||
| /* |  | ||||||
|  * MIT License |  | ||||||
|  * |  | ||||||
|  * Copyright (c) 2018 Jorge Aparicio |  | ||||||
|  * Copyright (c) 2018-2019 Andre Richter <andre.o.richter@gmail.com> |  | ||||||
|  * Copyright (c) 2019 Berkus Decker <berkus+vesper@metta.systems> |  | ||||||
|  * |  | ||||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy |  | ||||||
|  * of this software and associated documentation files (the "Software"), to deal |  | ||||||
|  * in the Software without restriction, including without limitation the rights |  | ||||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |  | ||||||
|  * copies of the Software, and to permit persons to whom the Software is |  | ||||||
|  * furnished to do so, subject to the following conditions: |  | ||||||
|  * |  | ||||||
|  * The above copyright notice and this permission notice shall be included in all |  | ||||||
|  * copies or substantial portions of the Software. |  | ||||||
|  * |  | ||||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |  | ||||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |  | ||||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |  | ||||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |  | ||||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |  | ||||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |  | ||||||
|  * SOFTWARE. |  | ||||||
|  */ |  | ||||||
| 
 |  | ||||||
| #![deny(missing_docs)] |  | ||||||
| #![deny(warnings)] |  | ||||||
| 
 |  | ||||||
| //! Low-level boot of the Raspberry's processor
 |  | ||||||
| //! http://infocenter.arm.com/help/topic/com.arm.doc.dai0527a/DAI0527A_baremetal_boot_code_for_ARMv8_A_processors.pdf
 |  | ||||||
| 
 |  | ||||||
| extern crate panic_abort; |  | ||||||
| 
 |  | ||||||
| /// Type check the user-supplied entry function.
 |  | ||||||
| #[macro_export] |  | ||||||
| macro_rules! entry { |  | ||||||
|     ($path:path) => { |  | ||||||
|         #[export_name = "main"] |  | ||||||
|         pub unsafe fn __main() -> ! { |  | ||||||
|             // type check the given path
 |  | ||||||
|             let f: fn() -> ! = $path; |  | ||||||
| 
 |  | ||||||
|             f() |  | ||||||
|         } |  | ||||||
|     }; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /// Reset function.
 |  | ||||||
| ///
 |  | ||||||
| /// Initializes the bss section before calling into the user's `main()`.
 |  | ||||||
| unsafe fn reset() -> ! { |  | ||||||
|     extern "C" { |  | ||||||
|         // Boundaries of the .bss section, provided by the linker script
 |  | ||||||
|         static mut __bss_start: u64; |  | ||||||
|         static mut __bss_end: u64; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     use cortex_a::regs::*; |  | ||||||
|     const STACK_START: u64 = 0x80_000; |  | ||||||
|     SP.set(STACK_START); |  | ||||||
| 
 |  | ||||||
|     // Zeroes the .bss section
 |  | ||||||
|     r0::zero_bss(&mut __bss_start, &mut __bss_end); |  | ||||||
| 
 |  | ||||||
|     extern "Rust" { |  | ||||||
|         fn main() -> !; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     main() |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /// Prepare and execute transition from EL2 to EL1.
 |  | ||||||
| #[inline] |  | ||||||
| fn setup_and_enter_el1_from_el2() -> ! { |  | ||||||
|     use cortex_a::{asm, regs::*}; |  | ||||||
| 
 |  | ||||||
|     const STACK_START: u64 = 0x80_000; |  | ||||||
| 
 |  | ||||||
|     // Enable timer counter registers for EL1
 |  | ||||||
|     CNTHCTL_EL2.write(CNTHCTL_EL2::EL1PCEN::SET + CNTHCTL_EL2::EL1PCTEN::SET); |  | ||||||
| 
 |  | ||||||
|     // No virtual offset for reading the counters
 |  | ||||||
|     CNTVOFF_EL2.set(0); |  | ||||||
| 
 |  | ||||||
|     // Set EL1 execution state to AArch64
 |  | ||||||
|     // TODO: Explain the SWIO bit (SWIO hardwired on Pi3)
 |  | ||||||
|     HCR_EL2.write(HCR_EL2::RW::EL1IsAarch64 + HCR_EL2::SWIO::SET); |  | ||||||
| 
 |  | ||||||
|     // Set up a simulated exception return.
 |  | ||||||
|     //
 |  | ||||||
|     // First, fake a saved program status, where all interrupts were
 |  | ||||||
|     // masked and SP_EL1 was used as a stack pointer.
 |  | ||||||
|     SPSR_EL2.write( |  | ||||||
|         SPSR_EL2::D::Masked |  | ||||||
|             + SPSR_EL2::A::Masked |  | ||||||
|             + SPSR_EL2::I::Masked |  | ||||||
|             + SPSR_EL2::F::Masked |  | ||||||
|             + SPSR_EL2::M::EL1h, // Use SP_EL1
 |  | ||||||
|     ); |  | ||||||
| 
 |  | ||||||
|     // Second, let the link register point to reset().
 |  | ||||||
|     ELR_EL2.set(reset as *const () as u64); |  | ||||||
| 
 |  | ||||||
|     // Set up SP_EL1 (stack pointer), which will be used by EL1 once
 |  | ||||||
|     // we "return" to it.
 |  | ||||||
|     SP_EL1.set(STACK_START); |  | ||||||
| 
 |  | ||||||
|     // Use `eret` to "return" to EL1. This will result in execution of
 |  | ||||||
|     // `reset()` in EL1.
 |  | ||||||
|     asm::eret() |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // Processors enter EL3 after reset.
 |  | ||||||
| // ref: http://infocenter.arm.com/help/topic/com.arm.doc.dai0527a/DAI0527A_baremetal_boot_code_for_ARMv8_A_processors.pdf
 |  | ||||||
| // section: 5.5.1
 |  | ||||||
| // However, GPU init code must be switching it down to EL2?
 |  | ||||||
| 
 |  | ||||||
| /// Entrypoint of the processor.
 |  | ||||||
| ///
 |  | ||||||
| /// Parks all cores except core0 and checks if we started in EL2. If
 |  | ||||||
| /// so, proceeds with setting up EL1.
 |  | ||||||
| ///
 |  | ||||||
| /// This is invoked from the linker script, does arch-specific init
 |  | ||||||
| /// and passes control to the kernel boot function reset().
 |  | ||||||
| #[link_section = ".text.boot"] |  | ||||||
| #[no_mangle] |  | ||||||
| pub unsafe extern "C" fn _boot_cores() -> ! { |  | ||||||
|     use cortex_a::{asm, regs::*}; |  | ||||||
| 
 |  | ||||||
|     // crate::arch::aarch64::jtag_dbg_wait();
 |  | ||||||
| 
 |  | ||||||
|     const CORE_0: u64 = 0; |  | ||||||
|     const CORE_MASK: u64 = 0x3; |  | ||||||
|     const EL1: u32 = CurrentEL::EL::EL1.value; |  | ||||||
|     const EL2: u32 = CurrentEL::EL::EL2.value; |  | ||||||
| 
 |  | ||||||
|     if CORE_0 == MPIDR_EL1.get() & CORE_MASK { |  | ||||||
|         if EL2 == CurrentEL.get() { |  | ||||||
|             setup_and_enter_el1_from_el2() |  | ||||||
|         } else if EL1 == CurrentEL.get() { |  | ||||||
|             reset() |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // if not core0 or EL2/EL1, infinitely wait for events
 |  | ||||||
|     loop { |  | ||||||
|         asm::wfe(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,103 +0,0 @@ | ||||||
| use super::{Frame, FrameAllocator}; |  | ||||||
| use multiboot2::{MemoryArea, MemoryAreaIter}; // replace with DTB?
 |  | ||||||
| 
 |  | ||||||
| pub struct AreaFrameAllocator { |  | ||||||
|     next_free_frame: Frame, |  | ||||||
|     current_area: Option<&'static MemoryArea>, |  | ||||||
|     areas: MemoryAreaIter, |  | ||||||
|     kernel_start: Frame, |  | ||||||
|     kernel_end: Frame, |  | ||||||
|     multiboot_start: Frame, |  | ||||||
|     multiboot_end: Frame, |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl FrameAllocator for AreaFrameAllocator { |  | ||||||
|     fn allocate_frame(&mut self) -> Option<Frame> { |  | ||||||
|         if let Some(_area) = self.current_area { |  | ||||||
|             // "Clone" the frame to return it if it's free. Frame doesn't
 |  | ||||||
|             // implement Clone, but we can construct an identical frame.
 |  | ||||||
|             let frame = Frame { |  | ||||||
|                 number: self.next_free_frame.number, |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             // the last frame of the current area
 |  | ||||||
|             let current_area_last_frame = Frame::containing_address(0x3f00_0000); |  | ||||||
|             // {
 |  | ||||||
|             //     let address = area.base_addr + area.length - 1;
 |  | ||||||
|             //     Frame::containing_address(address as usize)
 |  | ||||||
|             // };
 |  | ||||||
| 
 |  | ||||||
|             if frame > current_area_last_frame { |  | ||||||
|                 // all frames of current area are used, switch to next area
 |  | ||||||
|                 // self.choose_next_area();
 |  | ||||||
|                 unimplemented!(); |  | ||||||
|             } else if frame >= self.kernel_start && frame <= self.kernel_end { |  | ||||||
|                 // `frame` is used by the kernel
 |  | ||||||
|                 self.next_free_frame = Frame { |  | ||||||
|                     number: self.kernel_end.number + 1, |  | ||||||
|                 }; |  | ||||||
|             } else if frame >= self.multiboot_start && frame <= self.multiboot_end { |  | ||||||
|                 // `frame` is used by the multiboot information structure
 |  | ||||||
|                 self.next_free_frame = Frame { |  | ||||||
|                     number: self.multiboot_end.number + 1, |  | ||||||
|                 }; |  | ||||||
|             } else { |  | ||||||
|                 // frame is unused, increment `next_free_frame` and return it
 |  | ||||||
|                 self.next_free_frame.number += 1; |  | ||||||
|                 return Some(frame); |  | ||||||
|             } |  | ||||||
|             // `frame` was not valid, try it again with the updated `next_free_frame`
 |  | ||||||
|             self.allocate_frame() |  | ||||||
|         } else { |  | ||||||
|             None // no free frames left
 |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn deallocate_frame(&mut self, _frame: Frame) { |  | ||||||
|         unimplemented!() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // Fixme: no multiboot, but dtb instead with avail memory regions
 |  | ||||||
| // Need dtb parser here!
 |  | ||||||
| 
 |  | ||||||
| impl AreaFrameAllocator { |  | ||||||
|     pub fn new( |  | ||||||
|         kernel_start: usize, |  | ||||||
|         kernel_end: usize, |  | ||||||
|         multiboot_start: usize, |  | ||||||
|         multiboot_end: usize, |  | ||||||
|         memory_areas: MemoryAreaIter, |  | ||||||
|     ) -> AreaFrameAllocator { |  | ||||||
|         let mut allocator = AreaFrameAllocator { |  | ||||||
|             next_free_frame: Frame::containing_address(0), |  | ||||||
|             current_area: None, |  | ||||||
|             areas: memory_areas, |  | ||||||
|             kernel_start: Frame::containing_address(kernel_start), |  | ||||||
|             kernel_end: Frame::containing_address(kernel_end), |  | ||||||
|             multiboot_start: Frame::containing_address(multiboot_start), |  | ||||||
|             multiboot_end: Frame::containing_address(multiboot_end), |  | ||||||
|         }; |  | ||||||
|         // allocator.choose_next_area();
 |  | ||||||
|         allocator.next_free_frame = Frame::containing_address(0x100000); // start from 1Mb
 |  | ||||||
|         allocator |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn choose_next_area(&mut self) { |  | ||||||
|         self.current_area = self |  | ||||||
|             .areas |  | ||||||
|             .clone() |  | ||||||
|             .filter(|area| { |  | ||||||
|                 let address = area.base_addr + area.length - 1; |  | ||||||
|                 Frame::containing_address(address as usize) >= self.next_free_frame |  | ||||||
|             }) |  | ||||||
|             .min_by_key(|area| area.base_addr); |  | ||||||
| 
 |  | ||||||
|         if let Some(area) = self.current_area { |  | ||||||
|             let start_frame = Frame::containing_address(area.base_addr as usize); |  | ||||||
|             if self.next_free_frame < start_frame { |  | ||||||
|                 self.next_free_frame = start_frame; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,100 +0,0 @@ | ||||||
| /* |  | ||||||
|  * MIT License |  | ||||||
|  * |  | ||||||
|  * Copyright (c) 2019 Andre Richter <andre.o.richter@gmail.com> |  | ||||||
|  * |  | ||||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy |  | ||||||
|  * of this software and associated documentation files (the "Software"), to deal |  | ||||||
|  * in the Software without restriction, including without limitation the rights |  | ||||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |  | ||||||
|  * copies of the Software, and to permit persons to whom the Software is |  | ||||||
|  * furnished to do so, subject to the following conditions: |  | ||||||
|  * |  | ||||||
|  * The above copyright notice and this permission notice shall be included in all |  | ||||||
|  * copies or substantial portions of the Software. |  | ||||||
|  * |  | ||||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |  | ||||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |  | ||||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |  | ||||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |  | ||||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |  | ||||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |  | ||||||
|  * SOFTWARE. |  | ||||||
|  */ |  | ||||||
| 
 |  | ||||||
| use crate::println; |  | ||||||
| use core::alloc::{Alloc, AllocErr, Layout}; |  | ||||||
| use core::mem; |  | ||||||
| use core::ptr::NonNull; |  | ||||||
| use core::slice; |  | ||||||
| 
 |  | ||||||
| pub struct BumpAllocator { |  | ||||||
|     next: usize, |  | ||||||
|     pool_end: usize, |  | ||||||
|     name: &'static str, |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| unsafe impl Alloc for BumpAllocator { |  | ||||||
|     unsafe fn alloc(&mut self, layout: Layout) -> Result<NonNull<u8>, AllocErr> { |  | ||||||
|         let start = crate::memory::aligned_addr_unchecked(self.next, layout.align()); |  | ||||||
|         let end = start + layout.size(); |  | ||||||
| 
 |  | ||||||
|         if end <= self.pool_end { |  | ||||||
|             self.next = end; |  | ||||||
| 
 |  | ||||||
|             println!( |  | ||||||
|                 "[i] {}:\n      Allocated Addr {:#010X} Size {:#X}", |  | ||||||
|                 self.name, |  | ||||||
|                 start, |  | ||||||
|                 layout.size() |  | ||||||
|             ); |  | ||||||
| 
 |  | ||||||
|             Ok(NonNull::new_unchecked(start as *mut u8)) |  | ||||||
|         } else { |  | ||||||
|             Err(AllocErr) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // A bump allocator doesn't care
 |  | ||||||
|     unsafe fn dealloc(&mut self, _ptr: NonNull<u8>, _layout: Layout) {} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl BumpAllocator { |  | ||||||
|     pub const fn new(pool_start: usize, pool_end: usize, name: &'static str) -> Self { |  | ||||||
|         Self { |  | ||||||
|             next: pool_start, |  | ||||||
|             pool_end, |  | ||||||
|             name, |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// Allocate a zeroed slice
 |  | ||||||
|     pub fn alloc_slice_zeroed<'a, T>( |  | ||||||
|         &mut self, |  | ||||||
|         count_of_items: usize, |  | ||||||
|         alignment: usize, |  | ||||||
|     ) -> Result<&'a mut [T], ()> { |  | ||||||
|         let l; |  | ||||||
|         let size_in_byte = count_of_items * mem::size_of::<T>(); |  | ||||||
|         match Layout::from_size_align(size_in_byte, alignment) { |  | ||||||
|             Ok(layout) => l = layout, |  | ||||||
| 
 |  | ||||||
|             Err(_) => { |  | ||||||
|                 println!("[e] Layout Error!"); |  | ||||||
|                 return Err(()); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         let ptr; |  | ||||||
|         match unsafe { self.alloc_zeroed(l) } { |  | ||||||
|             Ok(i) => ptr = i.as_ptr(), |  | ||||||
| 
 |  | ||||||
|             Err(_) => { |  | ||||||
|                 println!("[e] Layout Error!"); |  | ||||||
|                 return Err(()); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         Ok(unsafe { slice::from_raw_parts_mut(ptr as *mut T, count_of_items) }) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,51 +0,0 @@ | ||||||
| use super::super::Frame; |  | ||||||
| 
 |  | ||||||
| pub struct Entry(u64); |  | ||||||
| 
 |  | ||||||
| bitflags! { |  | ||||||
|     pub struct EntryFlags: u64 { |  | ||||||
|         const VALID          = 1 <<  0; |  | ||||||
|         const TABLE          = 1 <<  1; // If set, a table entry, otherwise block entry
 |  | ||||||
|         // ATTR_INDEX_MASK = 7 << 2;
 |  | ||||||
|         const NON_SECURE     = 1 <<  5; // block/page descriptor lower attributes
 |  | ||||||
|         const ACCESS         = 1 << 10; // block/page descriptor lower attributes
 |  | ||||||
|         const NOT_GLOBAL     = 1 << 11; // nG, block/page descriptor lower attributes
 |  | ||||||
|         const DIRTY          = 1 << 51; // block/page descriptor upper attributes
 |  | ||||||
|         const CONTIGUOUS     = 1 << 52; // block/page descriptor upper attributes
 |  | ||||||
|         const EL1_EXEC_NEVER = 1 << 53; // block/page descriptor upper attributes
 |  | ||||||
|         const EXEC_NEVER     = 1 << 54; // block/page descriptor upper attributes
 |  | ||||||
|         const PXN_TABLE      = 1 << 59; // table descriptor, next level table attributes
 |  | ||||||
|         const XN_TABLE       = 1 << 60; // table descriptor, next level table attributes
 |  | ||||||
|         const AP_TABLE       = 1 << 61; // table descriptor, next level table attributes, 2 bits
 |  | ||||||
|         const NS_TABLE       = 1 << 63; // table descriptor, next level table attributes
 |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl Entry { |  | ||||||
|     pub fn is_unused(&self) -> bool { |  | ||||||
|         self.0 == 0 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub fn set_unused(&mut self) { |  | ||||||
|         self.0 = 0; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub fn flags(&self) -> EntryFlags { |  | ||||||
|         EntryFlags::from_bits_truncate(self.0) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub fn pointed_frame(&self) -> Option<Frame> { |  | ||||||
|         if self.flags().contains(EntryFlags::VALID) { |  | ||||||
|             Some(Frame::containing_address( |  | ||||||
|                 self.0 as usize & 0x0000_ffff_ffff_f000, |  | ||||||
|             )) |  | ||||||
|         } else { |  | ||||||
|             None |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub fn set(&mut self, frame: Frame, flags: EntryFlags) { |  | ||||||
|         assert!(frame.start_address() & !0x0000_ffff_ffff_f000 == 0); |  | ||||||
|         self.0 = (frame.start_address() as u64) | flags.bits(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,198 +0,0 @@ | ||||||
| // mod arch::aarch64::memory::paging
 |  | ||||||
| 
 |  | ||||||
| //! Some code was borrowed from [Phil Opp's Blog](https://os.phil-opp.com/page-tables/)
 |  | ||||||
| //! Paging is mostly based on https://os.phil-opp.com/page-tables/ and ARM ARM
 |  | ||||||
| 
 |  | ||||||
| // AArch64:
 |  | ||||||
| // Table D4-8-2021: check supported granule sizes, select alloc policy based on results.
 |  | ||||||
| // TTBR_ELx is the pdbr for specific page tables
 |  | ||||||
| 
 |  | ||||||
| // Page 2068 actual page descriptor formats
 |  | ||||||
| 
 |  | ||||||
| /* |  | ||||||
|  *  With 4k page granule, a virtual address is split into 4 lookup parts |  | ||||||
|  *  spanning 9 bits each: |  | ||||||
|  * |  | ||||||
|  *    _______________________________________________ |  | ||||||
|  *   |       |       |       |       |       |       | |  | ||||||
|  *   | signx |  Lv0  |  Lv1  |  Lv2  |  Lv3  |  off  | |  | ||||||
|  *   |_______|_______|_______|_______|_______|_______| |  | ||||||
|  *     63-48   47-39   38-30   29-21   20-12   11-00 |  | ||||||
|  * |  | ||||||
|  *             mask        page size |  | ||||||
|  * |  | ||||||
|  *    Lv0: FF8000000000       -- |  | ||||||
|  *    Lv1:   7FC0000000       1G |  | ||||||
|  *    Lv2:     3FE00000       2M |  | ||||||
|  *    Lv3:       1FF000       4K |  | ||||||
|  *    off:          FFF |  | ||||||
|  */ |  | ||||||
| 
 |  | ||||||
| pub use self::entry::*; |  | ||||||
| use self::table::{Level0, Table}; |  | ||||||
| use super::{Frame, FrameAllocator, PhysicalAddress, VirtualAddress}; |  | ||||||
| use core::ptr::Unique; |  | ||||||
| 
 |  | ||||||
| mod entry; |  | ||||||
| mod table; |  | ||||||
| 
 |  | ||||||
| pub const PAGE_SIZE: usize = 4096; |  | ||||||
| 
 |  | ||||||
| pub const ENTRY_COUNT: usize = 512; |  | ||||||
| 
 |  | ||||||
| /**
 |  | ||||||
|  * Page is an addressable unit of the virtual address space. |  | ||||||
|  */ |  | ||||||
| pub struct Page { |  | ||||||
|     number: usize, |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl Page { |  | ||||||
|     pub fn containing_address(address: VirtualAddress) -> Page { |  | ||||||
|         assert!( |  | ||||||
|             address < 0x0000_8000_0000_0000 || address >= 0xffff_8000_0000_0000, |  | ||||||
|             "invalid address: 0x{:x}", |  | ||||||
|             address |  | ||||||
|         ); |  | ||||||
|         Page { |  | ||||||
|             number: address / PAGE_SIZE, |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn start_address(&self) -> usize { |  | ||||||
|         self.number * PAGE_SIZE |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn l0_index(&self) -> usize { |  | ||||||
|         (self.number >> 27) & 0o777 |  | ||||||
|     } |  | ||||||
|     fn l1_index(&self) -> usize { |  | ||||||
|         (self.number >> 18) & 0o777 |  | ||||||
|     } |  | ||||||
|     fn l2_index(&self) -> usize { |  | ||||||
|         (self.number >> 9) & 0o777 |  | ||||||
|     } |  | ||||||
|     fn l3_index(&self) -> usize { |  | ||||||
|         (self.number >> 0) & 0o777 |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| pub struct ActivePageTable { |  | ||||||
|     l0: Unique<Table<Level0>>, |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl ActivePageTable { |  | ||||||
|     pub unsafe fn new() -> ActivePageTable { |  | ||||||
|         ActivePageTable { |  | ||||||
|             l0: Unique::new_unchecked(table::L0), |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn l0(&self) -> &Table<Level0> { |  | ||||||
|         unsafe { self.l0.as_ref() } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn l0_mut(&mut self) -> &mut Table<Level0> { |  | ||||||
|         unsafe { self.l0.as_mut() } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub fn translate(&self, virtual_address: VirtualAddress) -> Option<PhysicalAddress> { |  | ||||||
|         let offset = virtual_address % PAGE_SIZE; |  | ||||||
|         self.translate_page(Page::containing_address(virtual_address)) |  | ||||||
|             .map(|frame| frame.number * PAGE_SIZE + offset) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn translate_page(&self, page: Page) -> Option<Frame> { |  | ||||||
|         use self::entry::EntryFlags; |  | ||||||
| 
 |  | ||||||
|         let l1 = self.l0().next_table(page.l0_index()); |  | ||||||
| 
 |  | ||||||
|         let huge_page = || { |  | ||||||
|             l1.and_then(|l1| { |  | ||||||
|                 let l1_entry = &l1[page.l1_index()]; |  | ||||||
|                 // 1GiB page?
 |  | ||||||
|                 if let Some(start_frame) = l1_entry.pointed_frame() { |  | ||||||
|                     if !l1_entry.flags().contains(EntryFlags::TABLE) { |  | ||||||
|                         // address must be 1GiB aligned
 |  | ||||||
|                         assert!(start_frame.number % (ENTRY_COUNT * ENTRY_COUNT) == 0); |  | ||||||
|                         return Some(Frame { |  | ||||||
|                             number: start_frame.number |  | ||||||
|                                 + page.l2_index() * ENTRY_COUNT |  | ||||||
|                                 + page.l3_index(), |  | ||||||
|                         }); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 if let Some(l2) = l1.next_table(page.l1_index()) { |  | ||||||
|                     let l2_entry = &l2[page.l2_index()]; |  | ||||||
|                     // 2MiB page?
 |  | ||||||
|                     if let Some(start_frame) = l2_entry.pointed_frame() { |  | ||||||
|                         if !l2_entry.flags().contains(EntryFlags::TABLE) { |  | ||||||
|                             // address must be 2MiB aligned
 |  | ||||||
|                             assert!(start_frame.number % ENTRY_COUNT == 0); |  | ||||||
|                             return Some(Frame { |  | ||||||
|                                 number: start_frame.number + page.l3_index(), |  | ||||||
|                             }); |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 None |  | ||||||
|             }) |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         l1.and_then(|l1| l1.next_table(page.l1_index())) |  | ||||||
|             .and_then(|l2| l2.next_table(page.l2_index())) |  | ||||||
|             .and_then(|l3| l3[page.l3_index()].pointed_frame()) |  | ||||||
|             .or_else(huge_page) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub fn map_to<A>(&mut self, page: Page, frame: Frame, flags: EntryFlags, allocator: &mut A) |  | ||||||
|     where |  | ||||||
|         A: FrameAllocator, |  | ||||||
|     { |  | ||||||
|         let l0 = self.l0_mut(); |  | ||||||
|         let l1 = l0.next_table_create(page.l0_index(), allocator); |  | ||||||
|         let l2 = l1.next_table_create(page.l1_index(), allocator); |  | ||||||
|         let l3 = l2.next_table_create(page.l2_index(), allocator); |  | ||||||
| 
 |  | ||||||
|         assert!(l3[page.l3_index()].is_unused()); |  | ||||||
|         l3[page.l3_index()].set(frame, flags | EntryFlags::VALID); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub fn map<A>(&mut self, page: Page, flags: EntryFlags, allocator: &mut A) |  | ||||||
|     where |  | ||||||
|         A: FrameAllocator, |  | ||||||
|     { |  | ||||||
|         let frame = allocator.allocate_frame().expect("out of memory"); |  | ||||||
|         self.map_to(page, frame, flags, allocator) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub fn identity_map<A>(&mut self, frame: Frame, flags: EntryFlags, allocator: &mut A) |  | ||||||
|     where |  | ||||||
|         A: FrameAllocator, |  | ||||||
|     { |  | ||||||
|         let page = Page::containing_address(frame.start_address()); |  | ||||||
|         self.map_to(page, frame, flags, allocator) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn unmap<A>(&mut self, page: Page, _allocator: &mut A) |  | ||||||
|     where |  | ||||||
|         A: FrameAllocator, |  | ||||||
|     { |  | ||||||
|         // use aarch64::instructions::tlb;
 |  | ||||||
|         // use x86_64::VirtualAddress;
 |  | ||||||
| 
 |  | ||||||
|         assert!(self.translate(page.start_address()).is_some()); |  | ||||||
| 
 |  | ||||||
|         let l3 = self |  | ||||||
|             .l0_mut() |  | ||||||
|             .next_table_mut(page.l0_index()) |  | ||||||
|             .and_then(|l1| l1.next_table_mut(page.l1_index())) |  | ||||||
|             .and_then(|l2| l2.next_table_mut(page.l2_index())) |  | ||||||
|             .expect("mapping code does not support huge pages"); |  | ||||||
|         let _frame = l3[page.l3_index()].pointed_frame().unwrap(); |  | ||||||
|         l3[page.l3_index()].set_unused(); |  | ||||||
|         // tlb::flush(VirtualAddress(page.start_address()));
 |  | ||||||
|         // TODO free p(1,2,3) table if empty
 |  | ||||||
|         //allocator.deallocate_frame(frame);
 |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,121 +0,0 @@ | ||||||
| use super::super::FrameAllocator; |  | ||||||
| use super::ENTRY_COUNT; |  | ||||||
| use arch::aarch64::memory::paging::entry::*; |  | ||||||
| use core::marker::PhantomData; |  | ||||||
| use core::ops::{Index, IndexMut}; |  | ||||||
| 
 |  | ||||||
| pub const L0: *mut Table<Level0> = 0xffff_ffff_ffff_f000 as *mut _; // L0 0o177777_777_777_777_777_0000
 |  | ||||||
|                                                                     // L1 0o177777_777_777_777_XXX_0000, XXX is the L0 index
 |  | ||||||
|                                                                     // L2 0o177777_777_777_XXX_YYY_0000, YYY is the L1 index
 |  | ||||||
|                                                                     // L3 0o177777_777_XXX_YYY_ZZZ_0000, ZZZ is the L2 index
 |  | ||||||
| 
 |  | ||||||
| // L1 = (L0 << 9) | (XXX << 12)
 |  | ||||||
| // L2 = (L1 << 9) | (YYY << 12)
 |  | ||||||
| // L3 = (L2 << 9) | (ZZZ << 12)
 |  | ||||||
| 
 |  | ||||||
| pub struct Table<L: TableLevel> { |  | ||||||
|     entries: [Entry; ENTRY_COUNT], |  | ||||||
|     level: PhantomData<L>, |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<L> Table<L> |  | ||||||
| where |  | ||||||
|     L: TableLevel, |  | ||||||
| { |  | ||||||
|     pub fn zero(&mut self) { |  | ||||||
|         for entry in self.entries.iter_mut() { |  | ||||||
|             entry.set_unused(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<L> Table<L> |  | ||||||
| where |  | ||||||
|     L: HierarchicalLevel, |  | ||||||
| { |  | ||||||
|     fn next_table_address(&self, index: usize) -> Option<usize> { |  | ||||||
|         let entry_flags = self[index].flags(); |  | ||||||
|         if entry_flags.contains(EntryFlags::VALID | EntryFlags::TABLE) { |  | ||||||
|             let table_address = self as *const _ as usize; |  | ||||||
|             Some((table_address << 9) | (index << 12)) |  | ||||||
|         } else { |  | ||||||
|             None |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub fn next_table(&self, index: usize) -> Option<&Table<L::NextLevel>> { |  | ||||||
|         self.next_table_address(index) |  | ||||||
|             .map(|address| unsafe { &*(address as *const _) }) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub fn next_table_mut(&mut self, index: usize) -> Option<&mut Table<L::NextLevel>> { |  | ||||||
|         self.next_table_address(index) |  | ||||||
|             .map(|address| unsafe { &mut *(address as *mut _) }) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     pub fn next_table_create<A>( |  | ||||||
|         &mut self, |  | ||||||
|         index: usize, |  | ||||||
|         allocator: &mut A, |  | ||||||
|     ) -> &mut Table<L::NextLevel> |  | ||||||
|     where |  | ||||||
|         A: FrameAllocator, |  | ||||||
|     { |  | ||||||
|         if self.next_table(index).is_none() { |  | ||||||
|             assert!( |  | ||||||
|                 self.entries[index].flags().contains(EntryFlags::TABLE), |  | ||||||
|                 "mapping code does not support huge pages" |  | ||||||
|             ); |  | ||||||
|             let frame = allocator.allocate_frame().expect("no frames available"); |  | ||||||
|             self.entries[index].set(frame, EntryFlags::VALID /*| WRITABLE*/); |  | ||||||
|             self.next_table_mut(index).unwrap().zero(); |  | ||||||
|         } |  | ||||||
|         self.next_table_mut(index).unwrap() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<L> Index<usize> for Table<L> |  | ||||||
| where |  | ||||||
|     L: TableLevel, |  | ||||||
| { |  | ||||||
|     type Output = Entry; |  | ||||||
| 
 |  | ||||||
|     fn index(&self, index: usize) -> &Entry { |  | ||||||
|         &self.entries[index] |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<L> IndexMut<usize> for Table<L> |  | ||||||
| where |  | ||||||
|     L: TableLevel, |  | ||||||
| { |  | ||||||
|     fn index_mut(&mut self, index: usize) -> &mut Entry { |  | ||||||
|         &mut self.entries[index] |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| pub trait TableLevel {} |  | ||||||
| 
 |  | ||||||
| pub enum Level0 {} |  | ||||||
| pub enum Level1 {} |  | ||||||
| pub enum Level2 {} |  | ||||||
| pub enum Level3 {} |  | ||||||
| 
 |  | ||||||
| impl TableLevel for Level0 {} |  | ||||||
| impl TableLevel for Level1 {} |  | ||||||
| impl TableLevel for Level2 {} |  | ||||||
| impl TableLevel for Level3 {} |  | ||||||
| 
 |  | ||||||
| pub trait HierarchicalLevel: TableLevel { |  | ||||||
|     type NextLevel: TableLevel; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl HierarchicalLevel for Level0 { |  | ||||||
|     type NextLevel = Level1; |  | ||||||
| } |  | ||||||
| impl HierarchicalLevel for Level1 { |  | ||||||
|     type NextLevel = Level2; |  | ||||||
| } |  | ||||||
| impl HierarchicalLevel for Level2 { |  | ||||||
|     type NextLevel = Level3; |  | ||||||
| } |  | ||||||
|  | @ -1,366 +0,0 @@ | ||||||
| /* |  | ||||||
|  * MIT License |  | ||||||
|  * |  | ||||||
|  * Copyright (c) 2018-2019 Andre Richter <andre.o.richter@gmail.com> |  | ||||||
|  * Copyright (c) 2019 Berkus Decker <berkus+github@metta.systems> |  | ||||||
|  * |  | ||||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy |  | ||||||
|  * of this software and associated documentation files (the "Software"), to deal |  | ||||||
|  * in the Software without restriction, including without limitation the rights |  | ||||||
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |  | ||||||
|  * copies of the Software, and to permit persons to whom the Software is |  | ||||||
|  * furnished to do so, subject to the following conditions: |  | ||||||
|  * |  | ||||||
|  * The above copyright notice and this permission notice shall be included in all |  | ||||||
|  * copies or substantial portions of the Software. |  | ||||||
|  * |  | ||||||
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |  | ||||||
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |  | ||||||
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |  | ||||||
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |  | ||||||
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |  | ||||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |  | ||||||
|  * SOFTWARE. |  | ||||||
|  */ |  | ||||||
| 
 |  | ||||||
| //! MMU initialisation.
 |  | ||||||
| //! [ARMv8 memory addressing](https://static.docs.arm.com/100940/0100/armv8_a_address%20translation_100940_0100_en.pdf)
 |  | ||||||
| 
 |  | ||||||
| use crate::arch::aarch64::memory::{get_virt_addr_properties, AttributeFields}; |  | ||||||
| use crate::println; |  | ||||||
| use cortex_a::{barrier, regs::*}; |  | ||||||
| use register::register_bitfields; |  | ||||||
| 
 |  | ||||||
| /// Parse the ID_AA64MMFR0_EL1 register for runtime information about supported MMU features.
 |  | ||||||
| pub fn print_features() { |  | ||||||
|     let mmfr = ID_AA64MMFR0_EL1.extract(); |  | ||||||
| 
 |  | ||||||
|     if let Some(ID_AA64MMFR0_EL1::TGran4::Value::Supported) = |  | ||||||
|         mmfr.read_as_enum(ID_AA64MMFR0_EL1::TGran4) |  | ||||||
|     { |  | ||||||
|         println!("[i] MMU: 4 KiB granule supported!"); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if let Some(ID_AA64MMFR0_EL1::PARange::Value::Bits_40) = |  | ||||||
|         mmfr.read_as_enum(ID_AA64MMFR0_EL1::PARange) |  | ||||||
|     { |  | ||||||
|         println!("[i] MMU: Up to 40 Bit physical address range supported!"); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| register_bitfields! {u64, |  | ||||||
|     // AArch64 Reference Manual page 2150
 |  | ||||||
|     STAGE1_DESCRIPTOR [ |  | ||||||
|         /// Privileged execute-never
 |  | ||||||
|         PXN      OFFSET(53) NUMBITS(1) [ |  | ||||||
|             False = 0, |  | ||||||
|             True = 1 |  | ||||||
|         ], |  | ||||||
| 
 |  | ||||||
|         /// Various address fields, depending on use case
 |  | ||||||
|         LVL2_OUTPUT_ADDR_4KiB    OFFSET(21) NUMBITS(27) [], // [47:21]
 |  | ||||||
|         NEXT_LVL_TABLE_ADDR_4KiB OFFSET(12) NUMBITS(36) [], // [47:12]
 |  | ||||||
| 
 |  | ||||||
|         /// Access flag
 |  | ||||||
|         AF       OFFSET(10) NUMBITS(1) [ |  | ||||||
|             False = 0, |  | ||||||
|             True = 1 |  | ||||||
|         ], |  | ||||||
| 
 |  | ||||||
|         /// Shareability field
 |  | ||||||
|         SH       OFFSET(8) NUMBITS(2) [ |  | ||||||
|             OuterShareable = 0b10, |  | ||||||
|             InnerShareable = 0b11 |  | ||||||
|         ], |  | ||||||
| 
 |  | ||||||
|         /// Access Permissions
 |  | ||||||
|         AP       OFFSET(6) NUMBITS(2) [ |  | ||||||
|             RW_EL1 = 0b00, |  | ||||||
|             RW_EL1_EL0 = 0b01, |  | ||||||
|             RO_EL1 = 0b10, |  | ||||||
|             RO_EL1_EL0 = 0b11 |  | ||||||
|         ], |  | ||||||
| 
 |  | ||||||
|         /// Memory attributes index into the MAIR_EL1 register
 |  | ||||||
|         AttrIndx OFFSET(2) NUMBITS(3) [], |  | ||||||
| 
 |  | ||||||
|         TYPE     OFFSET(1) NUMBITS(1) [ |  | ||||||
|             Block = 0, |  | ||||||
|             Table = 1 |  | ||||||
|         ], |  | ||||||
| 
 |  | ||||||
|         VALID    OFFSET(0) NUMBITS(1) [ |  | ||||||
|             False = 0, |  | ||||||
|             True = 1 |  | ||||||
|         ] |  | ||||||
|     ] |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| const FOUR_KIB: usize = 4 * 1024; |  | ||||||
| const FOUR_KIB_SHIFT: usize = 12; // log2(4 * 1024)
 |  | ||||||
| 
 |  | ||||||
| const TWO_MIB: usize = 2 * 1024 * 1024; |  | ||||||
| const TWO_MIB_SHIFT: usize = 21; // log2(2 * 1024 * 1024)
 |  | ||||||
| 
 |  | ||||||
| /// A descriptor pointing to the next page table.
 |  | ||||||
| struct TableDescriptor(register::FieldValue<u64, STAGE1_DESCRIPTOR::Register>); |  | ||||||
| 
 |  | ||||||
| impl TableDescriptor { |  | ||||||
|     fn new(next_lvl_table_addr: usize) -> Result<TableDescriptor, &'static str> { |  | ||||||
|         if next_lvl_table_addr % FOUR_KIB != 0 { |  | ||||||
|             return Err("TableDescriptor: Address is not 4 KiB aligned."); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         let shifted = next_lvl_table_addr >> FOUR_KIB_SHIFT; |  | ||||||
| 
 |  | ||||||
|         Ok(TableDescriptor( |  | ||||||
|             STAGE1_DESCRIPTOR::VALID::True |  | ||||||
|                 + STAGE1_DESCRIPTOR::TYPE::Table |  | ||||||
|                 + STAGE1_DESCRIPTOR::NEXT_LVL_TABLE_ADDR_4KiB.val(shifted as u64), |  | ||||||
|         )) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn value(&self) -> u64 { |  | ||||||
|         self.0.value |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /// A function that maps the generic memory range attributes to HW-specific
 |  | ||||||
| /// attributes of the MMU.
 |  | ||||||
| fn into_mmu_attributes( |  | ||||||
|     attribute_fields: AttributeFields, |  | ||||||
| ) -> register::FieldValue<u64, STAGE1_DESCRIPTOR::Register> { |  | ||||||
|     use crate::memory::{AccessPermissions, MemAttributes}; |  | ||||||
| 
 |  | ||||||
|     // Memory attributes
 |  | ||||||
|     let mut desc = match attribute_fields.mem_attributes { |  | ||||||
|         MemAttributes::CacheableDRAM => { |  | ||||||
|             STAGE1_DESCRIPTOR::SH::InnerShareable + STAGE1_DESCRIPTOR::AttrIndx.val(mair::NORMAL) |  | ||||||
|         } |  | ||||||
|         MemAttributes::NonCacheableDRAM => { |  | ||||||
|             STAGE1_DESCRIPTOR::SH::InnerShareable |  | ||||||
|                 + STAGE1_DESCRIPTOR::AttrIndx.val(mair::NORMAL_NON_CACHEABLE) |  | ||||||
|         } |  | ||||||
|         MemAttributes::Device => { |  | ||||||
|             STAGE1_DESCRIPTOR::SH::OuterShareable |  | ||||||
|                 + STAGE1_DESCRIPTOR::AttrIndx.val(mair::DEVICE_NGNRE) |  | ||||||
|         } |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     // Access Permissions
 |  | ||||||
|     desc += match attribute_fields.acc_perms { |  | ||||||
|         AccessPermissions::ReadOnly => STAGE1_DESCRIPTOR::AP::RO_EL1, |  | ||||||
|         AccessPermissions::ReadWrite => STAGE1_DESCRIPTOR::AP::RW_EL1, |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     // Execute Never
 |  | ||||||
|     desc += if attribute_fields.execute_never { |  | ||||||
|         STAGE1_DESCRIPTOR::PXN::True |  | ||||||
|     } else { |  | ||||||
|         STAGE1_DESCRIPTOR::PXN::False |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     desc |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /// A Level2 block descriptor with 2 MiB aperture.
 |  | ||||||
| ///
 |  | ||||||
| /// The output points to physical memory.
 |  | ||||||
| struct Lvl2BlockDescriptor(register::FieldValue<u64, STAGE1_DESCRIPTOR::Register>); |  | ||||||
| 
 |  | ||||||
| impl Lvl2BlockDescriptor { |  | ||||||
|     fn new( |  | ||||||
|         output_addr: usize, |  | ||||||
|         attribute_fields: AttributeFields, |  | ||||||
|     ) -> Result<Lvl2BlockDescriptor, &'static str> { |  | ||||||
|         if output_addr % TWO_MIB != 0 { |  | ||||||
|             return Err("BlockDescriptor: Address is not 2 MiB aligned."); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         let shifted = output_addr >> TWO_MIB_SHIFT; |  | ||||||
| 
 |  | ||||||
|         Ok(Lvl2BlockDescriptor( |  | ||||||
|             STAGE1_DESCRIPTOR::VALID::True |  | ||||||
|                 + STAGE1_DESCRIPTOR::AF::True |  | ||||||
|                 + into_mmu_attributes(attribute_fields) |  | ||||||
|                 + STAGE1_DESCRIPTOR::TYPE::Block |  | ||||||
|                 + STAGE1_DESCRIPTOR::LVL2_OUTPUT_ADDR_4KiB.val(shifted as u64), |  | ||||||
|         )) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn value(&self) -> u64 { |  | ||||||
|         self.0.value |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /// A page descriptor with 4 KiB aperture.
 |  | ||||||
| ///
 |  | ||||||
| /// The output points to physical memory.
 |  | ||||||
| struct PageDescriptor(register::FieldValue<u64, STAGE1_DESCRIPTOR::Register>); |  | ||||||
| 
 |  | ||||||
| impl PageDescriptor { |  | ||||||
|     fn new( |  | ||||||
|         output_addr: usize, |  | ||||||
|         attribute_fields: AttributeFields, |  | ||||||
|     ) -> Result<PageDescriptor, &'static str> { |  | ||||||
|         if output_addr % FOUR_KIB != 0 { |  | ||||||
|             return Err("PageDescriptor: Address is not 4 KiB aligned."); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         let shifted = output_addr >> FOUR_KIB_SHIFT; |  | ||||||
| 
 |  | ||||||
|         Ok(PageDescriptor( |  | ||||||
|             STAGE1_DESCRIPTOR::VALID::True |  | ||||||
|                 + STAGE1_DESCRIPTOR::AF::True |  | ||||||
|                 + into_mmu_attributes(attribute_fields) |  | ||||||
|                 + STAGE1_DESCRIPTOR::TYPE::Table |  | ||||||
|                 + STAGE1_DESCRIPTOR::NEXT_LVL_TABLE_ADDR_4KiB.val(shifted as u64), |  | ||||||
|         )) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn value(&self) -> u64 { |  | ||||||
|         self.0.value |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| trait BaseAddr { |  | ||||||
|     fn base_addr_u64(&self) -> u64; |  | ||||||
|     fn base_addr_usize(&self) -> usize; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl BaseAddr for [u64; 512] { |  | ||||||
|     fn base_addr_u64(&self) -> u64 { |  | ||||||
|         self as *const u64 as u64 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn base_addr_usize(&self) -> usize { |  | ||||||
|         self as *const u64 as usize |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| const NUM_ENTRIES_4KIB: usize = 512; |  | ||||||
| 
 |  | ||||||
| // We need a wrapper struct here so that we can make use of the align attribute.
 |  | ||||||
| #[repr(C)] |  | ||||||
| #[repr(align(4096))] |  | ||||||
| struct PageTable { |  | ||||||
|     entries: [u64; NUM_ENTRIES_4KIB], |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static mut LVL2_TABLE: PageTable = PageTable { |  | ||||||
|     entries: [0; NUM_ENTRIES_4KIB], |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| static mut LVL3_TABLE: PageTable = PageTable { |  | ||||||
|     entries: [0; NUM_ENTRIES_4KIB], |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| /// Setup function for the MAIR_EL1 register.
 |  | ||||||
| fn set_up_mair() { |  | ||||||
|     // Define the three memory types that we will map. Normal DRAM, Uncached and device.
 |  | ||||||
|     MAIR_EL1.write( |  | ||||||
|         // Attribute 2
 |  | ||||||
|         MAIR_EL1::Attr2_HIGH::Device |  | ||||||
|             + MAIR_EL1::Attr2_LOW_DEVICE::Device_nGnRE |  | ||||||
|             // Attribute 1
 |  | ||||||
|             + MAIR_EL1::Attr1_HIGH::Memory_OuterNonCacheable |  | ||||||
|             + MAIR_EL1::Attr1_LOW_MEMORY::InnerNonCacheable |  | ||||||
|             // Attribute 0
 |  | ||||||
|             + MAIR_EL1::Attr0_HIGH::Memory_OuterWriteBack_NonTransient_ReadAlloc_WriteAlloc |  | ||||||
|             + MAIR_EL1::Attr0_LOW_MEMORY::InnerWriteBack_NonTransient_ReadAlloc_WriteAlloc, |  | ||||||
|     ); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // Three descriptive consts for indexing into the correct MAIR_EL1 attributes.
 |  | ||||||
| mod mair { |  | ||||||
|     pub const NORMAL: u64 = 0; |  | ||||||
|     pub const NORMAL_NON_CACHEABLE: u64 = 1; |  | ||||||
|     pub const DEVICE_NGNRE: u64 = 2; |  | ||||||
|     // DEVICE_GRE
 |  | ||||||
|     // DEVICE_NGNRNE
 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /// Set up identity mapped page tables for the first 1 gigabyte of address space.
 |  | ||||||
| /// default: 880 MB ARM ram, 128MB VC
 |  | ||||||
| pub unsafe fn init() -> Result<(), &'static str> { |  | ||||||
|     // Prepare the memory attribute indirection register.
 |  | ||||||
|     set_up_mair(); |  | ||||||
| 
 |  | ||||||
|     // Point the first 2 MiB of virtual addresses to the follow-up LVL3
 |  | ||||||
|     // page-table.
 |  | ||||||
|     LVL2_TABLE.entries[0] = match TableDescriptor::new(LVL3_TABLE.entries.base_addr_usize()) { |  | ||||||
|         Err(s) => return Err(s), |  | ||||||
|         Ok(d) => d.value(), |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     // Fill the rest of the LVL2 (2 MiB) entries as block descriptors.
 |  | ||||||
|     //
 |  | ||||||
|     // Notice the skip(1) which makes the iteration start at the second 2 MiB
 |  | ||||||
|     // block (0x20_0000).
 |  | ||||||
|     for (block_descriptor_nr, entry) in LVL2_TABLE.entries.iter_mut().enumerate().skip(1) { |  | ||||||
|         let virt_addr = block_descriptor_nr << TWO_MIB_SHIFT; |  | ||||||
| 
 |  | ||||||
|         let (output_addr, attribute_fields) = match get_virt_addr_properties(virt_addr) { |  | ||||||
|             Err(s) => return Err(s), |  | ||||||
|             Ok((a, b)) => (a, b), |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         let block_desc = match Lvl2BlockDescriptor::new(output_addr, attribute_fields) { |  | ||||||
|             Err(s) => return Err(s), |  | ||||||
|             Ok(desc) => desc, |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         *entry = block_desc.value(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // Finally, fill the single LVL3 table (4 KiB granule).
 |  | ||||||
|     for (page_descriptor_nr, entry) in LVL3_TABLE.entries.iter_mut().enumerate() { |  | ||||||
|         let virt_addr = page_descriptor_nr << FOUR_KIB_SHIFT; |  | ||||||
| 
 |  | ||||||
|         let (output_addr, attribute_fields) = match get_virt_addr_properties(virt_addr) { |  | ||||||
|             Err(s) => return Err(s), |  | ||||||
|             Ok((a, b)) => (a, b), |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         let page_desc = match PageDescriptor::new(output_addr, attribute_fields) { |  | ||||||
|             Err(s) => return Err(s), |  | ||||||
|             Ok(desc) => desc, |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         *entry = page_desc.value(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // Point to the LVL2 table base address in TTBR0.
 |  | ||||||
|     TTBR0_EL1.set_baddr(LVL2_TABLE.entries.base_addr_u64()); |  | ||||||
| 
 |  | ||||||
|     // Configure various settings of stage 1 of the EL1 translation regime.
 |  | ||||||
|     let ips = ID_AA64MMFR0_EL1.read(ID_AA64MMFR0_EL1::PARange); |  | ||||||
|     TCR_EL1.write( |  | ||||||
|         TCR_EL1::TBI0::Ignored |  | ||||||
|             + TCR_EL1::IPS.val(ips) |  | ||||||
|             + TCR_EL1::TG0::KiB_4 // 4 KiB granule
 |  | ||||||
|             + TCR_EL1::SH0::Inner |  | ||||||
|             + TCR_EL1::ORGN0::WriteBack_ReadAlloc_WriteAlloc_Cacheable |  | ||||||
|             + TCR_EL1::IRGN0::WriteBack_ReadAlloc_WriteAlloc_Cacheable |  | ||||||
|             + TCR_EL1::EPD0::EnableTTBR0Walks |  | ||||||
|             + TCR_EL1::T0SZ.val(34), // Start walks at level 2
 |  | ||||||
|     ); |  | ||||||
| 
 |  | ||||||
|     // Switch the MMU on.
 |  | ||||||
|     //
 |  | ||||||
|     // First, force all previous changes to be seen before the MMU is enabled.
 |  | ||||||
|     barrier::isb(barrier::SY); |  | ||||||
| 
 |  | ||||||
|     // Enable the MMU and turn on data and instruction caching.
 |  | ||||||
|     SCTLR_EL1.modify(SCTLR_EL1::M::Enable + SCTLR_EL1::C::Cacheable + SCTLR_EL1::I::Cacheable); |  | ||||||
| 
 |  | ||||||
|     // Force MMU init to complete before next instruction
 |  | ||||||
|     /* |  | ||||||
|      * Invalidate the local I-cache so that any instructions fetched |  | ||||||
|      * speculatively from the PoC are discarded, since they may have |  | ||||||
|      * been dynamically patched at the PoU. |  | ||||||
|      */ |  | ||||||
|     barrier::isb(barrier::SY); |  | ||||||
| 
 |  | ||||||
|     Ok(()) |  | ||||||
| } |  | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
		Reference in New Issue