mirror of
https://github.com/italicsjenga/agb.git
synced 2025-01-26 00:56:38 +11:00
Merge pull request #149 from gwilymk/pong-book
First 3 chapters on writing a pong game
This commit is contained in:
commit
311c26caad
25 changed files with 1255 additions and 65 deletions
20
.github/scripts/build-example-gba-files.sh
vendored
20
.github/scripts/build-example-gba-files.sh
vendored
|
@ -6,26 +6,28 @@ set -x # print every command before it runs
|
||||||
# Requires gbafix and arm-none-eabi-objcopy to already be installed
|
# Requires gbafix and arm-none-eabi-objcopy to already be installed
|
||||||
|
|
||||||
function build_rom() {
|
function build_rom() {
|
||||||
local GAME_NAME="$1"
|
local GAME_FOLDER="$1"
|
||||||
local INTERNAL_NAME="$2"
|
local INTERNAL_NAME="$2"
|
||||||
|
|
||||||
local TARGET_FOLDER="${CARGO_TARGET_DIR:-target}"
|
local GAME_NAME
|
||||||
|
GAME_NAME="$(basename "$GAME_FOLDER")"
|
||||||
|
|
||||||
|
local TARGET_FOLDER="${CARGO_TARGET_DIR:-$GAME_FOLDER/target}"
|
||||||
local GBA_FILE="$TARGET_FOLDER/$GAME_NAME.gba"
|
local GBA_FILE="$TARGET_FOLDER/$GAME_NAME.gba"
|
||||||
|
|
||||||
pushd "examples/$GAME_NAME"
|
(cd "$GAME_FOLDER" && cargo build --release --verbose --target thumbv4t-none-eabi)
|
||||||
cargo build --release --verbose --target thumbv4t-none-eabi
|
|
||||||
|
|
||||||
arm-none-eabi-objcopy -O binary "$TARGET_FOLDER/thumbv4t-none-eabi/release/$GAME_NAME" "$GBA_FILE"
|
arm-none-eabi-objcopy -O binary "$TARGET_FOLDER/thumbv4t-none-eabi/release/$GAME_NAME" "$GBA_FILE"
|
||||||
gbafix -p "-t${INTERNAL_NAME:0:12}" "-c${INTERNAL_NAME:0:4}" -mGC "$GBA_FILE"
|
gbafix -p "-t${INTERNAL_NAME:0:12}" "-c${INTERNAL_NAME:0:4}" -mGC "$GBA_FILE"
|
||||||
|
|
||||||
cp -v "$GBA_FILE" "../$GAME_NAME.gba"
|
cp -v "$GBA_FILE" "examples/$GAME_NAME.gba"
|
||||||
|
|
||||||
popd
|
|
||||||
}
|
}
|
||||||
|
|
||||||
mkdir -p examples/target
|
mkdir -p examples/target
|
||||||
|
|
||||||
build_rom "the-purple-night" "PURPLENIGHT"
|
build_rom "examples/the-purple-night" "PURPLENIGHT"
|
||||||
build_rom "the-hat-chooses-the-wizard" "HATWIZARD"
|
build_rom "examples/the-hat-chooses-the-wizard" "HATWIZARD"
|
||||||
|
|
||||||
|
build_rom "book/games/pong" "PONG"
|
||||||
|
|
||||||
zip examples/target/examples.zip examples/*.gba
|
zip examples/target/examples.zip examples/*.gba
|
|
@ -60,7 +60,7 @@ pub fn include_gfx(input: TokenStream) -> TokenStream {
|
||||||
});
|
});
|
||||||
|
|
||||||
let module = quote! {
|
let module = quote! {
|
||||||
pub mod #module_name {
|
mod #module_name {
|
||||||
const _: &[u8] = include_bytes!(#include_path);
|
const _: &[u8] = include_bytes!(#include_path);
|
||||||
|
|
||||||
#(#image_code)*
|
#(#image_code)*
|
||||||
|
|
|
@ -15,7 +15,30 @@ const PALETTE_SPRITE: MemoryMapped1DArray<u16, 256> =
|
||||||
const TILE_SPRITE: MemoryMapped1DArray<u32, { 1024 * 8 }> =
|
const TILE_SPRITE: MemoryMapped1DArray<u32, { 1024 * 8 }> =
|
||||||
unsafe { MemoryMapped1DArray::new(0x06010000) };
|
unsafe { MemoryMapped1DArray::new(0x06010000) };
|
||||||
|
|
||||||
/// Handles distributing objects and matricies along with operations that effect all objects.
|
/// Handles distributing objects and matrices along with operations that effect all objects.
|
||||||
|
/// You can create an instance of this using the Gba struct.
|
||||||
|
///
|
||||||
|
/// This handles distribution of sprites, ensuring that object ids are not reused and are
|
||||||
|
/// returned to the pool once you're done handling them.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// # #![no_std]
|
||||||
|
/// # #![no_main]
|
||||||
|
/// #
|
||||||
|
/// # extern crate agb;
|
||||||
|
/// #
|
||||||
|
/// # use agb::Gba;
|
||||||
|
/// #
|
||||||
|
/// # #[agb::entry]
|
||||||
|
/// # fn main() -> ! {
|
||||||
|
/// let mut gba = Gba::new();
|
||||||
|
/// let mut object = gba.display.object.get();
|
||||||
|
/// #
|
||||||
|
/// # loop {}
|
||||||
|
/// # }
|
||||||
|
/// ```
|
||||||
pub struct ObjectControl {
|
pub struct ObjectControl {
|
||||||
objects: RefCell<Bitarray<4>>,
|
objects: RefCell<Bitarray<4>>,
|
||||||
affines: AffineArena,
|
affines: AffineArena,
|
||||||
|
@ -27,6 +50,41 @@ struct ObjectLoan<'a> {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The standard object, without rotation.
|
/// The standard object, without rotation.
|
||||||
|
///
|
||||||
|
/// You should create this from an instance of ObjectControl created using the Gba struct. Note that
|
||||||
|
/// no changes made to this will be visible until `commit()` is called. You should call `commit()` during
|
||||||
|
/// vblank to ensure that you get no visual artifacts.
|
||||||
|
///
|
||||||
|
/// This struct implements a sort of builder pattern, allowing you to chain settings together.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// # #![no_std]
|
||||||
|
/// # #![no_main]
|
||||||
|
/// #
|
||||||
|
/// # extern crate agb;
|
||||||
|
/// #
|
||||||
|
/// # use agb::Gba;
|
||||||
|
/// use agb::display::object::Size;
|
||||||
|
///
|
||||||
|
/// # #[agb::entry]
|
||||||
|
/// # fn main() -> ! {
|
||||||
|
/// # let mut gba = Gba::new();
|
||||||
|
/// let mut object = gba.display.object.get();
|
||||||
|
///
|
||||||
|
/// let mut my_new_object = object.get_object_standard();
|
||||||
|
/// my_new_object.set_x(50)
|
||||||
|
/// .set_y(50)
|
||||||
|
/// .set_sprite_Size(Size::S8x8)
|
||||||
|
/// .set_tile_id(7)
|
||||||
|
/// .show();
|
||||||
|
///
|
||||||
|
/// // some time later in vblank
|
||||||
|
/// my_new_object.commit();
|
||||||
|
/// # loop {}
|
||||||
|
/// # }
|
||||||
|
/// ```
|
||||||
pub struct ObjectStandard<'a> {
|
pub struct ObjectStandard<'a> {
|
||||||
attributes: ObjectAttribute,
|
attributes: ObjectAttribute,
|
||||||
loan: ObjectLoan<'a>,
|
loan: ObjectLoan<'a>,
|
||||||
|
@ -91,49 +149,77 @@ impl ObjectStandard<'_> {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Sets the x coordinate of the sprite on screen.
|
/// Sets the x coordinate of the sprite on screen.
|
||||||
pub fn set_x(&mut self, x: u16) {
|
pub fn set_x(&mut self, x: u16) -> &mut Self {
|
||||||
self.attributes.set_x(x)
|
self.attributes.set_x(x);
|
||||||
}
|
|
||||||
/// Sets the y coordinate of the sprite on screen.
|
self
|
||||||
pub fn set_y(&mut self, y: u16) {
|
|
||||||
self.attributes.set_y(y)
|
|
||||||
}
|
|
||||||
/// Sets the index of the tile to use as the sprite. Potentially a temporary function.
|
|
||||||
pub fn set_tile_id(&mut self, id: u16) {
|
|
||||||
self.attributes.set_tile_id(id)
|
|
||||||
}
|
|
||||||
/// Sets whether the sprite is horizontally mirrored or not.
|
|
||||||
pub fn set_hflip(&mut self, hflip: bool) {
|
|
||||||
self.attributes.set_hflip(hflip)
|
|
||||||
}
|
|
||||||
/// Sets the sprite size, will read tiles in x major order to construct this.
|
|
||||||
pub fn set_sprite_size(&mut self, size: Size) {
|
|
||||||
self.attributes.set_size(size);
|
|
||||||
}
|
|
||||||
/// Show the object on screen.
|
|
||||||
pub fn show(&mut self) {
|
|
||||||
self.attributes.set_mode(Mode::Normal)
|
|
||||||
}
|
|
||||||
/// Hide the object and do not render.
|
|
||||||
pub fn hide(&mut self) {
|
|
||||||
self.attributes.set_mode(Mode::Hidden)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_palette(&mut self, palette: u16) {
|
/// Sets the y coordinate of the sprite on screen.
|
||||||
|
pub fn set_y(&mut self, y: u16) -> &mut Self {
|
||||||
|
self.attributes.set_y(y);
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets the index of the tile to use as the sprite. Potentially a temporary function.
|
||||||
|
pub fn set_tile_id(&mut self, id: u16) -> &mut Self {
|
||||||
|
self.attributes.set_tile_id(id);
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets whether the sprite is horizontally mirrored or not.
|
||||||
|
pub fn set_hflip(&mut self, hflip: bool) -> &mut Self {
|
||||||
|
self.attributes.set_hflip(hflip);
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets the sprite size, will read tiles in x major order to construct this.
|
||||||
|
pub fn set_sprite_size(&mut self, size: Size) -> &mut Self {
|
||||||
|
self.attributes.set_size(size);
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Show the object on screen.
|
||||||
|
pub fn show(&mut self) -> &mut Self {
|
||||||
|
self.attributes.set_mode(Mode::Normal);
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Hide the object and do not render.
|
||||||
|
pub fn hide(&mut self) -> &mut Self {
|
||||||
|
self.attributes.set_mode(Mode::Hidden);
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sets the palette to use for this sprite
|
||||||
|
pub fn set_palette(&mut self, palette: u16) -> &mut Self {
|
||||||
self.attributes.set_palette(palette);
|
self.attributes.set_palette(palette);
|
||||||
|
|
||||||
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Sets the x and y position of the object, performing casts as nessesary
|
/// Sets the x and y position of the object, performing casts as nessesary
|
||||||
/// to fit within the bits allocated for this purpose.
|
/// to fit within the bits allocated for this purpose.
|
||||||
pub fn set_position(&mut self, position: Vector2D<i32>) {
|
pub fn set_position(&mut self, position: Vector2D<i32>) -> &mut Self {
|
||||||
let x = position.x as u16;
|
let x = position.x as u16;
|
||||||
let y = position.y as u16;
|
let y = position.y as u16;
|
||||||
self.attributes.set_x(x);
|
self.attributes.set_x(x);
|
||||||
self.attributes.set_y(y);
|
self.attributes.set_y(y);
|
||||||
|
|
||||||
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_priority(&mut self, p: Priority) {
|
/// Sets the priority (used for z ordering) of this sprite
|
||||||
self.attributes.set_priority(p)
|
pub fn set_priority(&mut self, p: Priority) -> &mut Self {
|
||||||
|
self.attributes.set_priority(p);
|
||||||
|
|
||||||
|
self
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
172
agb/src/lib.rs
172
agb/src/lib.rs
|
@ -13,9 +13,130 @@
|
||||||
//! programming language. It attempts to be a high level abstraction over the
|
//! programming language. It attempts to be a high level abstraction over the
|
||||||
//! internal workings of the Game Boy Advance whilst still being high
|
//! internal workings of the Game Boy Advance whilst still being high
|
||||||
//! performance and memory efficient.
|
//! performance and memory efficient.
|
||||||
|
//!
|
||||||
|
//! To get started with agb, you should clone the [template repo](https://github.com/agbrs/template) and work from there.
|
||||||
|
|
||||||
|
/// This macro is used to convert a png or bmp into a format usable by the Game Boy Advance.
|
||||||
|
///
|
||||||
|
/// The macro expects to be linked to a `toml` file which contains a metadata about the image
|
||||||
|
/// and a link to the png or bmp itself. See the examples below for a full definition of the format.
|
||||||
|
///
|
||||||
|
/// # The manifest file format
|
||||||
|
///
|
||||||
|
/// The following is an example of the toml file you would need to create. Generally you will
|
||||||
|
/// find this in the `gfx` folder in the same level as the `src` folder (see the examples).
|
||||||
|
///
|
||||||
|
/// Suppose that the following is in `gfx/sprites.toml`.
|
||||||
|
///
|
||||||
|
/// ```toml
|
||||||
|
/// version = "1.0" # version included for compatibility reasons
|
||||||
|
///
|
||||||
|
/// [images.objects]
|
||||||
|
/// filename = "sprites.png"
|
||||||
|
/// tile_size = "16x16"
|
||||||
|
/// transparent_colour = "ff0044"
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// You then import this using:
|
||||||
|
/// ```rust,ignore
|
||||||
|
/// agb::include_gfx!("gfx/sprites.toml");
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// This will generate something along the lines of the following:
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// // module name comes from the name of the toml file, so `sprites` in this case because it is
|
||||||
|
/// // called `sprites.toml`
|
||||||
|
/// mod sprites {
|
||||||
|
/// const objects = /* ... */;
|
||||||
|
/// }
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// And objects will be an instance of [`TileData`][crate::display::tile_data::TileData]
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ## Loading sprites:
|
||||||
|
///
|
||||||
|
/// In `gfx/sprites.toml`:
|
||||||
|
/// ```toml
|
||||||
|
/// version = "1.0"
|
||||||
|
///
|
||||||
|
/// [image.sprites]
|
||||||
|
/// filename = "sprites.png"
|
||||||
|
/// tile_size = "16x16"
|
||||||
|
/// transparent_colour = "ff0044"
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// In `src/main.rs`
|
||||||
|
/// ```
|
||||||
|
/// mod gfx {
|
||||||
|
/// use agb::display::object::ObjectControl;
|
||||||
|
///
|
||||||
|
/// // Import the sprites into this module. This will create a `sprites` module
|
||||||
|
/// // and within that will be a constant called `sprites` which houses all the
|
||||||
|
/// // palette and tile data.
|
||||||
|
/// agb::include_gfx!("gfx/sprites.toml");
|
||||||
|
///
|
||||||
|
/// // Loads the sprites tile data and palette data into VRAM
|
||||||
|
/// pub fn load_sprite_data(object: &mut ObjectControl) {
|
||||||
|
/// object.set_sprite_palettes(sprites::sprites.palettes);
|
||||||
|
/// object.set_sprite_tilemap(sprites::sprites.tiles);
|
||||||
|
/// }
|
||||||
|
/// }
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// ## Loading tiles:
|
||||||
|
///
|
||||||
|
/// In `gfx/tiles.toml`:
|
||||||
|
/// ```toml
|
||||||
|
/// version = "1.0"
|
||||||
|
///
|
||||||
|
/// [image.background]
|
||||||
|
/// filename = "tile_sheet.png"
|
||||||
|
/// tile_size = "8x8"
|
||||||
|
/// transparent_colour = "2ce8f4"
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// In `src/main.rs`:
|
||||||
|
/// ```
|
||||||
|
/// mod gfx {
|
||||||
|
/// use agb::display::background::BackgroundDistributor;
|
||||||
|
///
|
||||||
|
/// agb::include_gfx!("gfx/tile_sheet.toml");
|
||||||
|
///
|
||||||
|
/// pub fn load_tile_sheet(tiled: &mut BackgroundDistributor) {
|
||||||
|
/// tiled.set_background_palettes(tile_sheet::background.palettes);
|
||||||
|
/// tiled.set_background_tilemap(tile_sheet::background.tiles);
|
||||||
|
/// }
|
||||||
|
/// }
|
||||||
|
/// ```
|
||||||
pub use agb_image_converter::include_gfx;
|
pub use agb_image_converter::include_gfx;
|
||||||
|
|
||||||
|
/// This macro declares the entry point to your game written using `agb`.
|
||||||
|
///
|
||||||
|
/// It is already included in the template, but your `main` function must be annotated with `#[agb::entry]`, take no arguments and never return.
|
||||||
|
/// Doing this will ensure that `agb` can correctly set up the environment to call your rust function on start up.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
/// ```
|
||||||
|
/// #![no_std]
|
||||||
|
/// #![no_main]
|
||||||
|
///
|
||||||
|
/// // Required to set panic handlers
|
||||||
|
/// extern crate agb;
|
||||||
|
///
|
||||||
|
/// use agb::Gba;
|
||||||
|
///
|
||||||
|
/// #[agb::entry]
|
||||||
|
/// fn main() -> ! {
|
||||||
|
/// let mut gba = Gba::new();
|
||||||
|
///
|
||||||
|
/// loop {}
|
||||||
|
/// }
|
||||||
|
/// ```
|
||||||
pub use agb_macros::entry;
|
pub use agb_macros::entry;
|
||||||
|
|
||||||
pub use agb_sound_converter::include_wav;
|
pub use agb_sound_converter::include_wav;
|
||||||
|
|
||||||
#[cfg(feature = "alloc")]
|
#[cfg(feature = "alloc")]
|
||||||
|
@ -29,10 +150,12 @@ mod bitarray;
|
||||||
pub mod display;
|
pub mod display;
|
||||||
/// Button inputs to the system.
|
/// Button inputs to the system.
|
||||||
pub mod input;
|
pub mod input;
|
||||||
|
#[doc(hidden)] // hide for now as the implementation in here is unsound
|
||||||
pub mod interrupt;
|
pub mod interrupt;
|
||||||
mod memory_mapped;
|
mod memory_mapped;
|
||||||
/// Implements logging to the mgba emulator.
|
/// Implements logging to the mgba emulator.
|
||||||
pub mod mgba;
|
pub mod mgba;
|
||||||
|
/// Implementation of fixnums for working with non-integer values.
|
||||||
pub mod number;
|
pub mod number;
|
||||||
mod single;
|
mod single;
|
||||||
/// Implements sound output.
|
/// Implements sound output.
|
||||||
|
@ -59,14 +182,57 @@ fn panic_implementation(info: &core::panic::PanicInfo) -> ! {
|
||||||
|
|
||||||
static mut GBASINGLE: single::Singleton<Gba> = single::Singleton::new(unsafe { Gba::single_new() });
|
static mut GBASINGLE: single::Singleton<Gba> = single::Singleton::new(unsafe { Gba::single_new() });
|
||||||
|
|
||||||
|
/// The Gba struct is used to control access to the Game Boy Advance's hardware in a way which makes it the
|
||||||
|
/// borrow checker's responsibility to ensure no clashes of global resources.
|
||||||
|
///
|
||||||
|
/// This is typically created once at the start of the main function and then the various fields are used
|
||||||
|
/// to ensure mutually exclusive use of the various hardware registers. It provides a gateway into the main
|
||||||
|
/// usage of `agb` library.
|
||||||
|
///
|
||||||
|
/// # Panics
|
||||||
|
///
|
||||||
|
/// Calling this twice will panic.
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// #![no_std]
|
||||||
|
/// #![no_main]
|
||||||
|
///
|
||||||
|
/// extern crate agb;
|
||||||
|
///
|
||||||
|
/// use agb::Gba;
|
||||||
|
///
|
||||||
|
/// #[agb::entry]
|
||||||
|
/// fn main() -> ! {
|
||||||
|
/// let mut gba = Gba::new();
|
||||||
|
///
|
||||||
|
/// // Do whatever you need to do with gba
|
||||||
|
///
|
||||||
|
/// loop {}
|
||||||
|
/// }
|
||||||
|
/// ```
|
||||||
|
#[non_exhaustive]
|
||||||
pub struct Gba {
|
pub struct Gba {
|
||||||
|
/// Manages access to the Game Boy Advance's display hardware
|
||||||
pub display: display::Display,
|
pub display: display::Display,
|
||||||
|
/// Manages access to the Game Boy Advance's beeps and boops sound hardware as part of the
|
||||||
|
/// original Game Boy's sound chip (the DMG).
|
||||||
pub sound: sound::dmg::Sound,
|
pub sound: sound::dmg::Sound,
|
||||||
|
/// Manages access to the Game Boy Advance's direct sound mixer for playing raw wav files.
|
||||||
pub mixer: sound::mixer::MixerController,
|
pub mixer: sound::mixer::MixerController,
|
||||||
|
/// Manages access to the Game Boy Advance's 4 timers.
|
||||||
pub timers: timer::TimerController,
|
pub timers: timer::TimerController,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Gba {
|
impl Gba {
|
||||||
|
/// Creates a new instance of the Gba struct.
|
||||||
|
///
|
||||||
|
/// Note that you can only create 1 instance, and trying to create a second will panic.
|
||||||
|
///
|
||||||
|
/// # Panics
|
||||||
|
///
|
||||||
|
/// Panics if you try to create the second instance.
|
||||||
pub fn new() -> Self {
|
pub fn new() -> Self {
|
||||||
unsafe { GBASINGLE.take() }
|
unsafe { GBASINGLE.take() }
|
||||||
}
|
}
|
||||||
|
@ -87,6 +253,7 @@ impl Default for Gba {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
pub trait Testable {
|
pub trait Testable {
|
||||||
fn run(&self, gba: &mut Gba);
|
fn run(&self, gba: &mut Gba);
|
||||||
}
|
}
|
||||||
|
@ -123,6 +290,7 @@ fn panic_implementation(info: &core::panic::PanicInfo) -> ! {
|
||||||
loop {}
|
loop {}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
pub fn test_runner(tests: &[&dyn Testable]) {
|
pub fn test_runner(tests: &[&dyn Testable]) {
|
||||||
let mut mgba = mgba::Mgba::new().unwrap();
|
let mut mgba = mgba::Mgba::new().unwrap();
|
||||||
mgba.print(
|
mgba.print(
|
||||||
|
@ -144,9 +312,9 @@ pub fn test_runner(tests: &[&dyn Testable]) {
|
||||||
.unwrap();
|
.unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
#[no_mangle]
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
pub extern "C" fn main() -> ! {
|
#[entry]
|
||||||
|
fn agb_test_main() -> ! {
|
||||||
test_main();
|
test_main();
|
||||||
loop {}
|
loop {}
|
||||||
}
|
}
|
||||||
|
|
10
book/games/pong/.cargo/config.toml
Normal file
10
book/games/pong/.cargo/config.toml
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
[unstable]
|
||||||
|
build-std = ["core", "alloc"]
|
||||||
|
build-std-features = ["compiler-builtins-mem"]
|
||||||
|
|
||||||
|
[build]
|
||||||
|
target = "thumbv4t-none-eabi"
|
||||||
|
|
||||||
|
[target.thumbv4t-none-eabi]
|
||||||
|
rustflags = ["-Clink-arg=-Tgba.ld"]
|
||||||
|
runner = "mgba-qt"
|
1
book/games/pong/.gitignore
vendored
Normal file
1
book/games/pong/.gitignore
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
target
|
336
book/games/pong/Cargo.lock
generated
Normal file
336
book/games/pong/Cargo.lock
generated
Normal file
|
@ -0,0 +1,336 @@
|
||||||
|
# This file is automatically @generated by Cargo.
|
||||||
|
# It is not intended for manual editing.
|
||||||
|
version = 3
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "adler32"
|
||||||
|
version = "1.2.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "aae1277d39aeec15cb388266ecc24b11c80469deae6067e17a1a7aa9e5c1f234"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "agb"
|
||||||
|
version = "0.8.0"
|
||||||
|
dependencies = [
|
||||||
|
"agb_image_converter",
|
||||||
|
"agb_macros",
|
||||||
|
"agb_sound_converter",
|
||||||
|
"bitflags",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "agb_image_converter"
|
||||||
|
version = "0.6.0"
|
||||||
|
dependencies = [
|
||||||
|
"image",
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"serde",
|
||||||
|
"syn",
|
||||||
|
"toml",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "agb_macros"
|
||||||
|
version = "0.1.0"
|
||||||
|
dependencies = [
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"rand",
|
||||||
|
"syn",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "agb_sound_converter"
|
||||||
|
version = "0.1.0"
|
||||||
|
dependencies = [
|
||||||
|
"hound",
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"siphasher",
|
||||||
|
"syn",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "autocfg"
|
||||||
|
version = "1.0.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "cdb031dd78e28731d87d56cc8ffef4a8f36ca26c38fe2de700543e627f8a464a"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "bitflags"
|
||||||
|
version = "1.3.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "bytemuck"
|
||||||
|
version = "1.7.3"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "439989e6b8c38d1b6570a384ef1e49c8848128f5a97f3914baef02920842712f"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "byteorder"
|
||||||
|
version = "1.4.3"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "14c189c53d098945499cdfa7ecc63567cf3886b3332b312a5b4585d8d3a6a610"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "cfg-if"
|
||||||
|
version = "1.0.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "color_quant"
|
||||||
|
version = "1.1.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "3d7b894f5411737b7867f4827955924d7c254fc9f4d91a6aad6b097804b1018b"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "crc32fast"
|
||||||
|
version = "1.3.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "738c290dfaea84fc1ca15ad9c168d083b05a714e1efddd8edaab678dc28d2836"
|
||||||
|
dependencies = [
|
||||||
|
"cfg-if",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "deflate"
|
||||||
|
version = "0.8.6"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "73770f8e1fe7d64df17ca66ad28994a0a623ea497fa69486e14984e715c5d174"
|
||||||
|
dependencies = [
|
||||||
|
"adler32",
|
||||||
|
"byteorder",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "getrandom"
|
||||||
|
version = "0.2.3"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "7fcd999463524c52659517fe2cea98493cfe485d10565e7b0fb07dbba7ad2753"
|
||||||
|
dependencies = [
|
||||||
|
"cfg-if",
|
||||||
|
"libc",
|
||||||
|
"wasi",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "hound"
|
||||||
|
version = "3.4.0"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "8a164bb2ceaeff4f42542bdb847c41517c78a60f5649671b2a07312b6e117549"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "image"
|
||||||
|
version = "0.23.14"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "24ffcb7e7244a9bf19d35bf2883b9c080c4ced3c07a9895572178cdb8f13f6a1"
|
||||||
|
dependencies = [
|
||||||
|
"bytemuck",
|
||||||
|
"byteorder",
|
||||||
|
"color_quant",
|
||||||
|
"num-iter",
|
||||||
|
"num-rational",
|
||||||
|
"num-traits",
|
||||||
|
"png",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "libc"
|
||||||
|
version = "0.2.112"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "1b03d17f364a3a042d5e5d46b053bbbf82c92c9430c592dd4c064dc6ee997125"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "miniz_oxide"
|
||||||
|
version = "0.3.7"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "791daaae1ed6889560f8c4359194f56648355540573244a5448a83ba1ecc7435"
|
||||||
|
dependencies = [
|
||||||
|
"adler32",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "num-integer"
|
||||||
|
version = "0.1.44"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "d2cc698a63b549a70bc047073d2949cce27cd1c7b0a4a862d08a8031bc2801db"
|
||||||
|
dependencies = [
|
||||||
|
"autocfg",
|
||||||
|
"num-traits",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "num-iter"
|
||||||
|
version = "0.1.42"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "b2021c8337a54d21aca0d59a92577a029af9431cb59b909b03252b9c164fad59"
|
||||||
|
dependencies = [
|
||||||
|
"autocfg",
|
||||||
|
"num-integer",
|
||||||
|
"num-traits",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "num-rational"
|
||||||
|
version = "0.3.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "12ac428b1cb17fce6f731001d307d351ec70a6d202fc2e60f7d4c5e42d8f4f07"
|
||||||
|
dependencies = [
|
||||||
|
"autocfg",
|
||||||
|
"num-integer",
|
||||||
|
"num-traits",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "num-traits"
|
||||||
|
version = "0.2.14"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "9a64b1ec5cda2586e284722486d802acf1f7dbdc623e2bfc57e65ca1cd099290"
|
||||||
|
dependencies = [
|
||||||
|
"autocfg",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "png"
|
||||||
|
version = "0.16.8"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "3c3287920cb847dee3de33d301c463fba14dda99db24214ddf93f83d3021f4c6"
|
||||||
|
dependencies = [
|
||||||
|
"bitflags",
|
||||||
|
"crc32fast",
|
||||||
|
"deflate",
|
||||||
|
"miniz_oxide",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "pong"
|
||||||
|
version = "0.1.0"
|
||||||
|
dependencies = [
|
||||||
|
"agb",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "ppv-lite86"
|
||||||
|
version = "0.2.16"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "eb9f9e6e233e5c4a35559a617bf40a4ec447db2e84c20b55a6f83167b7e57872"
|
||||||
|
|
||||||
|
[[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 = "quote"
|
||||||
|
version = "1.0.14"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "47aa80447ce4daf1717500037052af176af5d38cc3e571d9ec1c7353fc10c87d"
|
||||||
|
dependencies = [
|
||||||
|
"proc-macro2",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "rand"
|
||||||
|
version = "0.8.4"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "2e7573632e6454cf6b99d7aac4ccca54be06da05aca2ef7423d22d27d4d4bcd8"
|
||||||
|
dependencies = [
|
||||||
|
"libc",
|
||||||
|
"rand_chacha",
|
||||||
|
"rand_core",
|
||||||
|
"rand_hc",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "rand_chacha"
|
||||||
|
version = "0.3.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88"
|
||||||
|
dependencies = [
|
||||||
|
"ppv-lite86",
|
||||||
|
"rand_core",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "rand_core"
|
||||||
|
version = "0.6.3"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "d34f1408f55294453790c48b2f1ebbb1c5b4b7563eb1f418bcfcfdbb06ebb4e7"
|
||||||
|
dependencies = [
|
||||||
|
"getrandom",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "rand_hc"
|
||||||
|
version = "0.3.1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "d51e9f596de227fda2ea6c84607f5558e196eeaf43c986b724ba4fb8fdf497e7"
|
||||||
|
dependencies = [
|
||||||
|
"rand_core",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "serde"
|
||||||
|
version = "1.0.133"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "97565067517b60e2d1ea8b268e59ce036de907ac523ad83a0475da04e818989a"
|
||||||
|
dependencies = [
|
||||||
|
"serde_derive",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "serde_derive"
|
||||||
|
version = "1.0.133"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "ed201699328568d8d08208fdd080e3ff594e6c422e438b6705905da01005d537"
|
||||||
|
dependencies = [
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"syn",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "siphasher"
|
||||||
|
version = "0.3.7"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "533494a8f9b724d33625ab53c6c4800f7cc445895924a8ef649222dcb76e938b"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "syn"
|
||||||
|
version = "1.0.84"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "ecb2e6da8ee5eb9a61068762a32fa9619cc591ceb055b3687f4cd4051ec2e06b"
|
||||||
|
dependencies = [
|
||||||
|
"proc-macro2",
|
||||||
|
"quote",
|
||||||
|
"unicode-xid",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "toml"
|
||||||
|
version = "0.5.8"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "a31142970826733df8241ef35dc040ef98c679ab14d7c3e54d827099b3acecaa"
|
||||||
|
dependencies = [
|
||||||
|
"serde",
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "unicode-xid"
|
||||||
|
version = "0.2.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "8ccb82d61f80a663efe1f787a51b16b5a51e3314d6ac365b08639f52387b33f3"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "wasi"
|
||||||
|
version = "0.10.2+wasi-snapshot-preview1"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "fd6fbd9a79829dd1ad0cc20627bf1ed606756a7f77edff7b66b7064f9cb327c6"
|
14
book/games/pong/Cargo.toml
Normal file
14
book/games/pong/Cargo.toml
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
[package]
|
||||||
|
name = "pong"
|
||||||
|
version = "0.1.0"
|
||||||
|
authors = ["Gwilym Kuiper <gw@ilym.me>"]
|
||||||
|
edition = "2018"
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
agb = { version = "0.8.0", path = "../../../agb" }
|
||||||
|
|
||||||
|
[profile.release]
|
||||||
|
panic = "abort"
|
||||||
|
lto = true
|
109
book/games/pong/gba.ld
Normal file
109
book/games/pong/gba.ld
Normal file
|
@ -0,0 +1,109 @@
|
||||||
|
OUTPUT_FORMAT("elf32-littlearm", "elf32-bigarm", "elf32-littlearm")
|
||||||
|
OUTPUT_ARCH(arm)
|
||||||
|
|
||||||
|
ENTRY(__start)
|
||||||
|
EXTERN(__RUST_INTERRUPT_HANDLER)
|
||||||
|
|
||||||
|
MEMORY {
|
||||||
|
ewram (w!x) : ORIGIN = 0x02000000, LENGTH = 256K
|
||||||
|
iwram (w!x) : ORIGIN = 0x03000000, LENGTH = 32K
|
||||||
|
rom (rx) : ORIGIN = 0x08000000, LENGTH = 32M
|
||||||
|
}
|
||||||
|
|
||||||
|
__text_start = ORIGIN(rom);
|
||||||
|
|
||||||
|
SECTIONS {
|
||||||
|
. = __text_start;
|
||||||
|
|
||||||
|
.crt0 : {
|
||||||
|
KEEP (crt0.o(.text));
|
||||||
|
. = ALIGN(4);
|
||||||
|
} > rom
|
||||||
|
|
||||||
|
.text : {
|
||||||
|
*(.text .text*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
} > rom
|
||||||
|
__text_end = .;
|
||||||
|
|
||||||
|
.rodata : {
|
||||||
|
*(.rodata .rodata.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
} > rom
|
||||||
|
|
||||||
|
__iwram_rom_start = .;
|
||||||
|
.iwram : {
|
||||||
|
__iwram_data_start = ABSOLUTE(.);
|
||||||
|
|
||||||
|
*(.iwram .iwram.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
|
||||||
|
*(.text_iwram .text_iwram.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
|
||||||
|
__iwram_data_end = ABSOLUTE(.);
|
||||||
|
} > iwram AT>rom
|
||||||
|
|
||||||
|
. = __iwram_rom_start + (__iwram_data_end - __iwram_data_start);
|
||||||
|
|
||||||
|
__ewram_rom_start = .;
|
||||||
|
.ewram : {
|
||||||
|
__ewram_data_start = ABSOLUTE(.);
|
||||||
|
|
||||||
|
*(.ewram .ewram.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
|
||||||
|
*(.data .data.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
|
||||||
|
__ewram_data_end = ABSOLUTE(.);
|
||||||
|
} > ewram AT>rom
|
||||||
|
|
||||||
|
.bss : {
|
||||||
|
*(.bss .bss.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
} > iwram
|
||||||
|
|
||||||
|
__iwram_rom_length_bytes = __iwram_data_end - __iwram_data_start;
|
||||||
|
__iwram_rom_length_halfwords = (__iwram_rom_length_bytes + 1) / 2;
|
||||||
|
|
||||||
|
__ewram_rom_length_bytes = __ewram_data_end - __ewram_data_start;
|
||||||
|
__ewram_rom_length_halfwords = (__ewram_rom_length_bytes + 1) / 2;
|
||||||
|
|
||||||
|
/* debugging sections */
|
||||||
|
/* Stabs */
|
||||||
|
.stab 0 : { *(.stab) }
|
||||||
|
.stabstr 0 : { *(.stabstr) }
|
||||||
|
.stab.excl 0 : { *(.stab.excl) }
|
||||||
|
.stab.exclstr 0 : { *(.stab.exclstr) }
|
||||||
|
.stab.index 0 : { *(.stab.index) }
|
||||||
|
.stab.indexstr 0 : { *(.stab.indexstr) }
|
||||||
|
.comment 0 : { *(.comment) }
|
||||||
|
/* DWARF 1 */
|
||||||
|
.debug 0 : { *(.debug) }
|
||||||
|
.line 0 : { *(.line) }
|
||||||
|
/* GNU DWARF 1 extensions */
|
||||||
|
.debug_srcinfo 0 : { *(.debug_srcinfo) }
|
||||||
|
.debug_sfnames 0 : { *(.debug_sfnames) }
|
||||||
|
/* DWARF 1.1 and DWARF 2 */
|
||||||
|
.debug_aranges 0 : { *(.debug_aranges) }
|
||||||
|
.debug_pubnames 0 : { *(.debug_pubnames) }
|
||||||
|
/* DWARF 2 */
|
||||||
|
.debug_info 0 : { *(.debug_info) }
|
||||||
|
.debug_abbrev 0 : { *(.debug_abbrev) }
|
||||||
|
.debug_line 0 : { *(.debug_line) }
|
||||||
|
.debug_frame 0 : { *(.debug_frame) }
|
||||||
|
.debug_str 0 : { *(.debug_str) }
|
||||||
|
.debug_loc 0 : { *(.debug_loc) }
|
||||||
|
.debug_macinfo 0 : { *(.debug_macinfo) }
|
||||||
|
/* SGI/MIPS DWARF 2 extensions */
|
||||||
|
.debug_weaknames 0 : { *(.debug_weaknames) }
|
||||||
|
.debug_funcnames 0 : { *(.debug_funcnames) }
|
||||||
|
.debug_typenames 0 : { *(.debug_typenames) }
|
||||||
|
.debug_varnames 0 : { *(.debug_varnames) }
|
||||||
|
|
||||||
|
.debug_ranges 0 : { *(.debug_ranges) }
|
||||||
|
|
||||||
|
/* discard anything not already mentioned */
|
||||||
|
/DISCARD/ : { *(*) }
|
||||||
|
}
|
108
book/games/pong/gba_mb.ld
Normal file
108
book/games/pong/gba_mb.ld
Normal file
|
@ -0,0 +1,108 @@
|
||||||
|
OUTPUT_FORMAT("elf32-littlearm", "elf32-bigarm", "elf32-littlearm")
|
||||||
|
OUTPUT_ARCH(arm)
|
||||||
|
|
||||||
|
ENTRY(__start)
|
||||||
|
EXTERN(__RUST_INTERRUPT_HANDLER)
|
||||||
|
|
||||||
|
MEMORY {
|
||||||
|
ewram (w!x) : ORIGIN = 0x02000000, LENGTH = 256K
|
||||||
|
iwram (w!x) : ORIGIN = 0x03000000, LENGTH = 32K
|
||||||
|
}
|
||||||
|
|
||||||
|
__text_start = ORIGIN(ewram);
|
||||||
|
|
||||||
|
SECTIONS {
|
||||||
|
. = __text_start;
|
||||||
|
|
||||||
|
.crt0 : {
|
||||||
|
KEEP (crt0.o(.text));
|
||||||
|
. = ALIGN(4);
|
||||||
|
} > ewram
|
||||||
|
|
||||||
|
.text : {
|
||||||
|
*(.text .text*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
} > ewram
|
||||||
|
__text_end = .;
|
||||||
|
|
||||||
|
.rodata : {
|
||||||
|
*(.rodata .rodata.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
} > ewram
|
||||||
|
|
||||||
|
__iwram_rom_start = .;
|
||||||
|
.iwram : {
|
||||||
|
__iwram_data_start = ABSOLUTE(.);
|
||||||
|
|
||||||
|
*(.iwram .iwram.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
|
||||||
|
*(.text_iwram .text_iwram.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
|
||||||
|
__iwram_data_end = ABSOLUTE(.);
|
||||||
|
} > iwram AT>ewram
|
||||||
|
|
||||||
|
. = __iwram_rom_start + (__iwram_data_end - __iwram_data_start);
|
||||||
|
|
||||||
|
__ewram_rom_start = .;
|
||||||
|
.ewram : {
|
||||||
|
__ewram_data_start = ABSOLUTE(.);
|
||||||
|
|
||||||
|
*(.ewram .ewram.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
|
||||||
|
*(.data .data.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
|
||||||
|
__ewram_data_end = ABSOLUTE(.);
|
||||||
|
} > ewram AT>ewram
|
||||||
|
|
||||||
|
.bss : {
|
||||||
|
*(.bss .bss.*);
|
||||||
|
. = ALIGN(4);
|
||||||
|
} > iwram
|
||||||
|
|
||||||
|
__iwram_rom_length_bytes = __iwram_data_end - __iwram_data_start;
|
||||||
|
__iwram_rom_length_halfwords = (__iwram_rom_length_bytes + 1) / 2;
|
||||||
|
|
||||||
|
__ewram_rom_length_bytes = __ewram_data_end - __ewram_data_start;
|
||||||
|
__ewram_rom_length_halfwords = (__ewram_rom_length_bytes + 1) / 2;
|
||||||
|
|
||||||
|
/* debugging sections */
|
||||||
|
/* Stabs */
|
||||||
|
.stab 0 : { *(.stab) }
|
||||||
|
.stabstr 0 : { *(.stabstr) }
|
||||||
|
.stab.excl 0 : { *(.stab.excl) }
|
||||||
|
.stab.exclstr 0 : { *(.stab.exclstr) }
|
||||||
|
.stab.index 0 : { *(.stab.index) }
|
||||||
|
.stab.indexstr 0 : { *(.stab.indexstr) }
|
||||||
|
.comment 0 : { *(.comment) }
|
||||||
|
/* DWARF 1 */
|
||||||
|
.debug 0 : { *(.debug) }
|
||||||
|
.line 0 : { *(.line) }
|
||||||
|
/* GNU DWARF 1 extensions */
|
||||||
|
.debug_srcinfo 0 : { *(.debug_srcinfo) }
|
||||||
|
.debug_sfnames 0 : { *(.debug_sfnames) }
|
||||||
|
/* DWARF 1.1 and DWARF 2 */
|
||||||
|
.debug_aranges 0 : { *(.debug_aranges) }
|
||||||
|
.debug_pubnames 0 : { *(.debug_pubnames) }
|
||||||
|
/* DWARF 2 */
|
||||||
|
.debug_info 0 : { *(.debug_info) }
|
||||||
|
.debug_abbrev 0 : { *(.debug_abbrev) }
|
||||||
|
.debug_line 0 : { *(.debug_line) }
|
||||||
|
.debug_frame 0 : { *(.debug_frame) }
|
||||||
|
.debug_str 0 : { *(.debug_str) }
|
||||||
|
.debug_loc 0 : { *(.debug_loc) }
|
||||||
|
.debug_macinfo 0 : { *(.debug_macinfo) }
|
||||||
|
/* SGI/MIPS DWARF 2 extensions */
|
||||||
|
.debug_weaknames 0 : { *(.debug_weaknames) }
|
||||||
|
.debug_funcnames 0 : { *(.debug_funcnames) }
|
||||||
|
.debug_typenames 0 : { *(.debug_typenames) }
|
||||||
|
.debug_varnames 0 : { *(.debug_varnames) }
|
||||||
|
|
||||||
|
.debug_ranges 0 : { *(.debug_ranges) }
|
||||||
|
|
||||||
|
/* discard anything not already mentioned */
|
||||||
|
/DISCARD/ : { *(*) }
|
||||||
|
}
|
BIN
book/games/pong/gfx/sprites.aseprite
Normal file
BIN
book/games/pong/gfx/sprites.aseprite
Normal file
Binary file not shown.
BIN
book/games/pong/gfx/sprites.png
Normal file
BIN
book/games/pong/gfx/sprites.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 438 B |
6
book/games/pong/gfx/sprites.toml
Normal file
6
book/games/pong/gfx/sprites.toml
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
version = "1.0"
|
||||||
|
|
||||||
|
[image.sprites]
|
||||||
|
filename = "sprites.png"
|
||||||
|
tile_size = "16x16"
|
||||||
|
transparent_colour = "ff0044"
|
3
book/games/pong/rust-toolchain.toml
Normal file
3
book/games/pong/rust-toolchain.toml
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
[toolchain]
|
||||||
|
channel = "nightly"
|
||||||
|
components = ["rust-src", "clippy"]
|
78
book/games/pong/src/main.rs
Normal file
78
book/games/pong/src/main.rs
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
// Games made using `agb` are no_std which means you don't have access to the standard
|
||||||
|
// rust library. This is because the game boy advance doesn't really have an operating
|
||||||
|
// system, so most of the content of the standard library doesn't apply.
|
||||||
|
//
|
||||||
|
// Provided you haven't disabled it, agb does provide an allocator, so it is possible
|
||||||
|
// to use both the `core` and the `alloc` built in crates.
|
||||||
|
#![no_std]
|
||||||
|
// `agb` defines its own `main` function, so you must declare your game's main function
|
||||||
|
// using the #[agb::entry] proc macro. Failing to do so will cause failure in linking
|
||||||
|
// which won't be a particularly clear error message.
|
||||||
|
#![no_main]
|
||||||
|
|
||||||
|
// This is required in order to ensure that the panic handler defined in `agb` is set
|
||||||
|
// up correctly.
|
||||||
|
extern crate agb;
|
||||||
|
|
||||||
|
use agb::display::object::Size;
|
||||||
|
use agb::Gba;
|
||||||
|
|
||||||
|
// Put all the graphics related code in the gfx module
|
||||||
|
mod gfx {
|
||||||
|
use agb::display::object::ObjectControl;
|
||||||
|
|
||||||
|
// Import the sprites into this module. This will create a `sprites` module
|
||||||
|
// and within that will be a constant called `sprites` which houses all the
|
||||||
|
// palette and tile data.
|
||||||
|
agb::include_gfx!("gfx/sprites.toml");
|
||||||
|
|
||||||
|
// Loads the sprites tile data and palette data into VRAM
|
||||||
|
pub fn load_sprite_data(object: &mut ObjectControl) {
|
||||||
|
object.set_sprite_palettes(sprites::sprites.palettes);
|
||||||
|
object.set_sprite_tilemap(sprites::sprites.tiles);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// The main function must take 0 arguments and never return. The agb::entry decorator
|
||||||
|
// ensures that everything is in order. `agb` will call this after setting up the stack
|
||||||
|
// and interrupt handlers correctly.
|
||||||
|
#[agb::entry]
|
||||||
|
fn main() -> ! {
|
||||||
|
let mut gba = Gba::new();
|
||||||
|
|
||||||
|
let _tiled = gba.display.video.tiled0();
|
||||||
|
let mut object = gba.display.object.get();
|
||||||
|
gfx::load_sprite_data(&mut object);
|
||||||
|
object.enable();
|
||||||
|
|
||||||
|
let mut ball = object.get_object_standard();
|
||||||
|
|
||||||
|
ball.set_x(50)
|
||||||
|
.set_y(50)
|
||||||
|
.set_sprite_size(Size::S16x16)
|
||||||
|
.set_tile_id(4 * 2)
|
||||||
|
.show();
|
||||||
|
|
||||||
|
let mut ball_x = 50;
|
||||||
|
let mut ball_y = 50;
|
||||||
|
let mut x_velocity = 1;
|
||||||
|
let mut y_velocity = 1;
|
||||||
|
|
||||||
|
loop {
|
||||||
|
ball_x = (ball_x + x_velocity).clamp(0, agb::display::WIDTH - 16);
|
||||||
|
ball_y = (ball_y + y_velocity).clamp(0, agb::display::HEIGHT - 16);
|
||||||
|
|
||||||
|
if ball_x == 0 || ball_x == agb::display::WIDTH - 16 {
|
||||||
|
x_velocity = -x_velocity;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ball_y == 0 || ball_y == agb::display::HEIGHT - 16 {
|
||||||
|
y_velocity = -y_velocity;
|
||||||
|
}
|
||||||
|
|
||||||
|
ball.set_x(ball_x as u16).set_y(ball_y as u16);
|
||||||
|
|
||||||
|
agb::display::busy_wait_for_vblank();
|
||||||
|
ball.commit();
|
||||||
|
}
|
||||||
|
}
|
|
@ -8,3 +8,6 @@
|
||||||
- [Windows setup]()
|
- [Windows setup]()
|
||||||
- [Mac OS setup]()
|
- [Mac OS setup]()
|
||||||
- [Building the game](./setup/building.md)
|
- [Building the game](./setup/building.md)
|
||||||
|
- [Learn agb part I - pong](./pong/01_introduction.md)
|
||||||
|
- [The Gba struct](./pong/02_the_gba_struct.md)
|
||||||
|
- [Sprites](./pong/03_sprites.md)
|
34
book/src/pong/01_introduction.md
Normal file
34
book/src/pong/01_introduction.md
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
# Learn agb part I - pong
|
||||||
|
|
||||||
|
In this section, we'll make a simple pong style game for the Game Boy Advance using `agb`.
|
||||||
|
You will learn:
|
||||||
|
|
||||||
|
* How to use tiled graphics modes.
|
||||||
|
* How to import graphics using `agb`.
|
||||||
|
* What Game Boy Advance sprites are and how to put them on the screen.
|
||||||
|
* How to detect button input and react to it.
|
||||||
|
* How to add a static background.
|
||||||
|
* How to make a dynamic background for a score display.
|
||||||
|
* How to add music to your game.
|
||||||
|
* How to add sound effects to your game.
|
||||||
|
|
||||||
|
With this knowledge, you'll be well equipped to start making your own games!
|
||||||
|
|
||||||
|
## Getting started
|
||||||
|
|
||||||
|
To start, create a new repository based on the [agb template](https://github.com/agbrs/template).
|
||||||
|
We'll call this `pong`.
|
||||||
|
|
||||||
|
Then replace the `name` field in `Cargo.toml` with `pong`, to end up with something similar to:
|
||||||
|
|
||||||
|
```toml
|
||||||
|
[package]
|
||||||
|
name = "pong"
|
||||||
|
version = "0.1.0"
|
||||||
|
authors = ["Your name here"]
|
||||||
|
edition = "2018"
|
||||||
|
|
||||||
|
# ...
|
||||||
|
```
|
||||||
|
|
||||||
|
You are now ready to get started learning about how `agb` works.
|
42
book/src/pong/02_the_gba_struct.md
Normal file
42
book/src/pong/02_the_gba_struct.md
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
# The Gba struct
|
||||||
|
|
||||||
|
In this section, we'll cover the importance of the Gba struct and how to create it.
|
||||||
|
|
||||||
|
# The importance of the Gba struct
|
||||||
|
|
||||||
|
Almost all interaction with the Game Boy Advance's hardware goes through the [Gba singleton struct](https://docs.rs/agb/latest/agb/struct.Gba.html).
|
||||||
|
Your games using `agb` will typically create this in the `main` function and then handle the abstractions in there.
|
||||||
|
|
||||||
|
The Gba struct is used to take advantage of rust's borrow checker, and lean on it to ensure that access to the Game Boy Advance hardware is done 'sensibly'.
|
||||||
|
You won't have to worry about 2 bits of your code modifying data in the wrong way!
|
||||||
|
This struct is a 'singleton', so you cannot create 2 instances of it at once.
|
||||||
|
Attempting to do so will result in a panic which by default crashes the game.
|
||||||
|
|
||||||
|
# How all agb games start
|
||||||
|
|
||||||
|
Replace the content of the `main` function with the following:
|
||||||
|
|
||||||
|
```rust,ignore
|
||||||
|
# #![no_std]
|
||||||
|
# #![no_main]
|
||||||
|
# extern crate agb;
|
||||||
|
# #[agb::entry]
|
||||||
|
# fn main() -> ! {
|
||||||
|
let mut gba = agb::Gba::new();
|
||||||
|
|
||||||
|
loop {} // infinite loop for now
|
||||||
|
# }
|
||||||
|
```
|
||||||
|
|
||||||
|
and ignore warnings for now.
|
||||||
|
|
||||||
|
# Running your pong game
|
||||||
|
|
||||||
|
Although there isn't much to see at the moment (just a black screen), you can start the game by using `cargo run` or whatever worked for you in the introduction.
|
||||||
|
|
||||||
|
# What we did
|
||||||
|
|
||||||
|
This was a very simple but incredibly important part of any game using `agb`.
|
||||||
|
All interactions with the hardware are gated via the Gba struct, so it must be created at the start of your `main` function and never again.
|
||||||
|
|
||||||
|
You are now ready to learn about display modes and how to start getting things onto the screen!
|
175
book/src/pong/03_sprites.md
Normal file
175
book/src/pong/03_sprites.md
Normal file
|
@ -0,0 +1,175 @@
|
||||||
|
# Sprites
|
||||||
|
|
||||||
|
In this section, we'll put the sprites needed for our pong game onto the screen.
|
||||||
|
We'll cover what sprites are in the Game Boy Advance, and how to get them to show up on screen.
|
||||||
|
We'll briefly cover vblank and by the end of this section, you'll have a ball bouncing around the screen!
|
||||||
|
|
||||||
|
# Why do we need sprites in the first place?
|
||||||
|
|
||||||
|
The Game Boy Advance has a 240x160px screen, with 15-bit RGB colour support.
|
||||||
|
In order to manually set the colour for each pixel in the screen, you would need to update a total of 38,400 pixels per frame, or 2,304,000 pixels per second at 60 fps.
|
||||||
|
With a 16MHz processor, that means you would need to be able to calculate 1 pixel every 8 clock cycles, which is pretty much impossible.
|
||||||
|
You could get clever with how you update these pixels, but using the tools provided by the Game Boy Advance to put pixels on the screen, you'll have a much easier time.
|
||||||
|
|
||||||
|
So there are 2 ways that the Game Boy Advance allows you to get these pixels on screen much more easily.
|
||||||
|
Tiles and sprites.
|
||||||
|
Tiles are 8x8 pixels in size and can be placed in a grid on the screen.
|
||||||
|
You can also scroll the whole tile layer to arbitrary positions, but the tiles will remain in this 8x8 pixel grid.
|
||||||
|
We'll cover tiles in more detail later.
|
||||||
|
|
||||||
|
The other way you can draw things on screen is using sprites, which we'll cover in more detail in this section.
|
||||||
|
|
||||||
|
# Sprites on the Game Boy Advance
|
||||||
|
|
||||||
|
The Game Boy Advance supports 256 hardware sprites.
|
||||||
|
These can be in one of many sizes, ranging from square 8x8 to more exotic sizes like 8x32 pixels.
|
||||||
|
For our pong game, all the sprites will be 16x16 pixels to make things a bit simpler.
|
||||||
|
|
||||||
|
Sprites are stored in the Game Boy Advance in a special area of video memory called the 'Object Attribute Memory' (OAM).
|
||||||
|
This has space for the 'attributes' of the sprites (things like whether or not they are visible, the location, which tile to use etc) but it does not store the actual pixel data.
|
||||||
|
The pixel data is stored in a different part of video RAM (VRAM) and the OAM only stores which tiles to use from this area.
|
||||||
|
|
||||||
|
Since RAM is in short supply, and at the time was quite expensive, the tile data is stored as indexed palette data.
|
||||||
|
So rather than storing the full colour data for each pixel in the tile, the Game Boy Advance instead stores a 'palette' of colours and the tiles which make up the sprites are stored as indexes to the palette.
|
||||||
|
You don't need to worry about this though, because `agb` handles it for you, but it is important to keep in mind that each sprite can use a maximum of 16 colours out of the total sprite palette of 256 colours.
|
||||||
|
|
||||||
|
There are technically 2 types of sprite, regular and affine sprites.
|
||||||
|
For now, we will only be dealing with regular sprites.
|
||||||
|
|
||||||
|
# Import the sprite
|
||||||
|
|
||||||
|
As mentioned above, we'll need to convert the sprite data into a format that the Game Boy Advance will be able to understand (so palette information and tile data).
|
||||||
|
Once we've converted it, we'll need to import this tile data into the Game Boy Advance's memory on start up and then create a sprite in the OAM.
|
||||||
|
|
||||||
|
Firstly, you're going to need to import the sprites into your project.
|
||||||
|
Save the following image into a new folder called `gfx` in your project:
|
||||||
|
|
||||||
|
![pong sprites](sprites.png)
|
||||||
|
|
||||||
|
This contains 5 `16x16px` sprites.
|
||||||
|
The first is the end cap for the paddle.
|
||||||
|
The second is the centre part of the paddle, which could potentially be repeated a few times.
|
||||||
|
The third until the fifth is the ball, with various squashed states.
|
||||||
|
The background is a lovely shade of `#ff0044` which we will use for the transparency.
|
||||||
|
|
||||||
|
`agb` needs to know what the tile size is, and also what the transparent colour is so that it can properly produce data that the Game Boy Advance can understand.
|
||||||
|
So you need to create a manifest file for the image, which is declared in a toml file.
|
||||||
|
|
||||||
|
In the same `gfx` folder as the `sprites.png` file, also create a `sprites.toml` file with the following content:
|
||||||
|
|
||||||
|
```toml
|
||||||
|
version = "1.0"
|
||||||
|
|
||||||
|
[image.sprites]
|
||||||
|
filename = "sprites.png"
|
||||||
|
tile_size = "16x16"
|
||||||
|
transparent_colour = "ff0044"
|
||||||
|
```
|
||||||
|
|
||||||
|
Now let's create a module in the `main.rs` file which imports the sprite sheet and loads it into memory.
|
||||||
|
Anything sprite related is managed by the [`ObjectControl` struct](https://docs.rs/agb/0.8.0/agb/display/object/struct.ObjectControl.html).
|
||||||
|
So we use that to load the sprite tile map and palette data.
|
||||||
|
|
||||||
|
```rust
|
||||||
|
// Put all the graphics related code in the gfx module
|
||||||
|
mod gfx {
|
||||||
|
use agb::display::object::ObjectControl;
|
||||||
|
|
||||||
|
// Import the sprites into this module. This will create a `sprites` module
|
||||||
|
// and within that will be a constant called `sprites` which houses all the
|
||||||
|
// palette and tile data.
|
||||||
|
agb::include_gfx!("gfx/sprites.toml");
|
||||||
|
|
||||||
|
// Loads the sprites tile data and palette data into VRAM
|
||||||
|
pub fn load_sprite_data(object: &mut ObjectControl) {
|
||||||
|
object.set_sprite_palettes(sprites::sprites.palettes);
|
||||||
|
object.set_sprite_tilemap(sprites::sprites.tiles);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
This uses the `include_gfx!` macro which loads the sprite information file and grabs the relevant tile data from there.
|
||||||
|
|
||||||
|
Now, let's put this on screen by firstly creating the object manager and loading the sprite data from above:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
let mut gba = Gba::new();
|
||||||
|
|
||||||
|
// set background mode to mode 0 (we'll cover this in more detail later)
|
||||||
|
// for now, this is required in order for anything to show up on screen at all.
|
||||||
|
let _tiled = gba.display.video.tiled0();
|
||||||
|
|
||||||
|
// Get the OAM manager
|
||||||
|
let mut object = gba.display.object.get();
|
||||||
|
gfx::load_sprite_data(&mut object);
|
||||||
|
object.enable();
|
||||||
|
```
|
||||||
|
|
||||||
|
Then, let's create the ball object and put it at 50, 50 on the screen:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
// continued from above:
|
||||||
|
let mut ball = object.get_object_standard(); // (1)
|
||||||
|
|
||||||
|
ball.set_x(50) // (2)
|
||||||
|
.set_y(50)
|
||||||
|
.set_sprite_size(Size::S16x16)
|
||||||
|
.set_tile_id(4 * 2)
|
||||||
|
.show();
|
||||||
|
|
||||||
|
ball.commit(); // (3)
|
||||||
|
```
|
||||||
|
|
||||||
|
There are a few new things introduced here, so lets go through these 1 by 1.
|
||||||
|
|
||||||
|
1. The first thing we're doing is creating a 'standard' object.
|
||||||
|
These are of non-affine type.
|
||||||
|
2. We now set some properties on the ball.
|
||||||
|
The position and size are self explanatory.
|
||||||
|
Interesting here is that the tile id is 4 * 2.
|
||||||
|
This is because the tile id is calculated in 8x8 pixel chunks, and in our example we have 16x16 sprites so each sprite takes 4 tiles and this is the tile in position 2 on the tilesheet above (0 indexed).
|
||||||
|
The final call to `.show()` will make it actually visible as sprites are hidden by default.
|
||||||
|
3. The call to `.commit()` actually makes the change to object memory.
|
||||||
|
Until `.commit()` is called, no changes you made will actually be visible.
|
||||||
|
This is very handy because we might want to change sprite positions etc while the frame is rendering, and then move them in the short space of time we have before the next frame starts rendering (vblank).
|
||||||
|
|
||||||
|
If you run this you should now see the ball for this pong game somewhere in the top left of the screen.
|
||||||
|
|
||||||
|
# Making the sprite move
|
||||||
|
|
||||||
|
As mentioned before, you should `.commit()` your sprites only during `vblank` which is the (very short) period of time nothing is being rendered to screen.
|
||||||
|
`agb` provides a convenience function for waiting until this happens called `agb::display::busy_wait_for_vblank()`.
|
||||||
|
You shouldn't use this is a real game (we'll do it properly later on), but for now we can use this to wait for the correct time to `commit` our sprites to memory.
|
||||||
|
|
||||||
|
Making the sprite move 1 pixel every frame (so approximately 60 pixels per second) can be done as follows:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
// replace the call to ball.commit() with the following:
|
||||||
|
|
||||||
|
let mut ball_x = 50;
|
||||||
|
let mut ball_y = 50;
|
||||||
|
let mut x_velocity = 1;
|
||||||
|
let mut y_velocity = 1;
|
||||||
|
|
||||||
|
loop {
|
||||||
|
ball_x = (ball_x + x_velocity).clamp(0, agb::display::WIDTH - 16);
|
||||||
|
ball_y = (ball_y + y_velocity).clamp(0, agb::display::HEIGHT - 16);
|
||||||
|
|
||||||
|
if ball_x == 0 || ball_x == agb::display::WIDTH - 16 {
|
||||||
|
x_velocity = -x_velocity;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ball_y == 0 || ball_y == agb::display::HEIGHT - 16 {
|
||||||
|
y_velocity = -y_velocity;
|
||||||
|
}
|
||||||
|
|
||||||
|
ball.set_x(ball_x as u16).set_y(ball_y as u16);
|
||||||
|
|
||||||
|
agb::display::busy_wait_for_vblank();
|
||||||
|
ball.commit();
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
# What we did
|
||||||
|
|
||||||
|
In this section, we covered why sprites are important, how to create and manage them using the `ObjectControl` in `agb` and make a ball bounce around the screen.
|
BIN
book/src/pong/sprites.png
Normal file
BIN
book/src/pong/sprites.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 438 B |
16
examples/the-hat-chooses-the-wizard/Cargo.lock
generated
16
examples/the-hat-chooses-the-wizard/Cargo.lock
generated
|
@ -59,9 +59,9 @@ checksum = "cdb031dd78e28731d87d56cc8ffef4a8f36ca26c38fe2de700543e627f8a464a"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "bitflags"
|
name = "bitflags"
|
||||||
version = "1.2.1"
|
version = "1.3.2"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "cf1de2fe8c75bc145a2f577add951f8134889b4795d47466a54a5c846d691693"
|
checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "bytemuck"
|
name = "bytemuck"
|
||||||
|
@ -220,18 +220,18 @@ checksum = "c3ca011bd0129ff4ae15cd04c4eef202cadf6c51c21e47aba319b4e0501db741"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "proc-macro2"
|
name = "proc-macro2"
|
||||||
version = "1.0.28"
|
version = "1.0.36"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "5c7ed8b8c7b886ea3ed7dde405212185f423ab44682667c8c6dd14aa1d9f6612"
|
checksum = "c7342d5883fbccae1cc37a2353b09c87c9b0f3afd73f5fb9bba687a1f733b029"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"unicode-xid",
|
"unicode-xid",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "quote"
|
name = "quote"
|
||||||
version = "1.0.9"
|
version = "1.0.14"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "c3d0b9745dc2debf507c8422de05d7226cc1f0644216dfdfead988f9b1ab32a7"
|
checksum = "47aa80447ce4daf1717500037052af176af5d38cc3e571d9ec1c7353fc10c87d"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
]
|
]
|
||||||
|
@ -321,9 +321,9 @@ checksum = "533494a8f9b724d33625ab53c6c4800f7cc445895924a8ef649222dcb76e938b"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "syn"
|
name = "syn"
|
||||||
version = "1.0.74"
|
version = "1.0.84"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "1873d832550d4588c3dbc20f01361ab00bfe741048f71e3fecf145a7cc18b29c"
|
checksum = "ecb2e6da8ee5eb9a61068762a32fa9619cc591ceb055b3687f4cd4051ec2e06b"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
|
|
16
examples/the-purple-night/Cargo.lock
generated
16
examples/the-purple-night/Cargo.lock
generated
|
@ -68,9 +68,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "bitflags"
|
name = "bitflags"
|
||||||
version = "1.2.1"
|
version = "1.3.2"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "cf1de2fe8c75bc145a2f577add951f8134889b4795d47466a54a5c846d691693"
|
checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "bytemuck"
|
name = "bytemuck"
|
||||||
|
@ -250,18 +250,18 @@ checksum = "c3ca011bd0129ff4ae15cd04c4eef202cadf6c51c21e47aba319b4e0501db741"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "proc-macro2"
|
name = "proc-macro2"
|
||||||
version = "1.0.32"
|
version = "1.0.36"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "ba508cc11742c0dc5c1659771673afbab7a0efab23aa17e854cbab0837ed0b43"
|
checksum = "c7342d5883fbccae1cc37a2353b09c87c9b0f3afd73f5fb9bba687a1f733b029"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"unicode-xid",
|
"unicode-xid",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "quote"
|
name = "quote"
|
||||||
version = "1.0.10"
|
version = "1.0.14"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "38bc8cc6a5f2e3655e0899c1b848643b2562f853f114bfec7be120678e3ace05"
|
checksum = "47aa80447ce4daf1717500037052af176af5d38cc3e571d9ec1c7353fc10c87d"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
]
|
]
|
||||||
|
@ -340,9 +340,9 @@ checksum = "533494a8f9b724d33625ab53c6c4800f7cc445895924a8ef649222dcb76e938b"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "syn"
|
name = "syn"
|
||||||
version = "1.0.81"
|
version = "1.0.84"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "f2afee18b8beb5a596ecb4a2dce128c719b4ba399d34126b9e4396e3f9860966"
|
checksum = "ecb2e6da8ee5eb9a61068762a32fa9619cc591ceb055b3687f4cd4051ec2e06b"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
|
|
|
@ -1578,24 +1578,24 @@ impl<'a> FollowingBoss<'a> {
|
||||||
}
|
}
|
||||||
|
|
||||||
let frame = (self.timer / 8) % 12;
|
let frame = (self.timer / 8) % 12;
|
||||||
self.entity.sprite.set_tile_id((125 + frame as u16) * 4)
|
self.entity.sprite.set_tile_id((125 + frame as u16) * 4);
|
||||||
} else if self.timer < 120 {
|
} else if self.timer < 120 {
|
||||||
let frame = (self.timer / 20) % 12;
|
let frame = (self.timer / 20) % 12;
|
||||||
self.entity.sprite.set_tile_id((125 + frame as u16) * 4)
|
self.entity.sprite.set_tile_id((125 + frame as u16) * 4);
|
||||||
} else if self.following {
|
} else if self.following {
|
||||||
self.entity.velocity = difference / 16;
|
self.entity.velocity = difference / 16;
|
||||||
if difference.manhattan_distance() < 20.into() {
|
if difference.manhattan_distance() < 20.into() {
|
||||||
self.following = false;
|
self.following = false;
|
||||||
}
|
}
|
||||||
let frame = (self.timer / 8) % 12;
|
let frame = (self.timer / 8) % 12;
|
||||||
self.entity.sprite.set_tile_id((125 + frame as u16) * 4)
|
self.entity.sprite.set_tile_id((125 + frame as u16) * 4);
|
||||||
} else {
|
} else {
|
||||||
self.entity.velocity = (0, 0).into();
|
self.entity.velocity = (0, 0).into();
|
||||||
if difference.manhattan_distance() > 60.into() {
|
if difference.manhattan_distance() > 60.into() {
|
||||||
self.following = true;
|
self.following = true;
|
||||||
}
|
}
|
||||||
let frame = (self.timer / 16) % 12;
|
let frame = (self.timer / 16) % 12;
|
||||||
self.entity.sprite.set_tile_id((125 + frame as u16) * 4)
|
self.entity.sprite.set_tile_id((125 + frame as u16) * 4);
|
||||||
}
|
}
|
||||||
self.entity.update_position_without_collision();
|
self.entity.update_position_without_collision();
|
||||||
}
|
}
|
||||||
|
|
|
@ -78,7 +78,7 @@ if [ "$PROJECT" = "agb" ]; then
|
||||||
sed -i -e "s/^agb = \".*\"/agb = \"$VERSION\"/" template/Cargo.toml
|
sed -i -e "s/^agb = \".*\"/agb = \"$VERSION\"/" template/Cargo.toml
|
||||||
git add template/Cargo.toml
|
git add template/Cargo.toml
|
||||||
|
|
||||||
for EXAMPLE_DIR in examples/*/; do
|
for EXAMPLE_DIR in examples/*/ book/games/*/; do
|
||||||
sed -E -i -e "/agb =/ s/version = \"[^\"]+\"/version = \"$VERSION\"/" "$EXAMPLE_DIR/Cargo.toml"
|
sed -E -i -e "/agb =/ s/version = \"[^\"]+\"/version = \"$VERSION\"/" "$EXAMPLE_DIR/Cargo.toml"
|
||||||
(cd "$EXAMPLE_DIR" && cargo update)
|
(cd "$EXAMPLE_DIR" && cargo update)
|
||||||
git add "$EXAMPLE_DIR"/{Cargo.toml,Cargo.lock}
|
git add "$EXAMPLE_DIR"/{Cargo.toml,Cargo.lock}
|
||||||
|
|
|
@ -1,9 +1,24 @@
|
||||||
|
// Games made using `agb` are no_std which means you don't have access to the standard
|
||||||
|
// rust library. This is because the game boy advance doesn't really have an operating
|
||||||
|
// system, so most of the content of the standard library doesn't apply.
|
||||||
|
//
|
||||||
|
// Provided you haven't disabled it, agb does provide an allocator, so it is possible
|
||||||
|
// to use both the `core` and the `alloc` built in crates.
|
||||||
#![no_std]
|
#![no_std]
|
||||||
|
// `agb` defines its own `main` function, so you must declare your game's main function
|
||||||
|
// using the #[agb::entry] proc macro. Failing to do so will cause failure in linking
|
||||||
|
// which won't be a particularly clear error message.
|
||||||
#![no_main]
|
#![no_main]
|
||||||
|
|
||||||
|
// This is required in order to ensure that the panic handler defined in `agb` is set
|
||||||
|
// up correctly.
|
||||||
extern crate agb;
|
extern crate agb;
|
||||||
|
|
||||||
use agb::{display, syscall};
|
use agb::{display, syscall};
|
||||||
|
|
||||||
|
// The main function must take 0 arguments and never return. The agb::entry decorator
|
||||||
|
// ensures that everything is in order. `agb` will call this after setting up the stack
|
||||||
|
// and interrupt handlers correctly.
|
||||||
#[agb::entry]
|
#[agb::entry]
|
||||||
fn main() -> ! {
|
fn main() -> ! {
|
||||||
let mut gba = agb::Gba::new();
|
let mut gba = agb::Gba::new();
|
||||||
|
|
Loading…
Add table
Reference in a new issue