Render images on background thread to keep UI responsive
This commit is contained in:
parent
a82389224f
commit
a461faf89d
|
@ -35,7 +35,7 @@ rubato = "0.16"
|
||||||
serde = { version = "1", features = ["derive"] }
|
serde = { version = "1", features = ["derive"] }
|
||||||
serde_json = "1"
|
serde_json = "1"
|
||||||
thread-priority = "1"
|
thread-priority = "1"
|
||||||
tokio = { version = "1", features = ["io-util", "macros", "net", "rt", "sync"] }
|
tokio = { version = "1", features = ["io-util", "macros", "net", "rt", "sync", "time"] }
|
||||||
tracing = { version = "0.1", features = ["release_max_level_info"] }
|
tracing = { version = "0.1", features = ["release_max_level_info"] }
|
||||||
tracing-subscriber = { version = "0.3", features = ["env-filter"] }
|
tracing-subscriber = { version = "0.3", features = ["env-filter"] }
|
||||||
wgpu = "23"
|
wgpu = "23"
|
||||||
|
|
|
@ -20,6 +20,7 @@ use crate::{
|
||||||
input::MappingProvider,
|
input::MappingProvider,
|
||||||
memory::MemoryMonitor,
|
memory::MemoryMonitor,
|
||||||
persistence::Persistence,
|
persistence::Persistence,
|
||||||
|
vram::VramProcessor,
|
||||||
window::{
|
window::{
|
||||||
AboutWindow, AppWindow, BgMapWindow, CharacterDataWindow, GameWindow, GdbServerWindow,
|
AboutWindow, AppWindow, BgMapWindow, CharacterDataWindow, GameWindow, GdbServerWindow,
|
||||||
InputWindow,
|
InputWindow,
|
||||||
|
@ -45,6 +46,7 @@ pub struct Application {
|
||||||
mappings: MappingProvider,
|
mappings: MappingProvider,
|
||||||
controllers: ControllerManager,
|
controllers: ControllerManager,
|
||||||
memory: MemoryMonitor,
|
memory: MemoryMonitor,
|
||||||
|
vram: VramProcessor,
|
||||||
persistence: Persistence,
|
persistence: Persistence,
|
||||||
viewports: HashMap<ViewportId, Viewport>,
|
viewports: HashMap<ViewportId, Viewport>,
|
||||||
focused: Option<ViewportId>,
|
focused: Option<ViewportId>,
|
||||||
|
@ -63,6 +65,7 @@ impl Application {
|
||||||
let mappings = MappingProvider::new(persistence.clone());
|
let mappings = MappingProvider::new(persistence.clone());
|
||||||
let controllers = ControllerManager::new(client.clone(), &mappings);
|
let controllers = ControllerManager::new(client.clone(), &mappings);
|
||||||
let memory = MemoryMonitor::new(client.clone());
|
let memory = MemoryMonitor::new(client.clone());
|
||||||
|
let vram = VramProcessor::new();
|
||||||
{
|
{
|
||||||
let mappings = mappings.clone();
|
let mappings = mappings.clone();
|
||||||
let proxy = proxy.clone();
|
let proxy = proxy.clone();
|
||||||
|
@ -75,6 +78,7 @@ impl Application {
|
||||||
proxy,
|
proxy,
|
||||||
mappings,
|
mappings,
|
||||||
memory,
|
memory,
|
||||||
|
vram,
|
||||||
controllers,
|
controllers,
|
||||||
persistence,
|
persistence,
|
||||||
viewports: HashMap::new(),
|
viewports: HashMap::new(),
|
||||||
|
@ -205,11 +209,11 @@ impl ApplicationHandler<UserEvent> for Application {
|
||||||
self.open(event_loop, Box::new(about));
|
self.open(event_loop, Box::new(about));
|
||||||
}
|
}
|
||||||
UserEvent::OpenCharacterData(sim_id) => {
|
UserEvent::OpenCharacterData(sim_id) => {
|
||||||
let vram = CharacterDataWindow::new(sim_id, &mut self.memory);
|
let vram = CharacterDataWindow::new(sim_id, &mut self.memory, &mut self.vram);
|
||||||
self.open(event_loop, Box::new(vram));
|
self.open(event_loop, Box::new(vram));
|
||||||
}
|
}
|
||||||
UserEvent::OpenBgMap(sim_id) => {
|
UserEvent::OpenBgMap(sim_id) => {
|
||||||
let bgmap = BgMapWindow::new(sim_id, &mut self.memory);
|
let bgmap = BgMapWindow::new(sim_id, &mut self.memory, &mut self.vram);
|
||||||
self.open(event_loop, Box::new(bgmap));
|
self.open(event_loop, Box::new(bgmap));
|
||||||
}
|
}
|
||||||
UserEvent::OpenDebugger(sim_id) => {
|
UserEvent::OpenDebugger(sim_id) => {
|
||||||
|
|
269
src/vram.rs
269
src/vram.rs
|
@ -1,29 +1,103 @@
|
||||||
use std::{
|
use std::{
|
||||||
collections::{hash_map::Entry, HashMap, HashSet},
|
collections::HashMap,
|
||||||
hash::Hash,
|
ops::Deref,
|
||||||
sync::{Arc, Mutex},
|
sync::{Arc, Mutex, Weak},
|
||||||
|
thread,
|
||||||
|
time::Duration,
|
||||||
};
|
};
|
||||||
|
|
||||||
use egui::{
|
use egui::{
|
||||||
epaint::ImageDelta,
|
epaint::ImageDelta,
|
||||||
load::{LoadError, SizedTexture, TextureLoader, TexturePoll},
|
load::{LoadError, SizedTexture, TextureLoader, TexturePoll},
|
||||||
Color32, ColorImage, Context, TextureHandle, TextureOptions,
|
Color32, ColorImage, TextureHandle, TextureOptions,
|
||||||
};
|
};
|
||||||
use serde::{Deserialize, Serialize};
|
use tokio::{sync::mpsc, time::timeout};
|
||||||
|
|
||||||
pub trait VramResource: Sized + Clone + PartialEq + Eq + Hash {
|
pub struct VramProcessor {
|
||||||
fn to_uri(&self) -> String;
|
sender: mpsc::UnboundedSender<Box<dyn VramRendererImpl>>,
|
||||||
fn from_uri(uri: &str) -> Option<Self>;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: Serialize + for<'a> Deserialize<'a> + Clone + PartialEq + Eq + Hash> VramResource for T {
|
impl VramProcessor {
|
||||||
fn to_uri(&self) -> String {
|
pub fn new() -> Self {
|
||||||
format!("vram://{}", serde_json::to_string(self).unwrap())
|
let (sender, receiver) = mpsc::unbounded_channel();
|
||||||
|
thread::spawn(move || {
|
||||||
|
tokio::runtime::Builder::new_current_thread()
|
||||||
|
.enable_all()
|
||||||
|
.build()
|
||||||
|
.unwrap()
|
||||||
|
.block_on(async move {
|
||||||
|
let mut worker = VramProcessorWorker {
|
||||||
|
receiver,
|
||||||
|
renderers: vec![],
|
||||||
|
};
|
||||||
|
worker.run().await
|
||||||
|
})
|
||||||
|
});
|
||||||
|
Self { sender }
|
||||||
}
|
}
|
||||||
|
|
||||||
fn from_uri(uri: &str) -> Option<Self> {
|
pub fn add<const N: usize, R: VramRenderer<N> + 'static>(
|
||||||
let content = uri.strip_prefix("vram://")?;
|
&self,
|
||||||
serde_json::from_str(content).ok()
|
renderer: R,
|
||||||
|
) -> ([VramImageHandle; N], VramParams<R::Params>) {
|
||||||
|
let handles = renderer.sizes().map(|[width, height]| {
|
||||||
|
let data = Arc::new(Mutex::new(None));
|
||||||
|
VramImageHandle {
|
||||||
|
size: [width as f32, height as f32],
|
||||||
|
data,
|
||||||
|
}
|
||||||
|
});
|
||||||
|
let images = renderer
|
||||||
|
.sizes()
|
||||||
|
.map(|[width, height]| VramImage::new(width, height));
|
||||||
|
let sink = Arc::new(Mutex::new(R::Params::default()));
|
||||||
|
let _ = self.sender.send(Box::new(VramRendererWrapper {
|
||||||
|
renderer,
|
||||||
|
params: Arc::downgrade(&sink),
|
||||||
|
images,
|
||||||
|
sinks: handles.clone().map(|i| i.data),
|
||||||
|
}));
|
||||||
|
let params = VramParams {
|
||||||
|
value: R::Params::default(),
|
||||||
|
sink,
|
||||||
|
};
|
||||||
|
(handles, params)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct VramProcessorWorker {
|
||||||
|
receiver: mpsc::UnboundedReceiver<Box<dyn VramRendererImpl>>,
|
||||||
|
renderers: Vec<Box<dyn VramRendererImpl>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl VramProcessorWorker {
|
||||||
|
async fn run(&mut self) {
|
||||||
|
loop {
|
||||||
|
if self.renderers.is_empty() {
|
||||||
|
// if we have nothing to do, block until we have something to do
|
||||||
|
if self.receiver.recv_many(&mut self.renderers, 64).await == 0 {
|
||||||
|
// shutdown
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
while let Ok(renderer) = self.receiver.try_recv() {
|
||||||
|
self.renderers.push(renderer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
self.renderers
|
||||||
|
.retain_mut(|renderer| renderer.try_update().is_ok());
|
||||||
|
// wait up to 10 ms for more renderers
|
||||||
|
if timeout(
|
||||||
|
Duration::from_millis(10),
|
||||||
|
self.receiver.recv_many(&mut self.renderers, 64),
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
.is_ok()
|
||||||
|
{
|
||||||
|
while let Ok(renderer) = self.receiver.try_recv() {
|
||||||
|
self.renderers.push(renderer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -67,55 +141,107 @@ impl VramImage {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait VramImageLoader {
|
#[derive(Clone)]
|
||||||
type Resource: VramResource;
|
pub struct VramImageHandle {
|
||||||
|
size: [f32; 2],
|
||||||
fn id(&self) -> &str;
|
data: Arc<Mutex<Option<Arc<ColorImage>>>>,
|
||||||
fn add(&self, resource: &Self::Resource) -> Option<VramImage>;
|
|
||||||
fn update<'a>(
|
|
||||||
&'a self,
|
|
||||||
resources: impl Iterator<Item = (&'a Self::Resource, &'a mut VramImage)>,
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct VramTextureLoader<T: VramImageLoader> {
|
impl VramImageHandle {
|
||||||
id: String,
|
fn pull(&mut self) -> Option<Arc<ColorImage>> {
|
||||||
loader: Mutex<T>,
|
self.data.lock().unwrap().take()
|
||||||
cache: Mutex<HashMap<T::Resource, (TextureHandle, VramImage)>>,
|
}
|
||||||
seen: Mutex<HashSet<T::Resource>>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: VramImageLoader> VramTextureLoader<T> {
|
pub struct VramParams<T> {
|
||||||
pub fn new(loader: T) -> Self {
|
value: T,
|
||||||
|
sink: Arc<Mutex<T>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> Deref for VramParams<T> {
|
||||||
|
type Target = T;
|
||||||
|
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
&self.value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: Clone + Eq> VramParams<T> {
|
||||||
|
pub fn write(&mut self, value: T) {
|
||||||
|
if self.value != value {
|
||||||
|
self.value = value.clone();
|
||||||
|
*self.sink.lock().unwrap() = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait VramRenderer<const N: usize>: Send {
|
||||||
|
type Params: Clone + Default + Send;
|
||||||
|
fn sizes(&self) -> [[usize; 2]; N];
|
||||||
|
fn render(&mut self, params: &Self::Params, images: &mut [VramImage; N]);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct VramRendererWrapper<const N: usize, R: VramRenderer<N>> {
|
||||||
|
renderer: R,
|
||||||
|
params: Weak<Mutex<R::Params>>,
|
||||||
|
images: [VramImage; N],
|
||||||
|
sinks: [Arc<Mutex<Option<Arc<ColorImage>>>>; N],
|
||||||
|
}
|
||||||
|
|
||||||
|
trait VramRendererImpl: Send {
|
||||||
|
fn try_update(&mut self) -> Result<(), ()>;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<const N: usize, R: VramRenderer<N> + Send> VramRendererImpl for VramRendererWrapper<N, R> {
|
||||||
|
fn try_update(&mut self) -> Result<(), ()> {
|
||||||
|
let params = match self.params.upgrade() {
|
||||||
|
Some(params) => params.lock().unwrap().clone(),
|
||||||
|
None => {
|
||||||
|
// the UI isn't using this anymore
|
||||||
|
return Err(());
|
||||||
|
}
|
||||||
|
};
|
||||||
|
self.renderer.render(¶ms, &mut self.images);
|
||||||
|
|
||||||
|
for (image, sink) in self.images.iter_mut().zip(&self.sinks) {
|
||||||
|
if let Some(update) = image.take() {
|
||||||
|
sink.lock().unwrap().replace(update);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct VramTextureLoader {
|
||||||
|
cache: Mutex<HashMap<String, (VramImageHandle, Option<TextureHandle>)>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl VramTextureLoader {
|
||||||
|
pub fn new(renderers: impl IntoIterator<Item = (String, VramImageHandle)>) -> Self {
|
||||||
|
let mut cache = HashMap::new();
|
||||||
|
for (key, image) in renderers {
|
||||||
|
cache.insert(key, (image, None));
|
||||||
|
}
|
||||||
Self {
|
Self {
|
||||||
id: loader.id().to_string(),
|
cache: Mutex::new(cache),
|
||||||
loader: Mutex::new(loader),
|
|
||||||
cache: Mutex::new(HashMap::new()),
|
|
||||||
seen: Mutex::new(HashSet::new()),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn begin_pass(&self) {
|
|
||||||
let mut cache = self.cache.lock().unwrap();
|
|
||||||
let mut seen = self.seen.lock().unwrap();
|
|
||||||
cache.retain(|res, _| seen.contains(res));
|
|
||||||
seen.clear();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: VramImageLoader> TextureLoader for VramTextureLoader<T> {
|
impl TextureLoader for VramTextureLoader {
|
||||||
fn id(&self) -> &str {
|
fn id(&self) -> &str {
|
||||||
&self.id
|
concat!(module_path!(), "VramTextureLoader")
|
||||||
}
|
}
|
||||||
|
|
||||||
fn load(
|
fn load(
|
||||||
&self,
|
&self,
|
||||||
ctx: &Context,
|
ctx: &egui::Context,
|
||||||
uri: &str,
|
uri: &str,
|
||||||
texture_options: TextureOptions,
|
texture_options: TextureOptions,
|
||||||
_size_hint: egui::SizeHint,
|
_size_hint: egui::SizeHint,
|
||||||
) -> Result<TexturePoll, LoadError> {
|
) -> Result<TexturePoll, LoadError> {
|
||||||
let Some(resource) = T::Resource::from_uri(uri) else {
|
let mut cache = self.cache.lock().unwrap();
|
||||||
|
let Some((image, maybe_handle)) = cache.get_mut(uri) else {
|
||||||
return Err(LoadError::NotSupported);
|
return Err(LoadError::NotSupported);
|
||||||
};
|
};
|
||||||
if texture_options != TextureOptions::NEAREST {
|
if texture_options != TextureOptions::NEAREST {
|
||||||
|
@ -123,54 +249,45 @@ impl<T: VramImageLoader> TextureLoader for VramTextureLoader<T> {
|
||||||
"Only TextureOptions::NEAREST are supported".into(),
|
"Only TextureOptions::NEAREST are supported".into(),
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
let mut cache = self.cache.lock().unwrap();
|
match (image.pull(), maybe_handle.as_ref()) {
|
||||||
let mut seen = self.seen.lock().unwrap();
|
(Some(update), Some(handle)) => {
|
||||||
seen.insert(resource.clone());
|
let delta = ImageDelta::full(update, texture_options);
|
||||||
let loader = self.loader.lock().unwrap();
|
|
||||||
let resources = cache.iter_mut().map(|(k, v)| (k, &mut v.1));
|
|
||||||
loader.update(resources);
|
|
||||||
for (handle, image) in cache.values_mut() {
|
|
||||||
if let Some(data) = image.take() {
|
|
||||||
let delta = ImageDelta::full(data, TextureOptions::NEAREST);
|
|
||||||
ctx.tex_manager().write().set(handle.id(), delta);
|
ctx.tex_manager().write().set(handle.id(), delta);
|
||||||
}
|
let texture = SizedTexture::new(handle, image.size);
|
||||||
}
|
|
||||||
match cache.entry(resource) {
|
|
||||||
Entry::Occupied(entry) => {
|
|
||||||
let texture = SizedTexture::from_handle(&entry.get().0);
|
|
||||||
Ok(TexturePoll::Ready { texture })
|
Ok(TexturePoll::Ready { texture })
|
||||||
}
|
}
|
||||||
Entry::Vacant(entry) => {
|
(Some(update), None) => {
|
||||||
let Some(mut image) = loader.add(entry.key()) else {
|
let handle = ctx.load_texture(uri, update, texture_options);
|
||||||
return Err(LoadError::Loading("could not load texture".into()));
|
let texture = SizedTexture::new(&handle, image.size);
|
||||||
};
|
maybe_handle.replace(handle);
|
||||||
let Some(data) = image.take() else {
|
|
||||||
return Err(LoadError::Loading("could not load texture".into()));
|
|
||||||
};
|
|
||||||
let handle = ctx.load_texture(uri, data, TextureOptions::NEAREST);
|
|
||||||
let (handle, _) = entry.insert((handle, image));
|
|
||||||
let texture = SizedTexture::from_handle(handle);
|
|
||||||
Ok(TexturePoll::Ready { texture })
|
Ok(TexturePoll::Ready { texture })
|
||||||
}
|
}
|
||||||
|
(None, Some(handle)) => {
|
||||||
|
let texture = SizedTexture::new(handle, image.size);
|
||||||
|
Ok(TexturePoll::Ready { texture })
|
||||||
|
}
|
||||||
|
(None, None) => {
|
||||||
|
let size = image.size.into();
|
||||||
|
Ok(TexturePoll::Pending { size: Some(size) })
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn forget(&self, uri: &str) {
|
fn forget(&self, uri: &str) {
|
||||||
if let Some(resource) = T::Resource::from_uri(uri) {
|
let _ = uri;
|
||||||
self.cache.lock().unwrap().remove(&resource);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn forget_all(&self) {
|
fn forget_all(&self) {}
|
||||||
self.cache.lock().unwrap().clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
fn byte_size(&self) -> usize {
|
fn byte_size(&self) -> usize {
|
||||||
self.cache
|
self.cache
|
||||||
.lock()
|
.lock()
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.values()
|
.values()
|
||||||
.map(|h| h.0.byte_size())
|
.map(|(image, _)| {
|
||||||
|
let [width, height] = image.size;
|
||||||
|
width as usize * height as usize * 4
|
||||||
|
})
|
||||||
.sum()
|
.sum()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,12 +5,11 @@ use egui::{
|
||||||
TextureOptions, Ui, ViewportBuilder, ViewportId,
|
TextureOptions, Ui, ViewportBuilder, ViewportId,
|
||||||
};
|
};
|
||||||
use egui_extras::{Column, Size, StripBuilder, TableBuilder};
|
use egui_extras::{Column, Size, StripBuilder, TableBuilder};
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
emulator::SimId,
|
emulator::SimId,
|
||||||
memory::{MemoryMonitor, MemoryView},
|
memory::{MemoryMonitor, MemoryView},
|
||||||
vram::{VramImage, VramImageLoader, VramResource as _, VramTextureLoader},
|
vram::{VramImage, VramParams, VramProcessor, VramRenderer, VramTextureLoader},
|
||||||
window::{utils::UiExt, AppWindow},
|
window::{utils::UiExt, AppWindow},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -18,26 +17,32 @@ use super::utils::{parse_palette, CharacterGrid, GENERIC_PALETTE};
|
||||||
|
|
||||||
pub struct BgMapWindow {
|
pub struct BgMapWindow {
|
||||||
sim_id: SimId,
|
sim_id: SimId,
|
||||||
loader: Arc<VramTextureLoader<BgMapLoader>>,
|
loader: Arc<VramTextureLoader>,
|
||||||
bgmaps: MemoryView,
|
bgmaps: MemoryView,
|
||||||
cell_index: usize,
|
cell_index: usize,
|
||||||
cell_index_str: String,
|
cell_index_str: String,
|
||||||
|
generic_palette: bool,
|
||||||
|
params: VramParams<BgMapParams>,
|
||||||
scale: f32,
|
scale: f32,
|
||||||
show_grid: bool,
|
show_grid: bool,
|
||||||
generic_palette: bool,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl BgMapWindow {
|
impl BgMapWindow {
|
||||||
pub fn new(sim_id: SimId, memory: &mut MemoryMonitor) -> Self {
|
pub fn new(sim_id: SimId, memory: &mut MemoryMonitor, vram: &mut VramProcessor) -> Self {
|
||||||
|
let renderer = BgMapRenderer::new(sim_id, memory);
|
||||||
|
let ([cell, bgmap], params) = vram.add(renderer);
|
||||||
|
let loader =
|
||||||
|
VramTextureLoader::new([("vram://cell".into(), cell), ("vram://bgmap".into(), bgmap)]);
|
||||||
Self {
|
Self {
|
||||||
sim_id,
|
sim_id,
|
||||||
loader: Arc::new(VramTextureLoader::new(BgMapLoader::new(sim_id, memory))),
|
loader: Arc::new(loader),
|
||||||
bgmaps: memory.view(sim_id, 0x00020000, 0x1d800),
|
bgmaps: memory.view(sim_id, 0x00020000, 0x1d800),
|
||||||
cell_index: 0,
|
cell_index: 0,
|
||||||
cell_index_str: "0".into(),
|
cell_index_str: "0".into(),
|
||||||
|
generic_palette: false,
|
||||||
|
params,
|
||||||
scale: 1.0,
|
scale: 1.0,
|
||||||
show_grid: false,
|
show_grid: false,
|
||||||
generic_palette: false,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -88,11 +93,7 @@ impl BgMapWindow {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
let resource = BgMapResource::Cell {
|
let image = Image::new("vram://cell")
|
||||||
index: self.cell_index,
|
|
||||||
generic_palette: self.generic_palette,
|
|
||||||
};
|
|
||||||
let image = Image::new(resource.to_uri())
|
|
||||||
.maintain_aspect_ratio(true)
|
.maintain_aspect_ratio(true)
|
||||||
.tint(Color32::RED)
|
.tint(Color32::RED)
|
||||||
.texture_options(TextureOptions::NEAREST);
|
.texture_options(TextureOptions::NEAREST);
|
||||||
|
@ -154,14 +155,14 @@ impl BgMapWindow {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
self.params.write(BgMapParams {
|
||||||
|
cell_index: self.cell_index,
|
||||||
|
generic_palette: self.generic_palette,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
fn show_bgmap(&mut self, ui: &mut Ui) {
|
fn show_bgmap(&mut self, ui: &mut Ui) {
|
||||||
let resource = BgMapResource::BgMap {
|
let grid = CharacterGrid::new("vram://bgmap")
|
||||||
index: self.cell_index / 4096,
|
|
||||||
generic_palette: self.generic_palette,
|
|
||||||
};
|
|
||||||
let grid = CharacterGrid::new(resource.to_uri())
|
|
||||||
.with_scale(self.scale)
|
.with_scale(self.scale)
|
||||||
.with_grid(self.show_grid)
|
.with_grid(self.show_grid)
|
||||||
.with_selected(self.cell_index % 4096);
|
.with_selected(self.cell_index % 4096);
|
||||||
|
@ -192,7 +193,6 @@ impl AppWindow for BgMapWindow {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn show(&mut self, ctx: &Context) {
|
fn show(&mut self, ctx: &Context) {
|
||||||
self.loader.begin_pass();
|
|
||||||
CentralPanel::default().show(ctx, |ui| {
|
CentralPanel::default().show(ctx, |ui| {
|
||||||
ui.horizontal_top(|ui| {
|
ui.horizontal_top(|ui| {
|
||||||
StripBuilder::new(ui)
|
StripBuilder::new(ui)
|
||||||
|
@ -219,20 +219,20 @@ fn parse_cell(cell: u16) -> (usize, bool, bool, usize) {
|
||||||
(char_index, vflip, hflip, palette_index)
|
(char_index, vflip, hflip, palette_index)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, Clone, PartialEq, Eq, Hash)]
|
#[derive(Default, Clone, PartialEq, Eq)]
|
||||||
enum BgMapResource {
|
struct BgMapParams {
|
||||||
BgMap { index: usize, generic_palette: bool },
|
cell_index: usize,
|
||||||
Cell { index: usize, generic_palette: bool },
|
generic_palette: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct BgMapLoader {
|
struct BgMapRenderer {
|
||||||
chardata: MemoryView,
|
chardata: MemoryView,
|
||||||
bgmaps: MemoryView,
|
bgmaps: MemoryView,
|
||||||
brightness: MemoryView,
|
brightness: MemoryView,
|
||||||
palettes: MemoryView,
|
palettes: MemoryView,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl BgMapLoader {
|
impl BgMapRenderer {
|
||||||
pub fn new(sim_id: SimId, memory: &mut MemoryMonitor) -> Self {
|
pub fn new(sim_id: SimId, memory: &mut MemoryMonitor) -> Self {
|
||||||
Self {
|
Self {
|
||||||
chardata: memory.view(sim_id, 0x00078000, 0x8000),
|
chardata: memory.view(sim_id, 0x00078000, 0x8000),
|
||||||
|
@ -242,7 +242,7 @@ impl BgMapLoader {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn update_bgmap(&self, image: &mut VramImage, bgmap_index: usize, generic_palette: bool) {
|
fn render_bgmap(&self, image: &mut VramImage, bgmap_index: usize, generic_palette: bool) {
|
||||||
let chardata = self.chardata.borrow();
|
let chardata = self.chardata.borrow();
|
||||||
let bgmaps = self.bgmaps.borrow();
|
let bgmaps = self.bgmaps.borrow();
|
||||||
let brightness = self.brightness.borrow();
|
let brightness = self.brightness.borrow();
|
||||||
|
@ -284,7 +284,7 @@ impl BgMapLoader {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn update_bgmap_cell(&self, image: &mut VramImage, index: usize, generic_palette: bool) {
|
fn render_bgmap_cell(&self, image: &mut VramImage, index: usize, generic_palette: bool) {
|
||||||
let chardata = self.chardata.borrow();
|
let chardata = self.chardata.borrow();
|
||||||
let bgmaps = self.bgmaps.borrow();
|
let bgmaps = self.bgmaps.borrow();
|
||||||
let brightness = self.brightness.borrow();
|
let brightness = self.brightness.borrow();
|
||||||
|
@ -324,49 +324,19 @@ impl BgMapLoader {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl VramImageLoader for BgMapLoader {
|
impl VramRenderer<2> for BgMapRenderer {
|
||||||
type Resource = BgMapResource;
|
type Params = BgMapParams;
|
||||||
|
|
||||||
fn id(&self) -> &str {
|
fn sizes(&self) -> [[usize; 2]; 2] {
|
||||||
concat!(module_path!(), "::BgMapLoader")
|
[[8, 8], [8 * 64, 8 * 64]]
|
||||||
}
|
}
|
||||||
|
|
||||||
fn add(&self, resource: &Self::Resource) -> Option<VramImage> {
|
fn render(&mut self, params: &Self::Params, images: &mut [VramImage; 2]) {
|
||||||
match resource {
|
self.render_bgmap_cell(&mut images[0], params.cell_index, params.generic_palette);
|
||||||
BgMapResource::BgMap {
|
self.render_bgmap(
|
||||||
index,
|
&mut images[1],
|
||||||
generic_palette,
|
params.cell_index / 4096,
|
||||||
} => {
|
params.generic_palette,
|
||||||
let mut image = VramImage::new(64 * 8, 64 * 8);
|
);
|
||||||
self.update_bgmap(&mut image, *index, *generic_palette);
|
|
||||||
Some(image)
|
|
||||||
}
|
|
||||||
BgMapResource::Cell {
|
|
||||||
index,
|
|
||||||
generic_palette,
|
|
||||||
} => {
|
|
||||||
let mut image = VramImage::new(8, 8);
|
|
||||||
self.update_bgmap_cell(&mut image, *index, *generic_palette);
|
|
||||||
Some(image)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn update<'a>(
|
|
||||||
&'a self,
|
|
||||||
resources: impl Iterator<Item = (&'a Self::Resource, &'a mut VramImage)>,
|
|
||||||
) {
|
|
||||||
for (resource, image) in resources {
|
|
||||||
match resource {
|
|
||||||
BgMapResource::BgMap {
|
|
||||||
index,
|
|
||||||
generic_palette,
|
|
||||||
} => self.update_bgmap(image, *index, *generic_palette),
|
|
||||||
BgMapResource::Cell {
|
|
||||||
index,
|
|
||||||
generic_palette,
|
|
||||||
} => self.update_bgmap_cell(image, *index, *generic_palette),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,14 +10,15 @@ use serde::{Deserialize, Serialize};
|
||||||
use crate::{
|
use crate::{
|
||||||
emulator::SimId,
|
emulator::SimId,
|
||||||
memory::{MemoryMonitor, MemoryView},
|
memory::{MemoryMonitor, MemoryView},
|
||||||
vram::{VramImage, VramImageLoader, VramResource as _, VramTextureLoader},
|
vram::{VramImage, VramParams, VramProcessor, VramRenderer, VramTextureLoader},
|
||||||
window::{utils::UiExt as _, AppWindow},
|
window::{utils::UiExt as _, AppWindow},
|
||||||
};
|
};
|
||||||
|
|
||||||
use super::utils::{self, CharacterGrid};
|
use super::utils::{self, CharacterGrid};
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Serialize, Deserialize)]
|
#[derive(Debug, Default, Clone, Copy, PartialEq, Eq, Hash, Serialize, Deserialize)]
|
||||||
pub enum VramPalette {
|
pub enum VramPalette {
|
||||||
|
#[default]
|
||||||
Generic,
|
Generic,
|
||||||
Bg0,
|
Bg0,
|
||||||
Bg1,
|
Bg1,
|
||||||
|
@ -77,26 +78,34 @@ impl Display for VramPalette {
|
||||||
|
|
||||||
pub struct CharacterDataWindow {
|
pub struct CharacterDataWindow {
|
||||||
sim_id: SimId,
|
sim_id: SimId,
|
||||||
loader: Arc<VramTextureLoader<CharDataLoader>>,
|
loader: Arc<VramTextureLoader>,
|
||||||
brightness: MemoryView,
|
brightness: MemoryView,
|
||||||
palettes: MemoryView,
|
palettes: MemoryView,
|
||||||
palette: VramPalette,
|
palette: VramPalette,
|
||||||
index: usize,
|
index: usize,
|
||||||
index_str: String,
|
index_str: String,
|
||||||
|
params: VramParams<CharDataParams>,
|
||||||
scale: f32,
|
scale: f32,
|
||||||
show_grid: bool,
|
show_grid: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl CharacterDataWindow {
|
impl CharacterDataWindow {
|
||||||
pub fn new(sim_id: SimId, memory: &mut MemoryMonitor) -> Self {
|
pub fn new(sim_id: SimId, memory: &mut MemoryMonitor, vram: &mut VramProcessor) -> Self {
|
||||||
|
let renderer = CharDataRenderer::new(sim_id, memory);
|
||||||
|
let ([char, chardata], params) = vram.add(renderer);
|
||||||
|
let loader = VramTextureLoader::new([
|
||||||
|
("vram://char".into(), char),
|
||||||
|
("vram://chardata".into(), chardata),
|
||||||
|
]);
|
||||||
Self {
|
Self {
|
||||||
sim_id,
|
sim_id,
|
||||||
loader: Arc::new(VramTextureLoader::new(CharDataLoader::new(sim_id, memory))),
|
loader: Arc::new(loader),
|
||||||
brightness: memory.view(sim_id, 0x0005f824, 8),
|
brightness: memory.view(sim_id, 0x0005f824, 8),
|
||||||
palettes: memory.view(sim_id, 0x0005f860, 16),
|
palettes: memory.view(sim_id, 0x0005f860, 16),
|
||||||
palette: VramPalette::Generic,
|
palette: params.palette,
|
||||||
index: 0,
|
index: params.index,
|
||||||
index_str: "0".into(),
|
index_str: params.index.to_string(),
|
||||||
|
params,
|
||||||
scale: 4.0,
|
scale: 4.0,
|
||||||
show_grid: true,
|
show_grid: true,
|
||||||
}
|
}
|
||||||
|
@ -150,11 +159,7 @@ impl CharacterDataWindow {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
let resource = CharDataResource::Character {
|
let image = Image::new("vram://char")
|
||||||
palette: self.palette,
|
|
||||||
index: self.index,
|
|
||||||
};
|
|
||||||
let image = Image::new(resource.to_uri())
|
|
||||||
.maintain_aspect_ratio(true)
|
.maintain_aspect_ratio(true)
|
||||||
.tint(Color32::RED)
|
.tint(Color32::RED)
|
||||||
.texture_options(TextureOptions::NEAREST);
|
.texture_options(TextureOptions::NEAREST);
|
||||||
|
@ -207,6 +212,11 @@ impl CharacterDataWindow {
|
||||||
ui.checkbox(&mut self.show_grid, "Show grid");
|
ui.checkbox(&mut self.show_grid, "Show grid");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
self.params.write(CharDataParams {
|
||||||
|
palette: self.palette,
|
||||||
|
index: self.index,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
fn load_palette_colors(&self) -> [u8; 4] {
|
fn load_palette_colors(&self) -> [u8; 4] {
|
||||||
|
@ -220,10 +230,7 @@ impl CharacterDataWindow {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn show_chardata(&mut self, ui: &mut Ui) {
|
fn show_chardata(&mut self, ui: &mut Ui) {
|
||||||
let resource = CharDataResource::CharacterData {
|
let grid = CharacterGrid::new("vram://chardata")
|
||||||
palette: self.palette,
|
|
||||||
};
|
|
||||||
let grid = CharacterGrid::new(resource.to_uri())
|
|
||||||
.with_scale(self.scale)
|
.with_scale(self.scale)
|
||||||
.with_grid(self.show_grid)
|
.with_grid(self.show_grid)
|
||||||
.with_selected(self.index);
|
.with_selected(self.index);
|
||||||
|
@ -254,7 +261,6 @@ impl AppWindow for CharacterDataWindow {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn show(&mut self, ctx: &Context) {
|
fn show(&mut self, ctx: &Context) {
|
||||||
self.loader.begin_pass();
|
|
||||||
CentralPanel::default().show(ctx, |ui| {
|
CentralPanel::default().show(ctx, |ui| {
|
||||||
ui.horizontal_top(|ui| {
|
ui.horizontal_top(|ui| {
|
||||||
StripBuilder::new(ui)
|
StripBuilder::new(ui)
|
||||||
|
@ -279,6 +285,82 @@ enum CharDataResource {
|
||||||
CharacterData { palette: VramPalette },
|
CharacterData { palette: VramPalette },
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Default, PartialEq, Eq)]
|
||||||
|
struct CharDataParams {
|
||||||
|
palette: VramPalette,
|
||||||
|
index: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
struct CharDataRenderer {
|
||||||
|
chardata: MemoryView,
|
||||||
|
brightness: MemoryView,
|
||||||
|
palettes: MemoryView,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl VramRenderer<2> for CharDataRenderer {
|
||||||
|
type Params = CharDataParams;
|
||||||
|
|
||||||
|
fn sizes(&self) -> [[usize; 2]; 2] {
|
||||||
|
[[8, 8], [16 * 8, 128 * 8]]
|
||||||
|
}
|
||||||
|
|
||||||
|
fn render(&mut self, params: &Self::Params, image: &mut [VramImage; 2]) {
|
||||||
|
self.render_character(&mut image[0], params.palette, params.index);
|
||||||
|
self.render_character_data(&mut image[1], params.palette);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl CharDataRenderer {
|
||||||
|
pub fn new(sim_id: SimId, memory: &mut MemoryMonitor) -> Self {
|
||||||
|
Self {
|
||||||
|
chardata: memory.view(sim_id, 0x00078000, 0x8000),
|
||||||
|
brightness: memory.view(sim_id, 0x0005f824, 8),
|
||||||
|
palettes: memory.view(sim_id, 0x0005f860, 16),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn render_character(&self, image: &mut VramImage, palette: VramPalette, index: usize) {
|
||||||
|
if index >= 2048 {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let palette = self.load_palette(palette);
|
||||||
|
let chardata = self.chardata.borrow();
|
||||||
|
let character = chardata.range::<u16>(index * 8, 8);
|
||||||
|
for (row, pixels) in character.iter().enumerate() {
|
||||||
|
for col in 0..8 {
|
||||||
|
let char = (pixels >> (col * 2)) & 0x03;
|
||||||
|
image.write((col, row), palette[char as usize]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn render_character_data(&self, image: &mut VramImage, palette: VramPalette) {
|
||||||
|
let palette = self.load_palette(palette);
|
||||||
|
let chardata = self.chardata.borrow();
|
||||||
|
for (row, pixels) in chardata.range::<u16>(0, 8 * 2048).iter().enumerate() {
|
||||||
|
let char_index = row / 8;
|
||||||
|
let row_index = row % 8;
|
||||||
|
let x = (char_index % 16) * 8;
|
||||||
|
let y = (char_index / 16) * 8 + row_index;
|
||||||
|
for col in 0..8 {
|
||||||
|
let char = (pixels >> (col * 2)) & 0x03;
|
||||||
|
image.write((x + col, y), palette[char as usize]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn load_palette(&self, palette: VramPalette) -> [u8; 4] {
|
||||||
|
let Some(offset) = palette.offset() else {
|
||||||
|
return utils::GENERIC_PALETTE;
|
||||||
|
};
|
||||||
|
let palette = self.palettes.borrow().read(offset);
|
||||||
|
let brightnesses = self.brightness.borrow();
|
||||||
|
let brts = brightnesses.range(0, 8);
|
||||||
|
utils::parse_palette(palette, brts)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
struct CharDataLoader {
|
struct CharDataLoader {
|
||||||
chardata: MemoryView,
|
chardata: MemoryView,
|
||||||
brightness: MemoryView,
|
brightness: MemoryView,
|
||||||
|
@ -373,3 +455,4 @@ impl VramImageLoader for CharDataLoader {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
Loading…
Reference in New Issue