2023-10-05 10:49:12 +11:00
|
|
|
use std::{
|
|
|
|
path::PathBuf,
|
|
|
|
sync::{
|
|
|
|
mpsc::{self, Receiver, Sender},
|
2023-10-06 16:53:57 +11:00
|
|
|
Arc, Mutex,
|
2023-10-05 10:49:12 +11:00
|
|
|
},
|
2023-03-07 21:45:11 +11:00
|
|
|
};
|
2023-03-05 20:18:06 +11:00
|
|
|
|
2023-10-05 12:42:03 +11:00
|
|
|
use baseview::{Event, EventStatus, Size, Window, WindowEvent, WindowHandler, WindowOpenOptions};
|
2023-03-07 21:45:11 +11:00
|
|
|
use gb_emu_lib::{
|
2023-10-04 11:39:27 +11:00
|
|
|
connect::{JoypadButtons, JoypadState, Renderer, ResolutionData, HEIGHT, WIDTH},
|
|
|
|
renderer::{RendererBackend, RendererBackendManager, WindowOptions},
|
2023-03-05 20:18:06 +11:00
|
|
|
};
|
2023-03-08 15:19:10 +11:00
|
|
|
use keyboard_types::{Code, KeyState};
|
2023-03-05 20:18:06 +11:00
|
|
|
use nih_plug::prelude::*;
|
|
|
|
|
2023-10-06 09:32:07 +11:00
|
|
|
use crate::{access_config, Frame, FrameReceiver, JoypadInfo, JoypadSender, IS_CGB};
|
2023-03-08 11:50:19 +11:00
|
|
|
|
2023-03-07 21:45:11 +11:00
|
|
|
pub struct Emulator {
|
2023-03-08 11:50:19 +11:00
|
|
|
frame_receiver: Arc<FrameReceiver>,
|
2023-03-08 15:19:10 +11:00
|
|
|
joypad_sender: Arc<JoypadSender>,
|
2023-10-06 16:53:57 +11:00
|
|
|
manager: Arc<Mutex<Option<Arc<RendererBackendManager>>>>,
|
2023-10-05 12:42:03 +11:00
|
|
|
size: Size,
|
2023-03-07 21:45:11 +11:00
|
|
|
}
|
2023-03-05 20:18:06 +11:00
|
|
|
|
|
|
|
impl Emulator {
|
2023-10-05 10:49:12 +11:00
|
|
|
pub fn new(
|
|
|
|
frame_receiver: Arc<FrameReceiver>,
|
|
|
|
joypad_sender: Arc<JoypadSender>,
|
2023-10-06 16:53:57 +11:00
|
|
|
manager: Arc<Mutex<Option<Arc<RendererBackendManager>>>>,
|
2023-10-05 12:42:03 +11:00
|
|
|
size: Size,
|
2023-10-05 10:49:12 +11:00
|
|
|
) -> Self {
|
2023-03-08 15:19:10 +11:00
|
|
|
Self {
|
|
|
|
frame_receiver,
|
|
|
|
joypad_sender,
|
2023-10-06 16:53:57 +11:00
|
|
|
manager,
|
2023-10-05 12:42:03 +11:00
|
|
|
size,
|
2023-03-08 15:19:10 +11:00
|
|
|
}
|
2023-03-05 20:18:06 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Editor for Emulator {
|
|
|
|
fn spawn(
|
|
|
|
&self,
|
|
|
|
parent: ParentWindowHandle,
|
|
|
|
_context: Arc<dyn GuiContext>,
|
|
|
|
) -> Box<dyn std::any::Any + Send> {
|
2023-03-08 15:19:10 +11:00
|
|
|
let fr_cloned = self.frame_receiver.clone();
|
|
|
|
let js_cloned = self.joypad_sender.clone();
|
2023-03-24 13:01:39 +11:00
|
|
|
|
2023-10-04 11:39:27 +11:00
|
|
|
// let (size, scale) = if cfg!(target_os = "macos") {
|
|
|
|
// (
|
|
|
|
// Size::new((WIDTH * EXTRA_SCALE) as f64, (HEIGHT * EXTRA_SCALE) as f64),
|
|
|
|
// baseview::WindowScalePolicy::SystemScaleFactor,
|
|
|
|
// )
|
|
|
|
// } else {
|
|
|
|
// (
|
|
|
|
// Size::new(WIDTH as f64, HEIGHT as f64),
|
|
|
|
// baseview::WindowScalePolicy::ScaleFactor(EXTRA_SCALE as f64),
|
|
|
|
// )
|
|
|
|
// };
|
2023-03-24 13:01:39 +11:00
|
|
|
|
2023-10-06 09:32:07 +11:00
|
|
|
let config = access_config();
|
|
|
|
|
2023-10-05 11:05:41 +11:00
|
|
|
#[cfg(feature = "vulkan")]
|
2023-10-06 09:32:07 +11:00
|
|
|
let shader_path = if IS_CGB.get().is_some_and(|v| *v) {
|
|
|
|
config.emu_config.vulkan_config.cgb_shader_path.as_ref()
|
|
|
|
} else {
|
|
|
|
config.emu_config.vulkan_config.dmg_shader_path.as_ref()
|
|
|
|
}
|
|
|
|
.map(|p| config.config_dir.join(p));
|
|
|
|
|
2023-10-05 21:38:43 +11:00
|
|
|
let scale_factor = access_config().vst_config.scale_factor;
|
2023-10-05 12:42:03 +11:00
|
|
|
|
|
|
|
let size = Size::new(
|
|
|
|
(WIDTH * scale_factor) as f64,
|
|
|
|
(HEIGHT * scale_factor) as f64,
|
|
|
|
);
|
2023-10-05 10:49:12 +11:00
|
|
|
|
2023-10-06 16:53:57 +11:00
|
|
|
let m = self.manager.clone();
|
|
|
|
|
2023-03-05 20:18:06 +11:00
|
|
|
Window::open_parented(
|
|
|
|
&parent,
|
|
|
|
WindowOpenOptions {
|
|
|
|
title: String::from("gb-emu"),
|
2023-10-05 12:42:03 +11:00
|
|
|
size,
|
2023-10-04 11:39:27 +11:00
|
|
|
scale: baseview::WindowScalePolicy::SystemScaleFactor,
|
2023-03-05 20:18:06 +11:00
|
|
|
},
|
2023-10-05 12:42:03 +11:00
|
|
|
move |w| {
|
2023-10-05 11:05:41 +11:00
|
|
|
EmulatorWindow::new(
|
|
|
|
w,
|
|
|
|
fr_cloned,
|
|
|
|
js_cloned,
|
2023-10-06 16:53:57 +11:00
|
|
|
m,
|
2023-10-05 12:42:03 +11:00
|
|
|
size,
|
2023-10-05 11:05:41 +11:00
|
|
|
#[cfg(feature = "vulkan")]
|
|
|
|
shader_path,
|
|
|
|
)
|
2023-10-06 16:53:57 +11:00
|
|
|
.0
|
2023-10-05 11:05:41 +11:00
|
|
|
},
|
2023-03-05 20:18:06 +11:00
|
|
|
);
|
2023-03-08 15:19:10 +11:00
|
|
|
Box::new(Self::new(
|
|
|
|
self.frame_receiver.clone(),
|
|
|
|
self.joypad_sender.clone(),
|
2023-10-06 16:53:57 +11:00
|
|
|
self.manager.clone(),
|
2023-10-05 12:42:03 +11:00
|
|
|
size,
|
2023-03-08 15:19:10 +11:00
|
|
|
))
|
2023-03-05 20:18:06 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
fn size(&self) -> (u32, u32) {
|
2023-10-05 12:42:03 +11:00
|
|
|
(self.size.width as u32, self.size.height as u32)
|
2023-03-05 20:18:06 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
fn set_scale_factor(&self, _factor: f32) -> bool {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
|
|
|
|
fn param_value_changed(&self, _id: &str, _normalized_value: f32) {}
|
|
|
|
|
|
|
|
fn param_modulation_changed(&self, _id: &str, _modulation_offset: f32) {}
|
|
|
|
|
|
|
|
fn param_values_changed(&self) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct EmulatorWindow {
|
2023-10-04 11:39:27 +11:00
|
|
|
renderer: RendererBackend,
|
|
|
|
manager: Arc<RendererBackendManager>,
|
2023-03-08 11:50:19 +11:00
|
|
|
frame_receiver: Arc<FrameReceiver>,
|
2023-03-08 15:19:10 +11:00
|
|
|
joypad_sender: Arc<JoypadSender>,
|
2023-10-04 11:39:27 +11:00
|
|
|
current_resolution: ResolutionData,
|
2023-03-05 20:18:06 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
impl EmulatorWindow {
|
2023-03-08 15:19:10 +11:00
|
|
|
fn new(
|
|
|
|
window: &mut Window,
|
|
|
|
frame_receiver: Arc<FrameReceiver>,
|
|
|
|
joypad_sender: Arc<JoypadSender>,
|
2023-10-06 16:53:57 +11:00
|
|
|
manager: Arc<Mutex<Option<Arc<RendererBackendManager>>>>,
|
2023-10-05 12:42:03 +11:00
|
|
|
size: Size,
|
2023-10-05 11:05:41 +11:00
|
|
|
#[cfg(feature = "vulkan")] shader_path: Option<PathBuf>,
|
2023-10-06 16:53:57 +11:00
|
|
|
) -> (Self, Arc<RendererBackendManager>) {
|
2023-10-04 11:39:27 +11:00
|
|
|
let current_resolution = ResolutionData {
|
2023-10-05 12:42:03 +11:00
|
|
|
real_width: size.width as u32,
|
|
|
|
real_height: size.height as u32,
|
2023-10-04 11:39:27 +11:00
|
|
|
scaled_width: WIDTH as u32,
|
|
|
|
scaled_height: HEIGHT as u32,
|
|
|
|
};
|
|
|
|
|
2023-10-05 11:05:41 +11:00
|
|
|
#[cfg(feature = "vulkan")]
|
2023-10-06 16:53:57 +11:00
|
|
|
let window_options = WindowOptions { shader_path };
|
2023-10-05 11:05:41 +11:00
|
|
|
#[cfg(feature = "pixels")]
|
2023-10-06 16:53:57 +11:00
|
|
|
let window_options = WindowOptions {};
|
2023-03-05 20:18:06 +11:00
|
|
|
|
2023-10-06 16:53:57 +11:00
|
|
|
let mut guard = manager.lock().unwrap();
|
2023-03-05 20:18:06 +11:00
|
|
|
|
2023-10-06 16:53:57 +11:00
|
|
|
let m = guard.get_or_insert_with(|| {
|
|
|
|
use raw_window_handle::HasRawDisplayHandle;
|
|
|
|
Arc::new(RendererBackendManager::new(window.raw_display_handle()))
|
|
|
|
});
|
|
|
|
|
|
|
|
let renderer = RendererBackend::new(current_resolution, window, window_options, m.clone());
|
|
|
|
|
|
|
|
(
|
|
|
|
Self {
|
|
|
|
renderer,
|
|
|
|
manager: m.clone(),
|
|
|
|
frame_receiver,
|
|
|
|
joypad_sender,
|
|
|
|
current_resolution,
|
|
|
|
},
|
|
|
|
m.clone(),
|
|
|
|
)
|
2023-03-05 20:18:06 +11:00
|
|
|
}
|
|
|
|
}
|
2023-10-06 16:53:57 +11:00
|
|
|
|
2023-03-05 20:18:06 +11:00
|
|
|
impl WindowHandler for EmulatorWindow {
|
|
|
|
fn on_frame(&mut self, _window: &mut Window) {
|
2023-03-07 21:45:11 +11:00
|
|
|
if let Some(ref mut receiver) = *self.frame_receiver.lock().expect("failed to lock mutex") {
|
2023-03-09 11:52:24 +11:00
|
|
|
if let Some(ref buf) = receiver.try_iter().last() {
|
2023-10-04 11:39:27 +11:00
|
|
|
self.renderer.new_frame(buf);
|
2023-03-07 21:45:11 +11:00
|
|
|
}
|
|
|
|
}
|
2023-10-04 11:39:27 +11:00
|
|
|
|
|
|
|
self.renderer.render(self.current_resolution, &self.manager);
|
2023-03-05 20:18:06 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
fn on_event(&mut self, window: &mut Window, event: baseview::Event) -> EventStatus {
|
2023-03-08 15:19:10 +11:00
|
|
|
match event {
|
|
|
|
Event::Window(WindowEvent::Resized(info)) => {
|
2023-10-04 11:39:27 +11:00
|
|
|
let physical_size = info.physical_size();
|
|
|
|
self.current_resolution = ResolutionData {
|
|
|
|
real_width: physical_size.width,
|
|
|
|
real_height: physical_size.height,
|
|
|
|
scaled_width: WIDTH as u32,
|
|
|
|
scaled_height: HEIGHT as u32,
|
|
|
|
};
|
|
|
|
self.renderer.resize(self.current_resolution, window);
|
2023-03-08 15:19:10 +11:00
|
|
|
EventStatus::Captured
|
|
|
|
}
|
|
|
|
Event::Keyboard(event) => {
|
|
|
|
let status = event.state == KeyState::Down;
|
|
|
|
if let Some(button) = match event.code {
|
|
|
|
Code::Equal => Some(JoypadButtons::Start),
|
|
|
|
Code::Minus => Some(JoypadButtons::Select),
|
|
|
|
Code::Quote => Some(JoypadButtons::A),
|
|
|
|
Code::Semicolon => Some(JoypadButtons::B),
|
|
|
|
Code::KeyW | Code::ArrowUp => Some(JoypadButtons::Up),
|
|
|
|
Code::KeyA | Code::ArrowLeft => Some(JoypadButtons::Left),
|
|
|
|
Code::KeyS | Code::ArrowDown => Some(JoypadButtons::Down),
|
|
|
|
Code::KeyD | Code::ArrowRight => Some(JoypadButtons::Right),
|
|
|
|
_ => None,
|
|
|
|
} {
|
|
|
|
if let Some(ref mut sender) =
|
|
|
|
*self.joypad_sender.lock().expect("failed to lock mutex")
|
|
|
|
{
|
|
|
|
sender.send((button, status)).unwrap();
|
|
|
|
}
|
|
|
|
EventStatus::Captured
|
|
|
|
} else {
|
|
|
|
EventStatus::Ignored
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => EventStatus::Ignored,
|
2023-03-05 20:18:06 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-03-07 08:51:12 +11:00
|
|
|
|
2023-03-07 21:45:11 +11:00
|
|
|
pub struct EmulatorRenderer {
|
2023-09-10 10:56:21 +10:00
|
|
|
tx: Sender<Frame>,
|
2023-03-08 15:19:10 +11:00
|
|
|
joypad: JoypadState,
|
2023-09-10 10:56:21 +10:00
|
|
|
keys: Receiver<JoypadInfo>,
|
2023-03-07 21:45:11 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
impl EmulatorRenderer {
|
2023-09-10 10:56:21 +10:00
|
|
|
pub(super) fn new() -> (Self, Receiver<Frame>, Sender<JoypadInfo>) {
|
|
|
|
let (tx, rx) = mpsc::channel::<Frame>();
|
|
|
|
let (keys_tx, keys) = mpsc::channel::<JoypadInfo>();
|
2023-03-08 15:19:10 +11:00
|
|
|
(
|
|
|
|
Self {
|
|
|
|
tx,
|
|
|
|
joypad: JoypadState::default(),
|
|
|
|
keys,
|
|
|
|
},
|
|
|
|
rx,
|
|
|
|
keys_tx,
|
|
|
|
)
|
2023-03-07 21:45:11 +11:00
|
|
|
}
|
|
|
|
}
|
2023-03-07 08:51:12 +11:00
|
|
|
|
2023-03-08 11:01:18 +11:00
|
|
|
impl Renderer<[u8; 4]> for EmulatorRenderer {
|
2023-03-07 08:51:12 +11:00
|
|
|
fn prepare(&mut self, _width: usize, _height: usize) {}
|
|
|
|
|
2023-03-08 11:01:18 +11:00
|
|
|
fn display(&mut self, buffer: &[[u8; 4]]) {
|
2023-04-06 13:15:48 +10:00
|
|
|
let _ = self.tx.send(buffer.to_vec());
|
2023-03-07 21:45:11 +11:00
|
|
|
}
|
2023-03-07 08:51:12 +11:00
|
|
|
|
2023-03-08 15:19:10 +11:00
|
|
|
fn latest_joypad_state(&mut self) -> JoypadState {
|
|
|
|
while let Ok((key, state)) = self.keys.try_recv() {
|
|
|
|
match key {
|
|
|
|
JoypadButtons::Down => self.joypad.down = state,
|
|
|
|
JoypadButtons::Up => self.joypad.up = state,
|
|
|
|
JoypadButtons::Left => self.joypad.left = state,
|
|
|
|
JoypadButtons::Right => self.joypad.right = state,
|
|
|
|
JoypadButtons::Start => self.joypad.start = state,
|
|
|
|
JoypadButtons::Select => self.joypad.select = state,
|
|
|
|
JoypadButtons::B => self.joypad.b = state,
|
|
|
|
JoypadButtons::A => self.joypad.a = state,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
self.joypad
|
2023-03-07 08:51:12 +11:00
|
|
|
}
|
|
|
|
}
|