diff --git a/agb/src/agbabi/memcpy.s b/agb/src/agbabi/memcpy.s index 0c7412b5..8f6b6018 100644 --- a/agb/src/agbabi/memcpy.s +++ b/agb/src/agbabi/memcpy.s @@ -29,6 +29,10 @@ __aeabi_memcpy: bmi .Lcopy1 bcs .Lcopy2 + // If the number of bytes to copy is less than 4, we should just copy one byte at a time + cmp r2, #4 + bmi .Lcopy1 + .Lcopy4: // Copy half and byte head rsb r3, r0, #4 diff --git a/agb/src/agbabi/mod.rs b/agb/src/agbabi/mod.rs new file mode 100644 index 00000000..65f74c4f --- /dev/null +++ b/agb/src/agbabi/mod.rs @@ -0,0 +1,153 @@ +#[cfg(test)] +mod test { + mod memcpy { + use alloc::vec; + + use crate::Gba; + + extern "C" { + fn __agbabi_memcpy(dest: *mut u8, src: *const u8, n: usize); + fn __aeabi_memcpy4(dest: *mut u32, src: *const u32, n: usize); + } + + #[test_case] + fn test_memcpy4_with_different_sizes(_gba: &mut Gba) { + let mut input = vec![0u32; 70]; + let mut output = vec![0u32; 70]; + + for size in 0..68 { + for (i, value) in input.iter_mut().enumerate() { + *value = i as u32 * 6; + } + + output.fill(0); + + unsafe { + __aeabi_memcpy4(output.as_mut_ptr(), input.as_ptr(), size * 4); + } + + for i in 0..size { + assert_eq!(input[i], output[i], "Failed with size = {size} at i = {i}"); + } + + for (i, value) in output.iter().enumerate().skip(size) { + assert_eq!(*value, 0, "overrun with size = {size} at i = {i}"); + } + } + } + + #[test_case] + fn test_memcpy_bytes_with_different_sizes(_gba: &mut Gba) { + let mut input = vec![0u8; 70]; + let mut output = vec![0u8; 70]; + + for size in 0..68 { + for (i, value) in input.iter_mut().enumerate() { + *value = i as u8 * 3; + } + + output.fill(0); + + unsafe { + __agbabi_memcpy(output.as_mut_ptr(), input.as_ptr(), size); + } + + for i in 0..size { + assert_eq!(input[i], output[i], "Failed with size = {size} at i = {i}"); + } + + for (i, value) in output.iter().enumerate().skip(size) { + assert_eq!(*value, 0, "overrun with size = {size} at i = {i}"); + } + } + } + + #[test_case] + fn test_memcpy_bytes_output_offsetted_with_different_sizes(_gba: &mut Gba) { + let mut input = vec![0u8; 70]; + let mut output = vec![0u8; 70]; + + for size in 0..60 { + for (i, value) in input.iter_mut().enumerate() { + *value = i as u8 * 3; + } + + output.fill(0); + + unsafe { + __agbabi_memcpy(output.as_mut_ptr().add(1), input.as_ptr(), size); + } + + for i in 0..size { + assert_eq!( + input[i], + output[i + 1], + "Failed with size = {size} at i = {i}" + ); + } + + for (i, value) in output.iter().enumerate().skip(size + 1) { + assert_eq!(*value, 0, "overrun with size = {size} at i = {i}"); + } + } + } + + #[test_case] + fn test_memcpy_bytes_input_offsetted_with_different_sizes(_gba: &mut Gba) { + let mut input = vec![0u8; 70]; + let mut output = vec![0u8; 70]; + + for size in 0..60 { + for (i, value) in input.iter_mut().enumerate() { + *value = i as u8 * 3; + } + + output.fill(0); + + unsafe { + __agbabi_memcpy(output.as_mut_ptr(), input.as_ptr().add(1), size); + } + + for i in 0..size { + assert_eq!( + input[i + 1], + output[i], + "Failed with size = {size} at i = {i}" + ); + } + + for (i, value) in output.iter().enumerate().skip(size) { + assert_eq!(*value, 0, "overrun with size = {size} at i = {i}"); + } + } + } + + #[test_case] + fn test_memcpy_bytes_input_output_offsetted_with_different_sizes(_gba: &mut Gba) { + let mut input = vec![0u8; 70]; + let mut output = vec![0u8; 70]; + + for size in 0..60 { + for (i, value) in input.iter_mut().enumerate() { + *value = i as u8 * 3; + } + + output.fill(0); + + unsafe { + __agbabi_memcpy(output.as_mut_ptr().add(1), input.as_ptr().add(1), size); + } + + assert_eq!(output[0], 0); + + for i in 1..size + 1 { + assert_eq!(input[i], output[i], "Failed with size = {size} at i = {i}"); + } + + for (i, value) in output.iter().enumerate().skip(size + 1) { + assert_eq!(*value, 0, "overrun with size = {size} at i = {i}"); + } + } + } + } +} diff --git a/agb/src/lib.rs b/agb/src/lib.rs index 1ac38a4d..a45d570c 100644 --- a/agb/src/lib.rs +++ b/agb/src/lib.rs @@ -157,6 +157,7 @@ pub use agb_sound_converter::include_wav; extern crate alloc; mod agb_alloc; +mod agbabi; mod bitarray; /// Implements everything relating to things that are displayed on screen. pub mod display;