mirror of
https://github.com/italicsjenga/gba.git
synced 2025-01-11 11:31:31 +11:00
commit
5113a5a1f2
13
.travis.yml
13
.travis.yml
|
@ -11,7 +11,6 @@ rust:
|
||||||
|
|
||||||
before_script:
|
before_script:
|
||||||
- rustup component add rust-src
|
- rustup component add rust-src
|
||||||
- rustup component add clippy --toolchain=nightly
|
|
||||||
- (test -x $HOME/.cargo/bin/cargo-install-update || cargo install cargo-update)
|
- (test -x $HOME/.cargo/bin/cargo-install-update || cargo install cargo-update)
|
||||||
- (test -x $HOME/.cargo/bin/cargo-xbuild || cargo install cargo-xbuild)
|
- (test -x $HOME/.cargo/bin/cargo-xbuild || cargo install cargo-xbuild)
|
||||||
- (test -x $HOME/.cargo/bin/cargo-make || cargo install cargo-make)
|
- (test -x $HOME/.cargo/bin/cargo-make || cargo install cargo-make)
|
||||||
|
@ -31,15 +30,13 @@ script:
|
||||||
- export PATH="$PATH:/opt/devkitpro/tools/bin"
|
- export PATH="$PATH:/opt/devkitpro/tools/bin"
|
||||||
- cd ..
|
- cd ..
|
||||||
# Run all verificaions, both debug and release
|
# Run all verificaions, both debug and release
|
||||||
#- cargo clippy
|
- cargo test --lib
|
||||||
#- cargo clippy --release
|
- cargo test --lib --release
|
||||||
- cargo test --no-fail-fast --lib
|
- cargo test --tests
|
||||||
- cargo test --no-fail-fast --lib --release
|
- cargo test --tests --release
|
||||||
- cargo test --no-fail-fast --tests
|
|
||||||
- cargo test --no-fail-fast --tests --release
|
|
||||||
# Let cargo make take over the rest
|
# Let cargo make take over the rest
|
||||||
- cargo make justrelease
|
- cargo make justrelease
|
||||||
# Test build the book so that a failed book build kills this run
|
# Test a build of the book so that a failed book build kills this run
|
||||||
- cd book && mdbook build
|
- cd book && mdbook build
|
||||||
|
|
||||||
deploy:
|
deploy:
|
||||||
|
|
|
@ -2,4 +2,8 @@
|
||||||
# Rust GBA Guide
|
# Rust GBA Guide
|
||||||
|
|
||||||
* [Development Setup](development-setup.md)
|
* [Development Setup](development-setup.md)
|
||||||
|
* [Volatile](volatile.md)
|
||||||
|
* [The Hardware Memory Map](the-hardware-memory-map.md)
|
||||||
|
* [IO Registers](io-registers.md)
|
||||||
|
* [Bitmap Video](bitmap-video.md)
|
||||||
* [GBA Assembly](gba-asm.md)
|
* [GBA Assembly](gba-asm.md)
|
||||||
|
|
1
book/src/bitmap-video.md
Normal file
1
book/src/bitmap-video.md
Normal file
|
@ -0,0 +1 @@
|
||||||
|
# Bitmap Video
|
|
@ -171,12 +171,14 @@ fn main(_argc: isize, _argv: *const *const u8) -> isize {
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
Throw that into your project skeleton, build the program, and give it a run. You
|
Throw that into your project skeleton, build the program, and give it a run in
|
||||||
should see a red, green, and blue dot close-ish to the middle of the screen. If
|
an emulator. I suggest [mgba](https://mgba.io/2019/01/26/mgba-0.7.0/), it has
|
||||||
you don't, something _already_ went wrong. Double check things, phone a friend,
|
some developer tools we'll use later on. You should see a red, green, and blue
|
||||||
write your senators, try asking `Lokathor` or `Ketsuban` on the [Rust Community
|
dot close-ish to the middle of the screen. If you don't, something _already_
|
||||||
|
went wrong. Double check things, phone a friend, write your senators, try asking
|
||||||
|
`Lokathor` or `Ketsuban` on the [Rust Community
|
||||||
Discord](https://discordapp.com/invite/aVESxV8), until you're eventually able to
|
Discord](https://discordapp.com/invite/aVESxV8), until you're eventually able to
|
||||||
get your three dots going.
|
get your three dots going.
|
||||||
|
|
||||||
Of course, I'm sure you want to know why those numbers are the numbers to use.
|
Of course, I'm sure you want to know why those particular numbers are the
|
||||||
Well that's what the whole rest of the book is about!
|
numbers to use. Well that's what the whole rest of the book is about!
|
||||||
|
|
|
@ -17,21 +17,32 @@ sometimes. Accordingly, you should know how assembly works on the GBA.
|
||||||
Version](http://infocenter.arm.com/help/topic/com.arm.doc.ddi0210c/DDI0210B.pdf)
|
Version](http://infocenter.arm.com/help/topic/com.arm.doc.ddi0210c/DDI0210B.pdf)
|
||||||
of the documentation available, if you'd like that.
|
of the documentation available, if you'd like that.
|
||||||
|
|
||||||
|
* In addition to the `ARM7TDMI` book, which is specific to the GBA's CPU, you'll
|
||||||
|
need to find a copy of the ARM Architecture Reference Manual if you want
|
||||||
|
general ARM knowledge. The ARM Infocenter has the
|
||||||
|
[ARMv5](http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ddi0100i/index.html)
|
||||||
|
version of said manual hosted on their site. Unfortunately, they don't seem to
|
||||||
|
host the `ARMv4T` version of the manual any more.
|
||||||
|
|
||||||
* The [GBATek: ARM CPU
|
* The [GBATek: ARM CPU
|
||||||
Overview](https://problemkaputt.de/gbatek.htm#armcpuoverview) also has quite a
|
Overview](https://problemkaputt.de/gbatek.htm#armcpuoverview) also has quite a
|
||||||
bit of info. Some of it is a duplication of what you'd find in the ARM
|
bit of info. Some of it is a duplication of what you'd find in the ARM
|
||||||
Infocenter reference manual. Some of it is specific to the GBA's chip. Some of
|
Infocenter reference manuals. Some of it is information that's specific to the
|
||||||
it is specific to the ARM chips within the DS and DSi. It's a bit of a jumbled
|
GBA's layout and how the CPU interacts with other parts (such as how its
|
||||||
mess, and as with the rest of GBATEK, the explanations are in a "sparse" style
|
timings and the display adapter's timings line up). Some of it is specific to
|
||||||
(to put it nicely), so I wouldn't take it as your only source.
|
the ARM chips _within the DS and DSi_, so be careful to make sure that you
|
||||||
|
don't wander into the wrong section. GBATEK is always a bit of a jumbled mess,
|
||||||
|
and the explanations are often "sparse" (to put it nicely), so I'd advise that
|
||||||
|
you also look at the official ARM manuals.
|
||||||
|
|
||||||
* The [Compiler Explorer](https://rust.godbolt.org/z/ndCnk3) can be used to
|
* The [Compiler Explorer](https://rust.godbolt.org/z/ndCnk3) can be used to
|
||||||
quickly look at assembly versions of your Rust code. That link there will load
|
quickly look at assembly versions of your Rust code. That link there will load
|
||||||
up an essentially blank `no_std` file with `opt-level=3` set and targeting
|
up an essentially blank `no_std` file with `opt-level=3` set and targeting
|
||||||
`thumbv6m-none-eabi`. That's _not_ the same target as the GBA (it's two ISA
|
`thumbv6m-none-eabi`. That's _not_ the same target as the GBA (it's two ISA
|
||||||
revisions later, ARMv6 instead of ARMv4), but it's the closest CPU target that
|
revisions later, `ARMv6` instead of `ARMv4`), but it's the closest CPU target
|
||||||
is bundled with rustc, so it's the closest you can get with the compiler
|
that is bundled with `rustc`, so it's the closest you can get with the
|
||||||
explorer website. If you're very dedicated I suppose you could setup a [local
|
compiler explorer website. If you're very dedicated I suppose you could setup
|
||||||
|
a [local
|
||||||
instance](https://github.com/mattgodbolt/compiler-explorer#running-a-local-instance)
|
instance](https://github.com/mattgodbolt/compiler-explorer#running-a-local-instance)
|
||||||
of compiler explorer and then add the extra target definition and so on, but
|
of compiler explorer and then add the extra target definition and so on, but
|
||||||
that's _probably_ overkill.
|
that's _probably_ overkill.
|
||||||
|
|
1
book/src/io-registers.md
Normal file
1
book/src/io-registers.md
Normal file
|
@ -0,0 +1 @@
|
||||||
|
# IO Registers
|
379
book/src/the-hardware-memory-map.md
Normal file
379
book/src/the-hardware-memory-map.md
Normal file
|
@ -0,0 +1,379 @@
|
||||||
|
# The Hardware Memory Map
|
||||||
|
|
||||||
|
So we saw `hello_magic.rs` and then we learned what `volatile` was all about,
|
||||||
|
but we've still got a few things that are a bit mysterious. You can't just cast
|
||||||
|
a number into a pointer and start writing to it! That's totally crazy! That's
|
||||||
|
writing to un-allocated memory! Against the rules!
|
||||||
|
|
||||||
|
Well, _kinda_. It's true that you're not allowed to write _anywhere at all_, but
|
||||||
|
those locations were carefully selected locations.
|
||||||
|
|
||||||
|
You see, on a modern computer if you need to check if a key is pressed you ask
|
||||||
|
the Operating System (OS) to please go check for you. If you need to play a
|
||||||
|
sound, you ask the OS to please play the sound on a default sound output. If you
|
||||||
|
need to show a picture you ask the OS to give you access to the video driver so
|
||||||
|
that you can ask the video driver to please put some pixels on the screen.
|
||||||
|
That's mostly fine, except how does the OS actually do it? It doesn't have an OS
|
||||||
|
to go ask, it has to stop somewhere.
|
||||||
|
|
||||||
|
Ultimately, every piece of hardware is mapped into somewhere in the address
|
||||||
|
space of the CPU. You can't actually tell that this is the case as a normal user
|
||||||
|
because your program runs inside a virtualized address space. That way you can't
|
||||||
|
go writing into another program's memory and crash what they're doing or steal
|
||||||
|
their data (well, hopefully, it's obviously not perfect). Outside of the
|
||||||
|
virtualization layer the OS is running directly in the "true" address space, and
|
||||||
|
it can access the hardware on behalf of a program whenever it's asked to.
|
||||||
|
|
||||||
|
How does directly accessing the hardware work, _precisely_? It's just the same
|
||||||
|
as accessing the RAM. Each address holds some bits, and the CPU picks an address
|
||||||
|
and loads in the bits. Then the program gets the bits and has to decide what
|
||||||
|
they mean. The "driver" of a hardware device is just the layer that translates
|
||||||
|
between raw bits in the outside world and more meaningful values inside of the
|
||||||
|
program.
|
||||||
|
|
||||||
|
Of course, memory mapped hardware can change its bits at any time. The user can
|
||||||
|
press and release a key and you can't stop them. This is where `volatile` comes
|
||||||
|
in. Whenever there's memory mapped hardware you want to access it with
|
||||||
|
`volatile` operations so that you can be sure that you're sending the data every
|
||||||
|
time, and that you're getting fresh data every time.
|
||||||
|
|
||||||
|
## GBA Specifics
|
||||||
|
|
||||||
|
That's enough about the general concept of memory mapped hardware, let's get to
|
||||||
|
some GBA specifics. The GBA has the following sections in its memory map.
|
||||||
|
|
||||||
|
* BIOS
|
||||||
|
* External Work RAM (EWRAM)
|
||||||
|
* Internal Work RAM (IWRAM)
|
||||||
|
* IO Registers
|
||||||
|
* Palette RAM (PALRAM)
|
||||||
|
* Video RAM (VRAM)
|
||||||
|
* Object Attribute Memory (OAM)
|
||||||
|
* Game Pak ROM (ROM)
|
||||||
|
* Save RAM (SRAM)
|
||||||
|
|
||||||
|
Each of these has a few key points of interest:
|
||||||
|
|
||||||
|
* **Bus Width:** Also just called "bus", this is how many little wires are
|
||||||
|
_physically_ connecting a part of the address space to the CPU. If you need to
|
||||||
|
transfer more data than fits in the bus you have to do repeated transfers
|
||||||
|
until it all gets through.
|
||||||
|
* **Read/Write Modes:** Most parts of the address space can be read from in 8,
|
||||||
|
16, or 32 bits at a time (there's a few exceptions we'll see). However, a
|
||||||
|
significant portion of the address space can't accept 8 bit writes. Usually
|
||||||
|
this isn't a big deal, but standard `memcopy` routine switches to doing a
|
||||||
|
byte-by-byte copy in some situations, so we'll have to be careful about using
|
||||||
|
it in combination with those regions of the memory.
|
||||||
|
* **Access Speed:** On top of the bus width issue, not all memory can be
|
||||||
|
accessed at the same speed. The "fast" parts of memory can do a read or write
|
||||||
|
in 1 cycle, but the slower parts of memory can take a few cycles per access.
|
||||||
|
These are called "wait cycles". The exact timings depend on what you configure
|
||||||
|
the system to use, which is also limited by what your cartridge physically
|
||||||
|
supports. You'll often see timings broken down into `N` cycles (non-sequential
|
||||||
|
memory access) and `S` cycles (sequential memory access, often faster). There
|
||||||
|
are also `I` cycles (internal cycles) which happen whenever the CPU does an
|
||||||
|
internal operation that's more than one cycle to complete (like a multiply).
|
||||||
|
Don't worry, you don't have to count exact cycle timings unless you're on the
|
||||||
|
razor's edge of the GBA's abilities. For more normal games you just have to be
|
||||||
|
mindful of what you're doing and it'll be fine.
|
||||||
|
|
||||||
|
Let's briefly go over the major talking points of each memory region. All of
|
||||||
|
this information is also available in GBATEK, mostly in their [memory
|
||||||
|
map](http://www.akkit.org/info/gbatek.htm#gbamemorymap) section (though somewhat
|
||||||
|
spread through the rest of the document too).
|
||||||
|
|
||||||
|
Though I'm going to list the location range of each memory space below, most of
|
||||||
|
the hardware locations are actually mirrored at several points throughout the
|
||||||
|
address space.
|
||||||
|
|
||||||
|
### BIOS
|
||||||
|
|
||||||
|
* **Location:** `0x0` to `0x3FFF`
|
||||||
|
* **Bus:** 32-bit
|
||||||
|
* **Access:** Memory protected read-only (see text).
|
||||||
|
* **Wait Cycles:** None
|
||||||
|
|
||||||
|
The "basic input output system". This contains a grab bag of utilities that do
|
||||||
|
various tasks. The code is optimized for small size rather than great speed, so
|
||||||
|
you can sometimes write faster versions of these routines. Also, calling a bios
|
||||||
|
function has more overhead than a normal function call. You can think of bios
|
||||||
|
calls as being similar to system calls to the OS on a desktop computer. Useful,
|
||||||
|
but costly.
|
||||||
|
|
||||||
|
As a side note, not only is BIOS memory read only, but it's memory protected so
|
||||||
|
that you can't even read from bios memory unless the system is currently
|
||||||
|
executing a function that's in bios memory. If you try then the system just
|
||||||
|
gives back a nonsensical value that's not really what you asked for. If you
|
||||||
|
really want to know what's inside, there's actually a bug in one bios call
|
||||||
|
(`MidiKey2Freq`) that lets you read the bios section one byte at a time.
|
||||||
|
|
||||||
|
Also, there's not just one bios! Of course there's the official bios from
|
||||||
|
Nintendo that's used on actual hardware, but since that's code instead of
|
||||||
|
hardware it's protected by copyright. Since a bios is needed to run a GBA
|
||||||
|
emulator properly, people have come up with their own open source versions or
|
||||||
|
they simply make the emulator special case the bios and act _as if_ the function
|
||||||
|
call had done the right thing.
|
||||||
|
|
||||||
|
* The [TempGBA](https://github.com/Nebuleon/TempGBA) repository has an easy to
|
||||||
|
look at version written in assembly. It's API and effects are close enough to
|
||||||
|
the Nintendo version that most games will run just fine.
|
||||||
|
* You can also check out the [mGBA
|
||||||
|
bios](https://github.com/mgba-emu/mgba/blob/master/src/gba/bios.c) if you want
|
||||||
|
to see the C version of what various bios functions are doing.
|
||||||
|
|
||||||
|
### External Work RAM (EWRAM)
|
||||||
|
|
||||||
|
* **Location:** `0x200_0000` to `0x203_FFFF` (256k)
|
||||||
|
* **Bus:** 16-bit
|
||||||
|
* **Access:** Read-write, any size.
|
||||||
|
* **Wait Cycles:** 2
|
||||||
|
|
||||||
|
The external work ram is a sizable amount of space, but the 2 wait cycles per
|
||||||
|
access and 16-bit bus mean that you should probably think of it as being a
|
||||||
|
"heap" to avoid putting things in if you don't have to.
|
||||||
|
|
||||||
|
The GBA itself doesn't use this for anything, so any use is totally up to you.
|
||||||
|
|
||||||
|
At the moment, the linker script and `crt0.s` files provided with the `gba`
|
||||||
|
crate also have no defined use for the EWRAM, so it's 100% on you to decide how
|
||||||
|
you wanna use them.
|
||||||
|
|
||||||
|
(Note: There is an undocumented control register that lets you adjust the wait
|
||||||
|
cycles on EWRAM. Using it, you can turn EWRAM from the default 2 wait cycles
|
||||||
|
down to 1. However, not all GBA-like things support it. The GBA and GBA SP do,
|
||||||
|
the GBA Micro and DS do not. Emulators might or might not depending on the
|
||||||
|
particular emulator. See the [GBATEK system
|
||||||
|
control](https://problemkaputt.de/gbatek.htm#gbasystemcontrol) page for a full
|
||||||
|
description of that register, though probably only once you've read more of this
|
||||||
|
tutorial book and know how to make sense of IO registers and such.)
|
||||||
|
|
||||||
|
### Internal Work RAM (IWRAM)
|
||||||
|
|
||||||
|
* **Location:** `0x300_0000` to `0x300_7FFF` (32k)
|
||||||
|
* **Bus:** 32-bit
|
||||||
|
* **Access:** Read-write, any size.
|
||||||
|
* **Wait Cycles:** 0
|
||||||
|
|
||||||
|
This is where the "fast" memory for general purposes lives. By default the
|
||||||
|
system uses the 256 _bytes_ starting at `0x300_7F00` _and up_ for system and
|
||||||
|
interrupt purposes, while Rust's program stack starts at that same address _and
|
||||||
|
goes down_ from there.
|
||||||
|
|
||||||
|
Even though your stack exists in this space, it's totally reasonable to use the
|
||||||
|
bottom parts of this memory space for whatever quick scratch purposes, same as
|
||||||
|
EWRAM. 32k is fairly huge, and the stack going down from the top and the scratch
|
||||||
|
data going up from the bottom are unlikely to hit each other. If they do you
|
||||||
|
were probably well on your way to a stack overflow anyway.
|
||||||
|
|
||||||
|
The linker script and `crt0.s` file provided with the `gba` crate use the bottom
|
||||||
|
of IWRAM to store the `.data` and `.bss` [data
|
||||||
|
segments](https://en.wikipedia.org/wiki/Data_segment). That's where your global
|
||||||
|
variables get placed (both `static` and `static mut`). The `.data` segment holds
|
||||||
|
any variable that's initialized to non-zero, and the `.bss` section is for any
|
||||||
|
variable initialized to zero. When the GBA is powered on, some code in the
|
||||||
|
`crt0.s` file runs and copies the initial `.data` values into place within IWRAM
|
||||||
|
(all of `.bss` starts at 0, so there's no copy for those variables).
|
||||||
|
|
||||||
|
If you have no global variables at all, then you don't need to worry about those
|
||||||
|
details, but if you do have some global variables then you can use the _address
|
||||||
|
of_ the `__bss_end` symbol defined in the top of the `gba` crate as a marker for
|
||||||
|
where it's safe for you to start using IWRAM without overwriting your globals.
|
||||||
|
|
||||||
|
### IO Registers
|
||||||
|
|
||||||
|
* **Location:** `0x400_0000` to `0x400_03FE`
|
||||||
|
* **Bus:** 32-bit
|
||||||
|
* **Access:** different for each IO register
|
||||||
|
* **Wait Cycles:** 0
|
||||||
|
|
||||||
|
The IO Registers are where most of the magic happens, and it's where most of the
|
||||||
|
variety happens too. Each IO register is a specific width, usually 16-bit but
|
||||||
|
sometimes 32-bit. Most of them are fully read/write, but some of them are read
|
||||||
|
only or write only. Some of them have individual bits that are read only even
|
||||||
|
when the rest of the register is writable. Some of them can be written to, but
|
||||||
|
the write doesn't change the value you read back, it sets something else.
|
||||||
|
Really.
|
||||||
|
|
||||||
|
The IO registers are how you control every bit of hardware besides the CPU
|
||||||
|
itself. Reading the buttons, setting display modes, enabling timers, all of that
|
||||||
|
goes through different IO registers. Actually, even a few parts of the CPU's
|
||||||
|
operation can be controlled via IO register.
|
||||||
|
|
||||||
|
We'll go over IO registers more in the next section, including a few specific
|
||||||
|
registers, and then we'll constantly encounter more IO registers as we explore
|
||||||
|
each new topic through the rest of the book.
|
||||||
|
|
||||||
|
### Palette RAM (PALRAM)
|
||||||
|
|
||||||
|
* **Location:** `0x500_0000` to `0x500_03FF` (1k)
|
||||||
|
* **Bus:** 16-bit
|
||||||
|
* **Access:** Read any, single bytes mirrored (see text).
|
||||||
|
* **Wait Cycles:** Video Memory Wait (see text)
|
||||||
|
|
||||||
|
This is where the GBA stores color palette data. There's 256 slots for
|
||||||
|
Background color, and then 256 slots for Object color.
|
||||||
|
|
||||||
|
GBA colors are 15 bits each, with five bits per channel and the highest bit
|
||||||
|
being totally ignored, so we store them as `u16` values:
|
||||||
|
|
||||||
|
* `X_BBBBB_GGGGG_RRRRR`
|
||||||
|
|
||||||
|
Of note is the fact that the 256 palette slots can be viewed in two different
|
||||||
|
ways. There's two different formats for images in video memory: "8 bit per
|
||||||
|
pixel" (8bpp) and "4 bit per pixel mode" (4bpp).
|
||||||
|
|
||||||
|
* **8bpp:** Each pixel in the image is 8 bits and indexes directly into the full
|
||||||
|
256 entry palette array. An index of 0 means that pixel should be transparent,
|
||||||
|
so there's 255 possible colors.
|
||||||
|
* **4bpp:** Each pixel in the image is 4 bits and indexes into a "palbank" of 16
|
||||||
|
colors within the palette data. Some exterior control selects the palbank to
|
||||||
|
be used. An index of 0 still means that the pixel should be transparent, so
|
||||||
|
there's 15 possible colors.
|
||||||
|
|
||||||
|
Different images can use different modes all at once, as long as you can fit all
|
||||||
|
the colors you want to use into your palette layout.
|
||||||
|
|
||||||
|
PALRAM can't be written to in individual bytes. This isn't normally a problem at
|
||||||
|
all, because you wouldn't really want to write half of a color entry anyway. If
|
||||||
|
you do try to write a single byte then it gets "mirrored" into both halves of
|
||||||
|
the `u16` that would be associated with that address. For example, if you tried
|
||||||
|
to write `0x01u8` to either `0x500_0000` or `0x500_0001` then you'd actually
|
||||||
|
_effectively_ be writing `0x0101u16` to `0x500_0000`.
|
||||||
|
|
||||||
|
PALRAM follows what we'll call the "Video Memory Wait" rule: If you to access
|
||||||
|
the memory during a vertical blank or horizontal blank period there's 0 wait
|
||||||
|
cycles, and if you try to access the memory while the display controller is
|
||||||
|
drawing there is a 1 cycle wait inserted _if_ the display controller was using
|
||||||
|
that memory at that moment.
|
||||||
|
|
||||||
|
### Video RAM (VRAM)
|
||||||
|
|
||||||
|
* **Location:** `0x600_0000` to `0x601_7FFF` (96k or 64k+32k depending on mode)
|
||||||
|
* **Bus:** 16-bit
|
||||||
|
* **Access:** Read any, single bytes _sometimes_ mirrored (see text).
|
||||||
|
* **Wait Cycles:** Video Memory Wait (see text)
|
||||||
|
|
||||||
|
Video RAM is the memory for what you want the display controller to be
|
||||||
|
displaying. The GBA actually has 6 different display modes (numbered 0 through
|
||||||
|
5), and depending on the mode you're using the layout that you should imagine
|
||||||
|
VRAM having changes. Because there's so much involved here, I'll leave more
|
||||||
|
precise details to the following sections which talk about how to use VRAM in
|
||||||
|
each mode.
|
||||||
|
|
||||||
|
VRAM can't be written to in individual bytes. If you try to write a single byte
|
||||||
|
to background VRAM the byte gets mirrored like with PALRAM, and if you try with
|
||||||
|
object VRAM the write gets ignored entirely. Exactly what address ranges those
|
||||||
|
memory types are depends on video mode, but just don't bother with individual
|
||||||
|
byte writes to VRAM. If you want to change a single byte of data (and you might)
|
||||||
|
then the correct style is to read the full `u16`, mask out the old data, mask in
|
||||||
|
your new value, and then write the whole `u16`.
|
||||||
|
|
||||||
|
VRAM follows the same "Video Memory Wait" rule that PALRAM has.
|
||||||
|
|
||||||
|
### Object Attribute Memory (OAM)
|
||||||
|
|
||||||
|
* **Location:** `0x700_0000` to `0x700_03FF` (1k)
|
||||||
|
* **Bus:** 32-bit
|
||||||
|
* **Access:** Read any, single bytes no effect (see text).
|
||||||
|
* **Wait Cycles:** Video Memory Wait (see text)
|
||||||
|
|
||||||
|
This part of memory controls the "Objects" (OBJ) on the screen. An object is
|
||||||
|
_similar to_ the concept of a "sprite". However, because of an object's size
|
||||||
|
limitations, a single sprite might require more than one object to be drawn
|
||||||
|
properly. In general, if you want to think in terms of sprites at all, you
|
||||||
|
should think of sprites as being a logical / programming concept, and objects as
|
||||||
|
being a hardware concept.
|
||||||
|
|
||||||
|
While VRAM has the _image_ data for each object, this part of memory has the
|
||||||
|
_control_ data for each object. An objects "attributes" describe what part of
|
||||||
|
the VRAM to use, where to place is on the screen, any special graphical effects
|
||||||
|
to use, all that stuff. Each object has 6 bytes of attribute data (arranged as
|
||||||
|
three `u16` values), and there's a total of 128 objects (indexed 0 through 127).
|
||||||
|
|
||||||
|
But 6 bytes each times 128 entries out of 1024 bytes leaves us with 256 bytes
|
||||||
|
left over. What's the other space used for? Well, it's a little weird, but after
|
||||||
|
every three `u16` object attribute fields there's one `i16` "affine parameter"
|
||||||
|
field mixed in. It takes four such fields to make a complete set of affine
|
||||||
|
parameters (a 2x2 matrix), so we get a total of 32 affine parameter entries
|
||||||
|
across all of OAM. "Affine" might sound fancy but it just means a transformation
|
||||||
|
where anything that started parallel stays parallel after the transform. The
|
||||||
|
affine parameters can be used to scale, rotate, and/or skew a background or
|
||||||
|
object as it's being displayed on the screen. It takes more computing power than
|
||||||
|
the non-affine display, so you can't display as many different things at once
|
||||||
|
when using the affine modes.
|
||||||
|
|
||||||
|
OAM can't ever be written to with individual bytes. The write just has no effect
|
||||||
|
at all.
|
||||||
|
|
||||||
|
OAM follows the same "Video Memory Wait" rule that PALRAM has, **and** you can
|
||||||
|
also only freely access OAM during a horizontal blank if you set a special
|
||||||
|
"HBlank Interval Free" bit in one of the IO registers (the "Display Control"
|
||||||
|
register, which we'll talk about next lesson). The reason that you might _not_
|
||||||
|
want to set that bit is because when it's enabled you can't draw as many objects
|
||||||
|
at once. You don't lose the use of an exact number of objects, you actually lose
|
||||||
|
the use of a number of display adapter drawing cycles. Since not all objects
|
||||||
|
take the same number of cycles to render, it depends on what you're drawing.
|
||||||
|
GBATEK [has the details](https://problemkaputt.de/gbatek.htm#lcdobjoverview) if
|
||||||
|
you want to know precisely.
|
||||||
|
|
||||||
|
### Game Pak ROM (ROM)
|
||||||
|
|
||||||
|
* **Location:** Special (max of 32MB)
|
||||||
|
* **Bus:** 16-bit
|
||||||
|
* **Access:** Special
|
||||||
|
* **Wait Cycles:** Special
|
||||||
|
|
||||||
|
This is where your actual game is located! As you might guess, since each
|
||||||
|
cartridge is different, the details here depend quite a bit on the cartridge
|
||||||
|
that you use for your game. Even a simple statement like "you can't write to the
|
||||||
|
ROM region" isn't true for some carts if they have FlashROM.
|
||||||
|
|
||||||
|
The _most important_ thing to concern yourself with when considering the ROM
|
||||||
|
portion of memory is the 32MB limit. That's compiled code, images, sound,
|
||||||
|
everything put together. The total has to stay under 32MB.
|
||||||
|
|
||||||
|
The next most important thing to consider is that 16-bit bus. It means that we
|
||||||
|
compile our programs using "Thumb state" code instead of "ARM state" code.
|
||||||
|
Details about this can be found in the GBA Assembly section of the book, but
|
||||||
|
just be aware that there's two different types of assembly on the GBA. You can
|
||||||
|
switch between them, but the default for us is always Thumb state.
|
||||||
|
|
||||||
|
Another detail which you actually _don't_ have to think about much, but that you
|
||||||
|
might care if you're doing precise optimization, is that the ROM address space
|
||||||
|
is actually mirrored across three different locations:
|
||||||
|
|
||||||
|
* `0x800_0000` to `0x9FF_FFFF`: Wait State 0
|
||||||
|
* `0xA00_0000` to `0xBFF_FFFF`: Wait State 1
|
||||||
|
* `0xC00_0000` to `0xDFF_FFFF`: Wait State 2
|
||||||
|
|
||||||
|
These _don't_ mean 0, 1, and 2 wait cycles, they mean the wait cycles associated
|
||||||
|
with ROM mirrors 0, 1, and 2. On some carts the game will store different parts
|
||||||
|
of the data into different chips that are wired to be accessible through
|
||||||
|
different parts of the mirroring. The actual wait cycles used are even
|
||||||
|
configurable via an IO register called the
|
||||||
|
[WAITCNT](https://problemkaputt.de/gbatek.htm#gbasystemcontrol) ("Wait Control",
|
||||||
|
I don't know why C programmers have to give everything the worst names it's not
|
||||||
|
1980 any more).
|
||||||
|
|
||||||
|
### Save RAM (SRAM)
|
||||||
|
|
||||||
|
* **Location:** Special (max of 64k)
|
||||||
|
* **Bus:** 8-bit
|
||||||
|
* **Access:** Special
|
||||||
|
* **Wait Cycles:** Special
|
||||||
|
|
||||||
|
The Save RAM is also part of the cart that you've got your game on, so it also
|
||||||
|
depends on your hardware.
|
||||||
|
|
||||||
|
SRAM _starts_ at `0xE00_0000` and you can save up to however much the hardware
|
||||||
|
supports, to a maximum of 64k. However, you can only read and write SRAM one
|
||||||
|
_byte_ at a time. What's worse, while you can _write_ to SRAM using code
|
||||||
|
executing anywhere, you can only _read_ with code that's executing out of either
|
||||||
|
Internal or External Work RAM, not from with code that's executing out of ROM.
|
||||||
|
This means that you need to copy the code for doing the read into some scratch
|
||||||
|
space (either at startup or on the fly, doesn't matter) and call that function
|
||||||
|
you've carefully placed. It's a bit annoying, but soon enough a routine for it
|
||||||
|
all will be provided in the `gba` crate and we won't have to worry too much
|
||||||
|
about it.
|
||||||
|
|
||||||
|
(TODO: Provide the routine that I just claimed we would provide.)
|
48
book/src/volatile.md
Normal file
48
book/src/volatile.md
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
# Volatile
|
||||||
|
|
||||||
|
I know that you just got your first program running and you're probably excited
|
||||||
|
to learn more about GBA stuff, but first we have to cover a subject that's not
|
||||||
|
quite GBA specific.
|
||||||
|
|
||||||
|
In the `hello_magic.rs` file we had these lines
|
||||||
|
|
||||||
|
```rust
|
||||||
|
(0x600_0000 as *mut u16).offset(120 + 80 * 240).write_volatile(0x001F);
|
||||||
|
(0x600_0000 as *mut u16).offset(136 + 80 * 240).write_volatile(0x03E0);
|
||||||
|
(0x600_0000 as *mut u16).offset(120 + 96 * 240).write_volatile(0x7C00);
|
||||||
|
```
|
||||||
|
|
||||||
|
You've probably seen or heard of the
|
||||||
|
[write](https://doc.rust-lang.org/core/ptr/fn.write.html) function before, but
|
||||||
|
you'd be excused if you've never heard of its cousin function,
|
||||||
|
[write_volatile](https://doc.rust-lang.org/core/ptr/fn.write_volatile.html).
|
||||||
|
|
||||||
|
What's the difference? Well, when the compiler sees normal reads and writes, it
|
||||||
|
assumes that those go into plain old memory locations. CPU registers, RAM,
|
||||||
|
wherever it is that the value's being placed. The compiler assumes that it's
|
||||||
|
safe to optimize away some of the reads and writes, or maybe issue the reads and
|
||||||
|
writes in a different order from what you wrote. Normally this is okay, and it's
|
||||||
|
exactly what we want the compiler to be doing, quietly making things faster for us.
|
||||||
|
|
||||||
|
However, some of the time we access values from parts of memory where it's
|
||||||
|
important that each access happen, and in the exact order that we say. In our
|
||||||
|
`hello_magic.rs` example, we're writing directly into the video memory of the
|
||||||
|
display. The compiler sees that the rest of the Rust program never read out of
|
||||||
|
those locations, so it might think "oh, we can skip those writes, they're
|
||||||
|
pointless". It doesn't know that we're having a side effect besides just storing
|
||||||
|
some value at an address.
|
||||||
|
|
||||||
|
By declaring a particular read or write to be `volatile` then we can force the
|
||||||
|
compiler to issue that access. Further, we're guaranteed that all `volatile`
|
||||||
|
access will happen in exactly the order it appears in the program relative to
|
||||||
|
other `volatile` access. However, non-volatile access can still be re-ordered
|
||||||
|
relative to a volatile access. In other words, for parts of the memory that are
|
||||||
|
volatile, we must _always_ use a volatile read or write for our program to
|
||||||
|
perform properly.
|
||||||
|
|
||||||
|
For exactly this reason, we've got the [voladdress](https://docs.rs/voladdress/)
|
||||||
|
crate. It used to be part of the GBA crate, but it became big enough to break
|
||||||
|
out into a stand alone crate. It doesn't even do too much, it just makes it a
|
||||||
|
lot less error prone to accidentally forget to use volatile with our memory
|
||||||
|
mapped addresses. We just call `read` and `write` on any `VolAddress` that we
|
||||||
|
happen to see and the right thing will happen.
|
|
@ -133,7 +133,10 @@ extern "C" {
|
||||||
/// Memory in IWRAM _before_ this location is not free to use, you'll trash
|
/// Memory in IWRAM _before_ this location is not free to use, you'll trash
|
||||||
/// your globals and stuff. Memory here or after is freely available for use
|
/// your globals and stuff. Memory here or after is freely available for use
|
||||||
/// (careful that you don't run into your own stack of course).
|
/// (careful that you don't run into your own stack of course).
|
||||||
static __bss_end: u8;
|
///
|
||||||
|
/// The actual value is unimportant, you just want to use the _address of_
|
||||||
|
/// this location as the start of your IWRAM usage.
|
||||||
|
pub static __bss_end: u8;
|
||||||
}
|
}
|
||||||
|
|
||||||
newtype! {
|
newtype! {
|
||||||
|
|
Loading…
Reference in a new issue