VIP inspection tooling #4
			
				
			
		
		
		
	| 
						 | 
					@ -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) => {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										271
									
								
								src/vram.rs
								
								
								
								
							
							
						
						
									
										271
									
								
								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) {
 | 
					impl TextureLoader for VramTextureLoader {
 | 
				
			||||||
        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> {
 | 
					 | 
				
			||||||
    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