From ee576597c2c974a23960627f40150629e49f2709 Mon Sep 17 00:00:00 2001 From: lifning <> Date: Sun, 7 Aug 2022 01:49:32 -0700 Subject: [PATCH 01/49] Add rudimentary affine background layer support. --- agb-fixnum/src/lib.rs | 10 + agb-image-converter/src/lib.rs | 67 ++-- agb-image-converter/src/rust_generator.rs | 33 +- agb/examples/animated_background.rs | 2 +- agb/examples/chicken.rs | 2 +- agb/examples/dynamic_tiles.rs | 2 +- agb/examples/stereo_sound.rs | 2 +- agb/examples/text_render.rs | 2 +- agb/src/display/example_logo.rs | 3 +- agb/src/display/font.rs | 1 + agb/src/display/object.rs | 35 +- .../display/tiled/infinite_scrolled_map.rs | 5 +- agb/src/display/tiled/map.rs | 363 ++++++++++++++---- agb/src/display/tiled/mod.rs | 270 +++++++++++-- agb/src/display/tiled/tiled0.rs | 58 +-- agb/src/display/tiled/tiled1.rs | 47 +++ agb/src/display/tiled/tiled2.rs | 53 +++ agb/src/display/tiled/vram_manager.rs | 131 +++++-- agb/src/display/video.rs | 12 +- agb/src/lib.rs | 5 +- agb/src/memory_mapped.rs | 4 +- agb/src/syscall.rs | 145 ++++--- examples/hyperspace-roll/build.rs | 2 +- examples/hyperspace-roll/src/background.rs | 2 +- examples/hyperspace-roll/src/battle.rs | 2 +- examples/hyperspace-roll/src/customise.rs | 2 +- examples/hyperspace-roll/src/main.rs | 2 +- .../src/level_display.rs | 2 +- .../the-hat-chooses-the-wizard/src/main.rs | 2 +- .../src/splash_screen.rs | 2 +- 30 files changed, 940 insertions(+), 328 deletions(-) create mode 100644 agb/src/display/tiled/tiled1.rs create mode 100644 agb/src/display/tiled/tiled2.rs diff --git a/agb-fixnum/src/lib.rs b/agb-fixnum/src/lib.rs index 47036f28..b8b87469 100644 --- a/agb-fixnum/src/lib.rs +++ b/agb-fixnum/src/lib.rs @@ -566,6 +566,7 @@ impl Debug for Num { /// A vector of two points: (x, y) represened by integers or fixed point numbers #[derive(Clone, Copy, PartialEq, Eq, Debug)] +#[repr(C)] pub struct Vector2D { /// The x coordinate pub x: T, @@ -1211,6 +1212,15 @@ mod tests { } } + #[test] + fn test_only_frac_bits() { + let quarter: Num = num!(0.25); + let neg_quarter: Num = num!(-0.25); + + assert_eq!(quarter + quarter, num!(0.5)); + assert_eq!(neg_quarter + neg_quarter, num!(-0.5)); + } + #[test] fn test_vector_multiplication_and_division() { let a: Vector2D = (1, 2).into(); diff --git a/agb-image-converter/src/lib.rs b/agb-image-converter/src/lib.rs index 1cd8356b..2f41c577 100644 --- a/agb-image-converter/src/lib.rs +++ b/agb-image-converter/src/lib.rs @@ -132,7 +132,7 @@ pub fn include_aseprite_inner(input: TokenStream) -> TokenStream { let optimised_results = optimiser.optimise_palettes(); - let (palette_data, tile_data, assignments) = palete_tile_data(&optimised_results, &images); + let (palette_data, tile_data, assignments) = palette_tile_data(&optimised_results, &images); let palette_data = palette_data.iter().map(|colours| { quote! { @@ -273,7 +273,7 @@ fn add_to_optimiser( } } -fn palete_tile_data( +fn palette_tile_data( optimiser: &Palette16OptimisationResults, images: &[Image], ) -> (Vec>, Vec, Vec) { @@ -293,41 +293,54 @@ fn palete_tile_data( .collect(); let mut tile_data = Vec::new(); + let tile_size = TileSize::Tile8; for image in images { - let tile_size = 8; - let tiles_x = image.width / tile_size; - let tiles_y = image.height / tile_size; + add_image_to_tile_data(&mut tile_data, image, tile_size, &optimiser) + } - for y in 0..tiles_y { - for x in 0..tiles_x { - let palette_index = optimiser.assignments[y * tiles_x + x]; - let palette = &optimiser.optimised_palettes[palette_index]; + let tile_data = collapse_to_4bpp(&tile_data); - for inner_y in 0..tile_size / 8 { - for inner_x in 0..tile_size / 8 { - for j in inner_y * 8..inner_y * 8 + 8 { - for i in inner_x * 8..inner_x * 8 + 8 { - let colour = image.colour(x * tile_size + i, y * tile_size + j); - tile_data.push( - palette.colour_index(colour, optimiser.transparent_colour), - ); - } + let assignments = optimiser.assignments.clone(); + + (palette_data, tile_data, assignments) +} + +fn collapse_to_4bpp(tile_data: &[u8]) -> Vec { + tile_data + .chunks(2) + .map(|chunk| chunk[0] | (chunk[1] << 4)) + .collect() +} + +fn add_image_to_tile_data( + tile_data: &mut Vec, + image: &Image, + tile_size: TileSize, + optimiser: &&Palette16OptimisationResults, +) { + let tile_size = tile_size.to_size(); + let tiles_x = image.width / tile_size; + let tiles_y = image.height / tile_size; + + for y in 0..tiles_y { + for x in 0..tiles_x { + let palette_index = optimiser.assignments[y * tiles_x + x]; + let palette = &optimiser.optimised_palettes[palette_index]; + + for inner_y in 0..tile_size / 8 { + for inner_x in 0..tile_size / 8 { + for j in inner_y * 8..inner_y * 8 + 8 { + for i in inner_x * 8..inner_x * 8 + 8 { + let colour = image.colour(x * tile_size + i, y * tile_size + j); + tile_data + .push(palette.colour_index(colour, optimiser.transparent_colour)); } } } } } } - - let tile_data = tile_data - .chunks(2) - .map(|chunk| chunk[0] | (chunk[1] << 4)) - .collect(); - - let assignments = optimiser.assignments.clone(); - - (palette_data, tile_data, assignments) } fn flatten_group(expr: &Expr) -> &Expr { diff --git a/agb-image-converter/src/rust_generator.rs b/agb-image-converter/src/rust_generator.rs index f289ffeb..7f04b7eb 100644 --- a/agb-image-converter/src/rust_generator.rs +++ b/agb-image-converter/src/rust_generator.rs @@ -1,5 +1,5 @@ use crate::palette16::Palette16OptimisationResults; -use crate::TileSize; +use crate::{add_image_to_tile_data, collapse_to_4bpp, TileSize}; use crate::{image_loader::Image, ByteString}; use proc_macro2::TokenStream; @@ -34,36 +34,11 @@ pub(crate) fn generate_code( } }); - let tile_size = tile_size.to_size(); + let mut tile_data = Vec::new(); - let tiles_x = image.width / tile_size; - let tiles_y = image.height / tile_size; + add_image_to_tile_data(&mut tile_data, image, tile_size, &results); - let mut tile_data = vec![]; - - for y in 0..tiles_y { - for x in 0..tiles_x { - let palette_index = results.assignments[y * tiles_x + x]; - let palette = &results.optimised_palettes[palette_index]; - - for inner_y in 0..tile_size / 8 { - for inner_x in 0..tile_size / 8 { - for j in inner_y * 8..inner_y * 8 + 8 { - for i in inner_x * 8..inner_x * 8 + 8 { - let colour = image.colour(x * tile_size + i, y * tile_size + j); - tile_data - .push(palette.colour_index(colour, results.transparent_colour)); - } - } - } - } - } - } - - let tile_data: Vec<_> = tile_data - .chunks(2) - .map(|chunk| (chunk[1] << 4) | chunk[0]) - .collect(); + let tile_data = collapse_to_4bpp(&tile_data); let data = ByteString(&tile_data); diff --git a/agb/examples/animated_background.rs b/agb/examples/animated_background.rs index d20213d9..97ccf3cd 100644 --- a/agb/examples/animated_background.rs +++ b/agb/examples/animated_background.rs @@ -3,7 +3,7 @@ use agb::{ display::{ - tiled::{RegularBackgroundSize, TileFormat, TileSet, TileSetting}, + tiled::{RegularBackgroundSize, TileFormat, TileSet, TileSetting, TiledMap}, Priority, }, include_gfx, diff --git a/agb/examples/chicken.rs b/agb/examples/chicken.rs index 02536887..72bb5e63 100644 --- a/agb/examples/chicken.rs +++ b/agb/examples/chicken.rs @@ -2,7 +2,7 @@ #![no_main] use agb::{ - display::tiled::{TileFormat, TileSet, TileSetting}, + display::tiled::{TileFormat, TileSet, TileSetting, TiledMap}, display::{ object::{Object, ObjectController, Size, Sprite}, palette16::Palette16, diff --git a/agb/examples/dynamic_tiles.rs b/agb/examples/dynamic_tiles.rs index 53b013a4..171e3e0a 100644 --- a/agb/examples/dynamic_tiles.rs +++ b/agb/examples/dynamic_tiles.rs @@ -3,7 +3,7 @@ use agb::display::{ palette16::Palette16, - tiled::{RegularBackgroundSize, TileSetting}, + tiled::{RegularBackgroundSize, TileSetting, TiledMap}, Priority, }; diff --git a/agb/examples/stereo_sound.rs b/agb/examples/stereo_sound.rs index d813a237..ca293cf5 100644 --- a/agb/examples/stereo_sound.rs +++ b/agb/examples/stereo_sound.rs @@ -3,7 +3,7 @@ use agb::{ display::{ - tiled::{RegularBackgroundSize, RegularMap, TileSetting, VRamManager}, + tiled::{RegularBackgroundSize, RegularMap, TileSetting, TiledMap, VRamManager}, Font, Priority, }, include_font, include_wav, diff --git a/agb/examples/text_render.rs b/agb/examples/text_render.rs index fa398a1d..79470d3d 100644 --- a/agb/examples/text_render.rs +++ b/agb/examples/text_render.rs @@ -3,7 +3,7 @@ use agb::{ display::{ - tiled::{RegularBackgroundSize, TileSetting}, + tiled::{RegularBackgroundSize, TileSetting, TiledMap}, Font, Priority, }, include_font, diff --git a/agb/src/display/example_logo.rs b/agb/src/display/example_logo.rs index bec4df92..b21df025 100644 --- a/agb/src/display/example_logo.rs +++ b/agb/src/display/example_logo.rs @@ -1,4 +1,4 @@ -use super::tiled::{RegularMap, TileFormat, TileSet, TileSetting, VRamManager}; +use super::tiled::{RegularMap, TileFormat, TileSet, TileSetting, TiledMap, VRamManager}; crate::include_gfx!("gfx/agb_logo.toml"); @@ -21,6 +21,7 @@ pub fn display_logo(map: &mut RegularMap, vram: &mut VRamManager) { map.commit(vram); map.show(); } + #[cfg(test)] mod tests { use crate::display::{tiled::RegularBackgroundSize, Priority}; diff --git a/agb/src/display/font.rs b/agb/src/display/font.rs index cb589ec3..ac105391 100644 --- a/agb/src/display/font.rs +++ b/agb/src/display/font.rs @@ -210,6 +210,7 @@ impl<'a> Drop for TextRenderer<'a> { #[cfg(test)] mod tests { use super::*; + use crate::display::tiled::TiledMap; const FONT: Font = crate::include_font!("examples/font/yoster.ttf", 12); #[test_case] diff --git a/agb/src/display/object.rs b/agb/src/display/object.rs index 3888a8af..744be038 100644 --- a/agb/src/display/object.rs +++ b/agb/src/display/object.rs @@ -18,7 +18,7 @@ use super::{Priority, DISPLAY_CONTROL}; use crate::agb_alloc::block_allocator::BlockAllocator; use crate::agb_alloc::bump_allocator::StartEnd; use crate::dma; -use crate::fixnum::Vector2D; +use crate::fixnum::{Num, Vector2D}; use crate::hash_map::HashMap; use attributes::*; @@ -627,7 +627,7 @@ impl ObjectController { unsafe { (OBJECT_ATTRIBUTE_MEMORY as *mut u16) - .add((i as usize) * 4) + .add(i * 4) .write_volatile(HIDDEN_VALUE); } @@ -776,7 +776,7 @@ impl ObjectController { attrs.a2.set_tile_index(sprite.sprite_location); let shape_size = sprite.id.sprite().size.shape_size(); - attrs.a2.set_palete_bank(sprite.palette_location as u8); + attrs.a2.set_palette_bank(sprite.palette_location as u8); attrs.a0.set_shape(shape_size.0); attrs.a1a.set_size(shape_size.1); attrs.a1s.set_size(shape_size.1); @@ -877,7 +877,7 @@ impl<'a> Object<'a> { object_inner .attrs .a2 - .set_palete_bank(sprite.palette_location as u8); + .set_palette_bank(sprite.palette_location as u8); object_inner.attrs.a0.set_shape(shape_size.0); object_inner.attrs.a1a.set_size(shape_size.1); object_inner.attrs.a1s.set_size(shape_size.1); @@ -1193,6 +1193,31 @@ enum ColourMode { Eight, } +/// The parameters used for the PPU's affine transformation function +/// that can apply to objects and background layers in modes 1 and 2. +/// This can be obtained from X/Y scale and rotation angle with +/// [`agb::syscall::affine_matrix`]. +#[derive(Copy, Clone, Default, Debug, PartialEq, Eq)] +#[repr(C)] +pub struct AffineMatrixAttributes { + /// Adjustment made to *X* coordinate when drawing *horizontal* lines. + /// Also known as "dx". + /// Typically computed as `x_scale * cos(angle)`. + pub p_a: Num, + /// Adjustment made to *X* coordinate along *vertical* lines. + /// Also known as "dmx". + /// Typically computed as `y_scale * sin(angle)`. + pub p_b: Num, + /// Adjustment made to *Y* coordinate along *horizontal* lines. + /// Also known as "dy". + /// Typically computed as `-x_scale * sin(angle)`. + pub p_c: Num, + /// Adjustment made to *Y* coordinate along *vertical* lines. + /// Also known as "dmy". + /// Typically computed as `y_scale * cos(angle)`. + pub p_d: Num, +} + // this mod is not public, so the internal parts don't need documenting. #[allow(dead_code)] mod attributes { @@ -1232,7 +1257,7 @@ mod attributes { pub(super) struct ObjectAttribute2 { pub tile_index: B10, pub priority: Priority, - pub palete_bank: B4, + pub palette_bank: B4, } } diff --git a/agb/src/display/tiled/infinite_scrolled_map.rs b/agb/src/display/tiled/infinite_scrolled_map.rs index 4021494b..8497802c 100644 --- a/agb/src/display/tiled/infinite_scrolled_map.rs +++ b/agb/src/display/tiled/infinite_scrolled_map.rs @@ -1,6 +1,9 @@ use alloc::boxed::Box; -use super::{BackgroundID, MapLoan, RegularMap, TileSet, TileSetting, VRamManager}; +use super::{ + BackgroundID, BackgroundSizePrivate, MapLoan, RegularMap, TileSet, TileSetting, TiledMap, + VRamManager, +}; use crate::{ display, diff --git a/agb/src/display/tiled/map.rs b/agb/src/display/tiled/map.rs index b245ab23..0b5e5138 100644 --- a/agb/src/display/tiled/map.rs +++ b/agb/src/display/tiled/map.rs @@ -2,31 +2,184 @@ use core::cell::RefCell; use core::ops::{Deref, DerefMut}; use crate::bitarray::Bitarray; -use crate::display::{Priority, DISPLAY_CONTROL}; +use crate::display::{object::AffineMatrixAttributes, Priority, DISPLAY_CONTROL}; use crate::dma::dma_copy16; -use crate::fixnum::Vector2D; +use crate::fixnum::{Num, Number, Vector2D}; use crate::memory_mapped::MemoryMapped; -use super::{BackgroundID, RegularBackgroundSize, Tile, TileSet, TileSetting, VRamManager}; +use super::{ + AffineBackgroundSize, BackgroundID, BackgroundSize, BackgroundSizePrivate, + RegularBackgroundSize, Tile, TileFormat, TileIndex, TileSet, TileSetting, VRamManager, +}; +use crate::syscall::BgAffineSetData; use alloc::{vec, vec::Vec}; +pub(super) trait TiledMapPrivateConst: TiledMapTypes { + type TileType: Into + Copy + Default + Eq + PartialEq; + type AffineMatrix; + fn x_scroll(&self) -> Self::Position; + fn y_scroll(&self) -> Self::Position; + fn affine_matrix(&self) -> Self::AffineMatrix; + fn background_id(&self) -> usize; + fn screenblock(&self) -> usize; + fn priority(&self) -> Priority; + fn map_size(&self) -> Self::Size; + fn bg_x(&self) -> MemoryMapped; + fn bg_y(&self) -> MemoryMapped; + fn bg_affine_matrix(&self) -> MemoryMapped; + fn bg_control_register(&self) -> MemoryMapped { + unsafe { MemoryMapped::new(0x0400_0008 + 2 * self.background_id()) } + } + fn screenblock_memory(&self) -> *mut u16 { + (0x0600_0000 + 0x1000 * self.screenblock() as usize / 2) as *mut u16 + } +} + +trait TiledMapPrivate: TiledMapPrivateConst { + fn tiles_mut(&mut self) -> &mut [Self::TileType]; + fn tiles_dirty(&mut self) -> &mut bool; + fn x_scroll_mut(&mut self) -> &mut Self::Position; + fn y_scroll_mut(&mut self) -> &mut Self::Position; +} + +pub trait TiledMapTypes { + type Position: Number; + type Size: BackgroundSize + Copy; +} + +pub trait TiledMap: TiledMapTypes { + fn clear(&mut self, vram: &mut VRamManager); + fn show(&mut self); + fn hide(&mut self); + fn commit(&mut self, vram: &mut VRamManager); + fn size(&self) -> Self::Size; + + #[must_use] + fn scroll_pos(&self) -> Vector2D; + fn set_scroll_pos(&mut self, pos: Vector2D); +} + +impl TiledMap for T +where + T: TiledMapPrivateConst + TiledMapPrivate + TiledMapTypes, + T::Size: BackgroundSizePrivate, +{ + fn clear(&mut self, vram: &mut VRamManager) { + for tile in self.tiles_mut() { + if *tile != Default::default() { + vram.remove_tile((*tile).into()); + } + + *tile = Default::default(); + } + } + + fn show(&mut self) { + let mode = DISPLAY_CONTROL.get(); + let new_mode = mode | (1 << (self.background_id() + 0x08)) as u16; + DISPLAY_CONTROL.set(new_mode); + } + + fn hide(&mut self) { + let mode = DISPLAY_CONTROL.get(); + let new_mode = mode & !(1 << (self.background_id() + 0x08)) as u16; + DISPLAY_CONTROL.set(new_mode); + } + + fn commit(&mut self, vram: &mut VRamManager) { + let new_bg_control_value = (self.priority() as u16) + | ((self.screenblock() as u16) << 8) + | (self.map_size().size_flag() << 14); + + self.bg_control_register().set(new_bg_control_value); + self.bg_x().set(self.x_scroll()); + self.bg_y().set(self.y_scroll()); + self.bg_affine_matrix().set(self.affine_matrix()); + + let screenblock_memory = self.screenblock_memory(); + let x: TileIndex = unsafe { *self.tiles_mut().get_unchecked(0) }.into(); + let x = x.format().tile_size() / TileFormat::FourBpp.tile_size(); + if *self.tiles_dirty() { + unsafe { + dma_copy16( + self.tiles_mut().as_ptr() as *const u16, + screenblock_memory, + self.map_size().num_tiles() / x, + ); + } + } + + vram.gc(); + + *self.tiles_dirty() = false; + } + + fn size(&self) -> Self::Size { + self.map_size() + } + + #[must_use] + fn scroll_pos(&self) -> Vector2D { + (self.x_scroll(), self.y_scroll()).into() + } + + fn set_scroll_pos(&mut self, pos: Vector2D) { + *self.x_scroll_mut() = pos.x; + *self.y_scroll_mut() = pos.y; + } +} + pub struct RegularMap { background_id: u8, - screenblock: u8, + priority: Priority, + size: RegularBackgroundSize, + x_scroll: u16, y_scroll: u16, - priority: Priority, tiles: Vec, tiles_dirty: bool, - - size: RegularBackgroundSize, } pub const TRANSPARENT_TILE_INDEX: u16 = (1 << 10) - 1; +#[rustfmt::skip] +impl TiledMapPrivate for RegularMap { + fn tiles_mut(&mut self) -> &mut [Self::TileType] { &mut self.tiles } + fn tiles_dirty(&mut self) -> &mut bool { &mut self.tiles_dirty } + fn x_scroll_mut(&mut self) -> &mut Self::Position { &mut self.x_scroll } + fn y_scroll_mut(&mut self) -> &mut Self::Position { &mut self.y_scroll } +} + +impl TiledMapTypes for RegularMap { + type Position = u16; + type Size = RegularBackgroundSize; +} + +#[rustfmt::skip] +impl const TiledMapPrivateConst for RegularMap { + type TileType = Tile; + type AffineMatrix = (); + fn x_scroll(&self) -> Self::Position { self.x_scroll } + fn y_scroll(&self) -> Self::Position { self.y_scroll } + fn affine_matrix(&self) -> Self::AffineMatrix {} + fn background_id(&self) -> usize { self.background_id as usize } + fn screenblock(&self) -> usize { self.screenblock as usize } + fn priority(&self) -> Priority { self.priority } + fn map_size(&self) -> Self::Size { self.size } + fn bg_x(&self) -> MemoryMapped { + unsafe { MemoryMapped::new(0x0400_0010 + 4 * self.background_id as usize) } + } + fn bg_y(&self) -> MemoryMapped { + unsafe { MemoryMapped::new(0x0400_0012 + 4 * self.background_id as usize) } + } + fn bg_affine_matrix(&self) -> MemoryMapped { + unsafe { MemoryMapped::new(0) } + } +} + impl RegularMap { pub(crate) fn new( background_id: u8, @@ -36,16 +189,15 @@ impl RegularMap { ) -> Self { Self { background_id, - screenblock, + priority, + size, + x_scroll: 0, y_scroll: 0, - priority, tiles: vec![Default::default(); size.num_tiles()], tiles_dirty: true, - - size, } } @@ -56,11 +208,11 @@ impl RegularMap { tileset: &TileSet<'_>, tile_setting: TileSetting, ) { - let pos = self.size.gba_offset(pos); + let pos = self.map_size().gba_offset(pos); - let old_tile = self.tiles[pos]; + let old_tile = self.tiles_mut()[pos]; if old_tile != Tile::default() { - vram.remove_tile(old_tile.tile_index()); + vram.remove_tile(old_tile.into()); } let tile_index = tile_setting.index(); @@ -77,86 +229,129 @@ impl RegularMap { return; } - self.tiles[pos] = new_tile; - self.tiles_dirty = true; + self.tiles_mut()[pos] = new_tile; + *self.tiles_dirty() = true; } +} - pub fn clear(&mut self, vram: &mut VRamManager) { - for tile in self.tiles.iter_mut() { - if *tile != Tile::default() { - vram.remove_tile(tile.tile_index()); - } +pub struct AffineMap { + background_id: u8, + screenblock: u8, + priority: Priority, + size: AffineBackgroundSize, - *tile = Tile::default(); + bg_center: Vector2D>, + transform: BgAffineSetData, + + tiles: Vec, + tiles_dirty: bool, +} + +#[rustfmt::skip] +impl TiledMapPrivate for AffineMap { + fn tiles_mut(&mut self) -> &mut [Self::TileType] { &mut self.tiles } + fn tiles_dirty(&mut self) -> &mut bool { &mut self.tiles_dirty } + fn x_scroll_mut(&mut self) -> &mut Self::Position { &mut self.transform.position.x } + fn y_scroll_mut(&mut self) -> &mut Self::Position { &mut self.transform.position.y } +} + +impl TiledMapTypes for AffineMap { + type Position = Num; + type Size = AffineBackgroundSize; +} + +#[rustfmt::skip] +impl const TiledMapPrivateConst for AffineMap { + type TileType = u8; + type AffineMatrix = AffineMatrixAttributes; + fn x_scroll(&self) -> Self::Position { self.transform.position.x } + fn y_scroll(&self) -> Self::Position { self.transform.position.y } + fn affine_matrix(&self) -> Self::AffineMatrix { self.transform.matrix } + fn background_id(&self) -> usize { self.background_id as usize } + fn screenblock(&self) -> usize { self.screenblock as usize } + fn priority(&self) -> Priority { self.priority } + fn map_size(&self) -> Self::Size { self.size } + fn bg_x(&self) -> MemoryMapped { + unsafe { MemoryMapped::new(0x0400_0008 + 0x10 * self.background_id()) } + } + fn bg_y(&self) -> MemoryMapped { + unsafe { MemoryMapped::new(0x0400_000c + 0x10 * self.background_id()) } + } + fn bg_affine_matrix(&self) -> MemoryMapped { + unsafe { MemoryMapped::new(0x0400_0000 + 0x10 * self.background_id()) } + } +} + +impl AffineMap { + pub(crate) fn new( + background_id: u8, + screenblock: u8, + priority: Priority, + size: AffineBackgroundSize, + bg_center: Vector2D>, + ) -> Self { + Self { + background_id, + screenblock, + priority, + size, + + bg_center, + transform: Default::default(), + + tiles: vec![Default::default(); size.num_tiles()], + tiles_dirty: true, } } - pub fn show(&mut self) { - let mode = DISPLAY_CONTROL.get(); - let new_mode = mode | (1 << (self.background_id + 0x08)); - DISPLAY_CONTROL.set(new_mode); - } + pub fn set_tile( + &mut self, + vram: &mut VRamManager, + pos: Vector2D, + tileset: &TileSet<'_>, + tile_id: u8, + ) { + let pos = self.map_size().gba_offset(pos); - pub fn hide(&mut self) { - let mode = DISPLAY_CONTROL.get(); - let new_mode = mode & !(1 << (self.background_id + 0x08)); - DISPLAY_CONTROL.set(new_mode); - } - - pub fn commit(&mut self, vram: &mut VRamManager) { - let new_bg_control_value = (self.priority as u16) - | (u16::from(self.screenblock) << 8) - | (self.size.size_flag() << 14); - - self.bg_control_register().set(new_bg_control_value); - self.bg_h_offset().set(self.x_scroll); - self.bg_v_offset().set(self.y_scroll); - - let screenblock_memory = self.screenblock_memory(); - - if self.tiles_dirty { - unsafe { - dma_copy16( - self.tiles.as_ptr() as *const u16, - screenblock_memory, - self.size.num_tiles(), - ); - } + let old_tile = self.tiles_mut()[pos]; + if old_tile != 0 { + vram.remove_tile(old_tile.into()); } - vram.gc(); + let tile_index = tile_id as u16; - self.tiles_dirty = false; + let new_tile = if tile_index != TRANSPARENT_TILE_INDEX { + let new_tile_idx = vram.add_tile(tileset, tile_index); + new_tile_idx.raw_index() as u8 + } else { + 0 + }; + + if old_tile == new_tile { + // no need to mark as dirty if nothing changes + return; + } + + self.tiles_mut()[pos] = new_tile; + *self.tiles_dirty() = true; } - pub fn set_scroll_pos(&mut self, pos: Vector2D) { - self.x_scroll = pos.x; - self.y_scroll = pos.y; + pub fn set_transform_raw(&mut self, transform: BgAffineSetData) { + self.transform = transform; } - #[must_use] - pub fn scroll_pos(&self) -> Vector2D { - (self.x_scroll, self.y_scroll).into() - } - - pub(crate) fn size(&self) -> RegularBackgroundSize { - self.size - } - - const fn bg_control_register(&self) -> MemoryMapped { - unsafe { MemoryMapped::new(0x0400_0008 + 2 * self.background_id as usize) } - } - - const fn bg_h_offset(&self) -> MemoryMapped { - unsafe { MemoryMapped::new(0x0400_0010 + 4 * self.background_id as usize) } - } - - const fn bg_v_offset(&self) -> MemoryMapped { - unsafe { MemoryMapped::new(0x0400_0012 + 4 * self.background_id as usize) } - } - - const fn screenblock_memory(&self) -> *mut u16 { - (0x0600_0000 + 0x1000 * self.screenblock as usize / 2) as *mut u16 + pub fn set_transform( + &mut self, + display_center: Vector2D, + scale: Vector2D>, + rotation: Num, + ) { + self.set_transform_raw(crate::syscall::bg_affine_matrix( + self.bg_center, + display_center, + scale, + rotation, + )); } } @@ -165,7 +360,7 @@ pub struct MapLoan<'a, T> { background_id: u8, screenblock_id: u8, screenblock_length: u8, - regular_map_list: &'a RefCell>, + map_list: &'a RefCell>, screenblock_list: &'a RefCell>, } @@ -189,7 +384,7 @@ impl<'a, T> MapLoan<'a, T> { background_id: u8, screenblock_id: u8, screenblock_length: u8, - regular_map_list: &'a RefCell>, + map_list: &'a RefCell>, screenblock_list: &'a RefCell>, ) -> Self { MapLoan { @@ -197,7 +392,7 @@ impl<'a, T> MapLoan<'a, T> { background_id, screenblock_id, screenblock_length, - regular_map_list, + map_list, screenblock_list, } } @@ -210,7 +405,7 @@ impl<'a, T> MapLoan<'a, T> { impl<'a, T> Drop for MapLoan<'a, T> { fn drop(&mut self) { - self.regular_map_list + self.map_list .borrow_mut() .set(self.background_id as usize, false); diff --git a/agb/src/display/tiled/mod.rs b/agb/src/display/tiled/mod.rs index 529b8681..aadba54a 100644 --- a/agb/src/display/tiled/mod.rs +++ b/agb/src/display/tiled/mod.rs @@ -1,28 +1,76 @@ mod infinite_scrolled_map; mod map; mod tiled0; +mod tiled1; +mod tiled2; mod vram_manager; -use agb_fixnum::Vector2D; +use crate::bitarray::Bitarray; +use crate::display::Priority; +use agb_fixnum::{Num, Vector2D}; +use core::cell::RefCell; pub use infinite_scrolled_map::{InfiniteScrolledMap, PartialUpdateStatus}; -pub use map::{MapLoan, RegularMap}; +pub use map::{AffineMap, MapLoan, RegularMap, TiledMap}; pub use tiled0::Tiled0; +pub use tiled1::Tiled1; +pub use tiled2::Tiled2; pub use vram_manager::{DynamicTile, TileFormat, TileIndex, TileSet, VRamManager}; +// affine layers start at BG2 +pub(crate) const AFFINE_BG_ID_OFFSET: usize = 2; + #[derive(Clone, Copy, Debug, PartialEq, Eq)] +#[repr(u16)] pub enum RegularBackgroundSize { - Background32x32, - Background64x32, - Background32x64, - Background64x64, + Background32x32 = 0, + Background64x32 = 1, + Background32x64 = 2, + Background64x64 = 3, } #[derive(Debug, Clone, Copy, PartialEq, Eq)] pub struct BackgroundID(pub(crate) u8); -impl RegularBackgroundSize { +#[derive(Clone, Copy, Debug, PartialEq, Eq)] +#[repr(u16)] +pub enum AffineBackgroundSize { + Background16x16 = 0, + Background32x32 = 1, + Background64x64 = 2, + Background128x128 = 3, +} + +pub trait BackgroundSize { #[must_use] - pub fn width(&self) -> u32 { + fn width(&self) -> u32; + #[must_use] + fn height(&self) -> u32; +} + +pub(super) trait BackgroundSizePrivate: BackgroundSize + Sized { + fn size_flag(self) -> u16; + fn num_tiles(&self) -> usize { + (self.width() * self.height()) as usize + } + fn num_screen_blocks(&self) -> usize; + fn gba_offset(&self, pos: Vector2D) -> usize; + fn tile_pos_x(&self, x: i32) -> u16 { + ((x as u32) & (self.width() - 1)) as u16 + } + fn tile_pos_y(&self, y: i32) -> u16 { + ((y as u32) & (self.height() - 1)) as u16 + } + fn px_offset_x(&self, x: i32) -> u16 { + ((x as u32) & (self.width() * 8 - 1)) as u16 + } + fn px_offset_y(&self, y: i32) -> u16 { + ((y as u32) & (self.height() * 8 - 1)) as u16 + } +} + +impl BackgroundSize for RegularBackgroundSize { + #[must_use] + fn width(&self) -> u32 { match self { RegularBackgroundSize::Background32x64 | RegularBackgroundSize::Background32x32 => 32, RegularBackgroundSize::Background64x64 | RegularBackgroundSize::Background64x32 => 64, @@ -30,33 +78,26 @@ impl RegularBackgroundSize { } #[must_use] - pub fn height(&self) -> u32 { + fn height(&self) -> u32 { match self { RegularBackgroundSize::Background32x32 | RegularBackgroundSize::Background64x32 => 32, RegularBackgroundSize::Background32x64 | RegularBackgroundSize::Background64x64 => 64, } } +} - pub(crate) fn size_flag(self) -> u16 { - match self { - RegularBackgroundSize::Background32x32 => 0, - RegularBackgroundSize::Background64x32 => 1, - RegularBackgroundSize::Background32x64 => 2, - RegularBackgroundSize::Background64x64 => 3, - } +impl BackgroundSizePrivate for RegularBackgroundSize { + fn size_flag(self) -> u16 { + self as u16 } - pub(crate) fn num_tiles(self) -> usize { - (self.width() * self.height()) as usize - } - - pub(crate) fn num_screen_blocks(self) -> usize { + fn num_screen_blocks(&self) -> usize { self.num_tiles() / (32 * 32) } // This is hilariously complicated due to how the GBA stores the background screenblocks. // See https://www.coranac.com/tonc/text/regbg.htm#sec-map for an explanation - pub(crate) fn gba_offset(self, pos: Vector2D) -> usize { + fn gba_offset(&self, pos: Vector2D) -> usize { let x_mod = pos.x & (self.width() as u16 - 1); let y_mod = pos.y & (self.height() as u16 - 1); @@ -66,21 +107,43 @@ impl RegularBackgroundSize { pos as usize } +} - pub(crate) fn tile_pos_x(self, x: i32) -> u16 { - ((x as u32) & (self.width() - 1)) as u16 +impl BackgroundSize for AffineBackgroundSize { + #[must_use] + fn width(&self) -> u32 { + match self { + AffineBackgroundSize::Background16x16 => 16, + AffineBackgroundSize::Background32x32 => 32, + AffineBackgroundSize::Background64x64 => 64, + AffineBackgroundSize::Background128x128 => 128, + } } - pub(crate) fn tile_pos_y(self, y: i32) -> u16 { - ((y as u32) & (self.height() - 1)) as u16 + #[must_use] + fn height(&self) -> u32 { + self.width() + } +} + +impl BackgroundSizePrivate for AffineBackgroundSize { + fn size_flag(self) -> u16 { + self as u16 } - pub(crate) fn px_offset_x(self, x: i32) -> u16 { - ((x as u32) & (self.width() * 8 - 1)) as u16 + fn num_screen_blocks(&self) -> usize { + // technically 16x16 and 32x32 only use the first 1/8 and 1/2 of the SB, respectively + 1.max(self.num_tiles() / 2048) } - pub(crate) fn px_offset_y(self, y: i32) -> u16 { - ((y as u32) & (self.height() * 8 - 1)) as u16 + // Affine modes don't do the convoluted staggered block layout + fn gba_offset(&self, pos: Vector2D) -> usize { + let x_mod = pos.x & (self.width() as u16 - 1); + let y_mod = pos.y & (self.height() as u16 - 1); + + let pos = x_mod + (self.width() as u16 * y_mod); + + pos as usize } } @@ -90,11 +153,11 @@ struct Tile(u16); impl Tile { fn new(idx: TileIndex, setting: TileSetting) -> Self { - Self(idx.index() | setting.setting()) + Self(idx.raw_index() | setting.setting()) } fn tile_index(self) -> TileIndex { - TileIndex::new(self.0 as usize & ((1 << 10) - 1)) + TileIndex::new(self.0 as usize & ((1 << 10) - 1), TileFormat::FourBpp) } } @@ -126,6 +189,149 @@ impl TileSetting { } } +pub(self) fn find_screenblock_gap(screenblocks: &Bitarray<1>, gap: usize) -> usize { + let mut candidate = 0; + + 'outer: while candidate < 16 - gap { + let starting_point = candidate; + for attempt in starting_point..(starting_point + gap) { + if screenblocks.get(attempt) == Some(true) { + candidate = attempt + 1; + continue 'outer; + } + } + + return candidate; + } + + panic!( + "Failed to find screenblock gap of at least {} elements", + gap + ); +} + +pub(self) trait TiledMode { + const REGULAR_BACKGROUNDS: usize; + const AFFINE_BACKGROUNDS: usize; + fn screenblocks(&self) -> &RefCell>; + fn regular(&self) -> &RefCell>; + fn affine(&self) -> &RefCell>; +} + +pub trait RegularTiledMode { + fn regular_background( + &self, + priority: Priority, + size: RegularBackgroundSize, + ) -> MapLoan<'_, RegularMap>; +} + +pub trait AffineTiledMode { + fn affine_background( + &self, + priority: Priority, + size: AffineBackgroundSize, + ) -> MapLoan<'_, AffineMap>; +} + +// withoutboats: https://internals.rust-lang.org/t/const-generics-where-restrictions/12742/6 +pub struct If; +pub trait True {} +impl True for If {} + +impl RegularTiledMode for T +where + T: TiledMode, + If<{ T::REGULAR_BACKGROUNDS > 0 }>: True, +{ + fn regular_background( + &self, + priority: Priority, + size: RegularBackgroundSize, + ) -> MapLoan<'_, RegularMap> { + let mut regular = self.regular().borrow_mut(); + let new_background = regular.first_zero().unwrap(); + if new_background >= T::REGULAR_BACKGROUNDS { + panic!( + "can only have {} active regular backgrounds", + T::REGULAR_BACKGROUNDS + ); + } + + let num_screenblocks = size.num_screen_blocks(); + let mut screenblocks = self.screenblocks().borrow_mut(); + + let screenblock = find_screenblock_gap(&screenblocks, num_screenblocks); + for id in screenblock..(screenblock + num_screenblocks) { + screenblocks.set(id, true); + } + + let bg = RegularMap::new(new_background as u8, screenblock as u8 + 16, priority, size); + + regular.set(new_background, true); + + MapLoan::new( + bg, + new_background as u8, + screenblock as u8, + num_screenblocks as u8, + self.regular(), + self.screenblocks(), + ) + } +} + +impl AffineTiledMode for T +where + T: TiledMode, + If<{ T::AFFINE_BACKGROUNDS > 0 }>: True, +{ + fn affine_background( + &self, + priority: Priority, + size: AffineBackgroundSize, + ) -> MapLoan<'_, AffineMap> { + let mut affine = self.affine().borrow_mut(); + let new_background = affine.first_zero().unwrap(); + if new_background >= T::AFFINE_BACKGROUNDS + AFFINE_BG_ID_OFFSET { + panic!( + "can only have {} active affine backgrounds", + T::AFFINE_BACKGROUNDS + ); + } + + let num_screenblocks = size.num_screen_blocks(); + let mut screenblocks = self.screenblocks().borrow_mut(); + + let screenblock = find_screenblock_gap(&screenblocks, num_screenblocks); + for id in screenblock..(screenblock + num_screenblocks) { + screenblocks.set(id, true); + } + + let center_dim = Num::new(size.width() as i32 * 8 / 2); + let default_bg_center = (center_dim, center_dim).into(); + + let bg = AffineMap::new( + new_background as u8, + screenblock as u8 + 16, + priority, + size, + default_bg_center, + ); + + affine.set(new_background, true); + + MapLoan::new( + bg, + new_background as u8, + screenblock as u8, + num_screenblocks as u8, + self.affine(), + self.screenblocks(), + ) + } +} + #[cfg(test)] mod test { use super::*; diff --git a/agb/src/display/tiled/tiled0.rs b/agb/src/display/tiled/tiled0.rs index fbc523e2..e824a8e3 100644 --- a/agb/src/display/tiled/tiled0.rs +++ b/agb/src/display/tiled/tiled0.rs @@ -1,12 +1,11 @@ use core::cell::RefCell; +use super::{MapLoan, RegularBackgroundSize, RegularMap, RegularTiledMode, TiledMode}; use crate::{ bitarray::Bitarray, display::{set_graphics_mode, DisplayMode, Priority}, }; -use super::{MapLoan, RegularBackgroundSize, RegularMap}; - pub struct Tiled0 { regular: RefCell>, screenblocks: RefCell>, @@ -27,52 +26,23 @@ impl Tiled0 { priority: Priority, size: RegularBackgroundSize, ) -> MapLoan<'_, RegularMap> { - let mut regular = self.regular.borrow_mut(); - let new_background = regular.first_zero().unwrap(); - if new_background >= 4 { - panic!("can only have 4 active backgrounds"); - } - - let num_screenblocks = size.num_screen_blocks(); - let mut screenblocks = self.screenblocks.borrow_mut(); - - let screenblock = find_screenblock_gap(&screenblocks, num_screenblocks); - for id in screenblock..(screenblock + num_screenblocks) { - screenblocks.set(id, true); - } - - let bg = RegularMap::new(new_background as u8, screenblock as u8 + 16, priority, size); - - regular.set(new_background, true); - - MapLoan::new( - bg, - new_background as u8, - screenblock as u8, - num_screenblocks as u8, - &self.regular, - &self.screenblocks, - ) + self.regular_background(priority, size) } } -fn find_screenblock_gap(screenblocks: &Bitarray<1>, gap: usize) -> usize { - let mut candidate = 0; +impl TiledMode for Tiled0 { + const REGULAR_BACKGROUNDS: usize = 4; + const AFFINE_BACKGROUNDS: usize = 0; - 'outer: while candidate < 16 - gap { - let starting_point = candidate; - for attempt in starting_point..(starting_point + gap) { - if screenblocks.get(attempt) == Some(true) { - candidate = attempt + 1; - continue 'outer; - } - } - - return candidate; + fn screenblocks(&self) -> &RefCell> { + &self.screenblocks } - panic!( - "Failed to find screenblock gap of at least {} elements", - gap - ); + fn regular(&self) -> &RefCell> { + &self.regular + } + + fn affine(&self) -> &RefCell> { + unimplemented!() + } } diff --git a/agb/src/display/tiled/tiled1.rs b/agb/src/display/tiled/tiled1.rs new file mode 100644 index 00000000..2b4af6ee --- /dev/null +++ b/agb/src/display/tiled/tiled1.rs @@ -0,0 +1,47 @@ +use core::cell::RefCell; + +use super::TiledMode; +use crate::{ + bitarray::Bitarray, + display::{set_graphics_mode, tiled::AFFINE_BG_ID_OFFSET, DisplayMode}, +}; + +pub struct Tiled1 { + regular: RefCell>, + affine: RefCell>, + screenblocks: RefCell>, +} + +impl Tiled1 { + pub(crate) unsafe fn new() -> Self { + set_graphics_mode(DisplayMode::Tiled1); + + let affine = RefCell::new(Bitarray::new()); + for i in 0..AFFINE_BG_ID_OFFSET { + affine.borrow_mut().set(i, true); + } + + Self { + regular: Default::default(), + affine, + screenblocks: Default::default(), + } + } +} + +impl TiledMode for Tiled1 { + const REGULAR_BACKGROUNDS: usize = 2; + const AFFINE_BACKGROUNDS: usize = 1; + + fn screenblocks(&self) -> &RefCell> { + &self.screenblocks + } + + fn regular(&self) -> &RefCell> { + &self.regular + } + + fn affine(&self) -> &RefCell> { + &self.affine + } +} diff --git a/agb/src/display/tiled/tiled2.rs b/agb/src/display/tiled/tiled2.rs new file mode 100644 index 00000000..60a3c728 --- /dev/null +++ b/agb/src/display/tiled/tiled2.rs @@ -0,0 +1,53 @@ +use core::cell::RefCell; + +use super::{AffineBackgroundSize, AffineMap, AffineTiledMode, MapLoan, TiledMode}; +use crate::{ + bitarray::Bitarray, + display::{set_graphics_mode, tiled::AFFINE_BG_ID_OFFSET, DisplayMode, Priority}, +}; + +pub struct Tiled2 { + affine: RefCell>, + screenblocks: RefCell>, +} + +impl Tiled2 { + pub(crate) unsafe fn new() -> Self { + set_graphics_mode(DisplayMode::Tiled2); + + let affine = RefCell::new(Bitarray::new()); + for i in 0..AFFINE_BG_ID_OFFSET { + affine.borrow_mut().set(i, true); + } + + Self { + affine, + screenblocks: Default::default(), + } + } + + pub fn background( + &self, + priority: Priority, + size: AffineBackgroundSize, + ) -> MapLoan<'_, AffineMap> { + self.affine_background(priority, size) + } +} + +impl TiledMode for Tiled2 { + const REGULAR_BACKGROUNDS: usize = 0; + const AFFINE_BACKGROUNDS: usize = 2; + + fn screenblocks(&self) -> &RefCell> { + &self.screenblocks + } + + fn regular(&self) -> &RefCell> { + unimplemented!() + } + + fn affine(&self) -> &RefCell> { + &self.affine + } +} diff --git a/agb/src/display/tiled/vram_manager.rs b/agb/src/display/tiled/vram_manager.rs index 6caa8956..b9ee9ff4 100644 --- a/agb/src/display/tiled/vram_manager.rs +++ b/agb/src/display/tiled/vram_manager.rs @@ -2,6 +2,7 @@ use core::{alloc::Layout, ptr::NonNull}; use alloc::{slice, vec::Vec}; +use crate::display::tiled::Tile; use crate::{ agb_alloc::{block_allocator::BlockAllocator, bump_allocator::StartEnd}, display::palette16, @@ -22,18 +23,22 @@ static TILE_ALLOCATOR: BlockAllocator = unsafe { }) }; -const TILE_LAYOUT: Layout = unsafe { Layout::from_size_align_unchecked(8 * 8 / 2, 8 * 8 / 2) }; +const fn layout_of(format: TileFormat) -> Layout { + unsafe { Layout::from_size_align_unchecked(format.tile_size(), format.tile_size()) } +} #[derive(Clone, Copy, Debug)] pub enum TileFormat { FourBpp, + EightBpp, } impl TileFormat { /// Returns the size of the tile in bytes - fn tile_size(self) -> usize { + pub(crate) const fn tile_size(self) -> usize { match self { TileFormat::FourBpp => 8 * 8 / 2, + TileFormat::EightBpp => 8 * 8, } } } @@ -55,15 +60,50 @@ impl<'a> TileSet<'a> { } #[derive(Debug, Clone, Copy)] -pub struct TileIndex(u16); +pub enum TileIndex { + FourBpp(u16), + EightBpp(u8), +} impl TileIndex { - pub(crate) const fn new(index: usize) -> Self { - Self(index as u16) + pub(crate) const fn new(index: usize, format: TileFormat) -> Self { + match format { + TileFormat::FourBpp => Self::FourBpp(index as u16), + TileFormat::EightBpp => Self::EightBpp(index as u8), + } } - pub(crate) const fn index(self) -> u16 { - self.0 + pub(crate) const fn raw_index(self) -> u16 { + match self { + TileIndex::FourBpp(x) => x, + TileIndex::EightBpp(x) => x as u16, + } + } + + pub(crate) const fn format(self) -> TileFormat { + match self { + TileIndex::FourBpp(_) => TileFormat::FourBpp, + TileIndex::EightBpp(_) => TileFormat::EightBpp, + } + } + + fn refcount_key(self) -> usize { + match self { + TileIndex::FourBpp(x) => x as usize, + TileIndex::EightBpp(x) => x as usize * 2, + } + } +} + +impl From for TileIndex { + fn from(tile: Tile) -> Self { + tile.tile_index() + } +} + +impl From for TileIndex { + fn from(index: u8) -> TileIndex { + TileIndex::new(usize::from(index), TileFormat::EightBpp) } } @@ -159,7 +199,7 @@ impl DynamicTile<'_> { #[must_use] pub fn tile_index(&self) -> u16 { let difference = self.tile_data.as_ptr() as usize - TILE_RAM_START; - (difference / (8 * 8 / 2)) as u16 + (difference / TileFormat::FourBpp.tile_size()) as u16 } } @@ -182,24 +222,27 @@ impl VRamManager { } } - fn index_from_reference(reference: TileReference) -> usize { + fn index_from_reference(reference: TileReference, format: TileFormat) -> TileIndex { let difference = reference.0.as_ptr() as usize - TILE_RAM_START; - difference / (8 * 8 / 2) + TileIndex::new(difference / format.tile_size(), format) } fn reference_from_index(index: TileIndex) -> TileReference { - let ptr = (index.index() * (8 * 8 / 2)) as usize + TILE_RAM_START; + let ptr = (index.raw_index() as usize * index.format().tile_size()) + TILE_RAM_START; TileReference(NonNull::new(ptr as *mut _).unwrap()) } #[must_use] pub fn new_dynamic_tile<'a>(&mut self) -> DynamicTile<'a> { + // TODO: format param? let tile_format = TileFormat::FourBpp; - let new_reference: NonNull = - unsafe { TILE_ALLOCATOR.alloc(TILE_LAYOUT) }.unwrap().cast(); + let new_reference: NonNull = unsafe { TILE_ALLOCATOR.alloc(layout_of(tile_format)) } + .unwrap() + .cast(); let tile_reference = TileReference(new_reference); - let index = Self::index_from_reference(tile_reference); + let index = Self::index_from_reference(tile_reference, tile_format); + let key = index.refcount_key(); let tiles = unsafe { slice::from_raw_parts_mut(TILE_RAM_START as *mut u8, 1024 * tile_format.tile_size()) @@ -208,23 +251,21 @@ impl VRamManager { let tile_set = TileSet::new(tiles, tile_format); self.tile_set_to_vram.insert( - TileInTileSetReference::new(&tile_set, index as u16), + TileInTileSetReference::new(&tile_set, index.raw_index()), tile_reference, ); - self.reference_counts.resize( - self.reference_counts.len().max(index + 1), - Default::default(), - ); - self.reference_counts[index] = - TileReferenceCount::new(TileInTileSetReference::new(&tile_set, index as u16)); + self.reference_counts + .resize(self.reference_counts.len().max(key + 1), Default::default()); + self.reference_counts[key] = + TileReferenceCount::new(TileInTileSetReference::new(&tile_set, index.raw_index())); DynamicTile { tile_data: unsafe { slice::from_raw_parts_mut( tiles .as_mut_ptr() - .add((index * tile_format.tile_size()) as usize) + .add(index.raw_index() as usize * tile_format.tile_size()) .cast(), tile_format.tile_size() / core::mem::size_of::(), ) @@ -238,8 +279,9 @@ impl VRamManager { let pointer = NonNull::new(dynamic_tile.tile_data.as_mut_ptr() as *mut _).unwrap(); let tile_reference = TileReference(pointer); - let tile_index = Self::index_from_reference(tile_reference); - self.remove_tile(TileIndex::new(tile_index)); + // TODO: dynamic_tile.format? + let tile_index = Self::index_from_reference(tile_reference, TileFormat::FourBpp); + self.remove_tile(tile_index); } pub(crate) fn add_tile(&mut self, tile_set: &TileSet<'_>, tile: u16) -> TileIndex { @@ -248,37 +290,39 @@ impl VRamManager { .get(&TileInTileSetReference::new(tile_set, tile)); if let Some(reference) = reference { - let index = Self::index_from_reference(*reference); - self.reference_counts[index].increment_reference_count(); - return TileIndex::new(index); + let tile_index = Self::index_from_reference(*reference, tile_set.format); + let key = tile_index.refcount_key(); + self.reference_counts[key].increment_reference_count(); + return tile_index; } let new_reference: NonNull = - unsafe { TILE_ALLOCATOR.alloc(TILE_LAYOUT) }.unwrap().cast(); + unsafe { TILE_ALLOCATOR.alloc(layout_of(tile_set.format)) } + .unwrap() + .cast(); let tile_reference = TileReference(new_reference); self.copy_tile_to_location(tile_set, tile, tile_reference); - let index = Self::index_from_reference(tile_reference); + let index = Self::index_from_reference(tile_reference, tile_set.format); + let key = index.refcount_key(); self.tile_set_to_vram .insert(TileInTileSetReference::new(tile_set, tile), tile_reference); - self.reference_counts.resize( - self.reference_counts.len().max(index + 1), - Default::default(), - ); + self.reference_counts + .resize(self.reference_counts.len().max(key + 1), Default::default()); - self.reference_counts[index] = + self.reference_counts[key] = TileReferenceCount::new(TileInTileSetReference::new(tile_set, tile)); - TileIndex::new(index) + index } pub(crate) fn remove_tile(&mut self, tile_index: TileIndex) { - let index = tile_index.index() as usize; + let key = tile_index.refcount_key(); - let new_reference_count = self.reference_counts[index].decrement_reference_count(); + let new_reference_count = self.reference_counts[key].decrement_reference_count(); if new_reference_count != 0 { return; @@ -289,23 +333,26 @@ impl VRamManager { pub(crate) fn gc(&mut self) { for tile_index in self.indices_to_gc.drain(..) { - let index = tile_index.index() as usize; - if self.reference_counts[index].current_count() > 0 { + let key = tile_index.refcount_key() as usize; + if self.reference_counts[key].current_count() > 0 { continue; // it has since been added back } let tile_reference = Self::reference_from_index(tile_index); unsafe { - TILE_ALLOCATOR.dealloc_no_normalise(tile_reference.0.cast().as_ptr(), TILE_LAYOUT); + TILE_ALLOCATOR.dealloc_no_normalise( + tile_reference.0.cast().as_ptr(), + layout_of(tile_index.format()), + ); } - let tile_ref = self.reference_counts[index] + let tile_ref = self.reference_counts[key] .tile_in_tile_set .as_ref() .unwrap(); self.tile_set_to_vram.remove(tile_ref); - self.reference_counts[index].clear(); + self.reference_counts[key].clear(); } } diff --git a/agb/src/display/video.rs b/agb/src/display/video.rs index 9f62e3fa..d0edf2e1 100644 --- a/agb/src/display/video.rs +++ b/agb/src/display/video.rs @@ -1,7 +1,7 @@ use super::{ bitmap3::Bitmap3, bitmap4::Bitmap4, - tiled::{Tiled0, VRamManager}, + tiled::{Tiled0, Tiled1, Tiled2, VRamManager}, }; /// The video struct controls access to the video hardware. @@ -26,4 +26,14 @@ impl Video { pub fn tiled0(&mut self) -> (Tiled0, VRamManager) { (unsafe { Tiled0::new() }, VRamManager::new()) } + + /// Tiled 1 mode provides 2 regular tiled backgrounds and 1 affine tiled background + pub fn tiled1(&mut self) -> (Tiled1, VRamManager) { + (unsafe { Tiled1::new() }, VRamManager::new()) + } + + /// Tiled 2 mode provides 2 affine tiled backgrounds + pub fn tiled2(&mut self) -> (Tiled2, VRamManager) { + (unsafe { Tiled2::new() }, VRamManager::new()) + } } diff --git a/agb/src/lib.rs b/agb/src/lib.rs index b3cd8b4a..35a14b6d 100644 --- a/agb/src/lib.rs +++ b/agb/src/lib.rs @@ -13,6 +13,9 @@ #![feature(alloc_error_handler)] #![feature(allocator_api)] #![feature(asm_const)] +#![feature(const_trait_impl)] +#![allow(incomplete_features)] // generic_const_exprs, used for bg mode traits +#![feature(generic_const_exprs)] #![warn(clippy::all)] #![deny(clippy::must_use_candidate)] #![deny(clippy::trivially_copy_pass_by_ref)] @@ -80,7 +83,7 @@ /// # /// use agb::{ /// display::{ -/// tiled::{RegularBackgroundSize, TileFormat, TileSet, TileSetting, Tiled0, VRamManager}, +/// tiled::{RegularBackgroundSize, TileFormat, TileSet, TileSetting, Tiled0, TiledMap, VRamManager}, /// Priority, /// }, /// include_gfx, diff --git a/agb/src/memory_mapped.rs b/agb/src/memory_mapped.rs index 290d865f..617dc48d 100644 --- a/agb/src/memory_mapped.rs +++ b/agb/src/memory_mapped.rs @@ -16,7 +16,9 @@ impl MemoryMapped { } pub fn set(&self, val: T) { - unsafe { self.address.write_volatile(val) } + if core::mem::size_of::() != 0 { + unsafe { self.address.write_volatile(val) } + } } } diff --git a/agb/src/syscall.rs b/agb/src/syscall.rs index 69ff353f..59d37a52 100644 --- a/agb/src/syscall.rs +++ b/agb/src/syscall.rs @@ -1,11 +1,14 @@ +use agb_fixnum::Vector2D; use core::arch::asm; +use core::mem::MaybeUninit; -// use crate::display::object::AffineMatrixAttributes; +use crate::display::object::AffineMatrixAttributes; +use crate::fixnum::Num; #[allow(non_snake_case)] const fn swi_map(thumb_id: u32) -> u32 { - if cfg!(target_feature="thumb-mode") { + if cfg!(target_feature = "thumb-mode") { thumb_id } else { thumb_id << 16 @@ -136,55 +139,103 @@ pub fn arc_tan2(x: i16, y: i32) -> i16 { result } -// pub fn affine_matrix( -// x_scale: Num, -// y_scale: Num, -// rotation: u8, -// ) -> AffineMatrixAttributes { -// let mut result = AffineMatrixAttributes { -// p_a: 0, -// p_b: 0, -// p_c: 0, -// p_d: 0, -// }; +#[repr(C)] +pub struct BgAffineSetData { + pub matrix: AffineMatrixAttributes, + pub position: Vector2D>, +} +impl Default for BgAffineSetData { + fn default() -> Self { + Self { + matrix: AffineMatrixAttributes::default(), + position: (0, 0).into(), + } + } +} -// #[allow(dead_code)] -// #[repr(C, packed)] -// struct Input { -// x_scale: i16, -// y_scale: i16, -// rotation: u16, -// } +/// `rotation` is in revolutions. +#[must_use] +pub fn bg_affine_matrix( + bg_center: Vector2D>, + display_center: Vector2D, + scale: Vector2D>, + rotation: Num, +) -> BgAffineSetData { + #[repr(C, packed)] + struct Input { + bg_center: Vector2D>, + display_center: Vector2D, + scale: Vector2D>, + rotation: u16, + } -// let input = Input { -// y_scale: x_scale.to_raw(), -// x_scale: y_scale.to_raw(), -// rotation: rotation as u16, -// }; + let input = Input { + bg_center, + display_center, + scale, + rotation: u16::from(rotation.to_raw()) << 8, + }; -// unsafe { -// asm!("swi 0x0F", -// in("r0") &input as *const Input as usize, -// in("r1") &mut result as *mut AffineMatrixAttributes as usize, -// in("r2") 1, -// in("r3") 2, -// ) -// } + let mut output = MaybeUninit::uninit(); -// result -// } + unsafe { + asm!( + "swi {SWI}", + SWI = const { swi_map(0x0E) }, + in("r0") &input as *const Input, + in("r1") output.as_mut_ptr(), + in("r2") 1, + ); + } -// #[cfg(test)] -// mod tests { -// use super::*; + unsafe { output.assume_init() } +} -// #[test_case] -// fn affine(_gba: &mut crate::Gba) { -// // expect identity matrix -// let one: Num = 1.into(); +/// `rotation` is in revolutions. +#[must_use] +pub fn obj_affine_matrix( + scale: Vector2D>, + rotation: Num, +) -> AffineMatrixAttributes { + #[allow(dead_code)] + #[repr(C, packed)] + struct Input { + scale: Vector2D>, + rotation: u16, + } -// let aff = affine_matrix(one, one, 0); -// assert_eq!(aff.p_a, one.to_raw()); -// assert_eq!(aff.p_d, one.to_raw()); -// } -// } + let input = Input { + scale, + rotation: u16::from(rotation.to_raw()) << 8, + }; + + let mut output = MaybeUninit::uninit(); + + unsafe { + asm!( + "swi {SWI}", + SWI = const { swi_map(0x0F) }, + in("r0") &input as *const Input, + in("r1") output.as_mut_ptr(), + in("r2") 1, + in("r3") 2, + ); + } + + unsafe { output.assume_init() } +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test_case] + fn affine(_gba: &mut crate::Gba) { + // expect identity matrix + let one: Num = 1.into(); + + let aff = obj_affine_matrix((one, one).into(), Num::default()); + assert_eq!(aff.p_a, one); + assert_eq!(aff.p_d, one); + } +} diff --git a/examples/hyperspace-roll/build.rs b/examples/hyperspace-roll/build.rs index e71fdf55..f328e4d9 100644 --- a/examples/hyperspace-roll/build.rs +++ b/examples/hyperspace-roll/build.rs @@ -1 +1 @@ -fn main() {} \ No newline at end of file +fn main() {} diff --git a/examples/hyperspace-roll/src/background.rs b/examples/hyperspace-roll/src/background.rs index 91d16c95..c67cc709 100644 --- a/examples/hyperspace-roll/src/background.rs +++ b/examples/hyperspace-roll/src/background.rs @@ -1,5 +1,5 @@ use agb::{ - display::tiled::{RegularMap, TileFormat, TileSet, TileSetting, VRamManager}, + display::tiled::{RegularMap, TileFormat, TileSet, TileSetting, TiledMap, VRamManager}, include_gfx, rng, }; diff --git a/examples/hyperspace-roll/src/battle.rs b/examples/hyperspace-roll/src/battle.rs index 3fd3b870..691c7a85 100644 --- a/examples/hyperspace-roll/src/battle.rs +++ b/examples/hyperspace-roll/src/battle.rs @@ -2,7 +2,7 @@ use crate::sfx::Sfx; use crate::{ graphics::SELECT_BOX, level_generation::generate_attack, Agb, EnemyAttackType, Face, PlayerDice, }; -use agb::display::tiled::RegularMap; +use agb::display::tiled::{RegularMap, TiledMap}; use agb::{hash_map::HashMap, input::Button}; use alloc::vec; use alloc::vec::Vec; diff --git a/examples/hyperspace-roll/src/customise.rs b/examples/hyperspace-roll/src/customise.rs index 852ff5a5..c91f3cd9 100644 --- a/examples/hyperspace-roll/src/customise.rs +++ b/examples/hyperspace-roll/src/customise.rs @@ -2,7 +2,7 @@ use agb::{ display::{ object::{Object, ObjectController}, palette16::Palette16, - tiled::{RegularMap, TileSet, TileSetting}, + tiled::{RegularMap, TileSet, TileSetting, TiledMap}, HEIGHT, WIDTH, }, include_gfx, diff --git a/examples/hyperspace-roll/src/main.rs b/examples/hyperspace-roll/src/main.rs index 9acf3c04..49e43b73 100644 --- a/examples/hyperspace-roll/src/main.rs +++ b/examples/hyperspace-roll/src/main.rs @@ -12,7 +12,7 @@ use agb::display; use agb::display::object::ObjectController; -use agb::display::tiled::VRamManager; +use agb::display::tiled::{TiledMap, VRamManager}; use agb::display::Priority; use agb::interrupt::VBlank; diff --git a/examples/the-hat-chooses-the-wizard/src/level_display.rs b/examples/the-hat-chooses-the-wizard/src/level_display.rs index 990e7635..10ac10ec 100644 --- a/examples/the-hat-chooses-the-wizard/src/level_display.rs +++ b/examples/the-hat-chooses-the-wizard/src/level_display.rs @@ -1,5 +1,5 @@ use agb::display::{ - tiled::{RegularMap, TileSet, TileSetting, VRamManager}, + tiled::{RegularMap, TileSet, TileSetting, TiledMap, VRamManager}, HEIGHT, WIDTH, }; diff --git a/examples/the-hat-chooses-the-wizard/src/main.rs b/examples/the-hat-chooses-the-wizard/src/main.rs index 133afb4e..031b5cd7 100644 --- a/examples/the-hat-chooses-the-wizard/src/main.rs +++ b/examples/the-hat-chooses-the-wizard/src/main.rs @@ -11,7 +11,7 @@ use agb::{ object::{Graphics, Object, ObjectController, Tag, TagMap}, tiled::{ InfiniteScrolledMap, PartialUpdateStatus, RegularBackgroundSize, TileFormat, TileSet, - TileSetting, VRamManager, + TileSetting, TiledMap, VRamManager, }, Priority, HEIGHT, WIDTH, }, diff --git a/examples/the-hat-chooses-the-wizard/src/splash_screen.rs b/examples/the-hat-chooses-the-wizard/src/splash_screen.rs index 64243b9b..de6f9be9 100644 --- a/examples/the-hat-chooses-the-wizard/src/splash_screen.rs +++ b/examples/the-hat-chooses-the-wizard/src/splash_screen.rs @@ -1,6 +1,6 @@ use super::sfx::MusicBox; use agb::{ - display::tiled::{RegularMap, TileFormat, TileSet, TileSetting, VRamManager}, + display::tiled::{RegularMap, TiledMap, TileFormat, TileSet, TileSetting, VRamManager}, sound::mixer::Mixer, }; From d59e2672c09b9047006cd1ad0e283e7f73a0d605 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 11 Aug 2022 20:38:28 +0100 Subject: [PATCH 02/49] Remove features --- agb/src/lib.rs | 3 --- 1 file changed, 3 deletions(-) diff --git a/agb/src/lib.rs b/agb/src/lib.rs index 35a14b6d..c69b03b6 100644 --- a/agb/src/lib.rs +++ b/agb/src/lib.rs @@ -13,9 +13,6 @@ #![feature(alloc_error_handler)] #![feature(allocator_api)] #![feature(asm_const)] -#![feature(const_trait_impl)] -#![allow(incomplete_features)] // generic_const_exprs, used for bg mode traits -#![feature(generic_const_exprs)] #![warn(clippy::all)] #![deny(clippy::must_use_candidate)] #![deny(clippy::trivially_copy_pass_by_ref)] From 31f85cf6b962ec1c2772a12f45197fe8101f272e Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 11 Aug 2022 20:51:15 +0100 Subject: [PATCH 03/49] Simplify tiled0-2 --- agb/src/display/tiled/mod.rs | 27 +++++++++++++-------------- agb/src/display/tiled/tiled0.rs | 16 ++++++++-------- agb/src/display/tiled/tiled1.rs | 13 +++++++++---- agb/src/display/tiled/tiled2.rs | 13 ++++++------- 4 files changed, 36 insertions(+), 33 deletions(-) diff --git a/agb/src/display/tiled/mod.rs b/agb/src/display/tiled/mod.rs index aadba54a..164ce3a6 100644 --- a/agb/src/display/tiled/mod.rs +++ b/agb/src/display/tiled/mod.rs @@ -210,15 +210,21 @@ pub(self) fn find_screenblock_gap(screenblocks: &Bitarray<1>, gap: usize) -> usi ); } -pub(self) trait TiledMode { - const REGULAR_BACKGROUNDS: usize; - const AFFINE_BACKGROUNDS: usize; +trait TiledMode { fn screenblocks(&self) -> &RefCell>; +} + +trait CreatableRegularTiledMode: TiledMode { + const REGULAR_BACKGROUNDS: usize; fn regular(&self) -> &RefCell>; +} + +trait CreatableAffineTiledMode: TiledMode { + const AFFINE_BACKGROUNDS: usize; fn affine(&self) -> &RefCell>; } -pub trait RegularTiledMode { +trait RegularTiledMode { fn regular_background( &self, priority: Priority, @@ -226,7 +232,7 @@ pub trait RegularTiledMode { ) -> MapLoan<'_, RegularMap>; } -pub trait AffineTiledMode { +trait AffineTiledMode { fn affine_background( &self, priority: Priority, @@ -234,15 +240,9 @@ pub trait AffineTiledMode { ) -> MapLoan<'_, AffineMap>; } -// withoutboats: https://internals.rust-lang.org/t/const-generics-where-restrictions/12742/6 -pub struct If; -pub trait True {} -impl True for If {} - impl RegularTiledMode for T where - T: TiledMode, - If<{ T::REGULAR_BACKGROUNDS > 0 }>: True, + T: CreatableRegularTiledMode, { fn regular_background( &self, @@ -283,8 +283,7 @@ where impl AffineTiledMode for T where - T: TiledMode, - If<{ T::AFFINE_BACKGROUNDS > 0 }>: True, + T: CreatableAffineTiledMode, { fn affine_background( &self, diff --git a/agb/src/display/tiled/tiled0.rs b/agb/src/display/tiled/tiled0.rs index e824a8e3..66ff0054 100644 --- a/agb/src/display/tiled/tiled0.rs +++ b/agb/src/display/tiled/tiled0.rs @@ -1,6 +1,9 @@ use core::cell::RefCell; -use super::{MapLoan, RegularBackgroundSize, RegularMap, RegularTiledMode, TiledMode}; +use super::{ + CreatableRegularTiledMode, MapLoan, RegularBackgroundSize, RegularMap, RegularTiledMode, + TiledMode, +}; use crate::{ bitarray::Bitarray, display::{set_graphics_mode, DisplayMode, Priority}, @@ -31,18 +34,15 @@ impl Tiled0 { } impl TiledMode for Tiled0 { - const REGULAR_BACKGROUNDS: usize = 4; - const AFFINE_BACKGROUNDS: usize = 0; - fn screenblocks(&self) -> &RefCell> { &self.screenblocks } +} + +impl CreatableRegularTiledMode for Tiled0 { + const REGULAR_BACKGROUNDS: usize = 4; fn regular(&self) -> &RefCell> { &self.regular } - - fn affine(&self) -> &RefCell> { - unimplemented!() - } } diff --git a/agb/src/display/tiled/tiled1.rs b/agb/src/display/tiled/tiled1.rs index 2b4af6ee..84903aa2 100644 --- a/agb/src/display/tiled/tiled1.rs +++ b/agb/src/display/tiled/tiled1.rs @@ -1,6 +1,6 @@ use core::cell::RefCell; -use super::TiledMode; +use super::{CreatableAffineTiledMode, CreatableRegularTiledMode, TiledMode}; use crate::{ bitarray::Bitarray, display::{set_graphics_mode, tiled::AFFINE_BG_ID_OFFSET, DisplayMode}, @@ -30,16 +30,21 @@ impl Tiled1 { } impl TiledMode for Tiled1 { - const REGULAR_BACKGROUNDS: usize = 2; - const AFFINE_BACKGROUNDS: usize = 1; - fn screenblocks(&self) -> &RefCell> { &self.screenblocks } +} + +impl CreatableRegularTiledMode for Tiled1 { + const REGULAR_BACKGROUNDS: usize = 2; fn regular(&self) -> &RefCell> { &self.regular } +} + +impl CreatableAffineTiledMode for Tiled1 { + const AFFINE_BACKGROUNDS: usize = 1; fn affine(&self) -> &RefCell> { &self.affine diff --git a/agb/src/display/tiled/tiled2.rs b/agb/src/display/tiled/tiled2.rs index 60a3c728..bcef9f31 100644 --- a/agb/src/display/tiled/tiled2.rs +++ b/agb/src/display/tiled/tiled2.rs @@ -1,6 +1,8 @@ use core::cell::RefCell; -use super::{AffineBackgroundSize, AffineMap, AffineTiledMode, MapLoan, TiledMode}; +use super::{ + AffineBackgroundSize, AffineMap, AffineTiledMode, CreatableAffineTiledMode, MapLoan, TiledMode, +}; use crate::{ bitarray::Bitarray, display::{set_graphics_mode, tiled::AFFINE_BG_ID_OFFSET, DisplayMode, Priority}, @@ -36,16 +38,13 @@ impl Tiled2 { } impl TiledMode for Tiled2 { - const REGULAR_BACKGROUNDS: usize = 0; - const AFFINE_BACKGROUNDS: usize = 2; - fn screenblocks(&self) -> &RefCell> { &self.screenblocks } +} - fn regular(&self) -> &RefCell> { - unimplemented!() - } +impl CreatableAffineTiledMode for Tiled2 { + const AFFINE_BACKGROUNDS: usize = 2; fn affine(&self) -> &RefCell> { &self.affine From affea47a32ad4880482b2d4b72a04abbdc358e60 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 11 Aug 2022 20:52:35 +0100 Subject: [PATCH 04/49] Remove const and rustfmt::skip --- agb/src/display/tiled/map.rs | 92 +++++++++++++++++++++++++----------- 1 file changed, 65 insertions(+), 27 deletions(-) diff --git a/agb/src/display/tiled/map.rs b/agb/src/display/tiled/map.rs index 0b5e5138..85327c98 100644 --- a/agb/src/display/tiled/map.rs +++ b/agb/src/display/tiled/map.rs @@ -145,12 +145,19 @@ pub struct RegularMap { pub const TRANSPARENT_TILE_INDEX: u16 = (1 << 10) - 1; -#[rustfmt::skip] impl TiledMapPrivate for RegularMap { - fn tiles_mut(&mut self) -> &mut [Self::TileType] { &mut self.tiles } - fn tiles_dirty(&mut self) -> &mut bool { &mut self.tiles_dirty } - fn x_scroll_mut(&mut self) -> &mut Self::Position { &mut self.x_scroll } - fn y_scroll_mut(&mut self) -> &mut Self::Position { &mut self.y_scroll } + fn tiles_mut(&mut self) -> &mut [Self::TileType] { + &mut self.tiles + } + fn tiles_dirty(&mut self) -> &mut bool { + &mut self.tiles_dirty + } + fn x_scroll_mut(&mut self) -> &mut Self::Position { + &mut self.x_scroll + } + fn y_scroll_mut(&mut self) -> &mut Self::Position { + &mut self.y_scroll + } } impl TiledMapTypes for RegularMap { @@ -158,17 +165,28 @@ impl TiledMapTypes for RegularMap { type Size = RegularBackgroundSize; } -#[rustfmt::skip] -impl const TiledMapPrivateConst for RegularMap { +impl TiledMapPrivateConst for RegularMap { type TileType = Tile; type AffineMatrix = (); - fn x_scroll(&self) -> Self::Position { self.x_scroll } - fn y_scroll(&self) -> Self::Position { self.y_scroll } + fn x_scroll(&self) -> Self::Position { + self.x_scroll + } + fn y_scroll(&self) -> Self::Position { + self.y_scroll + } fn affine_matrix(&self) -> Self::AffineMatrix {} - fn background_id(&self) -> usize { self.background_id as usize } - fn screenblock(&self) -> usize { self.screenblock as usize } - fn priority(&self) -> Priority { self.priority } - fn map_size(&self) -> Self::Size { self.size } + fn background_id(&self) -> usize { + self.background_id as usize + } + fn screenblock(&self) -> usize { + self.screenblock as usize + } + fn priority(&self) -> Priority { + self.priority + } + fn map_size(&self) -> Self::Size { + self.size + } fn bg_x(&self) -> MemoryMapped { unsafe { MemoryMapped::new(0x0400_0010 + 4 * self.background_id as usize) } } @@ -247,12 +265,19 @@ pub struct AffineMap { tiles_dirty: bool, } -#[rustfmt::skip] impl TiledMapPrivate for AffineMap { - fn tiles_mut(&mut self) -> &mut [Self::TileType] { &mut self.tiles } - fn tiles_dirty(&mut self) -> &mut bool { &mut self.tiles_dirty } - fn x_scroll_mut(&mut self) -> &mut Self::Position { &mut self.transform.position.x } - fn y_scroll_mut(&mut self) -> &mut Self::Position { &mut self.transform.position.y } + fn tiles_mut(&mut self) -> &mut [Self::TileType] { + &mut self.tiles + } + fn tiles_dirty(&mut self) -> &mut bool { + &mut self.tiles_dirty + } + fn x_scroll_mut(&mut self) -> &mut Self::Position { + &mut self.transform.position.x + } + fn y_scroll_mut(&mut self) -> &mut Self::Position { + &mut self.transform.position.y + } } impl TiledMapTypes for AffineMap { @@ -260,17 +285,30 @@ impl TiledMapTypes for AffineMap { type Size = AffineBackgroundSize; } -#[rustfmt::skip] -impl const TiledMapPrivateConst for AffineMap { +impl TiledMapPrivateConst for AffineMap { type TileType = u8; type AffineMatrix = AffineMatrixAttributes; - fn x_scroll(&self) -> Self::Position { self.transform.position.x } - fn y_scroll(&self) -> Self::Position { self.transform.position.y } - fn affine_matrix(&self) -> Self::AffineMatrix { self.transform.matrix } - fn background_id(&self) -> usize { self.background_id as usize } - fn screenblock(&self) -> usize { self.screenblock as usize } - fn priority(&self) -> Priority { self.priority } - fn map_size(&self) -> Self::Size { self.size } + fn x_scroll(&self) -> Self::Position { + self.transform.position.x + } + fn y_scroll(&self) -> Self::Position { + self.transform.position.y + } + fn affine_matrix(&self) -> Self::AffineMatrix { + self.transform.matrix + } + fn background_id(&self) -> usize { + self.background_id as usize + } + fn screenblock(&self) -> usize { + self.screenblock as usize + } + fn priority(&self) -> Priority { + self.priority + } + fn map_size(&self) -> Self::Size { + self.size + } fn bg_x(&self) -> MemoryMapped { unsafe { MemoryMapped::new(0x0400_0008 + 0x10 * self.background_id()) } } From d106b0a82b9d4aef33cd72bec55f1ce76ba05a7a Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 11 Aug 2022 21:05:00 +0100 Subject: [PATCH 05/49] Simplify map.rs --- agb/src/display/tiled/map.rs | 69 +++++++++++++++++------------------- 1 file changed, 32 insertions(+), 37 deletions(-) diff --git a/agb/src/display/tiled/map.rs b/agb/src/display/tiled/map.rs index 85327c98..1933663a 100644 --- a/agb/src/display/tiled/map.rs +++ b/agb/src/display/tiled/map.rs @@ -15,9 +15,20 @@ use super::{ use crate::syscall::BgAffineSetData; use alloc::{vec, vec::Vec}; -pub(super) trait TiledMapPrivateConst: TiledMapTypes { +pub trait TiledMapTypes { + type Position: Number; + type Size: BackgroundSize + Copy; +} + +trait TiledMapPrivate: TiledMapTypes { type TileType: Into + Copy + Default + Eq + PartialEq; type AffineMatrix; + + fn tiles_mut(&mut self) -> &mut [Self::TileType]; + fn tiles_dirty(&mut self) -> &mut bool; + fn x_scroll_mut(&mut self) -> &mut Self::Position; + fn y_scroll_mut(&mut self) -> &mut Self::Position; + fn x_scroll(&self) -> Self::Position; fn y_scroll(&self) -> Self::Position; fn affine_matrix(&self) -> Self::AffineMatrix; @@ -36,18 +47,6 @@ pub(super) trait TiledMapPrivateConst: TiledMapTypes { } } -trait TiledMapPrivate: TiledMapPrivateConst { - fn tiles_mut(&mut self) -> &mut [Self::TileType]; - fn tiles_dirty(&mut self) -> &mut bool; - fn x_scroll_mut(&mut self) -> &mut Self::Position; - fn y_scroll_mut(&mut self) -> &mut Self::Position; -} - -pub trait TiledMapTypes { - type Position: Number; - type Size: BackgroundSize + Copy; -} - pub trait TiledMap: TiledMapTypes { fn clear(&mut self, vram: &mut VRamManager); fn show(&mut self); @@ -62,7 +61,7 @@ pub trait TiledMap: TiledMapTypes { impl TiledMap for T where - T: TiledMapPrivateConst + TiledMapPrivate + TiledMapTypes, + T: TiledMapPrivate, T::Size: BackgroundSizePrivate, { fn clear(&mut self, vram: &mut VRamManager) { @@ -115,16 +114,16 @@ where *self.tiles_dirty() = false; } - fn size(&self) -> Self::Size { + fn size(&self) -> T::Size { self.map_size() } #[must_use] - fn scroll_pos(&self) -> Vector2D { + fn scroll_pos(&self) -> Vector2D { (self.x_scroll(), self.y_scroll()).into() } - fn set_scroll_pos(&mut self, pos: Vector2D) { + fn set_scroll_pos(&mut self, pos: Vector2D) { *self.x_scroll_mut() = pos.x; *self.y_scroll_mut() = pos.y; } @@ -145,7 +144,15 @@ pub struct RegularMap { pub const TRANSPARENT_TILE_INDEX: u16 = (1 << 10) - 1; +impl TiledMapTypes for RegularMap { + type Position = u16; + type Size = RegularBackgroundSize; +} + impl TiledMapPrivate for RegularMap { + type TileType = Tile; + type AffineMatrix = (); + fn tiles_mut(&mut self) -> &mut [Self::TileType] { &mut self.tiles } @@ -158,16 +165,6 @@ impl TiledMapPrivate for RegularMap { fn y_scroll_mut(&mut self) -> &mut Self::Position { &mut self.y_scroll } -} - -impl TiledMapTypes for RegularMap { - type Position = u16; - type Size = RegularBackgroundSize; -} - -impl TiledMapPrivateConst for RegularMap { - type TileType = Tile; - type AffineMatrix = (); fn x_scroll(&self) -> Self::Position { self.x_scroll } @@ -265,7 +262,15 @@ pub struct AffineMap { tiles_dirty: bool, } +impl TiledMapTypes for AffineMap { + type Position = Num; + type Size = AffineBackgroundSize; +} + impl TiledMapPrivate for AffineMap { + type TileType = u8; + type AffineMatrix = AffineMatrixAttributes; + fn tiles_mut(&mut self) -> &mut [Self::TileType] { &mut self.tiles } @@ -278,16 +283,6 @@ impl TiledMapPrivate for AffineMap { fn y_scroll_mut(&mut self) -> &mut Self::Position { &mut self.transform.position.y } -} - -impl TiledMapTypes for AffineMap { - type Position = Num; - type Size = AffineBackgroundSize; -} - -impl TiledMapPrivateConst for AffineMap { - type TileType = u8; - type AffineMatrix = AffineMatrixAttributes; fn x_scroll(&self) -> Self::Position { self.transform.position.x } From 9a5ecc5cda66d7cb008ca45aea62735f7abbcde2 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 11 Aug 2022 21:08:06 +0100 Subject: [PATCH 06/49] Assert replace tile is passed two tiles of the same size --- agb/src/display/tiled/vram_manager.rs | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/agb/src/display/tiled/vram_manager.rs b/agb/src/display/tiled/vram_manager.rs index b9ee9ff4..6b3dbf43 100644 --- a/agb/src/display/tiled/vram_manager.rs +++ b/agb/src/display/tiled/vram_manager.rs @@ -27,7 +27,7 @@ const fn layout_of(format: TileFormat) -> Layout { unsafe { Layout::from_size_align_unchecked(format.tile_size(), format.tile_size()) } } -#[derive(Clone, Copy, Debug)] +#[derive(Clone, Copy, Debug, PartialEq, Eq)] pub enum TileFormat { FourBpp, EightBpp, @@ -363,6 +363,11 @@ impl VRamManager { target_tile_set: &TileSet<'_>, target_tile: u16, ) { + assert_eq!( + source_tile_set.format, target_tile_set.format, + "Must replace a tileset with the same format" + ); + if let Some(&reference) = self .tile_set_to_vram .get(&TileInTileSetReference::new(source_tile_set, source_tile)) From c87a9ff077b08f3c45818aac3e5fbe36189dd530 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 11 Aug 2022 21:16:26 +0100 Subject: [PATCH 07/49] Allow configuring backgrounds with a colour --- agb-image-converter/src/config.rs | 12 +++++++++++- agb-image-converter/src/lib.rs | 5 +++++ 2 files changed, 16 insertions(+), 1 deletion(-) diff --git a/agb-image-converter/src/config.rs b/agb-image-converter/src/config.rs index 5e0ce482..bccb252d 100644 --- a/agb-image-converter/src/config.rs +++ b/agb-image-converter/src/config.rs @@ -2,7 +2,7 @@ use serde::Deserialize; use std::collections::HashMap; use std::fs; -use crate::{Colour, TileSize}; +use crate::{Colour, Colours, TileSize}; pub(crate) fn parse(filename: &str) -> Box { let config_toml = @@ -23,6 +23,7 @@ pub(crate) fn parse(filename: &str) -> Box { pub(crate) trait Config { fn crate_prefix(&self) -> String; fn images(&self) -> HashMap; + fn colours(&self) -> Colours; } pub(crate) trait Image { @@ -35,6 +36,7 @@ pub(crate) trait Image { pub struct ConfigV1 { version: String, crate_prefix: Option, + colours: Option, image: HashMap, } @@ -52,6 +54,14 @@ impl Config for ConfigV1 { .map(|(filename, image)| (filename.clone(), image as &dyn Image)) .collect() } + + fn colours(&self) -> Colours { + match self.colours { + None | Some(16) => Colours::Colours16, + Some(256) => Colours::Colours256, + _ => panic!("colours must either not be set or 16 or 256"), + } + } } #[derive(Deserialize)] diff --git a/agb-image-converter/src/lib.rs b/agb-image-converter/src/lib.rs index 2f41c577..ddbe8638 100644 --- a/agb-image-converter/src/lib.rs +++ b/agb-image-converter/src/lib.rs @@ -30,6 +30,11 @@ pub(crate) enum TileSize { Tile32, } +pub(crate) enum Colours { + Colours16, + Colours256, +} + impl TileSize { fn to_size(self) -> usize { match self { From e97061a5afc23eecbfc723e03e792e95f171c0ed Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 11 Aug 2022 22:22:44 +0100 Subject: [PATCH 08/49] Optimise all palettes together --- agb-image-converter/src/lib.rs | 76 ++++++++----- agb-image-converter/src/rust_generator.rs | 12 ++- examples/hyperspace-roll/gfx/backgrounds.toml | 25 +++++ .../hyperspace-roll/gfx/descriptions.toml | 11 -- examples/hyperspace-roll/gfx/help.toml | 6 -- examples/hyperspace-roll/gfx/stars.toml | 10 -- examples/hyperspace-roll/src/background.rs | 101 +++++++++++++++--- examples/hyperspace-roll/src/customise.rs | 55 ++-------- 8 files changed, 177 insertions(+), 119 deletions(-) create mode 100644 examples/hyperspace-roll/gfx/backgrounds.toml delete mode 100644 examples/hyperspace-roll/gfx/descriptions.toml delete mode 100644 examples/hyperspace-roll/gfx/help.toml delete mode 100644 examples/hyperspace-roll/gfx/stars.toml diff --git a/agb-image-converter/src/lib.rs b/agb-image-converter/src/lib.rs index ddbe8638..d482b81f 100644 --- a/agb-image-converter/src/lib.rs +++ b/agb-image-converter/src/lib.rs @@ -1,10 +1,11 @@ -use palette16::Palette16OptimisationResults; +use palette16::{Palette16OptimisationResults, Palette16Optimiser}; use proc_macro::TokenStream; use proc_macro2::Literal; use syn::parse::Parser; use syn::{parse_macro_input, punctuated::Punctuated, LitStr}; use syn::{Expr, ExprLit, Lit}; +use std::collections::HashMap; use std::path::PathBuf; use std::{iter, path::Path, str}; @@ -68,9 +69,46 @@ pub fn include_gfx(input: TokenStream) -> TokenStream { let include_path = path.to_string_lossy(); let images = config.images(); - let image_code = images.iter().map(|(image_name, &image)| { - convert_image(image, parent, image_name, &config.crate_prefix()) - }); + + let mut optimiser = Palette16Optimiser::new(None); + let mut assignment_offsets = HashMap::new(); + let mut assignment_offset = 0; + + for (name, settings) in images.iter() { + let image_filename = &parent.join(&settings.filename()); + let image = Image::load_from_file(image_filename); + + let tile_size = settings.tilesize().to_size(); + if image.width % tile_size != 0 || image.height % tile_size != 0 { + panic!("Image size not a multiple of tile size"); + } + + add_to_optimiser( + &mut optimiser, + &image, + tile_size, + settings.transparent_colour(), + ); + + let num_tiles = image.width * image.height / settings.tilesize().to_size().pow(2); + assignment_offsets.insert(name, assignment_offset); + assignment_offset += num_tiles; + } + + let optimisation_results = optimiser.optimise_palettes(); + + let mut image_code = vec![]; + + for (image_name, &image) in images.iter() { + image_code.push(convert_image( + image, + parent, + image_name, + &config.crate_prefix(), + &optimisation_results, + assignment_offsets[image_name], + )); + } let module = quote! { mod #module_name { @@ -217,38 +255,23 @@ fn convert_image( parent: &Path, variable_name: &str, crate_prefix: &str, + optimisation_results: &Palette16OptimisationResults, + assignment_offset: usize, ) -> proc_macro2::TokenStream { let image_filename = &parent.join(&settings.filename()); let image = Image::load_from_file(image_filename); - let tile_size = settings.tilesize().to_size(); - if image.width % tile_size != 0 || image.height % tile_size != 0 { - panic!("Image size not a multiple of tile size"); - } - - let optimiser = optimiser_for_image(&image, tile_size, settings.transparent_colour()); - let optimisation_results = optimiser.optimise_palettes(); - rust_generator::generate_code( variable_name, - &optimisation_results, + optimisation_results, &image, &image_filename.to_string_lossy(), settings.tilesize(), crate_prefix.to_owned(), + assignment_offset, ) } -fn optimiser_for_image( - image: &Image, - tile_size: usize, - transparent_colour: Option, -) -> palette16::Palette16Optimiser { - let mut palette_optimiser = palette16::Palette16Optimiser::new(transparent_colour); - add_to_optimiser(&mut palette_optimiser, image, tile_size, transparent_colour); - palette_optimiser -} - fn add_to_optimiser( palette_optimiser: &mut palette16::Palette16Optimiser, image: &Image, @@ -301,7 +324,7 @@ fn palette_tile_data( let tile_size = TileSize::Tile8; for image in images { - add_image_to_tile_data(&mut tile_data, image, tile_size, &optimiser) + add_image_to_tile_data(&mut tile_data, image, tile_size, &optimiser, 0) } let tile_data = collapse_to_4bpp(&tile_data); @@ -322,7 +345,8 @@ fn add_image_to_tile_data( tile_data: &mut Vec, image: &Image, tile_size: TileSize, - optimiser: &&Palette16OptimisationResults, + optimiser: &Palette16OptimisationResults, + assignment_offset: usize, ) { let tile_size = tile_size.to_size(); let tiles_x = image.width / tile_size; @@ -330,7 +354,7 @@ fn add_image_to_tile_data( for y in 0..tiles_y { for x in 0..tiles_x { - let palette_index = optimiser.assignments[y * tiles_x + x]; + let palette_index = optimiser.assignments[y * tiles_x + x + assignment_offset]; let palette = &optimiser.optimised_palettes[palette_index]; for inner_y in 0..tile_size / 8 { diff --git a/agb-image-converter/src/rust_generator.rs b/agb-image-converter/src/rust_generator.rs index 7f04b7eb..0a668036 100644 --- a/agb-image-converter/src/rust_generator.rs +++ b/agb-image-converter/src/rust_generator.rs @@ -14,6 +14,7 @@ pub(crate) fn generate_code( image_filename: &str, tile_size: TileSize, crate_prefix: String, + assignment_offset: usize, ) -> TokenStream { let crate_prefix = format_ident!("{}", crate_prefix); let output_variable_name = format_ident!("{}", output_variable_name); @@ -36,13 +37,20 @@ pub(crate) fn generate_code( let mut tile_data = Vec::new(); - add_image_to_tile_data(&mut tile_data, image, tile_size, &results); + add_image_to_tile_data(&mut tile_data, image, tile_size, results, assignment_offset); let tile_data = collapse_to_4bpp(&tile_data); let data = ByteString(&tile_data); - let assignments = results.assignments.iter().map(|&x| x as u8); + let num_tiles = image.width * image.height / tile_size.to_size().pow(2); + + let assignments = results + .assignments + .iter() + .skip(assignment_offset) + .take(num_tiles) + .map(|&x| x as u8); quote! { #[allow(non_upper_case_globals)] diff --git a/examples/hyperspace-roll/gfx/backgrounds.toml b/examples/hyperspace-roll/gfx/backgrounds.toml new file mode 100644 index 00000000..21ed1005 --- /dev/null +++ b/examples/hyperspace-roll/gfx/backgrounds.toml @@ -0,0 +1,25 @@ +version = "1.0" + +[image.stars] +filename = "stars.png" +tile_size = "8x8" +transparent_colour = "121105" + +[image.title] +filename = "title-screen.png" +tile_size = "8x8" + +[image.help] +filename = "help-text.png" +tile_size = "8x8" +transparent_colour = "121105" + +[image.descriptions1] +filename = "descriptions1.png" +tile_size = "8x8" +transparent_colour = "121105" + +[image.descriptions2] +filename = "descriptions2.png" +tile_size = "8x8" +transparent_colour = "121105" \ No newline at end of file diff --git a/examples/hyperspace-roll/gfx/descriptions.toml b/examples/hyperspace-roll/gfx/descriptions.toml deleted file mode 100644 index 9ce038e0..00000000 --- a/examples/hyperspace-roll/gfx/descriptions.toml +++ /dev/null @@ -1,11 +0,0 @@ -version = "1.0" - -[image.descriptions1] -filename = "descriptions1.png" -tile_size = "8x8" -transparent_colour = "121105" - -[image.descriptions2] -filename = "descriptions2.png" -tile_size = "8x8" -transparent_colour = "121105" \ No newline at end of file diff --git a/examples/hyperspace-roll/gfx/help.toml b/examples/hyperspace-roll/gfx/help.toml deleted file mode 100644 index 149b68d9..00000000 --- a/examples/hyperspace-roll/gfx/help.toml +++ /dev/null @@ -1,6 +0,0 @@ -version = "1.0" - -[image.help] -filename = "help-text.png" -tile_size = "8x8" -transparent_colour = "121105" diff --git a/examples/hyperspace-roll/gfx/stars.toml b/examples/hyperspace-roll/gfx/stars.toml deleted file mode 100644 index 354d1df4..00000000 --- a/examples/hyperspace-roll/gfx/stars.toml +++ /dev/null @@ -1,10 +0,0 @@ -version = "1.0" - -[image.stars] -filename = "stars.png" -tile_size = "8x8" -transparent_colour = "121105" - -[image.title] -filename = "title-screen.png" -tile_size = "8x8" \ No newline at end of file diff --git a/examples/hyperspace-roll/src/background.rs b/examples/hyperspace-roll/src/background.rs index c67cc709..037fa20a 100644 --- a/examples/hyperspace-roll/src/background.rs +++ b/examples/hyperspace-roll/src/background.rs @@ -5,17 +5,23 @@ use agb::{ use crate::sfx::Sfx; -include_gfx!("gfx/stars.toml"); - -include_gfx!("gfx/help.toml"); +include_gfx!("gfx/backgrounds.toml"); pub fn load_palettes(vram: &mut VRamManager) { - vram.set_background_palettes(&[ - stars::stars.palettes[0].clone(), - crate::customise::DESCRIPTIONS_1_PALETTE.clone(), - crate::customise::DESCRIPTIONS_2_PALETTE.clone(), - help::help.palettes[0].clone(), - ]); + vram.set_background_palettes(backgrounds::stars.palettes); +} + +fn description_tileset() -> (TileSet<'static>, TileSet<'static>) { + let descriptions_1_tileset = TileSet::new( + backgrounds::descriptions1.tiles, + agb::display::tiled::TileFormat::FourBpp, + ); + let descriptions_2_tileset = TileSet::new( + backgrounds::descriptions2.tiles, + agb::display::tiled::TileFormat::FourBpp, + ); + + (descriptions_1_tileset, descriptions_2_tileset) } pub(crate) fn load_help_text( @@ -24,18 +30,69 @@ pub(crate) fn load_help_text( help_text_line: u16, at_tile: (u16, u16), ) { - let help_tileset = TileSet::new(help::help.tiles, agb::display::tiled::TileFormat::FourBpp); + let help_tileset = TileSet::new( + backgrounds::help.tiles, + agb::display::tiled::TileFormat::FourBpp, + ); for x in 0..16 { + let tile_id = help_text_line * 16 + x; background.set_tile( vram, (x + at_tile.0, at_tile.1).into(), &help_tileset, - TileSetting::new(help_text_line * 16 + x, false, false, 3), + TileSetting::new( + tile_id, + false, + false, + backgrounds::help.palette_assignments[tile_id as usize], + ), ) } } +pub(crate) fn load_description( + upgrade: usize, + descriptions_map: &mut RegularMap, + vram: &mut VRamManager, +) { + let (descriptions_1_tileset, descriptions_2_tileset) = description_tileset(); + + for y in 0..11 { + for x in 0..8 { + if upgrade < 10 { + let tile_id = y * 8 + x + 8 * 11 * upgrade as u16; + + descriptions_map.set_tile( + vram, + (x, y).into(), + &descriptions_1_tileset, + TileSetting::new( + tile_id, + false, + false, + backgrounds::descriptions1.palette_assignments[tile_id as usize], + ), + ) + } else { + let tile_id = y * 8 + x + 8 * 11 * (upgrade as u16 - 10); + + descriptions_map.set_tile( + vram, + (x, y).into(), + &descriptions_2_tileset, + TileSetting::new( + tile_id, + false, + false, + backgrounds::descriptions2.palette_assignments[tile_id as usize], + ), + ) + } + } + } +} + // Expects a 64x32 map fn create_background_map(map: &mut RegularMap, vram: &mut VRamManager, stars_tileset: &TileSet) { for x in 0..64u16 { @@ -47,7 +104,16 @@ fn create_background_map(map: &mut RegularMap, vram: &mut VRamManager, stars_til } else { rng::gen().rem_euclid(64) as u16 }; - let tile_setting = TileSetting::new(tile_id, false, false, 0); + let tile_setting = TileSetting::new( + tile_id, + false, + false, + if blank { + 0 + } else { + backgrounds::stars.palette_assignments[tile_id as usize] + }, + ); map.set_tile(vram, (x, y).into(), stars_tileset, tile_setting); } @@ -58,8 +124,11 @@ fn create_background_map(map: &mut RegularMap, vram: &mut VRamManager, stars_til pub fn show_title_screen(background: &mut RegularMap, vram: &mut VRamManager, sfx: &mut Sfx) { background.set_scroll_pos((0_u16, 0_u16).into()); - vram.set_background_palettes(stars::title.palettes); - let tile_set = TileSet::new(stars::title.tiles, agb::display::tiled::TileFormat::FourBpp); + vram.set_background_palettes(backgrounds::title.palettes); + let tile_set = TileSet::new( + backgrounds::title.tiles, + agb::display::tiled::TileFormat::FourBpp, + ); background.hide(); for x in 0..30u16 { @@ -73,7 +142,7 @@ pub fn show_title_screen(background: &mut RegularMap, vram: &mut VRamManager, sf tile_id, false, false, - stars::title.palette_assignments[tile_id as usize], + backgrounds::title.palette_assignments[tile_id as usize], ), ); } @@ -100,7 +169,7 @@ impl<'a> StarBackground<'a> { background2: &'a mut RegularMap, vram: &'_ mut VRamManager, ) -> Self { - let stars_tileset = TileSet::new(stars::stars.tiles, TileFormat::FourBpp); + let stars_tileset = TileSet::new(backgrounds::stars.tiles, TileFormat::FourBpp); create_background_map(background1, vram, &stars_tileset); create_background_map(background2, vram, &stars_tileset); diff --git a/examples/hyperspace-roll/src/customise.rs b/examples/hyperspace-roll/src/customise.rs index c91f3cd9..a6d9e5b6 100644 --- a/examples/hyperspace-roll/src/customise.rs +++ b/examples/hyperspace-roll/src/customise.rs @@ -1,26 +1,20 @@ use agb::{ display::{ object::{Object, ObjectController}, - palette16::Palette16, - tiled::{RegularMap, TileSet, TileSetting, TiledMap}, + tiled::{RegularMap, TiledMap}, HEIGHT, WIDTH, }, - include_gfx, input::{Button, Tri}, }; use alloc::vec::Vec; use crate::{ + background::load_description, graphics::{FACE_SPRITES, MODIFIED_BOX, SELECTED_BOX, SELECT_BOX}, Agb, Die, Face, PlayerDice, }; -include_gfx!("gfx/descriptions.toml"); - -pub const DESCRIPTIONS_1_PALETTE: &Palette16 = &descriptions::descriptions1.palettes[0]; -pub const DESCRIPTIONS_2_PALETTE: &Palette16 = &descriptions::descriptions2.palettes[0]; - enum CustomiseState { Dice, Face, @@ -174,15 +168,6 @@ pub(crate) fn customise_screen( help_background.set_scroll_pos((u16::MAX - 148, u16::MAX - 34).into()); crate::background::load_help_text(&mut agb.vram, help_background, 0, (0, 0)); - let descriptions_1_tileset = TileSet::new( - descriptions::descriptions1.tiles, - agb::display::tiled::TileFormat::FourBpp, - ); - let descriptions_2_tileset = TileSet::new( - descriptions::descriptions2.tiles, - agb::display::tiled::TileFormat::FourBpp, - ); - // create the dice let mut _net = create_net(&agb.obj, &player_dice.dice[0], &[]); @@ -301,37 +286,11 @@ pub(crate) fn customise_screen( if (upgrades[cursor.upgrade] as u32) < 17 { if cursor.upgrade != old_updade { - for y in 0..11 { - for x in 0..8 { - if (upgrades[cursor.upgrade] as usize) < 10 { - descriptions_map.set_tile( - &mut agb.vram, - (x, y).into(), - &descriptions_1_tileset, - TileSetting::new( - y * 8 + x + 8 * 11 * upgrades[cursor.upgrade] as u16, - false, - false, - 1, - ), - ) - } else { - descriptions_map.set_tile( - &mut agb.vram, - (x, y).into(), - &descriptions_2_tileset, - TileSetting::new( - y * 8 - + x - + 8 * 11 * (upgrades[cursor.upgrade] as u16 - 10), - false, - false, - 2, - ), - ) - } - } - } + load_description( + upgrades[cursor.upgrade] as usize, + descriptions_map, + &mut agb.vram, + ); } descriptions_map.show(); } else { From 702286979d21685e059780184e1f6cb4982fedd2 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 11 Aug 2022 22:23:06 +0100 Subject: [PATCH 09/49] Fix warning --- agb-image-converter/src/lib.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/agb-image-converter/src/lib.rs b/agb-image-converter/src/lib.rs index d482b81f..ee0c5b57 100644 --- a/agb-image-converter/src/lib.rs +++ b/agb-image-converter/src/lib.rs @@ -324,7 +324,7 @@ fn palette_tile_data( let tile_size = TileSize::Tile8; for image in images { - add_image_to_tile_data(&mut tile_data, image, tile_size, &optimiser, 0) + add_image_to_tile_data(&mut tile_data, image, tile_size, optimiser, 0) } let tile_data = collapse_to_4bpp(&tile_data); From 916a3d4b20d62b0749065b5c46b7dbc01f5d246e Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 11 Aug 2022 22:47:52 +0100 Subject: [PATCH 10/49] Start tracking the colours for a 256 colour image --- agb-image-converter/src/config.rs | 20 ++++---- agb-image-converter/src/lib.rs | 40 +++++++++------ agb-image-converter/src/palette16.rs | 17 +++++++ agb-image-converter/src/palette256.rs | 72 +++++++++++++++++++++++++++ 4 files changed, 125 insertions(+), 24 deletions(-) create mode 100644 agb-image-converter/src/palette256.rs diff --git a/agb-image-converter/src/config.rs b/agb-image-converter/src/config.rs index bccb252d..eaa2cd20 100644 --- a/agb-image-converter/src/config.rs +++ b/agb-image-converter/src/config.rs @@ -23,20 +23,19 @@ pub(crate) fn parse(filename: &str) -> Box { pub(crate) trait Config { fn crate_prefix(&self) -> String; fn images(&self) -> HashMap; - fn colours(&self) -> Colours; } pub(crate) trait Image { fn filename(&self) -> String; fn transparent_colour(&self) -> Option; fn tilesize(&self) -> TileSize; + fn colours(&self) -> Colours; } #[derive(Deserialize)] pub struct ConfigV1 { version: String, crate_prefix: Option, - colours: Option, image: HashMap, } @@ -54,14 +53,6 @@ impl Config for ConfigV1 { .map(|(filename, image)| (filename.clone(), image as &dyn Image)) .collect() } - - fn colours(&self) -> Colours { - match self.colours { - None | Some(16) => Colours::Colours16, - Some(256) => Colours::Colours256, - _ => panic!("colours must either not be set or 16 or 256"), - } - } } #[derive(Deserialize)] @@ -69,6 +60,7 @@ pub struct ImageV1 { filename: String, transparent_colour: Option, tile_size: TileSizeV1, + colours: Option, } impl Image for ImageV1 { @@ -95,6 +87,14 @@ impl Image for ImageV1 { fn tilesize(&self) -> TileSize { self.tile_size.into() } + + fn colours(&self) -> Colours { + match self.colours { + None | Some(16) => Colours::Colours16, + Some(256) => Colours::Colours256, + _ => panic!("colours must either not be set or 16 or 256"), + } + } } #[derive(Deserialize, Clone, Copy)] diff --git a/agb-image-converter/src/lib.rs b/agb-image-converter/src/lib.rs index ee0c5b57..235de782 100644 --- a/agb-image-converter/src/lib.rs +++ b/agb-image-converter/src/lib.rs @@ -1,4 +1,5 @@ use palette16::{Palette16OptimisationResults, Palette16Optimiser}; +use palette256::Palette256; use proc_macro::TokenStream; use proc_macro2::Literal; use syn::parse::Parser; @@ -17,6 +18,7 @@ mod config; mod font_loader; mod image_loader; mod palette16; +mod palette256; mod rust_generator; use image::GenericImageView; @@ -74,28 +76,38 @@ pub fn include_gfx(input: TokenStream) -> TokenStream { let mut assignment_offsets = HashMap::new(); let mut assignment_offset = 0; + let mut palette256 = Palette256::new(); + for (name, settings) in images.iter() { let image_filename = &parent.join(&settings.filename()); let image = Image::load_from_file(image_filename); - let tile_size = settings.tilesize().to_size(); - if image.width % tile_size != 0 || image.height % tile_size != 0 { - panic!("Image size not a multiple of tile size"); + match settings.colours() { + Colours::Colours16 => { + let tile_size = settings.tilesize().to_size(); + if image.width % tile_size != 0 || image.height % tile_size != 0 { + panic!("Image size not a multiple of tile size"); + } + + add_to_optimiser( + &mut optimiser, + &image, + tile_size, + settings.transparent_colour(), + ); + + let num_tiles = image.width * image.height / settings.tilesize().to_size().pow(2); + assignment_offsets.insert(name, assignment_offset); + assignment_offset += num_tiles; + } + Colours::Colours256 => { + palette256.add_image(&image); + } } - - add_to_optimiser( - &mut optimiser, - &image, - tile_size, - settings.transparent_colour(), - ); - - let num_tiles = image.width * image.height / settings.tilesize().to_size().pow(2); - assignment_offsets.insert(name, assignment_offset); - assignment_offset += num_tiles; } let optimisation_results = optimiser.optimise_palettes(); + let optimisation_results = palette256.extend_results(&optimisation_results); let mut image_code = vec![]; diff --git a/agb-image-converter/src/palette16.rs b/agb-image-converter/src/palette16.rs index 01d0af0f..7cf1def4 100644 --- a/agb-image-converter/src/palette16.rs +++ b/agb-image-converter/src/palette16.rs @@ -28,6 +28,19 @@ impl Palette16 { true } + pub fn try_add_colour(&mut self, colour: Colour) -> bool { + if self.colours.contains(&colour) { + return true; + } + + if self.colours.len() == MAX_COLOURS_PER_PALETTE { + return false; + } + + self.colours.push(colour); + true + } + pub fn colour_index(&self, colour: Colour, transparent_colour: Option) -> u8 { let colour_to_search = match (transparent_colour, colour.is_transparent()) { (Some(transparent_colour), true) => transparent_colour, @@ -45,6 +58,10 @@ impl Palette16 { }) as u8 } + pub fn colours<'a>(&'a self) -> impl Iterator + 'a { + self.colours.iter() + } + fn union_length(&self, other: &Palette16) -> usize { self.colours .iter() diff --git a/agb-image-converter/src/palette256.rs b/agb-image-converter/src/palette256.rs new file mode 100644 index 00000000..fc9b47ff --- /dev/null +++ b/agb-image-converter/src/palette256.rs @@ -0,0 +1,72 @@ +use std::{collections::HashSet, iter::FromIterator}; + +use crate::{ + colour::Colour, + image_loader::Image, + palette16::{Palette16, Palette16OptimisationResults}, +}; + +pub struct Palette256 { + colours: HashSet, +} + +impl Palette256 { + pub fn new() -> Self { + Self { + colours: HashSet::new(), + } + } + + pub(crate) fn add_image(&mut self, image: &Image) { + for y in 0..image.height { + for x in 0..image.width { + self.colours.insert(image.colour(x, y)); + } + } + + assert!( + self.colours.len() <= 256, + "Must have at most 256 colours in the palette" + ); + } + + pub(crate) fn extend_results( + &self, + palette16: &Palette16OptimisationResults, + ) -> Palette16OptimisationResults { + let optimised_palette_colours: Vec<_> = palette16 + .optimised_palettes + .iter() + .flat_map(|p| p.colours()) + .cloned() + .collect(); + + let current_colours_set = HashSet::from_iter(optimised_palette_colours.iter().cloned()); + let new_colours: HashSet<_> = self + .colours + .symmetric_difference(¤t_colours_set) + .collect(); + + assert!( + new_colours.len() + optimised_palette_colours.len() <= 256, + "Cannot optimise 16 colour and 256 colour palettes together, produces too many colours" + ); + + let mut new_palettes = palette16.optimised_palettes.clone(); + new_palettes.resize_with(16, Palette16::new); + + for colour in new_colours { + for palette in new_palettes.iter_mut() { + if palette.try_add_colour(*colour) { + break; + } + } + } + + Palette16OptimisationResults { + optimised_palettes: new_palettes, + assignments: palette16.assignments.clone(), + transparent_colour: palette16.transparent_colour, + } + } +} From ff89f7f425d0c745f34b7f28bc6dbce551707801 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 11 Aug 2022 23:48:57 +0100 Subject: [PATCH 11/49] Get first example working --- agb-image-converter/src/lib.rs | 41 +++++++++++++- agb-image-converter/src/rust_generator.rs | 39 ++++++++----- agb/examples/affine_background.rs | 69 +++++++++++++++++++++++ agb/examples/affine_tiles.toml | 6 ++ agb/src/display/tiled/map.rs | 10 ++-- agb/src/syscall.rs | 6 +- 6 files changed, 147 insertions(+), 24 deletions(-) create mode 100644 agb/examples/affine_background.rs create mode 100644 agb/examples/affine_tiles.toml diff --git a/agb-image-converter/src/lib.rs b/agb-image-converter/src/lib.rs index 235de782..8d6e7183 100644 --- a/agb-image-converter/src/lib.rs +++ b/agb-image-converter/src/lib.rs @@ -112,13 +112,18 @@ pub fn include_gfx(input: TokenStream) -> TokenStream { let mut image_code = vec![]; for (image_name, &image) in images.iter() { + let assignment_offset = match image.colours() { + Colours::Colours16 => Some(assignment_offsets[image_name]), + _ => None, + }; + image_code.push(convert_image( image, parent, image_name, &config.crate_prefix(), &optimisation_results, - assignment_offsets[image_name], + assignment_offset, )); } @@ -268,7 +273,7 @@ fn convert_image( variable_name: &str, crate_prefix: &str, optimisation_results: &Palette16OptimisationResults, - assignment_offset: usize, + assignment_offset: Option, ) -> proc_macro2::TokenStream { let image_filename = &parent.join(&settings.filename()); let image = Image::load_from_file(image_filename); @@ -384,6 +389,38 @@ fn add_image_to_tile_data( } } +fn add_image_256_to_tile_data( + tile_data: &mut Vec, + image: &Image, + tile_size: TileSize, + optimiser: &Palette16OptimisationResults, +) { + let tile_size = tile_size.to_size(); + let tiles_x = image.width / tile_size; + let tiles_y = image.height / tile_size; + + let all_colours: Vec<_> = optimiser + .optimised_palettes + .iter() + .flat_map(|p| p.colours()) + .collect(); + + for y in 0..tiles_y { + for x in 0..tiles_x { + for inner_y in 0..tile_size / 8 { + for inner_x in 0..tile_size / 8 { + for j in inner_y * 8..inner_y * 8 + 8 { + for i in inner_x * 8..inner_x * 8 + 8 { + let colour = image.colour(x * tile_size + i, y * tile_size + j); + tile_data.push(all_colours.iter().position(|c| **c == colour).unwrap() as u8); + } + } + } + } + } + } +} + fn flatten_group(expr: &Expr) -> &Expr { match expr { Expr::Group(group) => &group.expr, diff --git a/agb-image-converter/src/rust_generator.rs b/agb-image-converter/src/rust_generator.rs index 0a668036..3e70466c 100644 --- a/agb-image-converter/src/rust_generator.rs +++ b/agb-image-converter/src/rust_generator.rs @@ -1,5 +1,5 @@ use crate::palette16::Palette16OptimisationResults; -use crate::{add_image_to_tile_data, collapse_to_4bpp, TileSize}; +use crate::{add_image_256_to_tile_data, add_image_to_tile_data, collapse_to_4bpp, TileSize}; use crate::{image_loader::Image, ByteString}; use proc_macro2::TokenStream; @@ -14,7 +14,7 @@ pub(crate) fn generate_code( image_filename: &str, tile_size: TileSize, crate_prefix: String, - assignment_offset: usize, + assignment_offset: Option, ) -> TokenStream { let crate_prefix = format_ident!("{}", crate_prefix); let output_variable_name = format_ident!("{}", output_variable_name); @@ -35,23 +35,34 @@ pub(crate) fn generate_code( } }); - let mut tile_data = Vec::new(); + let (tile_data, assignments) = if let Some(assignment_offset) = assignment_offset { + let mut tile_data = Vec::new(); - add_image_to_tile_data(&mut tile_data, image, tile_size, results, assignment_offset); + add_image_to_tile_data(&mut tile_data, image, tile_size, results, assignment_offset); - let tile_data = collapse_to_4bpp(&tile_data); + let tile_data = collapse_to_4bpp(&tile_data); + + let num_tiles = image.width * image.height / tile_size.to_size().pow(2); + + let assignments = results + .assignments + .iter() + .skip(assignment_offset) + .take(num_tiles) + .map(|&x| x as u8) + .collect(); + + (tile_data, assignments) + } else { + let mut tile_data = Vec::new(); + + add_image_256_to_tile_data(&mut tile_data, image, tile_size, results); + + (tile_data, vec![]) + }; let data = ByteString(&tile_data); - let num_tiles = image.width * image.height / tile_size.to_size().pow(2); - - let assignments = results - .assignments - .iter() - .skip(assignment_offset) - .take(num_tiles) - .map(|&x| x as u8); - quote! { #[allow(non_upper_case_globals)] pub const #output_variable_name: #crate_prefix::display::tile_data::TileData = { diff --git a/agb/examples/affine_background.rs b/agb/examples/affine_background.rs new file mode 100644 index 00000000..f5519727 --- /dev/null +++ b/agb/examples/affine_background.rs @@ -0,0 +1,69 @@ +#![no_std] +#![no_main] + +use agb::{ + display::{ + tiled::{AffineBackgroundSize, TileFormat, TileSet, TiledMap}, + Priority, + }, + fixnum::{num, Num}, + include_gfx, + input::Tri, +}; + +include_gfx!("examples/affine_tiles.toml"); + +#[agb::entry] +fn main(mut gba: agb::Gba) -> ! { + let (gfx, mut vram) = gba.display.video.tiled2(); + let vblank = agb::interrupt::VBlank::get(); + + let tileset = TileSet::new(affine_tiles::water_tiles.tiles, TileFormat::EightBpp); + + vram.set_background_palettes(affine_tiles::water_tiles.palettes); + + let mut bg = gfx.background(Priority::P0, AffineBackgroundSize::Background32x32); + + for y in 0..32u16 { + for x in 0..32u16 { + bg.set_tile(&mut vram, (x, y).into(), &tileset, 1); + } + } + + bg.commit(&mut vram); + bg.show(); + + let mut rotation = num!(0.); + let rotation_increase = num!(1.); + + let mut input = agb::input::ButtonController::new(); + + loop { + input.update(); + + let x_dir = match input.x_tri() { + Tri::Positive => (1, 0).into(), + Tri::Negative => (-1, 0).into(), + _ => (0, 0).into(), + }; + + let y_dir = match input.y_tri() { + Tri::Positive => (0, 1).into(), + Tri::Negative => (0, -1).into(), + _ => (0, 0).into(), + }; + + let new_scroll_pos = bg.scroll_pos() + x_dir + y_dir; + agb::println!("{:?}", new_scroll_pos); + bg.set_scroll_pos(new_scroll_pos); + bg.set_transform((0i16, 0i16).into(), (1, 1).into(), rotation); + + rotation += rotation_increase; + if rotation >= num!(255.) { + rotation = 0.into(); + } + + vblank.wait_for_vblank(); + bg.commit(&mut vram); + } +} diff --git a/agb/examples/affine_tiles.toml b/agb/examples/affine_tiles.toml new file mode 100644 index 00000000..65d608a2 --- /dev/null +++ b/agb/examples/affine_tiles.toml @@ -0,0 +1,6 @@ +version = "1.0" + +[image.water_tiles] +filename = "water_tiles.png" +tile_size = "8x8" +colours = 256 \ No newline at end of file diff --git a/agb/src/display/tiled/map.rs b/agb/src/display/tiled/map.rs index 1933663a..7af1f57b 100644 --- a/agb/src/display/tiled/map.rs +++ b/agb/src/display/tiled/map.rs @@ -278,16 +278,16 @@ impl TiledMapPrivate for AffineMap { &mut self.tiles_dirty } fn x_scroll_mut(&mut self) -> &mut Self::Position { - &mut self.transform.position.x + &mut self.bg_center.x } fn y_scroll_mut(&mut self) -> &mut Self::Position { - &mut self.transform.position.y + &mut self.bg_center.y } fn x_scroll(&self) -> Self::Position { - self.transform.position.x + self.bg_center.x } fn y_scroll(&self) -> Self::Position { - self.transform.position.y + self.bg_center.y } fn affine_matrix(&self) -> Self::AffineMatrix { self.transform.matrix @@ -377,7 +377,7 @@ impl AffineMap { &mut self, display_center: Vector2D, scale: Vector2D>, - rotation: Num, + rotation: Num, ) { self.set_transform_raw(crate::syscall::bg_affine_matrix( self.bg_center, diff --git a/agb/src/syscall.rs b/agb/src/syscall.rs index 59d37a52..a91b8b79 100644 --- a/agb/src/syscall.rs +++ b/agb/src/syscall.rs @@ -159,21 +159,21 @@ pub fn bg_affine_matrix( bg_center: Vector2D>, display_center: Vector2D, scale: Vector2D>, - rotation: Num, + rotation: Num, ) -> BgAffineSetData { #[repr(C, packed)] struct Input { bg_center: Vector2D>, display_center: Vector2D, scale: Vector2D>, - rotation: u16, + rotation: Num, } let input = Input { bg_center, display_center, scale, - rotation: u16::from(rotation.to_raw()) << 8, + rotation, }; let mut output = MaybeUninit::uninit(); From 97e3948f3265b002524186874c9e186189df2f35 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 11 Aug 2022 23:49:27 +0100 Subject: [PATCH 12/49] Fix warnings and remove debug logs --- agb/examples/affine_background.rs | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/agb/examples/affine_background.rs b/agb/examples/affine_background.rs index f5519727..9f95e400 100644 --- a/agb/examples/affine_background.rs +++ b/agb/examples/affine_background.rs @@ -6,7 +6,7 @@ use agb::{ tiled::{AffineBackgroundSize, TileFormat, TileSet, TiledMap}, Priority, }, - fixnum::{num, Num}, + fixnum::num, include_gfx, input::Tri, }; @@ -54,7 +54,6 @@ fn main(mut gba: agb::Gba) -> ! { }; let new_scroll_pos = bg.scroll_pos() + x_dir + y_dir; - agb::println!("{:?}", new_scroll_pos); bg.set_scroll_pos(new_scroll_pos); bg.set_transform((0i16, 0i16).into(), (1, 1).into(), rotation); From fcde651821ddd4e46a833fe440b67abc4a4c50c0 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Fri, 12 Aug 2022 00:02:10 +0100 Subject: [PATCH 13/49] Allow movement of the affine background --- agb/examples/affine_background.rs | 31 ++++++++++++++++++------------- 1 file changed, 18 insertions(+), 13 deletions(-) diff --git a/agb/examples/affine_background.rs b/agb/examples/affine_background.rs index 9f95e400..5ac2e289 100644 --- a/agb/examples/affine_background.rs +++ b/agb/examples/affine_background.rs @@ -38,24 +38,29 @@ fn main(mut gba: agb::Gba) -> ! { let mut input = agb::input::ButtonController::new(); + let mut scroll_x = 0; + let mut scroll_y = 0; + loop { input.update(); - let x_dir = match input.x_tri() { - Tri::Positive => (1, 0).into(), - Tri::Negative => (-1, 0).into(), - _ => (0, 0).into(), - }; + match input.x_tri() { + Tri::Positive => scroll_x += 1, + Tri::Negative => scroll_x -= 1, + _ => {} + } - let y_dir = match input.y_tri() { - Tri::Positive => (0, 1).into(), - Tri::Negative => (0, -1).into(), - _ => (0, 0).into(), - }; + match input.y_tri() { + Tri::Positive => scroll_y += 1, + Tri::Negative => scroll_y -= 1, + _ => {} + } - let new_scroll_pos = bg.scroll_pos() + x_dir + y_dir; - bg.set_scroll_pos(new_scroll_pos); - bg.set_transform((0i16, 0i16).into(), (1, 1).into(), rotation); + bg.set_transform( + (-scroll_x as i16, -scroll_y as i16).into(), + (1, 1).into(), + rotation, + ); rotation += rotation_increase; if rotation >= num!(255.) { From d26311e76be646e5cb9734bfb6fa8a9d4d41053e Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 22 Sep 2022 20:26:45 +0100 Subject: [PATCH 14/49] The transparent colour should always go first --- agb-image-converter/src/palette16.rs | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/agb-image-converter/src/palette16.rs b/agb-image-converter/src/palette16.rs index 7cf1def4..afb50c68 100644 --- a/agb-image-converter/src/palette16.rs +++ b/agb-image-converter/src/palette16.rs @@ -167,9 +167,7 @@ impl Palette16Optimiser { fn find_maximal_palette_for(&self, unsatisfied_palettes: &HashSet) -> Palette16 { let mut palette = Palette16::new(); - if let Some(transparent_colour) = self.transparent_colour { - palette.add_colour(transparent_colour); - } + palette.add_colour(self.transparent_colour.unwrap_or_else(|| Colour::from_rgb(255, 0, 255, 0))); loop { let mut colour_usage = vec![0; MAX_COLOURS]; From 4eaf8af72f35825f96c9509d05a0f81b5f01564c Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 22 Sep 2022 21:32:36 +0100 Subject: [PATCH 15/49] Add Neg implementation for Vector2D --- agb-fixnum/src/lib.rs | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/agb-fixnum/src/lib.rs b/agb-fixnum/src/lib.rs index 5d725415..99aa5d3a 100644 --- a/agb-fixnum/src/lib.rs +++ b/agb-fixnum/src/lib.rs @@ -1016,6 +1016,14 @@ impl Vector2D { } } +impl > Neg for Vector2D { + type Output = Self; + + fn neg(self) -> Self::Output { + (-self.x, -self.y).into() + } +} + #[cfg(test)] mod tests { From 496b4a49531e97f9b3a15a124ce4878565300eda Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 22 Sep 2022 21:33:59 +0100 Subject: [PATCH 16/49] repr packed --- agb/src/display/object.rs | 2 +- agb/src/syscall.rs | 19 +++++++++++-------- 2 files changed, 12 insertions(+), 9 deletions(-) diff --git a/agb/src/display/object.rs b/agb/src/display/object.rs index 744be038..4389a0fa 100644 --- a/agb/src/display/object.rs +++ b/agb/src/display/object.rs @@ -1198,7 +1198,7 @@ enum ColourMode { /// This can be obtained from X/Y scale and rotation angle with /// [`agb::syscall::affine_matrix`]. #[derive(Copy, Clone, Default, Debug, PartialEq, Eq)] -#[repr(C)] +#[repr(C, packed)] pub struct AffineMatrixAttributes { /// Adjustment made to *X* coordinate when drawing *horizontal* lines. /// Also known as "dx". diff --git a/agb/src/syscall.rs b/agb/src/syscall.rs index a91b8b79..e7d342bd 100644 --- a/agb/src/syscall.rs +++ b/agb/src/syscall.rs @@ -139,7 +139,7 @@ pub fn arc_tan2(x: i16, y: i32) -> i16 { result } -#[repr(C)] +#[repr(C, packed)] pub struct BgAffineSetData { pub matrix: AffineMatrixAttributes, pub position: Vector2D>, @@ -180,11 +180,12 @@ pub fn bg_affine_matrix( unsafe { asm!( - "swi {SWI}", - SWI = const { swi_map(0x0E) }, - in("r0") &input as *const Input, - in("r1") output.as_mut_ptr(), - in("r2") 1, + "swi {SWI}", + SWI = const { swi_map(0x0E) }, + in("r0") &input as *const Input, + in("r1") output.as_mut_ptr(), + in("r2") 1, + lateout("r3") _, ); } @@ -235,7 +236,9 @@ mod tests { let one: Num = 1.into(); let aff = obj_affine_matrix((one, one).into(), Num::default()); - assert_eq!(aff.p_a, one); - assert_eq!(aff.p_d, one); + let (p_a, p_d) = (aff.p_a, aff.p_d); + + assert_eq!(p_a, one); + assert_eq!(p_d, one); } } From 59673206b3ff89eed576a32083f81239818d23d4 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 22 Sep 2022 21:34:03 +0100 Subject: [PATCH 17/49] Make the example work the way I expect it to --- agb/examples/affine_background.rs | 12 +++++++----- agb/src/display/tiled/map.rs | 16 ++++++++-------- 2 files changed, 15 insertions(+), 13 deletions(-) diff --git a/agb/examples/affine_background.rs b/agb/examples/affine_background.rs index 5ac2e289..2e336b13 100644 --- a/agb/examples/affine_background.rs +++ b/agb/examples/affine_background.rs @@ -6,7 +6,7 @@ use agb::{ tiled::{AffineBackgroundSize, TileFormat, TileSet, TiledMap}, Priority, }, - fixnum::num, + fixnum::{num, Num}, include_gfx, input::Tri, }; @@ -33,7 +33,7 @@ fn main(mut gba: agb::Gba) -> ! { bg.commit(&mut vram); bg.show(); - let mut rotation = num!(0.); + let mut rotation: Num = num!(0.); let rotation_increase = num!(1.); let mut input = agb::input::ButtonController::new(); @@ -56,10 +56,12 @@ fn main(mut gba: agb::Gba) -> ! { _ => {} } + bg.set_scroll_pos((16 * 4, 16 * 4).into()); + let scroll_pos = (scroll_x as i16, scroll_y as i16); bg.set_transform( - (-scroll_x as i16, -scroll_y as i16).into(), - (1, 1).into(), - rotation, + scroll_pos, + (1, 1), + rotation, ); rotation += rotation_increase; diff --git a/agb/src/display/tiled/map.rs b/agb/src/display/tiled/map.rs index 7af1f57b..45b09646 100644 --- a/agb/src/display/tiled/map.rs +++ b/agb/src/display/tiled/map.rs @@ -284,10 +284,10 @@ impl TiledMapPrivate for AffineMap { &mut self.bg_center.y } fn x_scroll(&self) -> Self::Position { - self.bg_center.x + self.bg_center.x + self.transform.position.x } fn y_scroll(&self) -> Self::Position { - self.bg_center.y + self.bg_center.y + self.transform.position.y } fn affine_matrix(&self) -> Self::AffineMatrix { self.transform.matrix @@ -375,15 +375,15 @@ impl AffineMap { pub fn set_transform( &mut self, - display_center: Vector2D, - scale: Vector2D>, - rotation: Num, + display_center: impl Into>, + scale: impl Into>>, + rotation: impl Into>, ) { self.set_transform_raw(crate::syscall::bg_affine_matrix( self.bg_center, - display_center, - scale, - rotation, + display_center.into(), + scale.into(), + rotation.into(), )); } } From 83c28254a114946537666a066a568b0ede82a21c Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Thu, 22 Sep 2022 23:29:58 +0100 Subject: [PATCH 18/49] Update to be closer to the API I actually want but currently only works in release mode --- agb-fixnum/src/lib.rs | 4 +- agb/examples/affine_background.rs | 8 +- agb/src/display/object.rs | 13 +- .../display/tiled/infinite_scrolled_map.rs | 14 +- agb/src/display/tiled/map.rs | 158 +++++++++--------- agb/src/display/tiled/mod.rs | 13 +- agb/src/syscall.rs | 23 ++- examples/hyperspace-roll/src/background.rs | 103 ++---------- 8 files changed, 137 insertions(+), 199 deletions(-) diff --git a/agb-fixnum/src/lib.rs b/agb-fixnum/src/lib.rs index 99aa5d3a..cc5669a6 100644 --- a/agb-fixnum/src/lib.rs +++ b/agb-fixnum/src/lib.rs @@ -570,7 +570,7 @@ impl Debug for Num { } /// A vector of two points: (x, y) represened by integers or fixed point numbers -#[derive(Clone, Copy, PartialEq, Eq, Debug)] +#[derive(Clone, Copy, PartialEq, Eq, Debug, Default)] #[repr(C)] pub struct Vector2D { /// The x coordinate @@ -1016,7 +1016,7 @@ impl Vector2D { } } -impl > Neg for Vector2D { +impl> Neg for Vector2D { type Output = Self; fn neg(self) -> Self::Output { diff --git a/agb/examples/affine_background.rs b/agb/examples/affine_background.rs index 2e336b13..9c337f8a 100644 --- a/agb/examples/affine_background.rs +++ b/agb/examples/affine_background.rs @@ -56,13 +56,9 @@ fn main(mut gba: agb::Gba) -> ! { _ => {} } - bg.set_scroll_pos((16 * 4, 16 * 4).into()); let scroll_pos = (scroll_x as i16, scroll_y as i16); - bg.set_transform( - scroll_pos, - (1, 1), - rotation, - ); + bg.set_scroll_pos(scroll_pos.into()); + bg.set_transform((0, 0), (1, 1), 0); rotation += rotation_increase; if rotation >= num!(255.) { diff --git a/agb/src/display/object.rs b/agb/src/display/object.rs index 4389a0fa..4d76b9a5 100644 --- a/agb/src/display/object.rs +++ b/agb/src/display/object.rs @@ -1197,7 +1197,7 @@ enum ColourMode { /// that can apply to objects and background layers in modes 1 and 2. /// This can be obtained from X/Y scale and rotation angle with /// [`agb::syscall::affine_matrix`]. -#[derive(Copy, Clone, Default, Debug, PartialEq, Eq)] +#[derive(Copy, Clone, Debug, PartialEq, Eq)] #[repr(C, packed)] pub struct AffineMatrixAttributes { /// Adjustment made to *X* coordinate when drawing *horizontal* lines. @@ -1218,6 +1218,17 @@ pub struct AffineMatrixAttributes { pub p_d: Num, } +impl Default for AffineMatrixAttributes { + fn default() -> Self { + Self { + p_a: 1.into(), + p_b: Default::default(), + p_c: Default::default(), + p_d: 1.into(), + } + } +} + // this mod is not public, so the internal parts don't need documenting. #[allow(dead_code)] mod attributes { diff --git a/agb/src/display/tiled/infinite_scrolled_map.rs b/agb/src/display/tiled/infinite_scrolled_map.rs index 8497802c..773ff782 100644 --- a/agb/src/display/tiled/infinite_scrolled_map.rs +++ b/agb/src/display/tiled/infinite_scrolled_map.rs @@ -257,13 +257,9 @@ impl<'a> InfiniteScrolledMap<'a> { let y_end = div_ceil(self.current_pos.y + display::HEIGHT, 8) + 1; let offset = self.current_pos - (x_start * 8, y_start * 8).into(); - let offset_scroll = ( - self.map.size().tile_pos_x(offset.x), - self.map.size().tile_pos_y(offset.y), - ) - .into(); - self.map.set_scroll_pos(offset_scroll); + self.map + .set_scroll_pos((offset.x as i16, offset.y as i16).into()); self.offset = (x_start, y_start).into(); let copy_from = self.copied_up_to; @@ -386,11 +382,7 @@ impl<'a> InfiniteScrolledMap<'a> { } let current_scroll = self.map.scroll_pos(); - let new_scroll = ( - size.px_offset_x(i32::from(current_scroll.x) + difference.x), - size.px_offset_y(i32::from(current_scroll.y) + difference.y), - ) - .into(); + let new_scroll = current_scroll + (difference.x as i16, difference.y as i16).into(); self.map.set_scroll_pos(new_scroll); diff --git a/agb/src/display/tiled/map.rs b/agb/src/display/tiled/map.rs index 45b09646..f5b60d89 100644 --- a/agb/src/display/tiled/map.rs +++ b/agb/src/display/tiled/map.rs @@ -4,7 +4,7 @@ use core::ops::{Deref, DerefMut}; use crate::bitarray::Bitarray; use crate::display::{object::AffineMatrixAttributes, Priority, DISPLAY_CONTROL}; use crate::dma::dma_copy16; -use crate::fixnum::{Num, Number, Vector2D}; +use crate::fixnum::{Num, Vector2D}; use crate::memory_mapped::MemoryMapped; use super::{ @@ -16,7 +16,6 @@ use crate::syscall::BgAffineSetData; use alloc::{vec, vec::Vec}; pub trait TiledMapTypes { - type Position: Number; type Size: BackgroundSize + Copy; } @@ -26,19 +25,17 @@ trait TiledMapPrivate: TiledMapTypes { fn tiles_mut(&mut self) -> &mut [Self::TileType]; fn tiles_dirty(&mut self) -> &mut bool; - fn x_scroll_mut(&mut self) -> &mut Self::Position; - fn y_scroll_mut(&mut self) -> &mut Self::Position; - fn x_scroll(&self) -> Self::Position; - fn y_scroll(&self) -> Self::Position; - fn affine_matrix(&self) -> Self::AffineMatrix; fn background_id(&self) -> usize; fn screenblock(&self) -> usize; fn priority(&self) -> Priority; fn map_size(&self) -> Self::Size; - fn bg_x(&self) -> MemoryMapped; - fn bg_y(&self) -> MemoryMapped; - fn bg_affine_matrix(&self) -> MemoryMapped; + + fn update_bg_registers(&self); + + fn scroll_pos(&self) -> Vector2D; + fn set_scroll_pos(&mut self, new_pos: Vector2D); + fn bg_control_register(&self) -> MemoryMapped { unsafe { MemoryMapped::new(0x0400_0008 + 2 * self.background_id()) } } @@ -55,8 +52,8 @@ pub trait TiledMap: TiledMapTypes { fn size(&self) -> Self::Size; #[must_use] - fn scroll_pos(&self) -> Vector2D; - fn set_scroll_pos(&mut self, pos: Vector2D); + fn scroll_pos(&self) -> Vector2D; + fn set_scroll_pos(&mut self, pos: Vector2D); } impl TiledMap for T @@ -92,9 +89,7 @@ where | (self.map_size().size_flag() << 14); self.bg_control_register().set(new_bg_control_value); - self.bg_x().set(self.x_scroll()); - self.bg_y().set(self.y_scroll()); - self.bg_affine_matrix().set(self.affine_matrix()); + self.update_bg_registers(); let screenblock_memory = self.screenblock_memory(); let x: TileIndex = unsafe { *self.tiles_mut().get_unchecked(0) }.into(); @@ -119,13 +114,12 @@ where } #[must_use] - fn scroll_pos(&self) -> Vector2D { - (self.x_scroll(), self.y_scroll()).into() + fn scroll_pos(&self) -> Vector2D { + TiledMapPrivate::scroll_pos(self) } - fn set_scroll_pos(&mut self, pos: Vector2D) { - *self.x_scroll_mut() = pos.x; - *self.y_scroll_mut() = pos.y; + fn set_scroll_pos(&mut self, pos: Vector2D) { + TiledMapPrivate::set_scroll_pos(self, pos); } } @@ -135,8 +129,7 @@ pub struct RegularMap { priority: Priority, size: RegularBackgroundSize, - x_scroll: u16, - y_scroll: u16, + scroll: Vector2D, tiles: Vec, tiles_dirty: bool, @@ -145,7 +138,6 @@ pub struct RegularMap { pub const TRANSPARENT_TILE_INDEX: u16 = (1 << 10) - 1; impl TiledMapTypes for RegularMap { - type Position = u16; type Size = RegularBackgroundSize; } @@ -159,19 +151,7 @@ impl TiledMapPrivate for RegularMap { fn tiles_dirty(&mut self) -> &mut bool { &mut self.tiles_dirty } - fn x_scroll_mut(&mut self) -> &mut Self::Position { - &mut self.x_scroll - } - fn y_scroll_mut(&mut self) -> &mut Self::Position { - &mut self.y_scroll - } - fn x_scroll(&self) -> Self::Position { - self.x_scroll - } - fn y_scroll(&self) -> Self::Position { - self.y_scroll - } - fn affine_matrix(&self) -> Self::AffineMatrix {} + fn background_id(&self) -> usize { self.background_id as usize } @@ -184,14 +164,15 @@ impl TiledMapPrivate for RegularMap { fn map_size(&self) -> Self::Size { self.size } - fn bg_x(&self) -> MemoryMapped { - unsafe { MemoryMapped::new(0x0400_0010 + 4 * self.background_id as usize) } + fn update_bg_registers(&self) { + self.x_register().set(self.scroll.x); + self.y_register().set(self.scroll.y); } - fn bg_y(&self) -> MemoryMapped { - unsafe { MemoryMapped::new(0x0400_0012 + 4 * self.background_id as usize) } + fn scroll_pos(&self) -> Vector2D { + self.scroll } - fn bg_affine_matrix(&self) -> MemoryMapped { - unsafe { MemoryMapped::new(0) } + fn set_scroll_pos(&mut self, new_pos: Vector2D) { + self.scroll = new_pos; } } @@ -208,8 +189,7 @@ impl RegularMap { priority, size, - x_scroll: 0, - y_scroll: 0, + scroll: Default::default(), tiles: vec![Default::default(); size.num_tiles()], tiles_dirty: true, @@ -247,6 +227,14 @@ impl RegularMap { self.tiles_mut()[pos] = new_tile; *self.tiles_dirty() = true; } + + fn x_register(&self) -> MemoryMapped { + unsafe { MemoryMapped::new(0x0400_0010 + 4 * self.background_id as usize) } + } + + fn y_register(&self) -> MemoryMapped { + unsafe { MemoryMapped::new(0x0400_0012 + 4 * self.background_id as usize) } + } } pub struct AffineMap { @@ -255,7 +243,9 @@ pub struct AffineMap { priority: Priority, size: AffineBackgroundSize, - bg_center: Vector2D>, + scroll: Vector2D, + + transform_origin: Vector2D>, transform: BgAffineSetData, tiles: Vec, @@ -263,7 +253,6 @@ pub struct AffineMap { } impl TiledMapTypes for AffineMap { - type Position = Num; type Size = AffineBackgroundSize; } @@ -277,21 +266,6 @@ impl TiledMapPrivate for AffineMap { fn tiles_dirty(&mut self) -> &mut bool { &mut self.tiles_dirty } - fn x_scroll_mut(&mut self) -> &mut Self::Position { - &mut self.bg_center.x - } - fn y_scroll_mut(&mut self) -> &mut Self::Position { - &mut self.bg_center.y - } - fn x_scroll(&self) -> Self::Position { - self.bg_center.x + self.transform.position.x - } - fn y_scroll(&self) -> Self::Position { - self.bg_center.y + self.transform.position.y - } - fn affine_matrix(&self) -> Self::AffineMatrix { - self.transform.matrix - } fn background_id(&self) -> usize { self.background_id as usize } @@ -304,14 +278,23 @@ impl TiledMapPrivate for AffineMap { fn map_size(&self) -> Self::Size { self.size } - fn bg_x(&self) -> MemoryMapped { - unsafe { MemoryMapped::new(0x0400_0008 + 0x10 * self.background_id()) } + fn update_bg_registers(&self) { + let register_pos = self.transform.position + self.transform_origin; + self.bg_x().set(register_pos.x); + self.bg_y().set(register_pos.y); + self.bg_affine_matrix().set(self.transform.matrix); + + crate::println!( + "update: {:?} {:?}", + self.transform.matrix, + self.bg_affine_matrix().get() + ); } - fn bg_y(&self) -> MemoryMapped { - unsafe { MemoryMapped::new(0x0400_000c + 0x10 * self.background_id()) } + fn scroll_pos(&self) -> Vector2D { + self.scroll } - fn bg_affine_matrix(&self) -> MemoryMapped { - unsafe { MemoryMapped::new(0x0400_0000 + 0x10 * self.background_id()) } + fn set_scroll_pos(&mut self, new_pos: Vector2D) { + self.scroll = new_pos; } } @@ -321,7 +304,6 @@ impl AffineMap { screenblock: u8, priority: Priority, size: AffineBackgroundSize, - bg_center: Vector2D>, ) -> Self { Self { background_id, @@ -329,7 +311,9 @@ impl AffineMap { priority, size, - bg_center, + scroll: Default::default(), + + transform_origin: Default::default(), transform: Default::default(), tiles: vec![Default::default(); size.num_tiles()], @@ -369,22 +353,36 @@ impl AffineMap { *self.tiles_dirty() = true; } - pub fn set_transform_raw(&mut self, transform: BgAffineSetData) { - self.transform = transform; - } - pub fn set_transform( &mut self, - display_center: impl Into>, + transform_origin: impl Into>>, scale: impl Into>>, rotation: impl Into>, ) { - self.set_transform_raw(crate::syscall::bg_affine_matrix( - self.bg_center, - display_center.into(), - scale.into(), - rotation.into(), - )); + self.transform_origin = transform_origin.into(); + let scale = scale.into(); + let rotation = rotation.into(); + self.transform = + crate::syscall::bg_affine_matrix(self.transform_origin, self.scroll, scale, rotation); + + crate::println!( + "{:?}, {:?}, {:?}, {:?}", + self.transform_origin, + self.scroll, + scale, + rotation + ); + crate::println!("{:?}", self.transform.matrix); + } + + fn bg_x(&self) -> MemoryMapped> { + unsafe { MemoryMapped::new(0x0400_0008 + 0x10 * self.background_id()) } + } + fn bg_y(&self) -> MemoryMapped> { + unsafe { MemoryMapped::new(0x0400_000c + 0x10 * self.background_id()) } + } + fn bg_affine_matrix(&self) -> MemoryMapped { + unsafe { MemoryMapped::new(0x0400_0000 + 0x10 * self.background_id()) } } } diff --git a/agb/src/display/tiled/mod.rs b/agb/src/display/tiled/mod.rs index 164ce3a6..ea23aa4a 100644 --- a/agb/src/display/tiled/mod.rs +++ b/agb/src/display/tiled/mod.rs @@ -7,7 +7,7 @@ mod vram_manager; use crate::bitarray::Bitarray; use crate::display::Priority; -use agb_fixnum::{Num, Vector2D}; +use agb_fixnum::Vector2D; use core::cell::RefCell; pub use infinite_scrolled_map::{InfiniteScrolledMap, PartialUpdateStatus}; pub use map::{AffineMap, MapLoan, RegularMap, TiledMap}; @@ -307,16 +307,7 @@ where screenblocks.set(id, true); } - let center_dim = Num::new(size.width() as i32 * 8 / 2); - let default_bg_center = (center_dim, center_dim).into(); - - let bg = AffineMap::new( - new_background as u8, - screenblock as u8 + 16, - priority, - size, - default_bg_center, - ); + let bg = AffineMap::new(new_background as u8, screenblock as u8 + 16, priority, size); affine.set(new_background, true); diff --git a/agb/src/syscall.rs b/agb/src/syscall.rs index e7d342bd..dd123ace 100644 --- a/agb/src/syscall.rs +++ b/agb/src/syscall.rs @@ -185,7 +185,8 @@ pub fn bg_affine_matrix( in("r0") &input as *const Input, in("r1") output.as_mut_ptr(), in("r2") 1, - lateout("r3") _, + + clobber_abi("C") ); } @@ -231,7 +232,7 @@ mod tests { use super::*; #[test_case] - fn affine(_gba: &mut crate::Gba) { + fn affine_obj(_gba: &mut crate::Gba) { // expect identity matrix let one: Num = 1.into(); @@ -241,4 +242,22 @@ mod tests { assert_eq!(p_a, one); assert_eq!(p_d, one); } + + #[test_case] + fn affine_bg(_gba: &mut crate::Gba) { + // expect the identity matrix + let aff = bg_affine_matrix( + (0, 0).into(), + (0i16, 0i16).into(), + (1i16, 1i16).into(), + 0.into(), + ); + + let matrix = aff.matrix; + let (p_a, p_b, p_c, p_d) = (matrix.p_a, matrix.p_b, matrix.p_c, matrix.p_d); + assert_eq!(p_a, 1.into()); + assert_eq!(p_b, 0.into()); + assert_eq!(p_c, 0.into()); + assert_eq!(p_d, 1.into()); + } } diff --git a/examples/hyperspace-roll/src/background.rs b/examples/hyperspace-roll/src/background.rs index 037fa20a..91d16c95 100644 --- a/examples/hyperspace-roll/src/background.rs +++ b/examples/hyperspace-roll/src/background.rs @@ -1,27 +1,21 @@ use agb::{ - display::tiled::{RegularMap, TileFormat, TileSet, TileSetting, TiledMap, VRamManager}, + display::tiled::{RegularMap, TileFormat, TileSet, TileSetting, VRamManager}, include_gfx, rng, }; use crate::sfx::Sfx; -include_gfx!("gfx/backgrounds.toml"); +include_gfx!("gfx/stars.toml"); + +include_gfx!("gfx/help.toml"); pub fn load_palettes(vram: &mut VRamManager) { - vram.set_background_palettes(backgrounds::stars.palettes); -} - -fn description_tileset() -> (TileSet<'static>, TileSet<'static>) { - let descriptions_1_tileset = TileSet::new( - backgrounds::descriptions1.tiles, - agb::display::tiled::TileFormat::FourBpp, - ); - let descriptions_2_tileset = TileSet::new( - backgrounds::descriptions2.tiles, - agb::display::tiled::TileFormat::FourBpp, - ); - - (descriptions_1_tileset, descriptions_2_tileset) + vram.set_background_palettes(&[ + stars::stars.palettes[0].clone(), + crate::customise::DESCRIPTIONS_1_PALETTE.clone(), + crate::customise::DESCRIPTIONS_2_PALETTE.clone(), + help::help.palettes[0].clone(), + ]); } pub(crate) fn load_help_text( @@ -30,69 +24,18 @@ pub(crate) fn load_help_text( help_text_line: u16, at_tile: (u16, u16), ) { - let help_tileset = TileSet::new( - backgrounds::help.tiles, - agb::display::tiled::TileFormat::FourBpp, - ); + let help_tileset = TileSet::new(help::help.tiles, agb::display::tiled::TileFormat::FourBpp); for x in 0..16 { - let tile_id = help_text_line * 16 + x; background.set_tile( vram, (x + at_tile.0, at_tile.1).into(), &help_tileset, - TileSetting::new( - tile_id, - false, - false, - backgrounds::help.palette_assignments[tile_id as usize], - ), + TileSetting::new(help_text_line * 16 + x, false, false, 3), ) } } -pub(crate) fn load_description( - upgrade: usize, - descriptions_map: &mut RegularMap, - vram: &mut VRamManager, -) { - let (descriptions_1_tileset, descriptions_2_tileset) = description_tileset(); - - for y in 0..11 { - for x in 0..8 { - if upgrade < 10 { - let tile_id = y * 8 + x + 8 * 11 * upgrade as u16; - - descriptions_map.set_tile( - vram, - (x, y).into(), - &descriptions_1_tileset, - TileSetting::new( - tile_id, - false, - false, - backgrounds::descriptions1.palette_assignments[tile_id as usize], - ), - ) - } else { - let tile_id = y * 8 + x + 8 * 11 * (upgrade as u16 - 10); - - descriptions_map.set_tile( - vram, - (x, y).into(), - &descriptions_2_tileset, - TileSetting::new( - tile_id, - false, - false, - backgrounds::descriptions2.palette_assignments[tile_id as usize], - ), - ) - } - } - } -} - // Expects a 64x32 map fn create_background_map(map: &mut RegularMap, vram: &mut VRamManager, stars_tileset: &TileSet) { for x in 0..64u16 { @@ -104,16 +47,7 @@ fn create_background_map(map: &mut RegularMap, vram: &mut VRamManager, stars_til } else { rng::gen().rem_euclid(64) as u16 }; - let tile_setting = TileSetting::new( - tile_id, - false, - false, - if blank { - 0 - } else { - backgrounds::stars.palette_assignments[tile_id as usize] - }, - ); + let tile_setting = TileSetting::new(tile_id, false, false, 0); map.set_tile(vram, (x, y).into(), stars_tileset, tile_setting); } @@ -124,11 +58,8 @@ fn create_background_map(map: &mut RegularMap, vram: &mut VRamManager, stars_til pub fn show_title_screen(background: &mut RegularMap, vram: &mut VRamManager, sfx: &mut Sfx) { background.set_scroll_pos((0_u16, 0_u16).into()); - vram.set_background_palettes(backgrounds::title.palettes); - let tile_set = TileSet::new( - backgrounds::title.tiles, - agb::display::tiled::TileFormat::FourBpp, - ); + vram.set_background_palettes(stars::title.palettes); + let tile_set = TileSet::new(stars::title.tiles, agb::display::tiled::TileFormat::FourBpp); background.hide(); for x in 0..30u16 { @@ -142,7 +73,7 @@ pub fn show_title_screen(background: &mut RegularMap, vram: &mut VRamManager, sf tile_id, false, false, - backgrounds::title.palette_assignments[tile_id as usize], + stars::title.palette_assignments[tile_id as usize], ), ); } @@ -169,7 +100,7 @@ impl<'a> StarBackground<'a> { background2: &'a mut RegularMap, vram: &'_ mut VRamManager, ) -> Self { - let stars_tileset = TileSet::new(backgrounds::stars.tiles, TileFormat::FourBpp); + let stars_tileset = TileSet::new(stars::stars.tiles, TileFormat::FourBpp); create_background_map(background1, vram, &stars_tileset); create_background_map(background2, vram, &stars_tileset); From 63b0fa63c6ac98fb69dfaa72680f0970c76fe090 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 1 Oct 2022 15:28:53 +0100 Subject: [PATCH 19/49] packed(4) to ensure alignment --- agb/src/syscall.rs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/agb/src/syscall.rs b/agb/src/syscall.rs index dd123ace..3ab60830 100644 --- a/agb/src/syscall.rs +++ b/agb/src/syscall.rs @@ -139,7 +139,7 @@ pub fn arc_tan2(x: i16, y: i32) -> i16 { result } -#[repr(C, packed)] +#[repr(C, packed(4))] pub struct BgAffineSetData { pub matrix: AffineMatrixAttributes, pub position: Vector2D>, @@ -161,7 +161,7 @@ pub fn bg_affine_matrix( scale: Vector2D>, rotation: Num, ) -> BgAffineSetData { - #[repr(C, packed)] + #[repr(C, packed(4))] struct Input { bg_center: Vector2D>, display_center: Vector2D, @@ -200,7 +200,7 @@ pub fn obj_affine_matrix( rotation: Num, ) -> AffineMatrixAttributes { #[allow(dead_code)] - #[repr(C, packed)] + #[repr(C, packed(4))] struct Input { scale: Vector2D>, rotation: u16, From dd2030d35478cce3b4bf7e77d46e2f7f7bf52230 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 1 Oct 2022 15:29:42 +0100 Subject: [PATCH 20/49] Use the fact that tris turn into integers nicely --- agb/examples/affine_background.rs | 14 ++------------ 1 file changed, 2 insertions(+), 12 deletions(-) diff --git a/agb/examples/affine_background.rs b/agb/examples/affine_background.rs index 9c337f8a..04267053 100644 --- a/agb/examples/affine_background.rs +++ b/agb/examples/affine_background.rs @@ -43,18 +43,8 @@ fn main(mut gba: agb::Gba) -> ! { loop { input.update(); - - match input.x_tri() { - Tri::Positive => scroll_x += 1, - Tri::Negative => scroll_x -= 1, - _ => {} - } - - match input.y_tri() { - Tri::Positive => scroll_y += 1, - Tri::Negative => scroll_y -= 1, - _ => {} - } + scroll_x += input.x_tri() as i32; + scroll_y += input.y_tri() as i32; let scroll_pos = (scroll_x as i16, scroll_y as i16); bg.set_scroll_pos(scroll_pos.into()); From cde3bca0ea27e30c93f3d92979f338e96e85eb55 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 1 Oct 2022 15:30:37 +0100 Subject: [PATCH 21/49] Make the background rotate again --- agb/examples/affine_background.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/agb/examples/affine_background.rs b/agb/examples/affine_background.rs index 04267053..21966e0d 100644 --- a/agb/examples/affine_background.rs +++ b/agb/examples/affine_background.rs @@ -48,7 +48,7 @@ fn main(mut gba: agb::Gba) -> ! { let scroll_pos = (scroll_x as i16, scroll_y as i16); bg.set_scroll_pos(scroll_pos.into()); - bg.set_transform((0, 0), (1, 1), 0); + bg.set_transform((0, 0), (1, 1), rotation); rotation += rotation_increase; if rotation >= num!(255.) { From ccc3823085de892b47062368134e793a4565b106 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 1 Oct 2022 15:30:46 +0100 Subject: [PATCH 22/49] Remove unused import --- agb/examples/affine_background.rs | 1 - 1 file changed, 1 deletion(-) diff --git a/agb/examples/affine_background.rs b/agb/examples/affine_background.rs index 21966e0d..175145f5 100644 --- a/agb/examples/affine_background.rs +++ b/agb/examples/affine_background.rs @@ -8,7 +8,6 @@ use agb::{ }, fixnum::{num, Num}, include_gfx, - input::Tri, }; include_gfx!("examples/affine_tiles.toml"); From ffde16abd96f419baf2f0d72e5318a785a511f10 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 1 Oct 2022 15:31:19 +0100 Subject: [PATCH 23/49] Remove crate::println! --- agb/src/display/tiled/map.rs | 15 --------------- 1 file changed, 15 deletions(-) diff --git a/agb/src/display/tiled/map.rs b/agb/src/display/tiled/map.rs index f5b60d89..deb37d83 100644 --- a/agb/src/display/tiled/map.rs +++ b/agb/src/display/tiled/map.rs @@ -283,12 +283,6 @@ impl TiledMapPrivate for AffineMap { self.bg_x().set(register_pos.x); self.bg_y().set(register_pos.y); self.bg_affine_matrix().set(self.transform.matrix); - - crate::println!( - "update: {:?} {:?}", - self.transform.matrix, - self.bg_affine_matrix().get() - ); } fn scroll_pos(&self) -> Vector2D { self.scroll @@ -364,15 +358,6 @@ impl AffineMap { let rotation = rotation.into(); self.transform = crate::syscall::bg_affine_matrix(self.transform_origin, self.scroll, scale, rotation); - - crate::println!( - "{:?}, {:?}, {:?}, {:?}", - self.transform_origin, - self.scroll, - scale, - rotation - ); - crate::println!("{:?}", self.transform.matrix); } fn bg_x(&self) -> MemoryMapped> { From 9affb6dfb79dfc2330fc8602c1d3e3c8a6dfe98a Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 1 Oct 2022 15:35:24 +0100 Subject: [PATCH 24/49] Use negative numbers --- examples/hyperspace-roll/src/battle.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/hyperspace-roll/src/battle.rs b/examples/hyperspace-roll/src/battle.rs index 691c7a85..b0354242 100644 --- a/examples/hyperspace-roll/src/battle.rs +++ b/examples/hyperspace-roll/src/battle.rs @@ -488,7 +488,7 @@ pub(crate) fn battle_screen( agb.sfx.battle(); agb.sfx.frame(); - help_background.set_scroll_pos((u16::MAX - 16, u16::MAX - 97).into()); + help_background.set_scroll_pos((-16i16, -97i16).into()); crate::background::load_help_text(&mut agb.vram, help_background, 1, (0, 0)); crate::background::load_help_text(&mut agb.vram, help_background, 2, (0, 1)); From 12ed911f164dc1a0c2ffd157d6db23ffb8fe1cad Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 1 Oct 2022 16:52:23 +0100 Subject: [PATCH 25/49] Fix build errors --- examples/hyperspace-roll/src/background.rs | 67 ++++++++++++++----- examples/hyperspace-roll/src/customise.rs | 4 +- .../src/level_display.rs | 2 +- .../src/splash_screen.rs | 4 +- 4 files changed, 55 insertions(+), 22 deletions(-) diff --git a/examples/hyperspace-roll/src/background.rs b/examples/hyperspace-roll/src/background.rs index 91d16c95..4fcfd5db 100644 --- a/examples/hyperspace-roll/src/background.rs +++ b/examples/hyperspace-roll/src/background.rs @@ -1,20 +1,18 @@ use agb::{ - display::tiled::{RegularMap, TileFormat, TileSet, TileSetting, VRamManager}, + display::tiled::{RegularMap, TileFormat, TileSet, TileSetting, TiledMap, VRamManager}, include_gfx, rng, }; use crate::sfx::Sfx; -include_gfx!("gfx/stars.toml"); - -include_gfx!("gfx/help.toml"); +include_gfx!("gfx/backgrounds.toml"); pub fn load_palettes(vram: &mut VRamManager) { vram.set_background_palettes(&[ - stars::stars.palettes[0].clone(), - crate::customise::DESCRIPTIONS_1_PALETTE.clone(), - crate::customise::DESCRIPTIONS_2_PALETTE.clone(), - help::help.palettes[0].clone(), + backgrounds::stars.palettes[0].clone(), + backgrounds::descriptions1.palettes[0].clone(), + backgrounds::descriptions2.palettes[0].clone(), + backgrounds::help.palettes[0].clone(), ]); } @@ -24,7 +22,10 @@ pub(crate) fn load_help_text( help_text_line: u16, at_tile: (u16, u16), ) { - let help_tileset = TileSet::new(help::help.tiles, agb::display::tiled::TileFormat::FourBpp); + let help_tileset = TileSet::new( + backgrounds::help.tiles, + agb::display::tiled::TileFormat::FourBpp, + ); for x in 0..16 { background.set_tile( @@ -36,6 +37,35 @@ pub(crate) fn load_help_text( } } +pub(crate) fn load_description( + face_id: usize, + descriptions_map: &mut RegularMap, + vram: &mut VRamManager, +) { + let tileset = if face_id < 10 { + TileSet::new( + backgrounds::descriptions1.tiles, + agb::display::tiled::TileFormat::FourBpp, + ) + } else { + TileSet::new( + backgrounds::descriptions2.tiles, + agb::display::tiled::TileFormat::FourBpp, + ) + }; + + for y in 0..11 { + for x in 0..8 { + descriptions_map.set_tile( + vram, + (x, y).into(), + &tileset, + TileSetting::new(y * 8 + x + 8 * 11 * (face_id as u16 - 10), false, false, 2), + ) + } + } +} + // Expects a 64x32 map fn create_background_map(map: &mut RegularMap, vram: &mut VRamManager, stars_tileset: &TileSet) { for x in 0..64u16 { @@ -53,13 +83,16 @@ fn create_background_map(map: &mut RegularMap, vram: &mut VRamManager, stars_til } } - map.set_scroll_pos((0u16, rng::gen().rem_euclid(8) as u16).into()); + map.set_scroll_pos((0i16, rng::gen().rem_euclid(8) as i16).into()); } pub fn show_title_screen(background: &mut RegularMap, vram: &mut VRamManager, sfx: &mut Sfx) { - background.set_scroll_pos((0_u16, 0_u16).into()); - vram.set_background_palettes(stars::title.palettes); - let tile_set = TileSet::new(stars::title.tiles, agb::display::tiled::TileFormat::FourBpp); + background.set_scroll_pos((0i16, 0).into()); + vram.set_background_palettes(backgrounds::title.palettes); + let tile_set = TileSet::new( + backgrounds::title.tiles, + agb::display::tiled::TileFormat::FourBpp, + ); background.hide(); for x in 0..30u16 { @@ -73,7 +106,7 @@ pub fn show_title_screen(background: &mut RegularMap, vram: &mut VRamManager, sf tile_id, false, false, - stars::title.palette_assignments[tile_id as usize], + backgrounds::title.palette_assignments[tile_id as usize], ), ); } @@ -100,7 +133,7 @@ impl<'a> StarBackground<'a> { background2: &'a mut RegularMap, vram: &'_ mut VRamManager, ) -> Self { - let stars_tileset = TileSet::new(stars::stars.tiles, TileFormat::FourBpp); + let stars_tileset = TileSet::new(backgrounds::stars.tiles, TileFormat::FourBpp); create_background_map(background1, vram, &stars_tileset); create_background_map(background2, vram, &stars_tileset); @@ -116,13 +149,13 @@ impl<'a> StarBackground<'a> { pub fn update(&mut self) { if self.background1_timer == 0 { self.background1 - .set_scroll_pos(self.background1.scroll_pos() + (1u16, 0).into()); + .set_scroll_pos(self.background1.scroll_pos() + (1i16, 0).into()); self.background1_timer = 2; } if self.background2_timer == 0 { self.background2 - .set_scroll_pos(self.background2.scroll_pos() + (1u16, 0).into()); + .set_scroll_pos(self.background2.scroll_pos() + (1i16, 0).into()); self.background2_timer = 3; } diff --git a/examples/hyperspace-roll/src/customise.rs b/examples/hyperspace-roll/src/customise.rs index a6d9e5b6..ac0e9a96 100644 --- a/examples/hyperspace-roll/src/customise.rs +++ b/examples/hyperspace-roll/src/customise.rs @@ -163,9 +163,9 @@ pub(crate) fn customise_screen( ) -> PlayerDice { agb.sfx.customise(); agb.sfx.frame(); - descriptions_map.set_scroll_pos((u16::MAX - 174, u16::MAX - 52).into()); + descriptions_map.set_scroll_pos((-174i16, -52).into()); - help_background.set_scroll_pos((u16::MAX - 148, u16::MAX - 34).into()); + help_background.set_scroll_pos((-148i16, -34).into()); crate::background::load_help_text(&mut agb.vram, help_background, 0, (0, 0)); // create the dice diff --git a/examples/the-hat-chooses-the-wizard/src/level_display.rs b/examples/the-hat-chooses-the-wizard/src/level_display.rs index 10ac10ec..0defcc31 100644 --- a/examples/the-hat-chooses-the-wizard/src/level_display.rs +++ b/examples/the-hat-chooses-the-wizard/src/level_display.rs @@ -35,5 +35,5 @@ pub fn write_level( ); } - map.set_scroll_pos((-(WIDTH / 2 - 7 * 8 / 2) as u16, -(HEIGHT / 2 - 4) as u16).into()); + map.set_scroll_pos((-(WIDTH / 2 - 7 * 8 / 2) as i16, -(HEIGHT / 2 - 4) as i16).into()); } diff --git a/examples/the-hat-chooses-the-wizard/src/splash_screen.rs b/examples/the-hat-chooses-the-wizard/src/splash_screen.rs index de6f9be9..9af2e1fc 100644 --- a/examples/the-hat-chooses-the-wizard/src/splash_screen.rs +++ b/examples/the-hat-chooses-the-wizard/src/splash_screen.rs @@ -1,6 +1,6 @@ use super::sfx::MusicBox; use agb::{ - display::tiled::{RegularMap, TiledMap, TileFormat, TileSet, TileSetting, VRamManager}, + display::tiled::{RegularMap, TileFormat, TileSet, TileSetting, TiledMap, VRamManager}, sound::mixer::Mixer, }; @@ -18,7 +18,7 @@ pub fn show_splash_screen( map: &mut RegularMap, vram: &mut VRamManager, ) { - map.set_scroll_pos((0u16, 0u16).into()); + map.set_scroll_pos((0i16, 0i16).into()); let (tileset, palette) = match which { SplashScreen::Start => { let tileset = TileSet::new(splash_screens::splash.tiles, TileFormat::FourBpp); From cdad8afe702de1baba9361ebab88210a6806a153 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sun, 2 Oct 2022 19:59:21 +0100 Subject: [PATCH 26/49] Add some badges to the readme --- README.md | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/README.md b/README.md index df80be43..578a72b6 100644 --- a/README.md +++ b/README.md @@ -2,6 +2,11 @@ ## Rust for the Game Boy Advance +[![Docs](https://docs.rs/agb/badge.svg)](https://docs.rs/agb/latest/agb) +[![Build](https://github.com/agbrs/agb/actions/workflows/build-and-test.yml/badge.svg?query=branch%3Amaster)](https://github.com/agbrs/agb/actions/workflows/build-and-test.yml) +[![Licence](https://img.shields.io/crates/l/agb)](https://www.mozilla.org/en-US/MPL/2.0/) +[![Crates.io](https://img.shields.io/crates/v/agb)](https://crates.io/crates/agb) + ![AGB logo](.github/logo.png) This is a library for making games on the Game Boy Advance using the Rust From 5bb1c3b162b4e56083f5dd2f8986e1d4bd65a20a Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sun, 2 Oct 2022 20:10:46 +0100 Subject: [PATCH 27/49] Use the correct way to specify the branch --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 578a72b6..6da2f92a 100644 --- a/README.md +++ b/README.md @@ -3,7 +3,7 @@ ## Rust for the Game Boy Advance [![Docs](https://docs.rs/agb/badge.svg)](https://docs.rs/agb/latest/agb) -[![Build](https://github.com/agbrs/agb/actions/workflows/build-and-test.yml/badge.svg?query=branch%3Amaster)](https://github.com/agbrs/agb/actions/workflows/build-and-test.yml) +[![Build](https://github.com/agbrs/agb/actions/workflows/build-and-test.yml/badge.svg?branch=master)](https://github.com/agbrs/agb/actions/workflows/build-and-test.yml) [![Licence](https://img.shields.io/crates/l/agb)](https://www.mozilla.org/en-US/MPL/2.0/) [![Crates.io](https://img.shields.io/crates/v/agb)](https://crates.io/crates/agb) From 2ffc68c5c1c453c945307f7a7347029afda77eec Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sun, 2 Oct 2022 21:14:01 +0100 Subject: [PATCH 28/49] Start making the frequency field dynamic rather than a feature flag --- agb-sound-converter/Cargo.toml | 4 -- agb-sound-converter/src/lib.rs | 16 ------- agb/Cargo.toml | 6 --- agb/examples/mixer_32768.rs | 4 +- agb/examples/mixer_basic.rs | 4 +- agb/examples/stereo_sound.rs | 4 +- agb/src/sound/mixer/mod.rs | 42 +++++++++++++++++- agb/src/sound/mixer/sw_mixer.rs | 78 +++++++++++++++------------------ 8 files changed, 82 insertions(+), 76 deletions(-) diff --git a/agb-sound-converter/Cargo.toml b/agb-sound-converter/Cargo.toml index 5d6f316d..caea2f60 100644 --- a/agb-sound-converter/Cargo.toml +++ b/agb-sound-converter/Cargo.toml @@ -18,10 +18,6 @@ debug = true [lib] proc-macro = true -[features] -freq18157 = [] -freq32768 = [] - [dependencies] hound = "3.5" syn = "1" diff --git a/agb-sound-converter/src/lib.rs b/agb-sound-converter/src/lib.rs index b04f280b..5cbc2446 100644 --- a/agb-sound-converter/src/lib.rs +++ b/agb-sound-converter/src/lib.rs @@ -6,15 +6,6 @@ use quote::{quote, ToTokens}; use std::path::Path; use syn::parse_macro_input; -#[cfg(all(not(feature = "freq18157"), not(feature = "freq32768")))] -const FREQUENCY: u32 = 10512; -#[cfg(feature = "freq18157")] -const FREQUENCY: u32 = 18157; -#[cfg(feature = "freq32768")] -const FREQUENCY: u32 = 32768; -#[cfg(all(feature = "freq18157", feature = "freq32768"))] -compile_error!("Must have at most one of freq18157 or freq32768 features enabled"); - use quote::TokenStreamExt; struct ByteString<'a>(&'a [u8]); impl ToTokens for ByteString<'_> { @@ -37,13 +28,6 @@ pub fn include_wav(input: TokenStream) -> TokenStream { let wav_reader = hound::WavReader::open(&path) .unwrap_or_else(|_| panic!("Failed to load file {}", include_path)); - assert_eq!( - wav_reader.spec().sample_rate, - FREQUENCY, - "agb currently only supports sample rate of {}Hz", - FREQUENCY - ); - let samples: Vec = samples_from_reader(wav_reader).collect(); let samples = ByteString(&samples); diff --git a/agb/Cargo.toml b/agb/Cargo.toml index 83afc851..29070813 100644 --- a/agb/Cargo.toml +++ b/agb/Cargo.toml @@ -17,8 +17,6 @@ debug = true [features] default = ["testing"] -freq18157 = ["agb_sound_converter/freq18157"] -freq32768 = ["agb_sound_converter/freq32768"] testing = [] [dependencies] @@ -34,7 +32,3 @@ rustc-hash = { version = "1", default-features = false } [package.metadata.docs.rs] default-target = "thumbv6m-none-eabi" targets = [] - -[[example]] -name = "mixer_32768" -required-features = ["freq32768"] \ No newline at end of file diff --git a/agb/examples/mixer_32768.rs b/agb/examples/mixer_32768.rs index b10bf8d3..8dd38120 100644 --- a/agb/examples/mixer_32768.rs +++ b/agb/examples/mixer_32768.rs @@ -7,7 +7,7 @@ use agb::{ Font, Priority, }, include_font, include_wav, - sound::mixer::SoundChannel, + sound::mixer::{Frequency, SoundChannel}, Gba, }; @@ -40,7 +40,7 @@ fn main(mut gba: Gba) -> ! { let mut timer = timer_controller.timer2; timer.set_enabled(true); - let mut mixer = gba.mixer.mixer(); + let mut mixer = gba.mixer.mixer(Frequency::Hz32768); mixer.enable(); let _interrupt = mixer.setup_interrupt_handler(); diff --git a/agb/examples/mixer_basic.rs b/agb/examples/mixer_basic.rs index 1419562b..e3fc76f9 100644 --- a/agb/examples/mixer_basic.rs +++ b/agb/examples/mixer_basic.rs @@ -3,7 +3,7 @@ use agb::fixnum::Num; use agb::input::{Button, ButtonController, Tri}; -use agb::sound::mixer::SoundChannel; +use agb::sound::mixer::{Frequency, SoundChannel}; use agb::{fixnum::num, include_wav, Gba}; // Music - "Dead Code" by Josh Woodward, free download at http://joshwoodward.com @@ -14,7 +14,7 @@ fn main(mut gba: Gba) -> ! { let mut input = ButtonController::new(); let vblank_provider = agb::interrupt::VBlank::get(); - let mut mixer = gba.mixer.mixer(); + let mut mixer = gba.mixer.mixer(Frequency::Hz10512); mixer.enable(); let channel = SoundChannel::new(DEAD_CODE); diff --git a/agb/examples/stereo_sound.rs b/agb/examples/stereo_sound.rs index d813a237..6d87cfbd 100644 --- a/agb/examples/stereo_sound.rs +++ b/agb/examples/stereo_sound.rs @@ -7,7 +7,7 @@ use agb::{ Font, Priority, }, include_font, include_wav, - sound::mixer::SoundChannel, + sound::mixer::{Frequency, SoundChannel}, Gba, }; @@ -40,7 +40,7 @@ fn main(mut gba: Gba) -> ! { let mut timer = timer_controller.timer2; timer.set_enabled(true); - let mut mixer = gba.mixer.mixer(); + let mut mixer = gba.mixer.mixer(Frequency::Hz10512); mixer.enable(); let mut channel = SoundChannel::new(LET_IT_IN); diff --git a/agb/src/sound/mixer/mod.rs b/agb/src/sound/mixer/mod.rs index ac01dba7..645ad690 100644 --- a/agb/src/sound/mixer/mod.rs +++ b/agb/src/sound/mixer/mod.rs @@ -138,8 +138,8 @@ impl MixerController { } /// Get a [`Mixer`] in order to start producing sounds. - pub fn mixer(&mut self) -> Mixer { - Mixer::new() + pub fn mixer(&mut self, frequency: Frequency) -> Mixer { + Mixer::new(frequency) } } @@ -149,6 +149,44 @@ enum SoundPriority { Low, } +/// The supported frequencies within AGB. These are chosen to work well with +/// the hardware. Note that the higher the frequency, the better the quality of +/// the sound but the more CPU time sound mixing will take. +#[derive(Debug, Copy, Clone, PartialEq, Eq)] +#[non_exhaustive] +pub enum Frequency { + /// 10512Hz + Hz10512, + /// 18157Hz + Hz18157, + /// 32768Hz - note that this option requires the timer to do the buffer swapping + Hz32768, +} + +// These work perfectly with swapping the buffers every vblank +// list here: http://deku.gbadev.org/program/sound1.html +impl Frequency { + pub(crate) fn frequency(self) -> i32 { + use Frequency::*; + + match self { + Hz10512 => 10512, + Hz18157 => 18157, + Hz32768 => 32768, + } + } + + pub(crate) fn buffer_size(self) -> usize { + use Frequency::*; + + match self { + Hz10512 => 176, + Hz18157 => 18157, + Hz32768 => 32768, + } + } +} + /// Describes one sound which should be playing. This could be a sound effect or /// the background music. Use the factory methods on this to modify how it is played. /// diff --git a/agb/src/sound/mixer/sw_mixer.rs b/agb/src/sound/mixer/sw_mixer.rs index 80b5b139..2c6f501e 100644 --- a/agb/src/sound/mixer/sw_mixer.rs +++ b/agb/src/sound/mixer/sw_mixer.rs @@ -1,12 +1,14 @@ use core::cell::RefCell; -use core::intrinsics::transmute; +use alloc::boxed::Box; +use alloc::vec::Vec; use bare_metal::{CriticalSection, Mutex}; -use super::hw; use super::hw::LeftOrRight; +use super::{hw, Frequency}; use super::{SoundChannel, SoundPriority}; +use crate::InternalAllocator; use crate::{ fixnum::Num, interrupt::free, @@ -80,6 +82,7 @@ pub struct Mixer { buffer: MixerBuffer, channels: [Option; 8], indices: [i32; 8], + frequency: Frequency, timer: Timer, } @@ -108,9 +111,10 @@ pub struct Mixer { pub struct ChannelId(usize, i32); impl Mixer { - pub(super) fn new() -> Self { + pub(super) fn new(frequency: Frequency) -> Self { Self { - buffer: MixerBuffer::new(), + frequency, + buffer: MixerBuffer::new(frequency), channels: Default::default(), indices: Default::default(), @@ -123,7 +127,7 @@ impl Mixer { /// You must call this method in order to start playing sound. You can do as much set up before /// this as you like, but you will not get any sound out of the console until this method is called. pub fn enable(&mut self) { - hw::set_timer_counter_for_frequency_and_enable(&mut self.timer, constants::SOUND_FREQUENCY); + hw::set_timer_counter_for_frequency_and_enable(&mut self.timer, self.frequency.frequency()); hw::set_sound_control_register_for_mixer(); } @@ -184,7 +188,7 @@ impl Mixer { .set_cascade(true) .set_divider(Divider::Divider1) .set_interrupt(true) - .set_overflow_amount(constants::SOUND_BUFFER_SIZE as u16) + .set_overflow_amount(self.frequency.buffer_size() as u16) .set_enabled(true); add_interrupt_handler(timer1.interrupt(), move |cs| self.buffer.swap(cs)) @@ -313,47 +317,31 @@ impl Mixer { } } -// These work perfectly with swapping the buffers every vblank -// list here: http://deku.gbadev.org/program/sound1.html -#[cfg(all(not(feature = "freq18157"), not(feature = "freq32768")))] -mod constants { - pub const SOUND_FREQUENCY: i32 = 10512; - pub const SOUND_BUFFER_SIZE: usize = 176; -} - -#[cfg(feature = "freq18157")] -mod constants { - pub const SOUND_FREQUENCY: i32 = 18157; - pub const SOUND_BUFFER_SIZE: usize = 304; -} - -#[cfg(feature = "freq32768")] -mod constants { - pub const SOUND_FREQUENCY: i32 = 32768; - pub const SOUND_BUFFER_SIZE: usize = 560; -} - -fn set_asm_buffer_size() { +fn set_asm_buffer_size(frequency: Frequency) { extern "C" { static mut agb_rs__buffer_size: usize; } unsafe { - agb_rs__buffer_size = constants::SOUND_BUFFER_SIZE; + agb_rs__buffer_size = frequency.buffer_size(); } } -#[repr(C, align(4))] -struct SoundBuffer([i8; constants::SOUND_BUFFER_SIZE * 2]); +struct SoundBuffer(Box<[i8], InternalAllocator>); -impl Default for SoundBuffer { - fn default() -> Self { - Self([0; constants::SOUND_BUFFER_SIZE * 2]) +impl SoundBuffer { + fn new(frequency: Frequency) -> Self { + let my_size = frequency.buffer_size() * 2; + let mut v = Vec::with_capacity_in(my_size, InternalAllocator); + v.resize(my_size, 0); + + SoundBuffer(v.into_boxed_slice()) } } struct MixerBuffer { buffers: [SoundBuffer; 3], + frequency: Frequency, state: Mutex>, } @@ -391,16 +379,20 @@ impl MixerBufferState { } impl MixerBuffer { - fn new() -> Self { - set_asm_buffer_size(); - + fn new(frequency: Frequency) -> Self { MixerBuffer { - buffers: Default::default(), + buffers: [ + SoundBuffer::new(frequency), + SoundBuffer::new(frequency), + SoundBuffer::new(frequency), + ], state: Mutex::new(RefCell::new(MixerBufferState { active_buffer: 0, playing_buffer: 0, })), + + frequency, } } @@ -413,15 +405,17 @@ impl MixerBuffer { let (left_buffer, right_buffer) = self.buffers[buffer] .0 - .split_at(constants::SOUND_BUFFER_SIZE); + .split_at(self.frequency.buffer_size()); hw::enable_dma_for_sound(left_buffer, LeftOrRight::Left); hw::enable_dma_for_sound(right_buffer, LeftOrRight::Right); } fn write_channels<'a>(&mut self, channels: impl Iterator) { - let mut buffer: [Num; constants::SOUND_BUFFER_SIZE * 2] = - unsafe { transmute([0i16; constants::SOUND_BUFFER_SIZE * 2]) }; + set_asm_buffer_size(self.frequency); + + let mut buffer = Vec::with_capacity_in(self.frequency.buffer_size() * 2, InternalAllocator); + buffer.resize(self.frequency.buffer_size() * 2, 0.into()); for channel in channels { if channel.is_done { @@ -434,7 +428,7 @@ impl MixerBuffer { channel.playback_speed }; - if (channel.pos + playback_speed * constants::SOUND_BUFFER_SIZE).floor() + if (channel.pos + playback_speed * self.frequency.buffer_size()).floor() >= channel.data.len() { // TODO: This should probably play what's left rather than skip the last bit @@ -469,7 +463,7 @@ impl MixerBuffer { } } - channel.pos += playback_speed * constants::SOUND_BUFFER_SIZE; + channel.pos += playback_speed * self.frequency.buffer_size(); } let write_buffer_index = free(|cs| self.state.borrow(cs).borrow_mut().active_advanced()); From c6d961cd8d7490efbeda1f607a11ae9141a97f24 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sun, 2 Oct 2022 21:18:29 +0100 Subject: [PATCH 29/49] Almost works --- agb/src/sound/mixer/sw_mixer.rs | 16 +++++++++++----- 1 file changed, 11 insertions(+), 5 deletions(-) diff --git a/agb/src/sound/mixer/sw_mixer.rs b/agb/src/sound/mixer/sw_mixer.rs index 2c6f501e..4edc6107 100644 --- a/agb/src/sound/mixer/sw_mixer.rs +++ b/agb/src/sound/mixer/sw_mixer.rs @@ -341,6 +341,7 @@ impl SoundBuffer { struct MixerBuffer { buffers: [SoundBuffer; 3], + working_buffer: Box<[Num], InternalAllocator>, frequency: Frequency, state: Mutex>, @@ -380,6 +381,10 @@ impl MixerBufferState { impl MixerBuffer { fn new(frequency: Frequency) -> Self { + let mut working_buffer = + Vec::with_capacity_in(frequency.buffer_size() * 2, InternalAllocator); + working_buffer.resize(frequency.buffer_size() * 2, 0.into()); + MixerBuffer { buffers: [ SoundBuffer::new(frequency), @@ -387,6 +392,8 @@ impl MixerBuffer { SoundBuffer::new(frequency), ], + working_buffer: working_buffer.into_boxed_slice(), + state: Mutex::new(RefCell::new(MixerBufferState { active_buffer: 0, playing_buffer: 0, @@ -414,8 +421,7 @@ impl MixerBuffer { fn write_channels<'a>(&mut self, channels: impl Iterator) { set_asm_buffer_size(self.frequency); - let mut buffer = Vec::with_capacity_in(self.frequency.buffer_size() * 2, InternalAllocator); - buffer.resize(self.frequency.buffer_size() * 2, 0.into()); + self.working_buffer.fill(0.into()); for channel in channels { if channel.is_done { @@ -444,7 +450,7 @@ impl MixerBuffer { unsafe { agb_rs__mixer_add_stereo( channel.data.as_ptr().add(channel.pos.floor()), - buffer.as_mut_ptr(), + self.working_buffer.as_mut_ptr(), channel.volume, ); } @@ -455,7 +461,7 @@ impl MixerBuffer { unsafe { agb_rs__mixer_add( channel.data.as_ptr().add(channel.pos.floor()), - buffer.as_mut_ptr(), + self.working_buffer.as_mut_ptr(), playback_speed, left_amount, right_amount, @@ -471,7 +477,7 @@ impl MixerBuffer { let write_buffer = &mut self.buffers[write_buffer_index].0; unsafe { - agb_rs__mixer_collapse(write_buffer.as_mut_ptr(), buffer.as_ptr()); + agb_rs__mixer_collapse(write_buffer.as_mut_ptr(), self.working_buffer.as_ptr()); } } } From c90e8de893703ac6dcc69eafdbb7c62156db2386 Mon Sep 17 00:00:00 2001 From: Corwin Date: Sat, 8 Oct 2022 11:34:43 +0100 Subject: [PATCH 30/49] clippy autofix --- agb/src/agb_alloc/bump_allocator.rs | 2 +- agb/src/display/object.rs | 8 ++++---- agb/src/display/tiled/vram_manager.rs | 2 +- agb/src/save/flash.rs | 2 +- 4 files changed, 7 insertions(+), 7 deletions(-) diff --git a/agb/src/agb_alloc/bump_allocator.rs b/agb/src/agb_alloc/bump_allocator.rs index 717ba45e..0ea27d2d 100644 --- a/agb/src/agb_alloc/bump_allocator.rs +++ b/agb/src/agb_alloc/bump_allocator.rs @@ -43,7 +43,7 @@ impl BumpAllocator { let resulting_ptr = ptr + amount_to_add; let new_current_ptr = resulting_ptr + layout.size(); - if new_current_ptr as usize >= (self.start_end.borrow(cs).end)() { + if new_current_ptr >= (self.start_end.borrow(cs).end)() { return None; } diff --git a/agb/src/display/object.rs b/agb/src/display/object.rs index 3888a8af..08892e33 100644 --- a/agb/src/display/object.rs +++ b/agb/src/display/object.rs @@ -627,7 +627,7 @@ impl ObjectController { unsafe { (OBJECT_ATTRIBUTE_MEMORY as *mut u16) - .add((i as usize) * 4) + .add(i * 4) .write_volatile(HIDDEN_VALUE); } @@ -794,7 +794,7 @@ impl ObjectController { }); let loan = Loan { - index: index as u8, + index, phantom: PhantomData, }; @@ -916,8 +916,8 @@ impl<'a> Object<'a> { /// [ObjectController::commit] is called. pub fn set_x(&mut self, x: u16) -> &mut Self { let object_inner = unsafe { self.object_inner() }; - object_inner.attrs.a1a.set_x(x.rem_euclid(1 << 9) as u16); - object_inner.attrs.a1s.set_x(x.rem_euclid(1 << 9) as u16); + object_inner.attrs.a1a.set_x(x.rem_euclid(1 << 9)); + object_inner.attrs.a1s.set_x(x.rem_euclid(1 << 9)); self } diff --git a/agb/src/display/tiled/vram_manager.rs b/agb/src/display/tiled/vram_manager.rs index 6caa8956..9bd9cd0e 100644 --- a/agb/src/display/tiled/vram_manager.rs +++ b/agb/src/display/tiled/vram_manager.rs @@ -224,7 +224,7 @@ impl VRamManager { slice::from_raw_parts_mut( tiles .as_mut_ptr() - .add((index * tile_format.tile_size()) as usize) + .add(index * tile_format.tile_size()) .cast(), tile_format.tile_size() / core::mem::size_of::(), ) diff --git a/agb/src/save/flash.rs b/agb/src/save/flash.rs index 0384926b..28b21e50 100644 --- a/agb/src/save/flash.rs +++ b/agb/src/save/flash.rs @@ -51,7 +51,7 @@ fn set_bank(bank: u8) -> Result<(), Error> { Err(Error::OutOfBounds) } else if bank != CURRENT_BANK.read() { issue_flash_command(CMD_SET_BANK); - FLASH_PORT_BANK.set(bank as u8); + FLASH_PORT_BANK.set(bank); CURRENT_BANK.write(bank); Ok(()) } else { From 467ebb024028d2c250904efc96cc7f8f5cfaf2b6 Mon Sep 17 00:00:00 2001 From: Corwin Date: Sat, 8 Oct 2022 11:48:11 +0100 Subject: [PATCH 31/49] cargo clippy fixes in examples --- examples/hyperspace-roll/src/battle.rs | 2 +- examples/the-purple-night/src/main.rs | 12 ++++++------ 2 files changed, 7 insertions(+), 7 deletions(-) diff --git a/examples/hyperspace-roll/src/battle.rs b/examples/hyperspace-roll/src/battle.rs index 3fd3b870..b4f6cf96 100644 --- a/examples/hyperspace-roll/src/battle.rs +++ b/examples/hyperspace-roll/src/battle.rs @@ -207,7 +207,7 @@ impl RolledDice { let heal = *face_counts.entry(Face::Heal).or_default(); if heal != 0 { actions.push(Action::PlayerHeal { - amount: ((heal * (heal + 1)) / 2) as u32, + amount: (heal * (heal + 1)) / 2, }); } diff --git a/examples/the-purple-night/src/main.rs b/examples/the-purple-night/src/main.rs index 07fc59cc..54c9a6f6 100644 --- a/examples/the-purple-night/src/main.rs +++ b/examples/the-purple-night/src/main.rs @@ -129,7 +129,7 @@ impl<'a> Level<'a> { let factor: Number = Number::new(1) / Number::new(8); let (x, y) = (v * factor).floor().get(); - if (x < 0 || x > tilemap::WIDTH as i32) || (y < 0 || y > tilemap::HEIGHT as i32) { + if !(0..=tilemap::WIDTH).contains(&x) || !(0..=tilemap::HEIGHT).contains(&y) { return Some(Rect::new((x * 8, y * 8).into(), (8, 8).into())); } let position = tilemap::WIDTH as usize * y as usize + x as usize; @@ -1878,7 +1878,7 @@ enum MoveState { impl<'a> Game<'a> { fn has_just_reached_end(&self) -> bool { match self.boss { - BossState::NotSpawned => self.offset.x.floor() + 248 >= tilemap::WIDTH as i32 * 8, + BossState::NotSpawned => self.offset.x.floor() + 248 >= tilemap::WIDTH * 8, _ => false, } } @@ -1901,13 +1901,13 @@ impl<'a> Game<'a> { if self.has_just_reached_end() { sfx.boss(); - self.offset.x = (tilemap::WIDTH as i32 * 8 - 248).into(); + self.offset.x = (tilemap::WIDTH * 8 - 248).into(); self.move_state = MoveState::PinnedAtEnd; self.boss = BossState::Active(Boss::new(object_controller, self.offset)) } } MoveState::PinnedAtEnd => { - self.offset.x = (tilemap::WIDTH as i32 * 8 - 248).into(); + self.offset.x = (tilemap::WIDTH * 8 - 248).into(); } MoveState::FollowingPlayer => { Game::update_sunrise(vram, self.sunrise_timer); @@ -1917,8 +1917,8 @@ impl<'a> Game<'a> { let difference = self.player.entity.position.x - (self.offset.x + WIDTH / 2); self.offset.x += difference / 8; - if self.offset.x > (tilemap::WIDTH as i32 * 8 - 248).into() { - self.offset.x = (tilemap::WIDTH as i32 * 8 - 248).into(); + if self.offset.x > (tilemap::WIDTH * 8 - 248).into() { + self.offset.x = (tilemap::WIDTH * 8 - 248).into(); } else if self.offset.x < 8.into() { self.offset.x = 8.into(); self.move_state = MoveState::Ending; From 89af366b7a6e870f060ad090538ddba05e162126 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 16:43:25 +0100 Subject: [PATCH 32/49] Add transparent colour to the image itself --- agb-image-converter/src/colour.rs | 21 ++++++++++++++++++ agb-image-converter/src/config.rs | 33 +++++++++++++++++----------- agb-image-converter/src/palette16.rs | 7 ++++-- 3 files changed, 46 insertions(+), 15 deletions(-) diff --git a/agb-image-converter/src/colour.rs b/agb-image-converter/src/colour.rs index 917b3b99..b401b9bd 100644 --- a/agb-image-converter/src/colour.rs +++ b/agb-image-converter/src/colour.rs @@ -1,3 +1,5 @@ +use std::str::FromStr; + #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)] pub struct Colour { pub r: u8, @@ -20,3 +22,22 @@ impl Colour { self.a != 255 } } + +impl FromStr for Colour { + type Err = String; + + fn from_str(colour: &str) -> Result { + if colour.len() != 6 { + return Err(format!( + "Expected colour to be 6 characters, got {}", + colour + )); + } + + let r = u8::from_str_radix(&colour[0..2], 16).unwrap(); + let g = u8::from_str_radix(&colour[2..4], 16).unwrap(); + let b = u8::from_str_radix(&colour[4..6], 16).unwrap(); + + Ok(Colour::from_rgb(r, g, b, 255)) + } +} diff --git a/agb-image-converter/src/config.rs b/agb-image-converter/src/config.rs index eaa2cd20..075a9f7d 100644 --- a/agb-image-converter/src/config.rs +++ b/agb-image-converter/src/config.rs @@ -23,6 +23,7 @@ pub(crate) fn parse(filename: &str) -> Box { pub(crate) trait Config { fn crate_prefix(&self) -> String; fn images(&self) -> HashMap; + fn transparent_colour(&self) -> Option; } pub(crate) trait Image { @@ -36,6 +37,7 @@ pub(crate) trait Image { pub struct ConfigV1 { version: String, crate_prefix: Option, + transparent_colour: Option, image: HashMap, } @@ -53,6 +55,21 @@ impl Config for ConfigV1 { .map(|(filename, image)| (filename.clone(), image as &dyn Image)) .collect() } + + fn transparent_colour(&self) -> Option { + if let Some(colour) = &self + .transparent_colour + .as_ref() + .map(|colour| colour.parse().unwrap()) + { + return Some(*colour); + } + + self.images() + .values() + .flat_map(|image| image.transparent_colour()) + .next() + } } #[derive(Deserialize)] @@ -69,19 +86,9 @@ impl Image for ImageV1 { } fn transparent_colour(&self) -> Option { - if let Some(colour) = &self.transparent_colour { - if colour.len() != 6 { - panic!("Expected colour to be 6 characters, got {}", colour); - } - - let r = u8::from_str_radix(&colour[0..2], 16).unwrap(); - let g = u8::from_str_radix(&colour[2..4], 16).unwrap(); - let b = u8::from_str_radix(&colour[4..6], 16).unwrap(); - - return Some(Colour::from_rgb(r, g, b, 255)); - } - - None + self.transparent_colour + .as_ref() + .map(|colour| colour.parse().unwrap()) } fn tilesize(&self) -> TileSize { diff --git a/agb-image-converter/src/palette16.rs b/agb-image-converter/src/palette16.rs index afb50c68..2878bad6 100644 --- a/agb-image-converter/src/palette16.rs +++ b/agb-image-converter/src/palette16.rs @@ -58,7 +58,7 @@ impl Palette16 { }) as u8 } - pub fn colours<'a>(&'a self) -> impl Iterator + 'a { + pub fn colours(&self) -> impl Iterator { self.colours.iter() } @@ -167,7 +167,10 @@ impl Palette16Optimiser { fn find_maximal_palette_for(&self, unsatisfied_palettes: &HashSet) -> Palette16 { let mut palette = Palette16::new(); - palette.add_colour(self.transparent_colour.unwrap_or_else(|| Colour::from_rgb(255, 0, 255, 0))); + palette.add_colour( + self.transparent_colour + .unwrap_or_else(|| Colour::from_rgb(255, 0, 255, 0)), + ); loop { let mut colour_usage = vec![0; MAX_COLOURS]; From 1a51b3b50252e4d190340936f09dabb75303fcec Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 16:44:26 +0100 Subject: [PATCH 33/49] Remove the definition of transparent colour from image --- agb-image-converter/src/config.rs | 17 +++++++++-------- 1 file changed, 9 insertions(+), 8 deletions(-) diff --git a/agb-image-converter/src/config.rs b/agb-image-converter/src/config.rs index 075a9f7d..7379a6d5 100644 --- a/agb-image-converter/src/config.rs +++ b/agb-image-converter/src/config.rs @@ -28,7 +28,6 @@ pub(crate) trait Config { pub(crate) trait Image { fn filename(&self) -> String; - fn transparent_colour(&self) -> Option; fn tilesize(&self) -> TileSize; fn colours(&self) -> Colours; } @@ -65,7 +64,7 @@ impl Config for ConfigV1 { return Some(*colour); } - self.images() + self.image .values() .flat_map(|image| image.transparent_colour()) .next() @@ -85,12 +84,6 @@ impl Image for ImageV1 { self.filename.clone() } - fn transparent_colour(&self) -> Option { - self.transparent_colour - .as_ref() - .map(|colour| colour.parse().unwrap()) - } - fn tilesize(&self) -> TileSize { self.tile_size.into() } @@ -104,6 +97,14 @@ impl Image for ImageV1 { } } +impl ImageV1 { + fn transparent_colour(&self) -> Option { + self.transparent_colour + .as_ref() + .map(|colour| colour.parse().unwrap()) + } +} + #[derive(Deserialize, Clone, Copy)] pub enum TileSizeV1 { #[serde(rename = "8x8")] From ea75823900343dad7a6437dfad3697b3ce5e6b32 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 16:45:58 +0100 Subject: [PATCH 34/49] Use the correct transparent colour --- agb-image-converter/src/lib.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/agb-image-converter/src/lib.rs b/agb-image-converter/src/lib.rs index 8d6e7183..2d5506eb 100644 --- a/agb-image-converter/src/lib.rs +++ b/agb-image-converter/src/lib.rs @@ -72,7 +72,7 @@ pub fn include_gfx(input: TokenStream) -> TokenStream { let images = config.images(); - let mut optimiser = Palette16Optimiser::new(None); + let mut optimiser = Palette16Optimiser::new(config.transparent_colour()); let mut assignment_offsets = HashMap::new(); let mut assignment_offset = 0; @@ -93,7 +93,7 @@ pub fn include_gfx(input: TokenStream) -> TokenStream { &mut optimiser, &image, tile_size, - settings.transparent_colour(), + config.transparent_colour(), ); let num_tiles = image.width * image.height / settings.tilesize().to_size().pow(2); From b2e15c8e7b91e39585568998a4c0db6dd0ce1258 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 16:54:55 +0100 Subject: [PATCH 35/49] Fix hyperspace roll using the wrong palette assignments --- examples/hyperspace-roll/src/background.rs | 51 ++++++++++++++-------- 1 file changed, 32 insertions(+), 19 deletions(-) diff --git a/examples/hyperspace-roll/src/background.rs b/examples/hyperspace-roll/src/background.rs index 4fcfd5db..ed83f369 100644 --- a/examples/hyperspace-roll/src/background.rs +++ b/examples/hyperspace-roll/src/background.rs @@ -8,12 +8,7 @@ use crate::sfx::Sfx; include_gfx!("gfx/backgrounds.toml"); pub fn load_palettes(vram: &mut VRamManager) { - vram.set_background_palettes(&[ - backgrounds::stars.palettes[0].clone(), - backgrounds::descriptions1.palettes[0].clone(), - backgrounds::descriptions2.palettes[0].clone(), - backgrounds::help.palettes[0].clone(), - ]); + vram.set_background_palettes(backgrounds::stars.palettes); } pub(crate) fn load_help_text( @@ -28,11 +23,18 @@ pub(crate) fn load_help_text( ); for x in 0..16 { + let tile_id = help_text_line * 16 + x; + background.set_tile( vram, (x + at_tile.0, at_tile.1).into(), &help_tileset, - TileSetting::new(help_text_line * 16 + x, false, false, 3), + TileSetting::new( + tile_id, + false, + false, + backgrounds::help.palette_assignments[tile_id as usize], + ), ) } } @@ -42,25 +44,32 @@ pub(crate) fn load_description( descriptions_map: &mut RegularMap, vram: &mut VRamManager, ) { - let tileset = if face_id < 10 { - TileSet::new( - backgrounds::descriptions1.tiles, - agb::display::tiled::TileFormat::FourBpp, + let (tileset, palette_assignments) = if face_id < 10 { + ( + TileSet::new( + backgrounds::descriptions1.tiles, + agb::display::tiled::TileFormat::FourBpp, + ), + backgrounds::descriptions1.palette_assignments, ) } else { - TileSet::new( - backgrounds::descriptions2.tiles, - agb::display::tiled::TileFormat::FourBpp, + ( + TileSet::new( + backgrounds::descriptions2.tiles, + agb::display::tiled::TileFormat::FourBpp, + ), + backgrounds::descriptions2.palette_assignments, ) }; for y in 0..11 { for x in 0..8 { + let tile_id = y * 8 + x + 8 * 11 * (face_id as u16 - 10); descriptions_map.set_tile( vram, (x, y).into(), &tileset, - TileSetting::new(y * 8 + x + 8 * 11 * (face_id as u16 - 10), false, false, 2), + TileSetting::new(tile_id, false, false, palette_assignments[tile_id as usize]), ) } } @@ -72,12 +81,16 @@ fn create_background_map(map: &mut RegularMap, vram: &mut VRamManager, stars_til for y in 0..32u16 { let blank = rng::gen().rem_euclid(32) < 30; - let tile_id = if blank { - (1 << 10) - 1 + let (tile_id, palette_id) = if blank { + ((1 << 10) - 1, 0) } else { - rng::gen().rem_euclid(64) as u16 + let tile_id = rng::gen().rem_euclid(64) as u16; + ( + tile_id, + backgrounds::stars.palette_assignments[tile_id as usize], + ) }; - let tile_setting = TileSetting::new(tile_id, false, false, 0); + let tile_setting = TileSetting::new(tile_id, false, false, palette_id); map.set_tile(vram, (x, y).into(), stars_tileset, tile_setting); } From c5f6e10889c6f4f92d2f5df7faa30b543cfc7595 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 16:59:40 +0100 Subject: [PATCH 36/49] Update all toml files to the new transparent requirement --- agb/gfx/agb_logo.toml | 2 +- examples/hyperspace-roll/gfx/backgrounds.toml | 5 +---- .../gfx/splash_screens.toml | 4 ++-- .../the-hat-chooses-the-wizard/gfx/tile_sheet.toml | 4 ++-- examples/the-purple-night/gfx/background.toml | 3 ++- examples/the-purple-night/gfx/objects.toml | 11 ----------- 6 files changed, 8 insertions(+), 21 deletions(-) delete mode 100644 examples/the-purple-night/gfx/objects.toml diff --git a/agb/gfx/agb_logo.toml b/agb/gfx/agb_logo.toml index eb1f3304..9ef90025 100644 --- a/agb/gfx/agb_logo.toml +++ b/agb/gfx/agb_logo.toml @@ -2,8 +2,8 @@ version = "1.0" # Only needed for within the agb crate crate_prefix = "crate" +transparent_colour = "010101" [image.test_logo] filename = "test_logo.png" -transparent_colour = "010101" tile_size = "8x8" diff --git a/examples/hyperspace-roll/gfx/backgrounds.toml b/examples/hyperspace-roll/gfx/backgrounds.toml index 21ed1005..6fe2df1e 100644 --- a/examples/hyperspace-roll/gfx/backgrounds.toml +++ b/examples/hyperspace-roll/gfx/backgrounds.toml @@ -1,9 +1,9 @@ version = "1.0" +transparent_colour = "121105" [image.stars] filename = "stars.png" tile_size = "8x8" -transparent_colour = "121105" [image.title] filename = "title-screen.png" @@ -12,14 +12,11 @@ tile_size = "8x8" [image.help] filename = "help-text.png" tile_size = "8x8" -transparent_colour = "121105" [image.descriptions1] filename = "descriptions1.png" tile_size = "8x8" -transparent_colour = "121105" [image.descriptions2] filename = "descriptions2.png" tile_size = "8x8" -transparent_colour = "121105" \ No newline at end of file diff --git a/examples/the-hat-chooses-the-wizard/gfx/splash_screens.toml b/examples/the-hat-chooses-the-wizard/gfx/splash_screens.toml index 30156f38..fea8a855 100644 --- a/examples/the-hat-chooses-the-wizard/gfx/splash_screens.toml +++ b/examples/the-hat-chooses-the-wizard/gfx/splash_screens.toml @@ -1,11 +1,11 @@ version = "1.0" +transparent_colour = "2ce8f4" + [image.thanks_for_playing] filename = "thanks_for_playing.png" tile_size = "8x8" -transparent_colour = "2ce8f4" [image.splash] filename = "splash.png" tile_size = "8x8" -transparent_colour = "2ce8f4" diff --git a/examples/the-hat-chooses-the-wizard/gfx/tile_sheet.toml b/examples/the-hat-chooses-the-wizard/gfx/tile_sheet.toml index 373508d5..4ef16fc0 100644 --- a/examples/the-hat-chooses-the-wizard/gfx/tile_sheet.toml +++ b/examples/the-hat-chooses-the-wizard/gfx/tile_sheet.toml @@ -1,7 +1,7 @@ version = "1.0" +transparent_colour = "2ce8f4" + [image.background] filename = "tile_sheet.png" tile_size = "8x8" -transparent_colour = "2ce8f4" - diff --git a/examples/the-purple-night/gfx/background.toml b/examples/the-purple-night/gfx/background.toml index 0cb20336..c9a8ac22 100644 --- a/examples/the-purple-night/gfx/background.toml +++ b/examples/the-purple-night/gfx/background.toml @@ -1,6 +1,7 @@ version = "1.0" +transparent_colour = "53269a" + [image.background] filename = "background.png" tile_size = "8x8" -transparent_colour = "53269a" diff --git a/examples/the-purple-night/gfx/objects.toml b/examples/the-purple-night/gfx/objects.toml deleted file mode 100644 index 4f2626b5..00000000 --- a/examples/the-purple-night/gfx/objects.toml +++ /dev/null @@ -1,11 +0,0 @@ -version = "1.0" - -[image.objects] -filename = "objects.png" -tile_size = "16x16" -transparent_colour = "53269a" - -[image.boss] -filename = "boss.png" -tile_size = "32x32" -transparent_colour = "53269a" From 829a28af2bc8aa6c8de8556bf038b56bc71a8e83 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 17:02:09 +0100 Subject: [PATCH 37/49] Add the book to the workspace --- .vscode/agb.code-workspace | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.vscode/agb.code-workspace b/.vscode/agb.code-workspace index 376761be..95eb27cb 100644 --- a/.vscode/agb.code-workspace +++ b/.vscode/agb.code-workspace @@ -29,6 +29,9 @@ }, { "path": "../template" + }, + { + "path": "../book" } ] } \ No newline at end of file From faed7b456f2448103089fd50c837027178d6be72 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 17:12:50 +0100 Subject: [PATCH 38/49] Avoid double transforming --- agb/src/display/tiled/map.rs | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/agb/src/display/tiled/map.rs b/agb/src/display/tiled/map.rs index deb37d83..9abe79a8 100644 --- a/agb/src/display/tiled/map.rs +++ b/agb/src/display/tiled/map.rs @@ -245,7 +245,6 @@ pub struct AffineMap { scroll: Vector2D, - transform_origin: Vector2D>, transform: BgAffineSetData, tiles: Vec, @@ -279,7 +278,7 @@ impl TiledMapPrivate for AffineMap { self.size } fn update_bg_registers(&self) { - let register_pos = self.transform.position + self.transform_origin; + let register_pos = self.transform.position; self.bg_x().set(register_pos.x); self.bg_y().set(register_pos.y); self.bg_affine_matrix().set(self.transform.matrix); @@ -307,7 +306,6 @@ impl AffineMap { scroll: Default::default(), - transform_origin: Default::default(), transform: Default::default(), tiles: vec![Default::default(); size.num_tiles()], @@ -353,11 +351,10 @@ impl AffineMap { scale: impl Into>>, rotation: impl Into>, ) { - self.transform_origin = transform_origin.into(); let scale = scale.into(); let rotation = rotation.into(); self.transform = - crate::syscall::bg_affine_matrix(self.transform_origin, self.scroll, scale, rotation); + crate::syscall::bg_affine_matrix(transform_origin.into(), self.scroll, scale, rotation); } fn bg_x(&self) -> MemoryMapped> { From fe2e480f28c036ab1049ca4006ff73e042c91766 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 17:13:25 +0100 Subject: [PATCH 39/49] Remove pointless casts --- agb/src/display/tiled/map.rs | 2 +- agb/src/display/tiled/vram_manager.rs | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/agb/src/display/tiled/map.rs b/agb/src/display/tiled/map.rs index 9abe79a8..c094e966 100644 --- a/agb/src/display/tiled/map.rs +++ b/agb/src/display/tiled/map.rs @@ -40,7 +40,7 @@ trait TiledMapPrivate: TiledMapTypes { unsafe { MemoryMapped::new(0x0400_0008 + 2 * self.background_id()) } } fn screenblock_memory(&self) -> *mut u16 { - (0x0600_0000 + 0x1000 * self.screenblock() as usize / 2) as *mut u16 + (0x0600_0000 + 0x1000 * self.screenblock() / 2) as *mut u16 } } diff --git a/agb/src/display/tiled/vram_manager.rs b/agb/src/display/tiled/vram_manager.rs index 6b3dbf43..3464f54e 100644 --- a/agb/src/display/tiled/vram_manager.rs +++ b/agb/src/display/tiled/vram_manager.rs @@ -333,7 +333,7 @@ impl VRamManager { pub(crate) fn gc(&mut self) { for tile_index in self.indices_to_gc.drain(..) { - let key = tile_index.refcount_key() as usize; + let key = tile_index.refcount_key(); if self.reference_counts[key].current_count() > 0 { continue; // it has since been added back } From c0821c9229ef488260350d9d5a179f54670eb89a Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 17:14:58 +0100 Subject: [PATCH 40/49] Ensure packed(4) --- agb/src/display/object.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/agb/src/display/object.rs b/agb/src/display/object.rs index fba6bd04..f33c6430 100644 --- a/agb/src/display/object.rs +++ b/agb/src/display/object.rs @@ -1198,7 +1198,7 @@ enum ColourMode { /// This can be obtained from X/Y scale and rotation angle with /// [`agb::syscall::affine_matrix`]. #[derive(Copy, Clone, Debug, PartialEq, Eq)] -#[repr(C, packed)] +#[repr(C, packed(4))] pub struct AffineMatrixAttributes { /// Adjustment made to *X* coordinate when drawing *horizontal* lines. /// Also known as "dx". From 870478353cd23c41ffb6c3f04e59d5d5da86fec5 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 21:20:13 +0100 Subject: [PATCH 41/49] Use the correct buffer sizes --- agb/src/sound/mixer/mod.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/agb/src/sound/mixer/mod.rs b/agb/src/sound/mixer/mod.rs index 645ad690..c7fef302 100644 --- a/agb/src/sound/mixer/mod.rs +++ b/agb/src/sound/mixer/mod.rs @@ -181,8 +181,8 @@ impl Frequency { match self { Hz10512 => 176, - Hz18157 => 18157, - Hz32768 => 32768, + Hz18157 => 304, + Hz32768 => 560, } } } From f12502540936d25b1317bfb3a819bac98dc89a45 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 21:24:47 +0100 Subject: [PATCH 42/49] Add changelog entry for removing frequency features --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4db58884..e30b43b3 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -17,6 +17,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Changes - Many of the places that originally disabled IRQs now use the `sync` module, reducing the chance of missed interrupts. - HashMap iterators now implement `size_hint` which should result in slightly better generation of code using those iterators. +- Sound frequency is no longer a crate feature, instead set when initialising the sound mixer. ### Fixed - Fixed the fast magnitude function in agb_fixnum. This is also used in fast_normalise. Previously only worked for positive (x, y). From cc66b658474a04bd9a3f248329d11b93a237fd77 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 21:39:48 +0100 Subject: [PATCH 43/49] Update the docs --- .../display/tiled/infinite_scrolled_map.rs | 4 +- agb/src/sound/mixer/mod.rs | 47 +++++++++---------- agb/src/sound/mixer/sw_mixer.rs | 18 +++---- examples/hyperspace-roll/Cargo.toml | 2 +- examples/hyperspace-roll/src/main.rs | 4 +- .../the-hat-chooses-the-wizard/src/main.rs | 3 +- examples/the-purple-night/Cargo.toml | 2 +- examples/the-purple-night/src/main.rs | 3 +- 8 files changed, 42 insertions(+), 41 deletions(-) diff --git a/agb/src/display/tiled/infinite_scrolled_map.rs b/agb/src/display/tiled/infinite_scrolled_map.rs index 4021494b..391395ab 100644 --- a/agb/src/display/tiled/infinite_scrolled_map.rs +++ b/agb/src/display/tiled/infinite_scrolled_map.rs @@ -157,7 +157,7 @@ impl<'a> InfiniteScrolledMap<'a> { /// # ); /// # /// # let vblank = agb::interrupt::VBlank::get(); - /// # let mut mixer = gba.mixer.mixer(); + /// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// let start_position = agb::fixnum::Vector2D::new(10, 10); /// backdrop.init(&mut vram, start_position, &mut || { /// vblank.wait_for_vblank(); @@ -232,7 +232,7 @@ impl<'a> InfiniteScrolledMap<'a> { /// # ); /// # /// # let vblank = agb::interrupt::VBlank::get(); - /// # let mut mixer = gba.mixer.mixer(); + /// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// let start_position = agb::fixnum::Vector2D::new(10, 10); /// while backdrop.init_partial(&mut vram, start_position) == PartialUpdateStatus::Continue { /// vblank.wait_for_vblank(); diff --git a/agb/src/sound/mixer/mod.rs b/agb/src/sound/mixer/mod.rs index c7fef302..64b82c46 100644 --- a/agb/src/sound/mixer/mod.rs +++ b/agb/src/sound/mixer/mod.rs @@ -14,17 +14,11 @@ //! //! # Concepts //! -//! The mixer runs at a fixed frequency which is determined at compile time by enabling -//! certain features within the crate. The following features are currently available: -//! -//! | Feature | Frequency | -//! |---------|-----------| -//! | none | 10512Hz | -//! | freq18157 | 18157Hz | -//! | freq32768[^32768Hz] | 32768Hz | +//! The mixer runs at a fixed frequency which is determined at initialisation time by +//! passing certain [`Frequency`] options. //! //! All wav files you use within your application / game must use this _exact_ frequency. -//! You will get a compile error if you use the incorrect frequency for your file. +//! If you don't use this frequency, the sound will play either too slowly or too quickly. //! //! The mixer can play both mono and stereo sounds, but only mono sound effects can have //! effects applied to them (such as changing the speed at which they play or the panning). @@ -38,12 +32,17 @@ //! ```rust,no_run //! # #![no_std] //! # #![no_main] +//! use agb::sound::mixer::Frequency; //! # fn foo(gba: &mut agb::Gba) { -//! let mut mixer = gba.mixer.mixer(); +//! let mut mixer = gba.mixer.mixer(Frequency::Hz10512); //! mixer.enable(); //! # } //! ``` //! +//! Pass a frequency option. This option must be used for the entire lifetime of the `mixer` +//! variable. If you want to change frequency, you will need to drop this one and create a new +//! one. +//! //! ## Doing the per-frame work //! //! Then, you have a choice of whether you want to use interrupts or do the buffer swapping @@ -55,9 +54,10 @@ //! ```rust,no_run //! # #![no_std] //! # #![no_main] +//! use agb::sound::mixer::Frequency; //! # fn foo(gba: &mut agb::Gba) { -//! # let mut mixer = gba.mixer.mixer(); -//! # let vblank = agb::interrupt::VBlank::get(); +//! let mut mixer = gba.mixer.mixer(Frequency::Hz10512); +//! let vblank = agb::interrupt::VBlank::get(); //! // Somewhere in your main loop: //! mixer.frame(); //! vblank.wait_for_vblank(); @@ -70,10 +70,13 @@ //! ```rust,no_run //! # #![no_std] //! # #![no_main] +//! use agb::sound::mixer::Frequency; //! # fn foo(gba: &mut agb::Gba) { -//! # let mut mixer = gba.mixer.mixer(); -//! # let vblank = agb::interrupt::VBlank::get(); +//! let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz32768); +//! let vblank = agb::interrupt::VBlank::get(); //! // outside your main loop, close to initialisation +//! // you must assign this to a variable (not to _ or ignored) or rust will immediately drop it +//! // and prevent the interrupt handler from firing. //! let _mixer_interrupt = mixer.setup_interrupt_handler(); //! //! // inside your main loop @@ -99,7 +102,7 @@ //! # #![no_std] //! # #![no_main] //! # fn foo(gba: &mut agb::Gba) { -//! # let mut mixer = gba.mixer.mixer(); +//! # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); //! # let vblank = agb::interrupt::VBlank::get(); //! # use agb::{*, sound::mixer::*}; //! // Outside your main function in global scope: @@ -116,9 +119,6 @@ //! //! Once you have run [`play_sound`](Mixer::play_sound), the mixer will play that sound until //! it has finished. -//! -//! [^32768Hz]: You must use interrupts when using 32768Hz - mod hw; mod sw_mixer; @@ -159,11 +159,10 @@ pub enum Frequency { Hz10512, /// 18157Hz Hz18157, - /// 32768Hz - note that this option requires the timer to do the buffer swapping + /// 32768Hz - note that this option requires interrupts for buffer swapping Hz32768, } -// These work perfectly with swapping the buffers every vblank // list here: http://deku.gbadev.org/program/sound1.html impl Frequency { pub(crate) fn frequency(self) -> i32 { @@ -223,7 +222,7 @@ impl Frequency { /// /// // somewhere in code /// # fn foo(gba: &mut Gba) { -/// # let mut mixer = gba.mixer.mixer(); +/// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// let mut bgm = SoundChannel::new_high_priority(MY_BGM); /// bgm.stereo().should_loop(); /// let _ = mixer.play_sound(bgm); @@ -242,7 +241,7 @@ impl Frequency { /// /// // somewhere in code /// # fn foo(gba: &mut Gba) { -/// # let mut mixer = gba.mixer.mixer(); +/// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// let jump_sound = SoundChannel::new(JUMP_SOUND); /// let _ = mixer.play_sound(jump_sound); /// # } @@ -279,7 +278,7 @@ impl SoundChannel { /// # use agb::sound::mixer::*; /// # use agb::*; /// # fn foo(gba: &mut Gba) { - /// # let mut mixer = gba.mixer.mixer(); + /// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// // in global scope: /// const JUMP_SOUND: &[u8] = include_wav!("examples/sfx/jump.wav"); /// @@ -321,7 +320,7 @@ impl SoundChannel { /// # use agb::sound::mixer::*; /// # use agb::*; /// # fn foo(gba: &mut Gba) { - /// # let mut mixer = gba.mixer.mixer(); + /// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// // in global scope: /// const MY_BGM: &[u8] = include_wav!("examples/sfx/my_bgm.wav"); /// diff --git a/agb/src/sound/mixer/sw_mixer.rs b/agb/src/sound/mixer/sw_mixer.rs index 4edc6107..44c17c4f 100644 --- a/agb/src/sound/mixer/sw_mixer.rs +++ b/agb/src/sound/mixer/sw_mixer.rs @@ -48,7 +48,7 @@ extern "C" { /// # use agb::sound::mixer::*; /// # use agb::*; /// # fn foo(gba: &mut Gba) { -/// let mut mixer = gba.mixer.mixer(); +/// let mut mixer = gba.mixer.mixer(Frequency::Hz10512); /// # } /// ``` /// @@ -60,13 +60,13 @@ extern "C" { /// # use agb::sound::mixer::*; /// # use agb::*; /// # fn foo(gba: &mut Gba) { -/// # let mut mixer = gba.mixer.mixer(); +/// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// # let vblank = agb::interrupt::VBlank::get(); /// // Outside your main function in global scope: /// const MY_CRAZY_SOUND: &[u8] = include_wav!("examples/sfx/jump.wav"); /// /// // in your main function: -/// let mut mixer = gba.mixer.mixer(); +/// let mut mixer = gba.mixer.mixer(Frequency::Hz10512); /// let mut channel = SoundChannel::new(MY_CRAZY_SOUND); /// channel.stereo(); /// let _ = mixer.play_sound(channel); @@ -99,7 +99,7 @@ pub struct Mixer { /// # use agb::sound::mixer::*; /// # use agb::*; /// # fn foo(gba: &mut Gba) { -/// # let mut mixer = gba.mixer.mixer(); +/// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// # const MY_BGM: &[u8] = include_wav!("examples/sfx/my_bgm.wav"); /// let mut channel = SoundChannel::new_high_priority(MY_BGM); /// let bgm_channel_id = mixer.play_sound(channel).unwrap(); // will always be Some if high priority @@ -142,7 +142,7 @@ impl Mixer { /// # use agb::sound::mixer::*; /// # use agb::*; /// # fn foo(gba: &mut Gba) { - /// # let mut mixer = gba.mixer.mixer(); + /// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// # let vblank = agb::interrupt::VBlank::get(); /// loop { /// mixer.frame(); @@ -171,7 +171,7 @@ impl Mixer { /// # use agb::sound::mixer::*; /// # use agb::*; /// # fn foo(gba: &mut Gba) { - /// # let mut mixer = gba.mixer.mixer(); + /// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// # let vblank = agb::interrupt::VBlank::get(); /// // you must set this to a named variable to ensure that the scope is long enough /// let _mixer_interrupt = mixer.setup_interrupt_handler(); @@ -209,7 +209,7 @@ impl Mixer { /// # use agb::sound::mixer::*; /// # use agb::*; /// # fn foo(gba: &mut Gba) { - /// # let mut mixer = gba.mixer.mixer(); + /// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// # let vblank = agb::interrupt::VBlank::get(); /// loop { /// mixer.frame(); @@ -248,7 +248,7 @@ impl Mixer { /// # use agb::sound::mixer::*; /// # use agb::*; /// # fn foo(gba: &mut Gba) { - /// # let mut mixer = gba.mixer.mixer(); + /// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// # const MY_BGM: &[u8] = include_wav!("examples/sfx/my_bgm.wav"); /// let mut channel = SoundChannel::new_high_priority(MY_BGM); /// let bgm_channel_id = mixer.play_sound(channel).unwrap(); // will always be Some if high priority @@ -297,7 +297,7 @@ impl Mixer { /// # use agb::sound::mixer::*; /// # use agb::*; /// # fn foo(gba: &mut Gba) { - /// # let mut mixer = gba.mixer.mixer(); + /// # let mut mixer = gba.mixer.mixer(agb::sound::mixer::Frequency::Hz10512); /// # const MY_BGM: &[u8] = include_wav!("examples/sfx/my_bgm.wav"); /// let mut channel = SoundChannel::new_high_priority(MY_BGM); /// let bgm_channel_id = mixer.play_sound(channel).unwrap(); // will always be Some if high priority diff --git a/examples/hyperspace-roll/Cargo.toml b/examples/hyperspace-roll/Cargo.toml index 76c3f4bc..317e2525 100644 --- a/examples/hyperspace-roll/Cargo.toml +++ b/examples/hyperspace-roll/Cargo.toml @@ -7,7 +7,7 @@ edition = "2018" # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [dependencies] -agb = { version = "0.11.1", path = "../../agb", features = ["freq32768"] } +agb = { version = "0.11.1", path = "../../agb" } bare-metal = "1" [profile.dev] diff --git a/examples/hyperspace-roll/src/main.rs b/examples/hyperspace-roll/src/main.rs index 9ec4cc1a..3df47f7d 100644 --- a/examples/hyperspace-roll/src/main.rs +++ b/examples/hyperspace-roll/src/main.rs @@ -10,11 +10,11 @@ // which won't be a particularly clear error message. #![no_main] -use agb::display; use agb::display::object::ObjectController; use agb::display::tiled::VRamManager; use agb::display::Priority; use agb::interrupt::VBlank; +use agb::{display, sound::mixer::Frequency}; extern crate alloc; use alloc::vec; @@ -138,7 +138,7 @@ fn main(mut gba: agb::Gba) -> ! { let mut star_background = StarBackground::new(&mut background0, &mut background1, &mut vram); star_background.commit(&mut vram); - let mut mixer = gba.mixer.mixer(); + let mut mixer = gba.mixer.mixer(Frequency::Hz32768); mixer.enable(); let _interrupt_handler = mixer.setup_interrupt_handler(); diff --git a/examples/the-hat-chooses-the-wizard/src/main.rs b/examples/the-hat-chooses-the-wizard/src/main.rs index 133afb4e..eeffbe1d 100644 --- a/examples/the-hat-chooses-the-wizard/src/main.rs +++ b/examples/the-hat-chooses-the-wizard/src/main.rs @@ -17,6 +17,7 @@ use agb::{ }, fixnum::{FixedNum, Vector2D}, input::{self, Button, ButtonController}, + sound::mixer::Frequency, }; use alloc::boxed::Box; @@ -820,7 +821,7 @@ pub fn main(mut agb: agb::Gba) -> ! { vram.set_background_palettes(tile_sheet::background.palettes); let object = agb.display.object.get(); - let mut mixer = agb.mixer.mixer(); + let mut mixer = agb.mixer.mixer(Frequency::Hz10512); mixer.enable(); let mut music_box = sfx::MusicBox::new(); diff --git a/examples/the-purple-night/Cargo.toml b/examples/the-purple-night/Cargo.toml index 33f2e83d..eeac7e0f 100644 --- a/examples/the-purple-night/Cargo.toml +++ b/examples/the-purple-night/Cargo.toml @@ -7,7 +7,7 @@ edition = "2018" # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [dependencies] -agb = { path = "../../agb", version = "0.11.1", features = ["freq18157"]} +agb = { path = "../../agb", version = "0.11.1" } generational-arena = { version = "0.2", default-features = false } [build-dependencies] diff --git a/examples/the-purple-night/src/main.rs b/examples/the-purple-night/src/main.rs index 07fc59cc..8ca239ef 100644 --- a/examples/the-purple-night/src/main.rs +++ b/examples/the-purple-night/src/main.rs @@ -22,6 +22,7 @@ use agb::{ input::{Button, ButtonController, Tri}, interrupt::VBlank, rng, + sound::mixer::Frequency, }; use generational_arena::Arena; use sfx::Sfx; @@ -2205,7 +2206,7 @@ fn game_with_level(gba: &mut agb::Gba) { let vblank = agb::interrupt::VBlank::get(); vblank.wait_for_vblank(); - let mut mixer = gba.mixer.mixer(); + let mut mixer = gba.mixer.mixer(Frequency::Hz18157); mixer.enable(); let mut sfx = sfx::Sfx::new(&mut mixer); From 5c578df227e34cd47f0e0788d9cbbb5858d0c3fa Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 22:34:54 +0100 Subject: [PATCH 44/49] transparent colour definition isn't really needed here --- agb/gfx/agb_logo.toml | 1 - 1 file changed, 1 deletion(-) diff --git a/agb/gfx/agb_logo.toml b/agb/gfx/agb_logo.toml index 9ef90025..60787767 100644 --- a/agb/gfx/agb_logo.toml +++ b/agb/gfx/agb_logo.toml @@ -2,7 +2,6 @@ version = "1.0" # Only needed for within the agb crate crate_prefix = "crate" -transparent_colour = "010101" [image.test_logo] filename = "test_logo.png" From 9bddbdba1cde5e1d3c3c6f9a71b5f21a2479e450 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 22:40:20 +0100 Subject: [PATCH 45/49] Seal the TiledMap trait --- agb/src/display/tiled/map.rs | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/agb/src/display/tiled/map.rs b/agb/src/display/tiled/map.rs index c094e966..bfe3399e 100644 --- a/agb/src/display/tiled/map.rs +++ b/agb/src/display/tiled/map.rs @@ -15,7 +15,7 @@ use super::{ use crate::syscall::BgAffineSetData; use alloc::{vec, vec::Vec}; -pub trait TiledMapTypes { +pub trait TiledMapTypes: private::Sealed { type Size: BackgroundSize + Copy; } @@ -44,6 +44,8 @@ trait TiledMapPrivate: TiledMapTypes { } } +/// Trait which describes methods available on both tiled maps and affine maps. Note that +/// it is 'sealed' so you cannot implement this yourself. pub trait TiledMap: TiledMapTypes { fn clear(&mut self, vram: &mut VRamManager); fn show(&mut self); @@ -429,3 +431,10 @@ impl<'a, T> Drop for MapLoan<'a, T> { } } } + +mod private { + pub trait Sealed {} + + impl Sealed for super::RegularMap {} + impl Sealed for super::AffineMap {} +} From a5c86e8ce8ed851ed7f9e024c78c5a56db4c307c Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 22:44:37 +0100 Subject: [PATCH 46/49] Add changelog entries --- CHANGELOG.md | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4db58884..8c505068 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -13,10 +13,14 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Added a new agb::sync module that contains GBA-specific synchronization primitives. - Added support for save files. - Added implementation of `HashMap.retain()`. +- Added support for affine backgrounds (tiled modes 1 and 2) which allows for scaling, rotating etc of tiled backgrounds. +- Added support for 256 colour backgrounds (when working with affine ones). ### Changes - Many of the places that originally disabled IRQs now use the `sync` module, reducing the chance of missed interrupts. - HashMap iterators now implement `size_hint` which should result in slightly better generation of code using those iterators. +- Transparency of backgrounds is now set once in the toml file rather than once for every image. +- Palette generation now takes into account every single background a toml definition rather than one at a time. ### Fixed - Fixed the fast magnitude function in agb_fixnum. This is also used in fast_normalise. Previously only worked for positive (x, y). From 1e759dbeca922d9685eb7ea784a23edc1df83957 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 23:02:54 +0100 Subject: [PATCH 47/49] Move the palettes to a single constant rather than being repeated --- CHANGELOG.md | 2 +- agb-image-converter/src/lib.rs | 5 +++ agb-image-converter/src/rust_generator.rs | 35 +++++++++++-------- agb/examples/affine_background.rs | 2 +- agb/examples/animated_background.rs | 2 +- agb/src/display/example_logo.rs | 2 +- agb/src/display/tile_data.rs | 11 ++---- examples/hyperspace-roll/src/background.rs | 4 +-- .../the-hat-chooses-the-wizard/src/main.rs | 4 +-- .../src/splash_screen.rs | 21 ++++------- examples/the-purple-night/src/main.rs | 6 ++-- 11 files changed, 46 insertions(+), 48 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 8c505068..616deb65 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -20,7 +20,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Many of the places that originally disabled IRQs now use the `sync` module, reducing the chance of missed interrupts. - HashMap iterators now implement `size_hint` which should result in slightly better generation of code using those iterators. - Transparency of backgrounds is now set once in the toml file rather than once for every image. -- Palette generation now takes into account every single background a toml definition rather than one at a time. +- Palette generation now takes into account every single background a toml definition rather than one at a time, you can now find it in the PALETTES constant rather than in every individual image. ### Fixed - Fixed the fast magnitude function in agb_fixnum. This is also used in fast_normalise. Previously only worked for positive (x, y). diff --git a/agb-image-converter/src/lib.rs b/agb-image-converter/src/lib.rs index 2d5506eb..cf89fa6d 100644 --- a/agb-image-converter/src/lib.rs +++ b/agb-image-converter/src/lib.rs @@ -127,10 +127,15 @@ pub fn include_gfx(input: TokenStream) -> TokenStream { )); } + let palette_code = + rust_generator::generate_palette_code(&optimisation_results, &config.crate_prefix()); + let module = quote! { mod #module_name { const _: &[u8] = include_bytes!(#include_path); + #palette_code + #(#image_code)* } }; diff --git a/agb-image-converter/src/rust_generator.rs b/agb-image-converter/src/rust_generator.rs index 3e70466c..e15cae6d 100644 --- a/agb-image-converter/src/rust_generator.rs +++ b/agb-image-converter/src/rust_generator.rs @@ -7,19 +7,13 @@ use quote::{format_ident, quote}; use std::iter; -pub(crate) fn generate_code( - output_variable_name: &str, +pub(crate) fn generate_palette_code( results: &Palette16OptimisationResults, - image: &Image, - image_filename: &str, - tile_size: TileSize, - crate_prefix: String, - assignment_offset: Option, + crate_prefix: &str, ) -> TokenStream { let crate_prefix = format_ident!("{}", crate_prefix); - let output_variable_name = format_ident!("{}", output_variable_name); - let palette_data = results.optimised_palettes.iter().map(|palette| { + let palettes = results.optimised_palettes.iter().map(|palette| { let colours = palette .clone() .into_iter() @@ -35,6 +29,23 @@ pub(crate) fn generate_code( } }); + quote! { + pub const PALETTES: &[#crate_prefix::display::palette16::Palette16] = &[#(#palettes),*]; + } +} + +pub(crate) fn generate_code( + output_variable_name: &str, + results: &Palette16OptimisationResults, + image: &Image, + image_filename: &str, + tile_size: TileSize, + crate_prefix: String, + assignment_offset: Option, +) -> TokenStream { + let crate_prefix = format_ident!("{}", crate_prefix); + let output_variable_name = format_ident!("{}", output_variable_name); + let (tile_data, assignments) = if let Some(assignment_offset) = assignment_offset { let mut tile_data = Vec::new(); @@ -68,17 +79,13 @@ pub(crate) fn generate_code( pub const #output_variable_name: #crate_prefix::display::tile_data::TileData = { const _: &[u8] = include_bytes!(#image_filename); - const PALETTE_DATA: &[#crate_prefix::display::palette16::Palette16] = &[ - #(#palette_data),* - ]; - const TILE_DATA: &[u8] = #data; const PALETTE_ASSIGNMENT: &[u8] = &[ #(#assignments),* ]; - #crate_prefix::display::tile_data::TileData::new(PALETTE_DATA, TILE_DATA, PALETTE_ASSIGNMENT) + #crate_prefix::display::tile_data::TileData::new(TILE_DATA, PALETTE_ASSIGNMENT) }; } } diff --git a/agb/examples/affine_background.rs b/agb/examples/affine_background.rs index 175145f5..7d783c25 100644 --- a/agb/examples/affine_background.rs +++ b/agb/examples/affine_background.rs @@ -19,7 +19,7 @@ fn main(mut gba: agb::Gba) -> ! { let tileset = TileSet::new(affine_tiles::water_tiles.tiles, TileFormat::EightBpp); - vram.set_background_palettes(affine_tiles::water_tiles.palettes); + vram.set_background_palettes(affine_tiles::PALETTES); let mut bg = gfx.background(Priority::P0, AffineBackgroundSize::Background32x32); diff --git a/agb/examples/animated_background.rs b/agb/examples/animated_background.rs index 97ccf3cd..6d547ac0 100644 --- a/agb/examples/animated_background.rs +++ b/agb/examples/animated_background.rs @@ -18,7 +18,7 @@ fn main(mut gba: agb::Gba) -> ! { let tileset = TileSet::new(water_tiles::water_tiles.tiles, TileFormat::FourBpp); - vram.set_background_palettes(water_tiles::water_tiles.palettes); + vram.set_background_palettes(water_tiles::PALETTES); let mut bg = gfx.background(Priority::P0, RegularBackgroundSize::Background32x32); diff --git a/agb/src/display/example_logo.rs b/agb/src/display/example_logo.rs index fa4397d8..c6f57679 100644 --- a/agb/src/display/example_logo.rs +++ b/agb/src/display/example_logo.rs @@ -3,7 +3,7 @@ use super::tiled::{RegularMap, TileFormat, TileSet, TileSetting, TiledMap, VRamM crate::include_gfx!("gfx/agb_logo.toml"); pub fn display_logo(map: &mut RegularMap, vram: &mut VRamManager) { - vram.set_background_palettes(agb_logo::test_logo.palettes); + vram.set_background_palettes(agb_logo::PALETTES); let background_tilemap = TileSet::new(agb_logo::test_logo.tiles, TileFormat::FourBpp); diff --git a/agb/src/display/tile_data.rs b/agb/src/display/tile_data.rs index 364b57c2..cc51ef9b 100644 --- a/agb/src/display/tile_data.rs +++ b/agb/src/display/tile_data.rs @@ -1,20 +1,13 @@ -use crate::display::palette16::Palette16; - +#[non_exhaustive] pub struct TileData { - pub palettes: &'static [Palette16], pub tiles: &'static [u8], pub palette_assignments: &'static [u8], } impl TileData { #[must_use] - pub const fn new( - palettes: &'static [Palette16], - tiles: &'static [u8], - palette_assignments: &'static [u8], - ) -> Self { + pub const fn new(tiles: &'static [u8], palette_assignments: &'static [u8]) -> Self { TileData { - palettes, tiles, palette_assignments, } diff --git a/examples/hyperspace-roll/src/background.rs b/examples/hyperspace-roll/src/background.rs index ed83f369..3c3d6931 100644 --- a/examples/hyperspace-roll/src/background.rs +++ b/examples/hyperspace-roll/src/background.rs @@ -8,7 +8,7 @@ use crate::sfx::Sfx; include_gfx!("gfx/backgrounds.toml"); pub fn load_palettes(vram: &mut VRamManager) { - vram.set_background_palettes(backgrounds::stars.palettes); + vram.set_background_palettes(backgrounds::PALETTES); } pub(crate) fn load_help_text( @@ -101,7 +101,7 @@ fn create_background_map(map: &mut RegularMap, vram: &mut VRamManager, stars_til pub fn show_title_screen(background: &mut RegularMap, vram: &mut VRamManager, sfx: &mut Sfx) { background.set_scroll_pos((0i16, 0).into()); - vram.set_background_palettes(backgrounds::title.palettes); + vram.set_background_palettes(backgrounds::PALETTES); let tile_set = TileSet::new( backgrounds::title.tiles, agb::display::tiled::TileFormat::FourBpp, diff --git a/examples/the-hat-chooses-the-wizard/src/main.rs b/examples/the-hat-chooses-the-wizard/src/main.rs index 031b5cd7..e5592fa3 100644 --- a/examples/the-hat-chooses-the-wizard/src/main.rs +++ b/examples/the-hat-chooses-the-wizard/src/main.rs @@ -785,7 +785,7 @@ fn agb_main(mut gba: agb::Gba) -> ! { pub fn main(mut agb: agb::Gba) -> ! { let (tiled, mut vram) = agb.display.video.tiled0(); - vram.set_background_palettes(tile_sheet::background.palettes); + vram.set_background_palettes(tile_sheet::PALETTES); let mut splash_screen = tiled.background(Priority::P0, RegularBackgroundSize::Background32x32); let mut world_display = tiled.background(Priority::P0, RegularBackgroundSize::Background32x32); @@ -817,7 +817,7 @@ pub fn main(mut agb: agb::Gba) -> ! { world_display.commit(&mut vram); world_display.show(); - vram.set_background_palettes(tile_sheet::background.palettes); + vram.set_background_palettes(tile_sheet::PALETTES); let object = agb.display.object.get(); let mut mixer = agb.mixer.mixer(); diff --git a/examples/the-hat-chooses-the-wizard/src/splash_screen.rs b/examples/the-hat-chooses-the-wizard/src/splash_screen.rs index 9af2e1fc..7be40dda 100644 --- a/examples/the-hat-chooses-the-wizard/src/splash_screen.rs +++ b/examples/the-hat-chooses-the-wizard/src/splash_screen.rs @@ -19,20 +19,13 @@ pub fn show_splash_screen( vram: &mut VRamManager, ) { map.set_scroll_pos((0i16, 0i16).into()); - let (tileset, palette) = match which { - SplashScreen::Start => { - let tileset = TileSet::new(splash_screens::splash.tiles, TileFormat::FourBpp); + let tileset = match which { + SplashScreen::Start => TileSet::new(splash_screens::splash.tiles, TileFormat::FourBpp), - (tileset, splash_screens::splash.palettes) - } - SplashScreen::End => { - let tileset = TileSet::new( - splash_screens::thanks_for_playing.tiles, - TileFormat::FourBpp, - ); - - (tileset, splash_screens::thanks_for_playing.palettes) - } + SplashScreen::End => TileSet::new( + splash_screens::thanks_for_playing.tiles, + TileFormat::FourBpp, + ), }; let vblank = agb::interrupt::VBlank::get(); @@ -77,7 +70,7 @@ pub fn show_splash_screen( } map.commit(vram); - vram.set_background_palettes(palette); + vram.set_background_palettes(splash_screens::PALETTES); map.show(); loop { diff --git a/examples/the-purple-night/src/main.rs b/examples/the-purple-night/src/main.rs index 54c9a6f6..ce906f9a 100644 --- a/examples/the-purple-night/src/main.rs +++ b/examples/the-purple-night/src/main.rs @@ -2146,7 +2146,7 @@ impl<'a> Game<'a> { } fn update_sunrise(vram: &mut VRamManager, time: u16) { - let mut modified_palette = background::background.palettes[0].clone(); + let mut modified_palette = background::PALETTES[0].clone(); let a = modified_palette.colour(0); let b = modified_palette.colour(1); @@ -2160,7 +2160,7 @@ impl<'a> Game<'a> { } fn update_fade_out(vram: &mut VRamManager, time: u16) { - let mut modified_palette = background::background.palettes[0].clone(); + let mut modified_palette = background::PALETTES[0].clone(); let c = modified_palette.colour(2); @@ -2216,7 +2216,7 @@ fn game_with_level(gba: &mut agb::Gba) { loop { let (background, mut vram) = gba.display.video.tiled0(); - vram.set_background_palettes(background::background.palettes); + vram.set_background_palettes(background::PALETTES); let tileset = TileSet::new(background::background.tiles, TileFormat::FourBpp); From a09be563192fa8c243ac93b4cf05d75e4c5d62dc Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 23:07:48 +0100 Subject: [PATCH 48/49] Fix example --- agb/examples/mixer_32768.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/agb/examples/mixer_32768.rs b/agb/examples/mixer_32768.rs index 8dd38120..2d9f60d7 100644 --- a/agb/examples/mixer_32768.rs +++ b/agb/examples/mixer_32768.rs @@ -3,7 +3,7 @@ use agb::{ display::{ - tiled::{RegularBackgroundSize, RegularMap, TileSetting, VRamManager}, + tiled::{RegularBackgroundSize, RegularMap, TileSetting, TiledMap, VRamManager}, Font, Priority, }, include_font, include_wav, From f921da930a6712424f6712402fa352e9c03ca3f1 Mon Sep 17 00:00:00 2001 From: Gwilym Kuiper Date: Sat, 8 Oct 2022 23:15:54 +0100 Subject: [PATCH 49/49] Fix doc comment --- agb/src/lib.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/agb/src/lib.rs b/agb/src/lib.rs index d8f221b1..693717f9 100644 --- a/agb/src/lib.rs +++ b/agb/src/lib.rs @@ -92,7 +92,7 @@ /// # fn load_tileset(mut gfx: Tiled0, mut vram: VRamManager) { /// let tileset = TileSet::new(water_tiles::water_tiles.tiles, TileFormat::FourBpp); /// -/// vram.set_background_palettes(water_tiles::water_tiles.palettes); +/// vram.set_background_palettes(water_tiles::PALETTES); /// /// let mut bg = gfx.background(Priority::P0, RegularBackgroundSize::Background32x32); ///