Save and load SRAM at all the right times

This commit is contained in:
Simon Gellis 2024-12-08 14:39:40 -05:00
parent e9ae8bed1b
commit 0504e351ba
4 changed files with 51 additions and 11 deletions

9
Cargo.lock generated
View File

@ -1,6 +1,6 @@
# This file is automatically @generated by Cargo. # This file is automatically @generated by Cargo.
# It is not intended for manual editing. # It is not intended for manual editing.
version = 3 version = 4
[[package]] [[package]]
name = "ab_glyph" name = "ab_glyph"
@ -2200,6 +2200,12 @@ version = "1.20.2"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1261fe7e33c73b354eab43b1273a57c8f967d0391e80353e51f764ac02cf6775" checksum = "1261fe7e33c73b354eab43b1273a57c8f967d0391e80353e51f764ac02cf6775"
[[package]]
name = "oneshot"
version = "0.1.8"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e296cf87e61c9cfc1a61c3c63a0f7f286ed4554e0e22be84e8a38e1d264a2a29"
[[package]] [[package]]
name = "orbclient" name = "orbclient"
version = "0.3.48" version = "0.3.48"
@ -2684,6 +2690,7 @@ dependencies = [
"itertools", "itertools",
"num-derive", "num-derive",
"num-traits", "num-traits",
"oneshot",
"pollster 0.4.0", "pollster 0.4.0",
"rfd", "rfd",
"rtrb", "rtrb",

View File

@ -17,6 +17,7 @@ gilrs = "0.11"
itertools = "0.13" itertools = "0.13"
num-derive = "0.4" num-derive = "0.4"
num-traits = "0.2" num-traits = "0.2"
oneshot = "0.1"
pollster = "0.4" pollster = "0.4"
rfd = "0.15" rfd = "0.15"
rtrb = "0.3" rtrb = "0.3"

View File

@ -1,4 +1,4 @@
use std::{collections::HashSet, num::NonZero, sync::Arc, thread}; use std::{collections::HashSet, num::NonZero, sync::Arc, thread, time::Duration};
use egui::{ use egui::{
ahash::{HashMap, HashMapExt}, ahash::{HashMap, HashMapExt},
@ -15,7 +15,7 @@ use winit::{
use crate::{ use crate::{
controller::ControllerManager, controller::ControllerManager,
emulator::{EmulatorClient, SimId}, emulator::{EmulatorClient, EmulatorCommand, SimId},
input::MappingProvider, input::MappingProvider,
window::{AppWindow, GameWindow, InputWindow}, window::{AppWindow, GameWindow, InputWindow},
}; };
@ -170,6 +170,15 @@ impl ApplicationHandler<UserEvent> for Application {
viewport.window.request_redraw(); viewport.window.request_redraw();
} }
} }
fn exiting(&mut self, _event_loop: &ActiveEventLoop) {
let (sender, receiver) = oneshot::channel();
if self.client.send_command(EmulatorCommand::Exit(sender)) {
if let Err(err) = receiver.recv_timeout(Duration::from_secs(5)) {
eprintln!("could not gracefully exit: {}", err);
}
}
}
} }
struct Viewport { struct Viewport {

View File

@ -186,6 +186,8 @@ impl Emulator {
} }
fn reset_sim(&mut self, sim_id: SimId, new_cart: Option<Cart>) -> Result<()> { fn reset_sim(&mut self, sim_id: SimId, new_cart: Option<Cart>) -> Result<()> {
self.save_sram(sim_id)?;
let index = sim_id.to_index(); let index = sim_id.to_index();
while self.sims.len() <= index { while self.sims.len() <= index {
self.sims.push(Sim::new()); self.sims.push(Sim::new());
@ -226,6 +228,10 @@ impl Emulator {
pub fn pause_sim(&mut self, sim_id: SimId) -> Result<()> { pub fn pause_sim(&mut self, sim_id: SimId) -> Result<()> {
self.running[sim_id.to_index()].store(false, Ordering::Release); self.running[sim_id.to_index()].store(false, Ordering::Release);
self.save_sram(sim_id)
}
fn save_sram(&mut self, sim_id: SimId) -> Result<()> {
let sim = self.sims.get_mut(sim_id.to_index()); let sim = self.sims.get_mut(sim_id.to_index());
let cart = self.carts[sim_id.to_index()].as_mut(); let cart = self.carts[sim_id.to_index()].as_mut();
if let (Some(sim), Some(cart)) = (sim, cart) { if let (Some(sim), Some(cart)) = (sim, cart) {
@ -236,13 +242,15 @@ impl Emulator {
Ok(()) Ok(())
} }
pub fn stop_second_sim(&mut self) { pub fn stop_second_sim(&mut self) -> Result<()> {
self.save_sram(SimId::Player2)?;
self.renderers.remove(&SimId::Player2); self.renderers.remove(&SimId::Player2);
self.sims.truncate(1); self.sims.truncate(1);
self.sim_count.store(self.sims.len(), Ordering::Relaxed); self.sim_count.store(self.sims.len(), Ordering::Relaxed);
self.running[SimId::Player2.to_index()].store(false, Ordering::Release); self.running[SimId::Player2.to_index()].store(false, Ordering::Release);
self.has_game[SimId::Player2.to_index()].store(false, Ordering::Release); self.has_game[SimId::Player2.to_index()].store(false, Ordering::Release);
self.linked.store(false, Ordering::Release); self.linked.store(false, Ordering::Release);
Ok(())
} }
pub fn run(&mut self) { pub fn run(&mut self) {
@ -336,7 +344,9 @@ impl Emulator {
} }
} }
EmulatorCommand::StopSecondSim => { EmulatorCommand::StopSecondSim => {
self.stop_second_sim(); if let Err(error) = self.stop_second_sim() {
eprintln!("error stopping second sim: {}", error);
}
} }
EmulatorCommand::Pause => { EmulatorCommand::Pause => {
for sim_id in SimId::values() { for sim_id in SimId::values() {
@ -373,6 +383,14 @@ impl Emulator {
sim.set_keys(keys); sim.set_keys(keys);
} }
} }
EmulatorCommand::Exit(done) => {
for sim_id in SimId::values() {
if let Err(error) = self.save_sram(sim_id) {
eprintln!("error saving sram on exit: {}", error);
}
}
let _ = done.send(());
}
} }
} }
} }
@ -390,6 +408,7 @@ pub enum EmulatorCommand {
Unlink, Unlink,
Reset(SimId), Reset(SimId),
SetKeys(SimId, VBKey), SetKeys(SimId, VBKey),
Exit(oneshot::Sender<()>),
} }
#[derive(Clone)] #[derive(Clone)]
@ -418,12 +437,16 @@ impl EmulatorClient {
pub fn is_audio_enabled(&self, sim_id: SimId) -> bool { pub fn is_audio_enabled(&self, sim_id: SimId) -> bool {
self.audio_on[sim_id.to_index()].load(Ordering::Acquire) self.audio_on[sim_id.to_index()].load(Ordering::Acquire)
} }
pub fn send_command(&self, command: EmulatorCommand) { pub fn send_command(&self, command: EmulatorCommand) -> bool {
if let Err(err) = self.queue.send(command) { match self.queue.send(command) {
Ok(()) => true,
Err(err) => {
eprintln!( eprintln!(
"could not send command {:?} as emulator is shut down", "could not send command {:?} as emulator is shut down",
err.0 err.0
); );
false
}
} }
} }
} }