2021-04-19 22:21:44 +01:00
|
|
|
#![allow(clippy::all)]
|
|
|
|
|
2022-11-30 21:17:26 +00:00
|
|
|
mod bindings;
|
2021-04-19 22:21:44 +01:00
|
|
|
mod runner;
|
2022-11-30 21:17:26 +00:00
|
|
|
|
2021-04-19 22:21:44 +01:00
|
|
|
use anyhow::{anyhow, Error};
|
2021-06-04 10:30:18 +01:00
|
|
|
use image::io::Reader;
|
2022-04-23 15:33:57 +01:00
|
|
|
use image::GenericImage;
|
2021-04-19 22:21:44 +01:00
|
|
|
use io::Write;
|
|
|
|
use regex::Regex;
|
2021-06-04 10:30:18 +01:00
|
|
|
use runner::VideoBuffer;
|
2023-04-25 20:34:47 +01:00
|
|
|
use std::cell::Cell;
|
2021-04-19 22:21:44 +01:00
|
|
|
use std::io;
|
|
|
|
use std::path::Path;
|
2023-04-25 20:34:47 +01:00
|
|
|
use std::rc::Rc;
|
2021-04-19 22:21:44 +01:00
|
|
|
|
2023-04-25 20:34:47 +01:00
|
|
|
#[derive(PartialEq, Eq, Debug, Clone, Copy)]
|
2021-04-19 22:21:44 +01:00
|
|
|
enum Status {
|
|
|
|
Running,
|
|
|
|
Failed,
|
2022-11-17 21:25:06 +00:00
|
|
|
Success,
|
2021-04-19 22:21:44 +01:00
|
|
|
}
|
|
|
|
|
2023-04-25 20:34:47 +01:00
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
|
2021-07-03 22:19:10 +01:00
|
|
|
enum Timing {
|
|
|
|
None,
|
|
|
|
WaitFor(i32),
|
|
|
|
Difference(i32),
|
|
|
|
}
|
|
|
|
|
|
|
|
const TEST_RUNNER_TAG: u16 = 785;
|
|
|
|
|
2021-04-19 22:21:44 +01:00
|
|
|
fn test_file(file_to_run: &str) -> Status {
|
2023-04-25 20:34:47 +01:00
|
|
|
let finished = Rc::new(Cell::new(Status::Running));
|
2021-06-04 19:15:16 +01:00
|
|
|
let debug_reader_mutex = Regex::new(r"(?s)^\[(.*)\] GBA Debug: (.*)$").unwrap();
|
2021-07-03 22:19:10 +01:00
|
|
|
let tagged_cycles_reader = Regex::new(r"Cycles: (\d*) Tag: (\d*)").unwrap();
|
2021-04-19 22:21:44 +01:00
|
|
|
|
2021-07-03 16:33:26 +00:00
|
|
|
let mut mgba = runner::MGBA::new(file_to_run).unwrap();
|
2023-04-25 20:34:47 +01:00
|
|
|
|
|
|
|
{
|
|
|
|
let finished = finished.clone();
|
|
|
|
let video_buffer = mgba.get_video_buffer();
|
|
|
|
let number_of_cycles = Cell::new(Timing::None);
|
|
|
|
|
|
|
|
mgba.set_logger(move |message| {
|
|
|
|
if let Some(captures) = debug_reader_mutex.captures(message) {
|
|
|
|
let log_level = &captures[1];
|
|
|
|
let out = &captures[2];
|
|
|
|
|
|
|
|
if out.starts_with("image:") {
|
|
|
|
let image_path = out.strip_prefix("image:").unwrap();
|
|
|
|
match check_image_match(image_path, &video_buffer) {
|
|
|
|
Err(e) => {
|
|
|
|
println!("[failed]");
|
|
|
|
println!("{}", e);
|
|
|
|
finished.set(Status::Failed);
|
|
|
|
}
|
|
|
|
Ok(_) => {}
|
2021-06-04 10:30:18 +01:00
|
|
|
}
|
2023-04-25 20:34:47 +01:00
|
|
|
} else if out.ends_with("...") {
|
|
|
|
print!("{}", out);
|
|
|
|
io::stdout().flush().expect("can't flush stdout");
|
|
|
|
} else if out.starts_with("Cycles: ") {
|
|
|
|
if let Some(captures) = tagged_cycles_reader.captures(out) {
|
|
|
|
let num_cycles: i32 = captures[1].parse().unwrap();
|
|
|
|
let tag: u16 = captures[2].parse().unwrap();
|
|
|
|
|
|
|
|
if tag == TEST_RUNNER_TAG {
|
|
|
|
number_of_cycles.set(match number_of_cycles.get() {
|
|
|
|
Timing::WaitFor(n) => Timing::Difference(num_cycles - n),
|
|
|
|
Timing::None => Timing::WaitFor(num_cycles),
|
|
|
|
Timing::Difference(_) => Timing::WaitFor(num_cycles),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if out == "[ok]" {
|
|
|
|
if let Timing::Difference(cycles) = number_of_cycles.get() {
|
|
|
|
println!(
|
|
|
|
"[ok: {} c ≈ {} s]",
|
|
|
|
cycles,
|
|
|
|
((cycles as f64 / (16.78 * 1_000_000.0)) * 100.0).round() / 100.0
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
println!("{}", out);
|
2021-07-03 22:19:10 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
println!("{}", out);
|
|
|
|
}
|
2021-04-19 22:21:44 +01:00
|
|
|
|
2023-04-25 20:34:47 +01:00
|
|
|
if log_level == "FATAL" {
|
|
|
|
finished.set(Status::Failed);
|
|
|
|
}
|
2021-04-19 22:21:44 +01:00
|
|
|
|
2023-04-25 20:34:47 +01:00
|
|
|
if out == "Tests finished successfully" {
|
|
|
|
finished.set(Status::Success);
|
|
|
|
}
|
2021-04-19 22:21:44 +01:00
|
|
|
}
|
2023-04-25 20:34:47 +01:00
|
|
|
});
|
|
|
|
}
|
2021-04-19 22:21:44 +01:00
|
|
|
|
|
|
|
loop {
|
|
|
|
mgba.advance_frame();
|
2023-04-25 20:34:47 +01:00
|
|
|
if finished.get() != Status::Running {
|
2021-04-19 22:21:44 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-25 20:34:47 +01:00
|
|
|
return finished.get();
|
2021-04-19 22:21:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
fn main() -> Result<(), Error> {
|
|
|
|
let args: Vec<String> = std::env::args().collect();
|
|
|
|
let file_to_run = args.get(1).expect("you should provide file to run");
|
|
|
|
|
|
|
|
if !Path::new(file_to_run).exists() {
|
|
|
|
return Err(anyhow!("File to run should exist!"));
|
|
|
|
}
|
|
|
|
|
|
|
|
let output = test_file(file_to_run);
|
|
|
|
|
|
|
|
match output {
|
|
|
|
Status::Failed => Err(anyhow!("Tests failed!")),
|
2022-11-17 21:25:06 +00:00
|
|
|
Status::Success => Ok(()),
|
2021-04-19 22:21:44 +01:00
|
|
|
_ => {
|
|
|
|
unreachable!("very bad thing happened");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn gba_colour_to_rgba(colour: u32) -> [u8; 4] {
|
|
|
|
[
|
2021-06-04 10:30:18 +01:00
|
|
|
((colour >> 0) & 0xFF) as u8,
|
|
|
|
((colour >> 8) & 0xFF) as u8,
|
|
|
|
((colour >> 16) & 0xFF) as u8,
|
2021-04-19 22:21:44 +01:00
|
|
|
255,
|
|
|
|
]
|
|
|
|
}
|
2021-06-04 10:30:18 +01:00
|
|
|
|
|
|
|
fn rgba_to_gba_to_rgba(c: [u8; 4]) -> [u8; 4] {
|
|
|
|
let mut n = c.clone();
|
|
|
|
n.iter_mut()
|
|
|
|
.for_each(|a| *a = ((((*a as u32 >> 3) << 3) * 0x21) >> 5) as u8);
|
|
|
|
n
|
|
|
|
}
|
|
|
|
|
|
|
|
fn check_image_match(image_path: &str, video_buffer: &VideoBuffer) -> Result<(), Error> {
|
|
|
|
let expected_image = Reader::open(image_path)?.decode()?;
|
2022-04-23 15:33:57 +01:00
|
|
|
let expected = expected_image.to_rgba8();
|
2021-06-04 10:30:18 +01:00
|
|
|
|
|
|
|
let (buf_dim_x, buf_dim_y) = video_buffer.get_size();
|
|
|
|
let (exp_dim_x, exp_dim_y) = expected.dimensions();
|
|
|
|
if (buf_dim_x != exp_dim_x) || (buf_dim_y != exp_dim_y) {
|
|
|
|
return Err(anyhow!("image sizes do not match"));
|
|
|
|
}
|
2022-04-23 15:33:57 +01:00
|
|
|
|
2021-06-04 10:30:18 +01:00
|
|
|
for y in 0..buf_dim_y {
|
|
|
|
for x in 0..buf_dim_x {
|
|
|
|
let video_pixel = video_buffer.get_pixel(x, y);
|
|
|
|
let image_pixel = expected.get_pixel(x, y);
|
|
|
|
let video_pixel = gba_colour_to_rgba(video_pixel);
|
|
|
|
let image_pixel = rgba_to_gba_to_rgba(image_pixel.0);
|
|
|
|
if image_pixel != video_pixel {
|
2022-04-23 15:33:57 +01:00
|
|
|
let output_file = write_video_buffer(video_buffer);
|
|
|
|
|
|
|
|
return Err(anyhow!(
|
|
|
|
"images do not match, actual output written to {}",
|
|
|
|
output_file
|
|
|
|
));
|
2021-06-04 10:30:18 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2022-04-23 15:33:57 +01:00
|
|
|
|
|
|
|
fn write_video_buffer(video_buffer: &VideoBuffer) -> String {
|
|
|
|
let (width, height) = video_buffer.get_size();
|
|
|
|
let mut output_image = image::DynamicImage::new_rgba8(width, height);
|
|
|
|
|
|
|
|
for y in 0..height {
|
|
|
|
for x in 0..width {
|
|
|
|
let pixel = video_buffer.get_pixel(x, y);
|
|
|
|
let pixel_as_rgba = gba_colour_to_rgba(pixel);
|
|
|
|
|
|
|
|
output_image.put_pixel(x, y, pixel_as_rgba.into())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let output_folder = std::env::temp_dir();
|
|
|
|
let output_file = "mgba-test-runner-output.png"; // TODO make this random
|
|
|
|
|
|
|
|
let output_file = output_folder.join(output_file);
|
|
|
|
let _ = output_image.save_with_format(&output_file, image::ImageFormat::Png);
|
|
|
|
|
|
|
|
output_file.to_string_lossy().into_owned()
|
|
|
|
}
|