Compare commits
	
		
			No commits in common. "ad5f1e8abe31b875d32eebed900a363c31615181" and "24d1c54299bb9795ce11f605e345ee812abc57ac" have entirely different histories.
		
	
	
		
			ad5f1e8abe
			...
			24d1c54299
		
	
		
					 10 changed files with 684 additions and 835 deletions
				
			
		
							
								
								
									
										637
									
								
								Cargo.lock
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										637
									
								
								Cargo.lock
									
										
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										14
									
								
								Cargo.toml
									
										
									
									
									
								
							
							
						
						
									
										14
									
								
								Cargo.toml
									
										
									
									
									
								
							| 
						 | 
				
			
			@ -11,14 +11,12 @@ log = "0.4"
 | 
			
		|||
env_logger = "0.11"
 | 
			
		||||
clap = { version = "4.5", features = ["derive"] }
 | 
			
		||||
# for drawing pixels onto the surface of the window
 | 
			
		||||
pixels = "0.15"
 | 
			
		||||
pixels = "0.14"
 | 
			
		||||
# I should not need this as a direct dependency, but then I cannot spell the types needed to use font-kit...
 | 
			
		||||
pathfinder_geometry = "0.5.1"
 | 
			
		||||
font-kit = "0.14.2"
 | 
			
		||||
thiserror = "2.0"
 | 
			
		||||
 | 
			
		||||
[dependencies.servicepoint]
 | 
			
		||||
version = "0.13.0"
 | 
			
		||||
version = "0.12.0"
 | 
			
		||||
features = ["all_compressions"]
 | 
			
		||||
 | 
			
		||||
[dependencies.winit]
 | 
			
		||||
| 
						 | 
				
			
			@ -26,3 +24,11 @@ version = "0.30"
 | 
			
		|||
features = ["rwh_05"]
 | 
			
		||||
default-features = true
 | 
			
		||||
 | 
			
		||||
[target.'cfg(target_os = "linux")'.dependencies.font-kit]
 | 
			
		||||
version = "0.14.2"
 | 
			
		||||
features = ["loader-freetype-default", "source-fontconfig-dlopen"]
 | 
			
		||||
default-features = false
 | 
			
		||||
 | 
			
		||||
[target.'cfg(target_os = "macos")'.dependencies.font-kit]
 | 
			
		||||
version = "0.14.2"
 | 
			
		||||
default-features = true
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										14
									
								
								flake.lock
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										14
									
								
								flake.lock
									
										
									
										generated
									
									
									
								
							| 
						 | 
				
			
			@ -7,11 +7,11 @@
 | 
			
		|||
        ]
 | 
			
		||||
      },
 | 
			
		||||
      "locked": {
 | 
			
		||||
        "lastModified": 1736429655,
 | 
			
		||||
        "narHash": "sha256-BwMekRuVlSB9C0QgwKMICiJ5EVbLGjfe4qyueyNQyGI=",
 | 
			
		||||
        "lastModified": 1721727458,
 | 
			
		||||
        "narHash": "sha256-r/xppY958gmZ4oTfLiHN0ZGuQ+RSTijDblVgVLFi1mw=",
 | 
			
		||||
        "owner": "nix-community",
 | 
			
		||||
        "repo": "naersk",
 | 
			
		||||
        "rev": "0621e47bd95542b8e1ce2ee2d65d6a1f887a13ce",
 | 
			
		||||
        "rev": "3fb418eaf352498f6b6c30592e3beb63df42ef11",
 | 
			
		||||
        "type": "github"
 | 
			
		||||
      },
 | 
			
		||||
      "original": {
 | 
			
		||||
| 
						 | 
				
			
			@ -37,16 +37,16 @@
 | 
			
		|||
    },
 | 
			
		||||
    "nixpkgs": {
 | 
			
		||||
      "locked": {
 | 
			
		||||
        "lastModified": 1736200483,
 | 
			
		||||
        "narHash": "sha256-JO+lFN2HsCwSLMUWXHeOad6QUxOuwe9UOAF/iSl1J4I=",
 | 
			
		||||
        "lastModified": 1732749044,
 | 
			
		||||
        "narHash": "sha256-T38FQOg0BV5M8FN1712fovzNakSOENEYs+CSkg31C9Y=",
 | 
			
		||||
        "owner": "nixos",
 | 
			
		||||
        "repo": "nixpkgs",
 | 
			
		||||
        "rev": "3f0a8ac25fb674611b98089ca3a5dd6480175751",
 | 
			
		||||
        "rev": "0c5b4ecbed5b155b705336aa96d878e55acd8685",
 | 
			
		||||
        "type": "github"
 | 
			
		||||
      },
 | 
			
		||||
      "original": {
 | 
			
		||||
        "owner": "nixos",
 | 
			
		||||
        "ref": "nixos-24.11",
 | 
			
		||||
        "ref": "nixos-24.05",
 | 
			
		||||
        "repo": "nixpkgs",
 | 
			
		||||
        "type": "github"
 | 
			
		||||
      }
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										28
									
								
								flake.nix
									
										
									
									
									
								
							
							
						
						
									
										28
									
								
								flake.nix
									
										
									
									
									
								
							| 
						 | 
				
			
			@ -2,7 +2,7 @@
 | 
			
		|||
  description = "Flake for servicepoint-simulator";
 | 
			
		||||
 | 
			
		||||
  inputs = {
 | 
			
		||||
    nixpkgs.url = "github:nixos/nixpkgs/nixos-24.11";
 | 
			
		||||
    nixpkgs.url = "github:nixos/nixpkgs/nixos-24.05";
 | 
			
		||||
    nix-filter.url = "github:numtide/nix-filter";
 | 
			
		||||
    naersk = {
 | 
			
		||||
      url = "github:nix-community/naersk";
 | 
			
		||||
| 
						 | 
				
			
			@ -70,25 +70,13 @@
 | 
			
		|||
              with pkgs;
 | 
			
		||||
              [
 | 
			
		||||
                xe
 | 
			
		||||
                xz
 | 
			
		||||
                lzma
 | 
			
		||||
              ]
 | 
			
		||||
              ++ lib.optionals pkgs.stdenv.isLinux (
 | 
			
		||||
              ++ (lib.optionals pkgs.stdenv.isLinux (
 | 
			
		||||
                with pkgs;
 | 
			
		||||
                [
 | 
			
		||||
                  # gpu
 | 
			
		||||
                  libGL
 | 
			
		||||
                  vulkan-headers
 | 
			
		||||
                  vulkan-loader
 | 
			
		||||
                  vulkan-tools vulkan-tools-lunarg
 | 
			
		||||
                  vulkan-extension-layer
 | 
			
		||||
                  vulkan-validation-layers
 | 
			
		||||
 | 
			
		||||
                  # keyboard
 | 
			
		||||
                  libxkbcommon
 | 
			
		||||
 | 
			
		||||
                   # font loading
 | 
			
		||||
                  fontconfig
 | 
			
		||||
                  freetype
 | 
			
		||||
                  libGL
 | 
			
		||||
 | 
			
		||||
                  # WINIT_UNIX_BACKEND=wayland
 | 
			
		||||
                  wayland
 | 
			
		||||
| 
						 | 
				
			
			@ -100,15 +88,15 @@
 | 
			
		|||
                  xorg.libX11
 | 
			
		||||
                  xorg.libX11.dev
 | 
			
		||||
                ]
 | 
			
		||||
              )
 | 
			
		||||
              ++ lib.optionals pkgs.stdenv.isDarwin (
 | 
			
		||||
              ))
 | 
			
		||||
              ++ (lib.optionals pkgs.stdenv.isDarwin (
 | 
			
		||||
                with pkgs.darwin.apple_sdk.frameworks;
 | 
			
		||||
                [
 | 
			
		||||
                  Carbon
 | 
			
		||||
                  QuartzCore
 | 
			
		||||
                  AppKit
 | 
			
		||||
                ]
 | 
			
		||||
              );
 | 
			
		||||
              ));
 | 
			
		||||
 | 
			
		||||
            postInstall = ''
 | 
			
		||||
              wrapProgram $out/bin/servicepoint-simulator \
 | 
			
		||||
| 
						 | 
				
			
			@ -139,7 +127,7 @@
 | 
			
		|||
        {
 | 
			
		||||
          default = pkgs.mkShell rec {
 | 
			
		||||
            inputsFrom = [ self.packages.${system}.default ];
 | 
			
		||||
            packages = [ rust-toolchain pkgs.gdb ];
 | 
			
		||||
            packages = [ rust-toolchain ];
 | 
			
		||||
            LD_LIBRARY_PATH = "${pkgs.lib.makeLibraryPath (builtins.concatMap (d: d.buildInputs) inputsFrom)}";
 | 
			
		||||
            RUST_SRC_PATH = "${pkgs.rust.packages.stable.rustPlatform.rustLibSrc}";
 | 
			
		||||
          };
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,17 +1,16 @@
 | 
			
		|||
use log::{debug, error, info, trace, warn};
 | 
			
		||||
use servicepoint::{
 | 
			
		||||
    Bitmap, BrightnessGrid, CharGrid, Command, Cp437Grid, Grid, Origin, Tiles,
 | 
			
		||||
    PIXEL_COUNT, PIXEL_WIDTH, TILE_SIZE,
 | 
			
		||||
};
 | 
			
		||||
use std::sync::{RwLock, RwLockWriteGuard};
 | 
			
		||||
 | 
			
		||||
use crate::font::Cp437Font;
 | 
			
		||||
use crate::font_renderer::FontRenderer8x8;
 | 
			
		||||
use log::{debug, error, info, trace, warn};
 | 
			
		||||
use servicepoint::{
 | 
			
		||||
    Bitmap, BrightnessGrid, Command, Cp437Grid, Grid, Origin, Tiles,
 | 
			
		||||
    PIXEL_COUNT, PIXEL_WIDTH, TILE_SIZE,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
use crate::font::BitmapFont;
 | 
			
		||||
 | 
			
		||||
pub(crate) fn execute_command(
 | 
			
		||||
    command: Command,
 | 
			
		||||
    cp436_font: &Cp437Font,
 | 
			
		||||
    utf8_font: &FontRenderer8x8,
 | 
			
		||||
    font: &BitmapFont,
 | 
			
		||||
    display_ref: &RwLock<Bitmap>,
 | 
			
		||||
    luma_ref: &RwLock<BrightnessGrid>,
 | 
			
		||||
) -> bool {
 | 
			
		||||
| 
						 | 
				
			
			@ -31,7 +30,7 @@ pub(crate) fn execute_command(
 | 
			
		|||
        }
 | 
			
		||||
        Command::Cp437Data(origin, grid) => {
 | 
			
		||||
            let mut display = display_ref.write().unwrap();
 | 
			
		||||
            print_cp437_data(origin, &grid, cp436_font, &mut display);
 | 
			
		||||
            print_cp437_data(origin, &grid, font, &mut display);
 | 
			
		||||
        }
 | 
			
		||||
        #[allow(deprecated)]
 | 
			
		||||
        Command::BitmapLegacy => {
 | 
			
		||||
| 
						 | 
				
			
			@ -100,10 +99,6 @@ pub(crate) fn execute_command(
 | 
			
		|||
        Command::FadeOut => {
 | 
			
		||||
            error!("command not implemented: {command:?}")
 | 
			
		||||
        }
 | 
			
		||||
        Command::Utf8Data(origin, grid) => {
 | 
			
		||||
            let mut display = display_ref.write().unwrap();
 | 
			
		||||
            print_utf8_data(origin, &grid, utf8_font, &mut display);
 | 
			
		||||
        }
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    true
 | 
			
		||||
| 
						 | 
				
			
			@ -121,7 +116,7 @@ fn check_bitmap_valid(offset: u16, payload_len: usize) -> bool {
 | 
			
		|||
fn print_cp437_data(
 | 
			
		||||
    origin: Origin<Tiles>,
 | 
			
		||||
    grid: &Cp437Grid,
 | 
			
		||||
    font: &Cp437Font,
 | 
			
		||||
    font: &BitmapFont,
 | 
			
		||||
    display: &mut RwLockWriteGuard<Bitmap>,
 | 
			
		||||
) {
 | 
			
		||||
    let Origin { x, y, .. } = origin;
 | 
			
		||||
| 
						 | 
				
			
			@ -150,33 +145,6 @@ fn print_cp437_data(
 | 
			
		|||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fn print_utf8_data(
 | 
			
		||||
    origin: Origin<Tiles>,
 | 
			
		||||
    grid: &CharGrid,
 | 
			
		||||
    font: &FontRenderer8x8,
 | 
			
		||||
    display: &mut RwLockWriteGuard<Bitmap>,
 | 
			
		||||
) {
 | 
			
		||||
    let Origin { x, y, .. } = origin;
 | 
			
		||||
    for char_y in 0usize..grid.height() {
 | 
			
		||||
        for char_x in 0usize..grid.width() {
 | 
			
		||||
            let char = grid.get(char_x, char_y);
 | 
			
		||||
            trace!("drawing {char}");
 | 
			
		||||
 | 
			
		||||
            let tile_x = char_x + x;
 | 
			
		||||
            let tile_y = char_y + y;
 | 
			
		||||
 | 
			
		||||
            if let Err(e) = font.render(
 | 
			
		||||
                char,
 | 
			
		||||
                display,
 | 
			
		||||
                Origin::new(tile_x * TILE_SIZE, tile_y * TILE_SIZE),
 | 
			
		||||
            ) {
 | 
			
		||||
                error!("stopping drawing text because char draw failed: {e}");
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fn print_pixel_grid(
 | 
			
		||||
    offset_x: usize,
 | 
			
		||||
    offset_y: usize,
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										334
									
								
								src/font.rs
									
										
									
									
									
								
							
							
						
						
									
										334
									
								
								src/font.rs
									
										
									
									
									
								
							| 
						 | 
				
			
			@ -1,297 +1,73 @@
 | 
			
		|||
use servicepoint::{Bitmap, DataRef, TILE_SIZE};
 | 
			
		||||
use crate::static_font;
 | 
			
		||||
use font_kit::canvas::{Canvas, Format, RasterizationOptions};
 | 
			
		||||
use font_kit::font::Font;
 | 
			
		||||
use font_kit::hinting::HintingOptions;
 | 
			
		||||
use pathfinder_geometry::transform2d::Transform2F;
 | 
			
		||||
use pathfinder_geometry::vector::{vec2f, vec2i};
 | 
			
		||||
use servicepoint::{Bitmap, Grid, TILE_SIZE};
 | 
			
		||||
 | 
			
		||||
const DEFAULT_FONT_FILE: &[u8] = include_bytes!("../Web437_IBM_BIOS.woff");
 | 
			
		||||
 | 
			
		||||
const CHAR_COUNT: usize = u8::MAX as usize + 1;
 | 
			
		||||
 | 
			
		||||
pub struct Cp437Font {
 | 
			
		||||
pub struct BitmapFont {
 | 
			
		||||
    bitmaps: [Bitmap; CHAR_COUNT],
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
impl Cp437Font {
 | 
			
		||||
impl BitmapFont {
 | 
			
		||||
    pub fn new(bitmaps: [Bitmap; CHAR_COUNT]) -> Self {
 | 
			
		||||
        Self { bitmaps }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    pub fn load(font: Font, size: usize) -> BitmapFont {
 | 
			
		||||
        let mut bitmaps =
 | 
			
		||||
            core::array::from_fn(|_| Bitmap::new(TILE_SIZE, TILE_SIZE));
 | 
			
		||||
        let mut canvas =
 | 
			
		||||
            Canvas::new(vec2i(size as i32, size as i32), Format::A8);
 | 
			
		||||
        let size_f = size as f32;
 | 
			
		||||
        let transform = Transform2F::default();
 | 
			
		||||
 | 
			
		||||
        for char_code in u8::MIN..=u8::MAX {
 | 
			
		||||
            let char = char_code as char;
 | 
			
		||||
            let glyph_id = match font.glyph_for_char(char) {
 | 
			
		||||
                None => continue,
 | 
			
		||||
                Some(val) => val,
 | 
			
		||||
            };
 | 
			
		||||
 | 
			
		||||
            canvas.pixels.fill(0);
 | 
			
		||||
            font.rasterize_glyph(
 | 
			
		||||
                &mut canvas,
 | 
			
		||||
                glyph_id,
 | 
			
		||||
                size_f,
 | 
			
		||||
                Transform2F::from_translation(vec2f(0f32, size_f)) * transform,
 | 
			
		||||
                HintingOptions::None,
 | 
			
		||||
                RasterizationOptions::GrayscaleAa,
 | 
			
		||||
            )
 | 
			
		||||
            .unwrap();
 | 
			
		||||
 | 
			
		||||
            assert_eq!(canvas.pixels.len(), size * size);
 | 
			
		||||
            assert_eq!(canvas.stride, size);
 | 
			
		||||
 | 
			
		||||
            let bitmap = &mut bitmaps[char_code as usize];
 | 
			
		||||
            for y in 0..TILE_SIZE {
 | 
			
		||||
                for x in 0..TILE_SIZE {
 | 
			
		||||
                    let index = x + y * TILE_SIZE;
 | 
			
		||||
                    let canvas_val = canvas.pixels[index] != 0;
 | 
			
		||||
                    bitmap.set(x, y, canvas_val);
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        Self::new(bitmaps)
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    pub fn get_bitmap(&self, char_code: u8) -> &Bitmap {
 | 
			
		||||
        &self.bitmaps[char_code as usize]
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
impl Default for Cp437Font {
 | 
			
		||||
impl Default for BitmapFont {
 | 
			
		||||
    fn default() -> Self {
 | 
			
		||||
        load_static()
 | 
			
		||||
        static_font::load_static()
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Font from the display firmware `cape-cccb-apd/cp437font_linear.h`
 | 
			
		||||
pub(crate) const CP437_FONT_LINEAR: [u64; 256] = [
 | 
			
		||||
    0x0000000000000000, // 0x00
 | 
			
		||||
    0x003854ba82aa4438, // 0x01
 | 
			
		||||
    0x003844bafed67c38, // 0x02
 | 
			
		||||
    0x0010387cfeee4400, // 0x03
 | 
			
		||||
    0x0010387cfe7c3810, // 0x04
 | 
			
		||||
    0x003810d6fefe3838, // 0x05
 | 
			
		||||
    0x003810d6fe7c3810, // 0x06
 | 
			
		||||
    0x0000387c7c7c3800, // 0x07
 | 
			
		||||
    0x00fec6828282c6fe, // 0x08
 | 
			
		||||
    0x0000384444443800, // 0x09
 | 
			
		||||
    0x00fec6bababac6fe, // 0x0a
 | 
			
		||||
    0x007088888a7a061e, // 0x0b
 | 
			
		||||
    0x1038103844444438, // 0x0c
 | 
			
		||||
    0x0030301010141418, // 0x0d
 | 
			
		||||
    0xc0c64642724e720e, // 0x0e
 | 
			
		||||
    0x00927c44c6447c92, // 0x0f
 | 
			
		||||
    0x00c0f0fcfefcf0c0, // 0x10
 | 
			
		||||
    0x00061e7efe7e1e06, // 0x11
 | 
			
		||||
    0x1038541010543810, // 0x12
 | 
			
		||||
    0x2800282828282828, // 0x13
 | 
			
		||||
    0x000e0a0a7a8a8a7e, // 0x14
 | 
			
		||||
    0x0038441c28704438, // 0x15
 | 
			
		||||
    0x000000ffff000000, // 0x16
 | 
			
		||||
    0xfe10385410543810, // 0x17
 | 
			
		||||
    0x1010101010543810, // 0x18
 | 
			
		||||
    0x1038541010101010, // 0x19
 | 
			
		||||
    0x00000804fe040800, // 0x1a
 | 
			
		||||
    0x00002040fe402000, // 0x1b
 | 
			
		||||
    0xffffc0c0c0c0c0c0, // 0x1c
 | 
			
		||||
    0x00002844fe442800, // 0x1d
 | 
			
		||||
    0x00fefe7c7c383810, // 0x1e
 | 
			
		||||
    0x001038387c7cfefe, // 0x1f
 | 
			
		||||
    0x0000000000000000, // 0x20
 | 
			
		||||
    0x1000101010101010, // 0x21
 | 
			
		||||
    0x0000000000505028, // 0x22
 | 
			
		||||
    0x00247e24247e2400, // 0x23
 | 
			
		||||
    0x1038541830543810, // 0x24
 | 
			
		||||
    0x00844a2a54a8a442, // 0x25
 | 
			
		||||
    0x003a444a32484830, // 0x26
 | 
			
		||||
    0x0000000000201010, // 0x27
 | 
			
		||||
    0x0810202020201008, // 0x28
 | 
			
		||||
    0x2010080808081020, // 0x29
 | 
			
		||||
    0x0010543854100000, // 0x2a
 | 
			
		||||
    0x0010107c10100000, // 0x2b
 | 
			
		||||
    0x2010100000000000, // 0x2c
 | 
			
		||||
    0x0000007c00000000, // 0x2d
 | 
			
		||||
    0x0010000000000000, // 0x2e
 | 
			
		||||
    0x4020201010080804, // 0x2f
 | 
			
		||||
    0x0038444454444438, // 0x30
 | 
			
		||||
    0x007c101010503010, // 0x31
 | 
			
		||||
    0x007c201008044438, // 0x32
 | 
			
		||||
    0x003844043810087c, // 0x33
 | 
			
		||||
    0x0004047e24140c04, // 0x34
 | 
			
		||||
    0x003844047840407c, // 0x35
 | 
			
		||||
    0x003844447840201c, // 0x36
 | 
			
		||||
    0x004020100804047c, // 0x37
 | 
			
		||||
    0x0038444438444438, // 0x38
 | 
			
		||||
    0x007008043c444438, // 0x39
 | 
			
		||||
    0x0000100000100000, // 0x3a
 | 
			
		||||
    0x2010100000100000, // 0x3b
 | 
			
		||||
    0x0006186018060000, // 0x3c
 | 
			
		||||
    0x00007c007c000000, // 0x3d
 | 
			
		||||
    0x00c0300c30c00000, // 0x3e
 | 
			
		||||
    0x1000384038044438, // 0x3f
 | 
			
		||||
    0x001c2248564a221c, // 0x40
 | 
			
		||||
    0x0042427e24241818, // 0x41
 | 
			
		||||
    0x007c42427c444478, // 0x42
 | 
			
		||||
    0x001c22404040221c, // 0x43
 | 
			
		||||
    0x0078444242424478, // 0x44
 | 
			
		||||
    0x007e40407e40407e, // 0x45
 | 
			
		||||
    0x004040407e40407e, // 0x46
 | 
			
		||||
    0x001c22424e40221c, // 0x47
 | 
			
		||||
    0x004242427e424242, // 0x48
 | 
			
		||||
    0x007c10101010107c, // 0x49
 | 
			
		||||
    0x003844040404047c, // 0x4a
 | 
			
		||||
    0x0042444870484442, // 0x4b
 | 
			
		||||
    0x007e404040404040, // 0x4c
 | 
			
		||||
    0x0082828292aac682, // 0x4d
 | 
			
		||||
    0x004242464a526242, // 0x4e
 | 
			
		||||
    0x0018244242422418, // 0x4f
 | 
			
		||||
    0x004040407c42427c, // 0x50
 | 
			
		||||
    0x001a244a42422418, // 0x51
 | 
			
		||||
    0x004244487c42427c, // 0x52
 | 
			
		||||
    0x003c42023c40423c, // 0x53
 | 
			
		||||
    0x00101010101010fe, // 0x54
 | 
			
		||||
    0x003c424242424242, // 0x55
 | 
			
		||||
    0x0010282844448282, // 0x56
 | 
			
		||||
    0x0044446caa929282, // 0x57
 | 
			
		||||
    0x0082442810284482, // 0x58
 | 
			
		||||
    0x0010101010284482, // 0x59
 | 
			
		||||
    0x007e20100804027e, // 0x5a
 | 
			
		||||
    0x3820202020202038, // 0x5b
 | 
			
		||||
    0x0408081010202040, // 0x5c
 | 
			
		||||
    0x3808080808080838, // 0x5d
 | 
			
		||||
    0x0000000000442810, // 0x5e
 | 
			
		||||
    0x007e000000000000, // 0x5f
 | 
			
		||||
    0x0000000000080810, // 0x60
 | 
			
		||||
    0x003c443c04380000, // 0x61
 | 
			
		||||
    0x0038444444784040, // 0x62
 | 
			
		||||
    0x0038444044380000, // 0x63
 | 
			
		||||
    0x003c4444443c0404, // 0x64
 | 
			
		||||
    0x003c407844380000, // 0x65
 | 
			
		||||
    0x2020202078202418, // 0x66
 | 
			
		||||
    0x78043c44443e0000, // 0x67
 | 
			
		||||
    0x0044444464584040, // 0x68
 | 
			
		||||
    0x001c101010700010, // 0x69
 | 
			
		||||
    0x38440404047c0010, // 0x6a
 | 
			
		||||
    0x0022243828242020, // 0x6b
 | 
			
		||||
    0x0018242020202020, // 0x6c
 | 
			
		||||
    0x0054545454780000, // 0x6d
 | 
			
		||||
    0x0044444464580000, // 0x6e
 | 
			
		||||
    0x0038444444380000, // 0x6f
 | 
			
		||||
    0x4040784444780000, // 0x70
 | 
			
		||||
    0x04043c44443c0000, // 0x71
 | 
			
		||||
    0x0040404064580000, // 0x72
 | 
			
		||||
    0x00384418201c0000, // 0x73
 | 
			
		||||
    0x0018242020782020, // 0x74
 | 
			
		||||
    0x0038444444440000, // 0x75
 | 
			
		||||
    0x0010282844440000, // 0x76
 | 
			
		||||
    0x0028285454440000, // 0x77
 | 
			
		||||
    0x0044281028440000, // 0x78
 | 
			
		||||
    0x38043c4444440000, // 0x79
 | 
			
		||||
    0x007c2010087c0000, // 0x7a
 | 
			
		||||
    0x0c0808083008080c, // 0x7b
 | 
			
		||||
    0x1010101010101010, // 0x7c
 | 
			
		||||
    0x301010100c101030, // 0x7d
 | 
			
		||||
    0x0000004c32000000, // 0x7e
 | 
			
		||||
    0xfe82828282442810, // 0x7f
 | 
			
		||||
    0x18083c428080423c, // 0x80
 | 
			
		||||
    0x0038444444440028, // 0x81
 | 
			
		||||
    0x003c407844381008, // 0x82
 | 
			
		||||
    0x003c443c04382810, // 0x83
 | 
			
		||||
    0x003c443c04380044, // 0x84
 | 
			
		||||
    0x003c443c04381020, // 0x85
 | 
			
		||||
    0x003c443c04382838, // 0x86
 | 
			
		||||
    0x1038444044380000, // 0x87
 | 
			
		||||
    0x003c407844382810, // 0x88
 | 
			
		||||
    0x003c407844380044, // 0x89
 | 
			
		||||
    0x003c407844381020, // 0x8a
 | 
			
		||||
    0x001c101010700028, // 0x8b
 | 
			
		||||
    0x001c101010702810, // 0x8c
 | 
			
		||||
    0x001c101010701020, // 0x8d
 | 
			
		||||
    0x0082827c44282892, // 0x8e
 | 
			
		||||
    0x0082827c44281038, // 0x8f
 | 
			
		||||
    0x00fe80fe80fe1008, // 0x90
 | 
			
		||||
    0x007e907c126c0000, // 0x91
 | 
			
		||||
    0x008e88784e28281e, // 0x92
 | 
			
		||||
    0x0038444438002810, // 0x93
 | 
			
		||||
    0x0038444444380028, // 0x94
 | 
			
		||||
    0x0038444438001020, // 0x95
 | 
			
		||||
    0x0038444444002810, // 0x96
 | 
			
		||||
    0x0038444444001020, // 0x97
 | 
			
		||||
    0x38043c4444440028, // 0x98
 | 
			
		||||
    0x0038448282443882, // 0x99
 | 
			
		||||
    0x0038448282820082, // 0x9a
 | 
			
		||||
    0x1038444044381010, // 0x9b
 | 
			
		||||
    0x00fc4240f0404438, // 0x9c
 | 
			
		||||
    0x107c107c10284482, // 0x9d
 | 
			
		||||
    0x001c22f840f8221c, // 0x9e
 | 
			
		||||
    0x205010103810120c, // 0x9f
 | 
			
		||||
    0x003c443c04381008, // 0xa0
 | 
			
		||||
    0x001c101010701008, // 0xa1
 | 
			
		||||
    0x0038444438001008, // 0xa2
 | 
			
		||||
    0x0038444444001008, // 0xa3
 | 
			
		||||
    0x0044446458004834, // 0xa4
 | 
			
		||||
    0x00464a5262424834, // 0xa5
 | 
			
		||||
    0x0000007090701060, // 0xa6
 | 
			
		||||
    0x0000007088888870, // 0xa7
 | 
			
		||||
    0x3844403804380010, // 0xa8
 | 
			
		||||
    0x0040407c00000000, // 0xa9
 | 
			
		||||
    0x0004047c00000000, // 0xaa
 | 
			
		||||
    0x0e84422c5048c442, // 0xab
 | 
			
		||||
    0x049e542c5448c442, // 0xac
 | 
			
		||||
    0x1010101010100010, // 0xad
 | 
			
		||||
    0x0024489048240000, // 0xae
 | 
			
		||||
    0x0048241224480000, // 0xaf
 | 
			
		||||
    0x1122448811224488, // 0xb0
 | 
			
		||||
    0x55aa55aa55aa55aa, // 0xb1
 | 
			
		||||
    0xddbb77eeddbb77ee, // 0xb2
 | 
			
		||||
    0x1010101010101010, // 0xb3
 | 
			
		||||
    0x101020c020101010, // 0xb4
 | 
			
		||||
    0x1020c000c0201010, // 0xb5
 | 
			
		||||
    0x2828488848282828, // 0xb6
 | 
			
		||||
    0x282850e000000000, // 0xb7
 | 
			
		||||
    0x1030d020c0000000, // 0xb8
 | 
			
		||||
    0x2848880888482828, // 0xb9
 | 
			
		||||
    0x2828282828282828, // 0xba
 | 
			
		||||
    0x2828c810e0000000, // 0xbb
 | 
			
		||||
    0x000000e010c82828, // 0xbc
 | 
			
		||||
    0x000000e050282828, // 0xbd
 | 
			
		||||
    0x0000c020d0301010, // 0xbe
 | 
			
		||||
    0x101020c000000000, // 0xbf
 | 
			
		||||
    0x0000000708101010, // 0xc0
 | 
			
		||||
    0x000000c728101010, // 0xc1
 | 
			
		||||
    0x101028c700000000, // 0xc2
 | 
			
		||||
    0x1010080708101010, // 0xc3
 | 
			
		||||
    0x000000ff00000000, // 0xc4
 | 
			
		||||
    0x101028c728101010, // 0xc5
 | 
			
		||||
    0x1008070007081010, // 0xc6
 | 
			
		||||
    0x2828242324282828, // 0xc7
 | 
			
		||||
    0x00000f1027282828, // 0xc8
 | 
			
		||||
    0x282827100f000000, // 0xc9
 | 
			
		||||
    0x0000ff0083442828, // 0xca
 | 
			
		||||
    0x28448300ff000000, // 0xcb
 | 
			
		||||
    0x2824232023242828, // 0xcc
 | 
			
		||||
    0x0000ff00ff000000, // 0xcd
 | 
			
		||||
    0x2844932893442828, // 0xce
 | 
			
		||||
    0x0000ff00c7281010, // 0xcf
 | 
			
		||||
    0x0000008344282828, // 0xd0
 | 
			
		||||
    0x1028c700ff000000, // 0xd1
 | 
			
		||||
    0x2828448300000000, // 0xd2
 | 
			
		||||
    0x0000000f14282828, // 0xd3
 | 
			
		||||
    0x0000070817181010, // 0xd4
 | 
			
		||||
    0x1018170807000000, // 0xd5
 | 
			
		||||
    0x2828140f00000000, // 0xd6
 | 
			
		||||
    0x2828448344282828, // 0xd7
 | 
			
		||||
    0x1028c700c7281010, // 0xd8
 | 
			
		||||
    0x000000c020101010, // 0xd9
 | 
			
		||||
    0x1010080700000000, // 0xda
 | 
			
		||||
    0xffffffffffffffff, // 0xdb
 | 
			
		||||
    0xffffffff00000000, // 0xdc
 | 
			
		||||
    0xf0f0f0f0f0f0f0f0, // 0xdd
 | 
			
		||||
    0x0f0f0f0f0f0f0f0f, // 0xde
 | 
			
		||||
    0x00000000ffffffff, // 0xdf
 | 
			
		||||
    0x0076888888740200, // 0xe0
 | 
			
		||||
    0x5844444458484830, // 0xe1
 | 
			
		||||
    0x00e04040404242fe, // 0xe2
 | 
			
		||||
    0x00242828a87c0000, // 0xe3
 | 
			
		||||
    0x00fe8240204082fe, // 0xe4
 | 
			
		||||
    0x00384444443e0000, // 0xe5
 | 
			
		||||
    0x405a644444440000, // 0xe6
 | 
			
		||||
    0x0010282020fc0000, // 0xe7
 | 
			
		||||
    0xfe103854543810fe, // 0xe8
 | 
			
		||||
    0x003844aabaaa4438, // 0xe9
 | 
			
		||||
    0x00ee448282824438, // 0xea
 | 
			
		||||
    0x007088887012221c, // 0xeb
 | 
			
		||||
    0x00006c92926c0000, // 0xec
 | 
			
		||||
    0x10107c92924c0000, // 0xed
 | 
			
		||||
    0x0038403040380000, // 0xee
 | 
			
		||||
    0x0082828282824438, // 0xef
 | 
			
		||||
    0x00fe00fe00fe0000, // 0xf0
 | 
			
		||||
    0x007c10107c101000, // 0xf1
 | 
			
		||||
    0x007e006018061860, // 0xf2
 | 
			
		||||
    0x007e000618601806, // 0xf3
 | 
			
		||||
    0x101010101010120c, // 0xf4
 | 
			
		||||
    0x6090101010101010, // 0xf5
 | 
			
		||||
    0x0010007c00100000, // 0xf6
 | 
			
		||||
    0x000c926c92600000, // 0xf7
 | 
			
		||||
    0x0000000030484830, // 0xf8
 | 
			
		||||
    0x0000103810000000, // 0xf9
 | 
			
		||||
    0x0000001000000000, // 0xfa
 | 
			
		||||
    0x10102828a4440202, // 0xfb
 | 
			
		||||
    0x00000000484848b0, // 0xfc
 | 
			
		||||
    0x0000000070201060, // 0xfd
 | 
			
		||||
    0x00007c7c7c7c7c00, // 0xfe
 | 
			
		||||
    0x0000000000000000, // 0xff
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
fn load_static() -> Cp437Font {
 | 
			
		||||
    let mut bitmaps =
 | 
			
		||||
        core::array::from_fn(|_| Bitmap::new(TILE_SIZE, TILE_SIZE));
 | 
			
		||||
 | 
			
		||||
    for (char_code, bitmap) in bitmaps.iter_mut().enumerate() {
 | 
			
		||||
        let bits = CP437_FONT_LINEAR[char_code];
 | 
			
		||||
        let mut bytes = bits.to_be_bytes();
 | 
			
		||||
        bytes.reverse();
 | 
			
		||||
        bitmap.data_ref_mut().copy_from_slice(bytes.as_slice());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    Cp437Font::new(bitmaps)
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,94 +0,0 @@
 | 
			
		|||
use crate::font_renderer::RenderError::{GlyphNotFound, OutOfBounds};
 | 
			
		||||
use font_kit::canvas::{Canvas, Format, RasterizationOptions};
 | 
			
		||||
use font_kit::error::GlyphLoadingError;
 | 
			
		||||
use font_kit::family_name::FamilyName;
 | 
			
		||||
use font_kit::font::Font;
 | 
			
		||||
use font_kit::hinting::HintingOptions;
 | 
			
		||||
use font_kit::properties::Properties;
 | 
			
		||||
use font_kit::source::SystemSource;
 | 
			
		||||
use pathfinder_geometry::transform2d::Transform2F;
 | 
			
		||||
use pathfinder_geometry::vector::{vec2f, vec2i};
 | 
			
		||||
use servicepoint::{Bitmap, Grid, Origin, Pixels, TILE_SIZE};
 | 
			
		||||
use std::sync::Mutex;
 | 
			
		||||
 | 
			
		||||
pub struct FontRenderer8x8 {
 | 
			
		||||
    font: Font,
 | 
			
		||||
    canvas: Mutex<Canvas>,
 | 
			
		||||
    fallback_char: Option<u32>,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#[derive(Debug, thiserror::Error)]
 | 
			
		||||
pub enum RenderError {
 | 
			
		||||
    #[error("Glyph not found for '{0}'")]
 | 
			
		||||
    GlyphNotFound(char),
 | 
			
		||||
    #[error(transparent)]
 | 
			
		||||
    GlyphLoadingError(#[from] GlyphLoadingError),
 | 
			
		||||
    #[error("out of bounds at {0} {1}")]
 | 
			
		||||
    OutOfBounds(usize, usize),
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
impl FontRenderer8x8 {
 | 
			
		||||
    pub fn new(font: Font, fallback_char: Option<char>) -> Self {
 | 
			
		||||
        let canvas =
 | 
			
		||||
            Canvas::new(vec2i(TILE_SIZE as i32, TILE_SIZE as i32), Format::A8);
 | 
			
		||||
        assert_eq!(canvas.pixels.len(), TILE_SIZE * TILE_SIZE);
 | 
			
		||||
        assert_eq!(canvas.stride, TILE_SIZE);
 | 
			
		||||
        let fallback_char = fallback_char.and_then(|c| font.glyph_for_char(c));
 | 
			
		||||
        let result = Self {
 | 
			
		||||
            font,
 | 
			
		||||
            fallback_char,
 | 
			
		||||
            canvas: Mutex::new(canvas),
 | 
			
		||||
        };
 | 
			
		||||
        result
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    pub fn render(
 | 
			
		||||
        &self,
 | 
			
		||||
        char: char,
 | 
			
		||||
        bitmap: &mut Bitmap,
 | 
			
		||||
        offset: Origin<Pixels>,
 | 
			
		||||
    ) -> Result<(), RenderError> {
 | 
			
		||||
        let mut canvas = self.canvas.lock().unwrap();
 | 
			
		||||
        let glyph_id = self.font.glyph_for_char(char).or(self.fallback_char);
 | 
			
		||||
        let glyph_id = match glyph_id {
 | 
			
		||||
            None => return Err(GlyphNotFound(char)),
 | 
			
		||||
            Some(val) => val,
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
        canvas.pixels.fill(0);
 | 
			
		||||
        self.font.rasterize_glyph(
 | 
			
		||||
            &mut canvas,
 | 
			
		||||
            glyph_id,
 | 
			
		||||
            TILE_SIZE as f32,
 | 
			
		||||
            Transform2F::from_translation(vec2f(0f32, TILE_SIZE as f32))
 | 
			
		||||
                * Transform2F::default(),
 | 
			
		||||
            HintingOptions::None,
 | 
			
		||||
            RasterizationOptions::Bilevel,
 | 
			
		||||
        )?;
 | 
			
		||||
 | 
			
		||||
        for y in 0..TILE_SIZE {
 | 
			
		||||
            for x in 0..TILE_SIZE {
 | 
			
		||||
                let index = x + y * TILE_SIZE;
 | 
			
		||||
                let canvas_val = canvas.pixels[index] != 0;
 | 
			
		||||
                let bitmap_x = (offset.x + x) as isize;
 | 
			
		||||
                let bitmap_y = (offset.y + y) as isize;
 | 
			
		||||
                if !bitmap.set_optional(bitmap_x, bitmap_y, canvas_val) {
 | 
			
		||||
                    return Err(OutOfBounds(x, y));
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        Ok(())
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
impl Default for FontRenderer8x8 {
 | 
			
		||||
    fn default() -> Self {
 | 
			
		||||
        let utf8_font = SystemSource::new()
 | 
			
		||||
            .select_best_match(&[FamilyName::Monospace], &Properties::new())
 | 
			
		||||
            .unwrap()
 | 
			
		||||
            .load()
 | 
			
		||||
            .unwrap();
 | 
			
		||||
        FontRenderer8x8::new(utf8_font, Some('?'))
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										54
									
								
								src/gui.rs
									
										
									
									
									
								
							
							
						
						
									
										54
									
								
								src/gui.rs
									
										
									
									
									
								
							| 
						 | 
				
			
			@ -20,9 +20,9 @@ pub struct App<'t> {
 | 
			
		|||
    display: &'t RwLock<Bitmap>,
 | 
			
		||||
    luma: &'t RwLock<BrightnessGrid>,
 | 
			
		||||
    window: Option<Window>,
 | 
			
		||||
    pixels: Option<Pixels>,
 | 
			
		||||
    stop_udp_tx: Sender<()>,
 | 
			
		||||
    cli: &'t Cli,
 | 
			
		||||
    logical_size: LogicalSize<u16>,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const SPACER_HEIGHT: usize = 4;
 | 
			
		||||
| 
						 | 
				
			
			@ -40,42 +40,18 @@ impl<'t> App<'t> {
 | 
			
		|||
        stop_udp_tx: Sender<()>,
 | 
			
		||||
        cli: &'t Cli,
 | 
			
		||||
    ) -> Self {
 | 
			
		||||
        let logical_size = {
 | 
			
		||||
            let height = if cli.spacers {
 | 
			
		||||
                let num_spacers = (PIXEL_HEIGHT / TILE_SIZE) - 1;
 | 
			
		||||
                PIXEL_HEIGHT + num_spacers * SPACER_HEIGHT
 | 
			
		||||
            } else {
 | 
			
		||||
                PIXEL_HEIGHT
 | 
			
		||||
            };
 | 
			
		||||
            LogicalSize::new(PIXEL_WIDTH as u16, height as u16)
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
        App {
 | 
			
		||||
            display,
 | 
			
		||||
            luma,
 | 
			
		||||
            stop_udp_tx,
 | 
			
		||||
            pixels: None,
 | 
			
		||||
            window: None,
 | 
			
		||||
            cli,
 | 
			
		||||
            logical_size,
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fn draw(&mut self) {
 | 
			
		||||
        let window = self.window.as_ref().unwrap();
 | 
			
		||||
        let mut pixels = {
 | 
			
		||||
            let window_size = window.inner_size();
 | 
			
		||||
            let surface_texture = SurfaceTexture::new(
 | 
			
		||||
                window_size.width,
 | 
			
		||||
                window_size.height,
 | 
			
		||||
                &window,
 | 
			
		||||
            );
 | 
			
		||||
            Pixels::new(
 | 
			
		||||
                self.logical_size.width as u32,
 | 
			
		||||
                self.logical_size.height as u32,
 | 
			
		||||
                surface_texture,
 | 
			
		||||
            )
 | 
			
		||||
            .unwrap()
 | 
			
		||||
        };
 | 
			
		||||
        let pixels = self.pixels.as_mut().unwrap();
 | 
			
		||||
 | 
			
		||||
        let mut frame = pixels.frame_mut().chunks_exact_mut(4);
 | 
			
		||||
        let display = self.display.read().unwrap();
 | 
			
		||||
| 
						 | 
				
			
			@ -119,13 +95,35 @@ impl<'t> App<'t> {
 | 
			
		|||
 | 
			
		||||
impl ApplicationHandler<AppEvents> for App<'_> {
 | 
			
		||||
    fn resumed(&mut self, event_loop: &ActiveEventLoop) {
 | 
			
		||||
        let height = if self.cli.spacers {
 | 
			
		||||
            let num_spacers = (PIXEL_HEIGHT / TILE_SIZE) - 1;
 | 
			
		||||
            PIXEL_HEIGHT + num_spacers * SPACER_HEIGHT
 | 
			
		||||
        } else {
 | 
			
		||||
            PIXEL_HEIGHT
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
        let size = LogicalSize::new(PIXEL_WIDTH as u16, height as u16);
 | 
			
		||||
        let attributes = Window::default_attributes()
 | 
			
		||||
            .with_title("servicepoint-simulator")
 | 
			
		||||
            .with_inner_size(self.logical_size)
 | 
			
		||||
            .with_inner_size(size)
 | 
			
		||||
            .with_transparent(false);
 | 
			
		||||
 | 
			
		||||
        let window = event_loop.create_window(attributes).unwrap();
 | 
			
		||||
        self.window = Some(window);
 | 
			
		||||
        let window = self.window.as_ref().unwrap();
 | 
			
		||||
 | 
			
		||||
        let pixels = {
 | 
			
		||||
            let window_size = window.inner_size();
 | 
			
		||||
            let surface_texture = SurfaceTexture::new(
 | 
			
		||||
                window_size.width,
 | 
			
		||||
                window_size.height,
 | 
			
		||||
                &window,
 | 
			
		||||
            );
 | 
			
		||||
            Pixels::new(size.width as u32, size.height as u32, surface_texture)
 | 
			
		||||
                .unwrap()
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
        self.pixels = Some(pixels);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fn user_event(&mut self, event_loop: &ActiveEventLoop, event: AppEvents) {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										14
									
								
								src/main.rs
									
										
									
									
									
								
							
							
						
						
									
										14
									
								
								src/main.rs
									
										
									
									
									
								
							| 
						 | 
				
			
			@ -11,14 +11,13 @@ use servicepoint::*;
 | 
			
		|||
use winit::event_loop::{ControlFlow, EventLoop};
 | 
			
		||||
 | 
			
		||||
use crate::execute_command::execute_command;
 | 
			
		||||
use crate::font::Cp437Font;
 | 
			
		||||
use crate::font_renderer::FontRenderer8x8;
 | 
			
		||||
use crate::font::BitmapFont;
 | 
			
		||||
use crate::gui::{App, AppEvents};
 | 
			
		||||
 | 
			
		||||
mod execute_command;
 | 
			
		||||
mod font;
 | 
			
		||||
mod font_renderer;
 | 
			
		||||
mod gui;
 | 
			
		||||
mod static_font;
 | 
			
		||||
 | 
			
		||||
#[derive(Parser, Debug)]
 | 
			
		||||
struct Cli {
 | 
			
		||||
| 
						 | 
				
			
			@ -57,14 +56,14 @@ fn main() {
 | 
			
		|||
    luma.fill(Brightness::MAX);
 | 
			
		||||
    let luma = RwLock::new(luma);
 | 
			
		||||
 | 
			
		||||
    run(&display, &luma, socket, Cp437Font::default(), &cli);
 | 
			
		||||
    run(&display, &luma, socket, BitmapFont::default(), &cli);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
fn run(
 | 
			
		||||
    display_ref: &RwLock<Bitmap>,
 | 
			
		||||
    luma_ref: &RwLock<BrightnessGrid>,
 | 
			
		||||
    socket: UdpSocket,
 | 
			
		||||
    cp437_font: Cp437Font,
 | 
			
		||||
    font: BitmapFont,
 | 
			
		||||
    cli: &Cli,
 | 
			
		||||
) {
 | 
			
		||||
    let (stop_udp_tx, stop_udp_rx) = mpsc::channel();
 | 
			
		||||
| 
						 | 
				
			
			@ -81,7 +80,6 @@ fn run(
 | 
			
		|||
    std::thread::scope(move |scope| {
 | 
			
		||||
        let udp_thread = scope.spawn(move || {
 | 
			
		||||
            let mut buf = [0; 8985];
 | 
			
		||||
            let utf8_font = FontRenderer8x8::default();
 | 
			
		||||
 | 
			
		||||
            while stop_udp_rx.try_recv().is_err() {
 | 
			
		||||
                let (amount, _) = match socket.recv_from(&mut buf) {
 | 
			
		||||
| 
						 | 
				
			
			@ -100,7 +98,7 @@ fn run(
 | 
			
		|||
                    );
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
                let package = match servicepoint::Packet::try_from(&buf[..amount]) {
 | 
			
		||||
                let package = match servicepoint::packet::Packet::try_from(&buf[..amount]) {
 | 
			
		||||
                    Err(_) =>  {
 | 
			
		||||
                        warn!("could not load packet with length {amount} into header");
 | 
			
		||||
                        continue;
 | 
			
		||||
| 
						 | 
				
			
			@ -116,7 +114,7 @@ fn run(
 | 
			
		|||
                    Ok(val) => val,
 | 
			
		||||
                };
 | 
			
		||||
 | 
			
		||||
                if !execute_command(command, &cp437_font, &utf8_font, display_ref, luma_ref) {
 | 
			
		||||
                if !execute_command(command, &font, display_ref, luma_ref) {
 | 
			
		||||
                    // hard reset
 | 
			
		||||
                    event_proxy
 | 
			
		||||
                        .send_event(AppEvents::UdpThreadClosed)
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										276
									
								
								src/static_font.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										276
									
								
								src/static_font.rs
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,276 @@
 | 
			
		|||
use crate::font::BitmapFont;
 | 
			
		||||
use servicepoint::{Bitmap, DataRef, TILE_SIZE};
 | 
			
		||||
 | 
			
		||||
/// Font from the display firmware `cape-cccb-apd/cp437font_linear.h`
 | 
			
		||||
pub(crate) const CP437_FONT_LINEAR: [u64; 256] = [
 | 
			
		||||
    0x0000000000000000, // 0x00
 | 
			
		||||
    0x003854ba82aa4438, // 0x01
 | 
			
		||||
    0x003844bafed67c38, // 0x02
 | 
			
		||||
    0x0010387cfeee4400, // 0x03
 | 
			
		||||
    0x0010387cfe7c3810, // 0x04
 | 
			
		||||
    0x003810d6fefe3838, // 0x05
 | 
			
		||||
    0x003810d6fe7c3810, // 0x06
 | 
			
		||||
    0x0000387c7c7c3800, // 0x07
 | 
			
		||||
    0x00fec6828282c6fe, // 0x08
 | 
			
		||||
    0x0000384444443800, // 0x09
 | 
			
		||||
    0x00fec6bababac6fe, // 0x0a
 | 
			
		||||
    0x007088888a7a061e, // 0x0b
 | 
			
		||||
    0x1038103844444438, // 0x0c
 | 
			
		||||
    0x0030301010141418, // 0x0d
 | 
			
		||||
    0xc0c64642724e720e, // 0x0e
 | 
			
		||||
    0x00927c44c6447c92, // 0x0f
 | 
			
		||||
    0x00c0f0fcfefcf0c0, // 0x10
 | 
			
		||||
    0x00061e7efe7e1e06, // 0x11
 | 
			
		||||
    0x1038541010543810, // 0x12
 | 
			
		||||
    0x2800282828282828, // 0x13
 | 
			
		||||
    0x000e0a0a7a8a8a7e, // 0x14
 | 
			
		||||
    0x0038441c28704438, // 0x15
 | 
			
		||||
    0x000000ffff000000, // 0x16
 | 
			
		||||
    0xfe10385410543810, // 0x17
 | 
			
		||||
    0x1010101010543810, // 0x18
 | 
			
		||||
    0x1038541010101010, // 0x19
 | 
			
		||||
    0x00000804fe040800, // 0x1a
 | 
			
		||||
    0x00002040fe402000, // 0x1b
 | 
			
		||||
    0xffffc0c0c0c0c0c0, // 0x1c
 | 
			
		||||
    0x00002844fe442800, // 0x1d
 | 
			
		||||
    0x00fefe7c7c383810, // 0x1e
 | 
			
		||||
    0x001038387c7cfefe, // 0x1f
 | 
			
		||||
    0x0000000000000000, // 0x20
 | 
			
		||||
    0x1000101010101010, // 0x21
 | 
			
		||||
    0x0000000000505028, // 0x22
 | 
			
		||||
    0x00247e24247e2400, // 0x23
 | 
			
		||||
    0x1038541830543810, // 0x24
 | 
			
		||||
    0x00844a2a54a8a442, // 0x25
 | 
			
		||||
    0x003a444a32484830, // 0x26
 | 
			
		||||
    0x0000000000201010, // 0x27
 | 
			
		||||
    0x0810202020201008, // 0x28
 | 
			
		||||
    0x2010080808081020, // 0x29
 | 
			
		||||
    0x0010543854100000, // 0x2a
 | 
			
		||||
    0x0010107c10100000, // 0x2b
 | 
			
		||||
    0x2010100000000000, // 0x2c
 | 
			
		||||
    0x0000007c00000000, // 0x2d
 | 
			
		||||
    0x0010000000000000, // 0x2e
 | 
			
		||||
    0x4020201010080804, // 0x2f
 | 
			
		||||
    0x0038444454444438, // 0x30
 | 
			
		||||
    0x007c101010503010, // 0x31
 | 
			
		||||
    0x007c201008044438, // 0x32
 | 
			
		||||
    0x003844043810087c, // 0x33
 | 
			
		||||
    0x0004047e24140c04, // 0x34
 | 
			
		||||
    0x003844047840407c, // 0x35
 | 
			
		||||
    0x003844447840201c, // 0x36
 | 
			
		||||
    0x004020100804047c, // 0x37
 | 
			
		||||
    0x0038444438444438, // 0x38
 | 
			
		||||
    0x007008043c444438, // 0x39
 | 
			
		||||
    0x0000100000100000, // 0x3a
 | 
			
		||||
    0x2010100000100000, // 0x3b
 | 
			
		||||
    0x0006186018060000, // 0x3c
 | 
			
		||||
    0x00007c007c000000, // 0x3d
 | 
			
		||||
    0x00c0300c30c00000, // 0x3e
 | 
			
		||||
    0x1000384038044438, // 0x3f
 | 
			
		||||
    0x001c2248564a221c, // 0x40
 | 
			
		||||
    0x0042427e24241818, // 0x41
 | 
			
		||||
    0x007c42427c444478, // 0x42
 | 
			
		||||
    0x001c22404040221c, // 0x43
 | 
			
		||||
    0x0078444242424478, // 0x44
 | 
			
		||||
    0x007e40407e40407e, // 0x45
 | 
			
		||||
    0x004040407e40407e, // 0x46
 | 
			
		||||
    0x001c22424e40221c, // 0x47
 | 
			
		||||
    0x004242427e424242, // 0x48
 | 
			
		||||
    0x007c10101010107c, // 0x49
 | 
			
		||||
    0x003844040404047c, // 0x4a
 | 
			
		||||
    0x0042444870484442, // 0x4b
 | 
			
		||||
    0x007e404040404040, // 0x4c
 | 
			
		||||
    0x0082828292aac682, // 0x4d
 | 
			
		||||
    0x004242464a526242, // 0x4e
 | 
			
		||||
    0x0018244242422418, // 0x4f
 | 
			
		||||
    0x004040407c42427c, // 0x50
 | 
			
		||||
    0x001a244a42422418, // 0x51
 | 
			
		||||
    0x004244487c42427c, // 0x52
 | 
			
		||||
    0x003c42023c40423c, // 0x53
 | 
			
		||||
    0x00101010101010fe, // 0x54
 | 
			
		||||
    0x003c424242424242, // 0x55
 | 
			
		||||
    0x0010282844448282, // 0x56
 | 
			
		||||
    0x0044446caa929282, // 0x57
 | 
			
		||||
    0x0082442810284482, // 0x58
 | 
			
		||||
    0x0010101010284482, // 0x59
 | 
			
		||||
    0x007e20100804027e, // 0x5a
 | 
			
		||||
    0x3820202020202038, // 0x5b
 | 
			
		||||
    0x0408081010202040, // 0x5c
 | 
			
		||||
    0x3808080808080838, // 0x5d
 | 
			
		||||
    0x0000000000442810, // 0x5e
 | 
			
		||||
    0x007e000000000000, // 0x5f
 | 
			
		||||
    0x0000000000080810, // 0x60
 | 
			
		||||
    0x003c443c04380000, // 0x61
 | 
			
		||||
    0x0038444444784040, // 0x62
 | 
			
		||||
    0x0038444044380000, // 0x63
 | 
			
		||||
    0x003c4444443c0404, // 0x64
 | 
			
		||||
    0x003c407844380000, // 0x65
 | 
			
		||||
    0x2020202078202418, // 0x66
 | 
			
		||||
    0x78043c44443e0000, // 0x67
 | 
			
		||||
    0x0044444464584040, // 0x68
 | 
			
		||||
    0x001c101010700010, // 0x69
 | 
			
		||||
    0x38440404047c0010, // 0x6a
 | 
			
		||||
    0x0022243828242020, // 0x6b
 | 
			
		||||
    0x0018242020202020, // 0x6c
 | 
			
		||||
    0x0054545454780000, // 0x6d
 | 
			
		||||
    0x0044444464580000, // 0x6e
 | 
			
		||||
    0x0038444444380000, // 0x6f
 | 
			
		||||
    0x4040784444780000, // 0x70
 | 
			
		||||
    0x04043c44443c0000, // 0x71
 | 
			
		||||
    0x0040404064580000, // 0x72
 | 
			
		||||
    0x00384418201c0000, // 0x73
 | 
			
		||||
    0x0018242020782020, // 0x74
 | 
			
		||||
    0x0038444444440000, // 0x75
 | 
			
		||||
    0x0010282844440000, // 0x76
 | 
			
		||||
    0x0028285454440000, // 0x77
 | 
			
		||||
    0x0044281028440000, // 0x78
 | 
			
		||||
    0x38043c4444440000, // 0x79
 | 
			
		||||
    0x007c2010087c0000, // 0x7a
 | 
			
		||||
    0x0c0808083008080c, // 0x7b
 | 
			
		||||
    0x1010101010101010, // 0x7c
 | 
			
		||||
    0x301010100c101030, // 0x7d
 | 
			
		||||
    0x0000004c32000000, // 0x7e
 | 
			
		||||
    0xfe82828282442810, // 0x7f
 | 
			
		||||
    0x18083c428080423c, // 0x80
 | 
			
		||||
    0x0038444444440028, // 0x81
 | 
			
		||||
    0x003c407844381008, // 0x82
 | 
			
		||||
    0x003c443c04382810, // 0x83
 | 
			
		||||
    0x003c443c04380044, // 0x84
 | 
			
		||||
    0x003c443c04381020, // 0x85
 | 
			
		||||
    0x003c443c04382838, // 0x86
 | 
			
		||||
    0x1038444044380000, // 0x87
 | 
			
		||||
    0x003c407844382810, // 0x88
 | 
			
		||||
    0x003c407844380044, // 0x89
 | 
			
		||||
    0x003c407844381020, // 0x8a
 | 
			
		||||
    0x001c101010700028, // 0x8b
 | 
			
		||||
    0x001c101010702810, // 0x8c
 | 
			
		||||
    0x001c101010701020, // 0x8d
 | 
			
		||||
    0x0082827c44282892, // 0x8e
 | 
			
		||||
    0x0082827c44281038, // 0x8f
 | 
			
		||||
    0x00fe80fe80fe1008, // 0x90
 | 
			
		||||
    0x007e907c126c0000, // 0x91
 | 
			
		||||
    0x008e88784e28281e, // 0x92
 | 
			
		||||
    0x0038444438002810, // 0x93
 | 
			
		||||
    0x0038444444380028, // 0x94
 | 
			
		||||
    0x0038444438001020, // 0x95
 | 
			
		||||
    0x0038444444002810, // 0x96
 | 
			
		||||
    0x0038444444001020, // 0x97
 | 
			
		||||
    0x38043c4444440028, // 0x98
 | 
			
		||||
    0x0038448282443882, // 0x99
 | 
			
		||||
    0x0038448282820082, // 0x9a
 | 
			
		||||
    0x1038444044381010, // 0x9b
 | 
			
		||||
    0x00fc4240f0404438, // 0x9c
 | 
			
		||||
    0x107c107c10284482, // 0x9d
 | 
			
		||||
    0x001c22f840f8221c, // 0x9e
 | 
			
		||||
    0x205010103810120c, // 0x9f
 | 
			
		||||
    0x003c443c04381008, // 0xa0
 | 
			
		||||
    0x001c101010701008, // 0xa1
 | 
			
		||||
    0x0038444438001008, // 0xa2
 | 
			
		||||
    0x0038444444001008, // 0xa3
 | 
			
		||||
    0x0044446458004834, // 0xa4
 | 
			
		||||
    0x00464a5262424834, // 0xa5
 | 
			
		||||
    0x0000007090701060, // 0xa6
 | 
			
		||||
    0x0000007088888870, // 0xa7
 | 
			
		||||
    0x3844403804380010, // 0xa8
 | 
			
		||||
    0x0040407c00000000, // 0xa9
 | 
			
		||||
    0x0004047c00000000, // 0xaa
 | 
			
		||||
    0x0e84422c5048c442, // 0xab
 | 
			
		||||
    0x049e542c5448c442, // 0xac
 | 
			
		||||
    0x1010101010100010, // 0xad
 | 
			
		||||
    0x0024489048240000, // 0xae
 | 
			
		||||
    0x0048241224480000, // 0xaf
 | 
			
		||||
    0x1122448811224488, // 0xb0
 | 
			
		||||
    0x55aa55aa55aa55aa, // 0xb1
 | 
			
		||||
    0xddbb77eeddbb77ee, // 0xb2
 | 
			
		||||
    0x1010101010101010, // 0xb3
 | 
			
		||||
    0x101020c020101010, // 0xb4
 | 
			
		||||
    0x1020c000c0201010, // 0xb5
 | 
			
		||||
    0x2828488848282828, // 0xb6
 | 
			
		||||
    0x282850e000000000, // 0xb7
 | 
			
		||||
    0x1030d020c0000000, // 0xb8
 | 
			
		||||
    0x2848880888482828, // 0xb9
 | 
			
		||||
    0x2828282828282828, // 0xba
 | 
			
		||||
    0x2828c810e0000000, // 0xbb
 | 
			
		||||
    0x000000e010c82828, // 0xbc
 | 
			
		||||
    0x000000e050282828, // 0xbd
 | 
			
		||||
    0x0000c020d0301010, // 0xbe
 | 
			
		||||
    0x101020c000000000, // 0xbf
 | 
			
		||||
    0x0000000708101010, // 0xc0
 | 
			
		||||
    0x000000c728101010, // 0xc1
 | 
			
		||||
    0x101028c700000000, // 0xc2
 | 
			
		||||
    0x1010080708101010, // 0xc3
 | 
			
		||||
    0x000000ff00000000, // 0xc4
 | 
			
		||||
    0x101028c728101010, // 0xc5
 | 
			
		||||
    0x1008070007081010, // 0xc6
 | 
			
		||||
    0x2828242324282828, // 0xc7
 | 
			
		||||
    0x00000f1027282828, // 0xc8
 | 
			
		||||
    0x282827100f000000, // 0xc9
 | 
			
		||||
    0x0000ff0083442828, // 0xca
 | 
			
		||||
    0x28448300ff000000, // 0xcb
 | 
			
		||||
    0x2824232023242828, // 0xcc
 | 
			
		||||
    0x0000ff00ff000000, // 0xcd
 | 
			
		||||
    0x2844932893442828, // 0xce
 | 
			
		||||
    0x0000ff00c7281010, // 0xcf
 | 
			
		||||
    0x0000008344282828, // 0xd0
 | 
			
		||||
    0x1028c700ff000000, // 0xd1
 | 
			
		||||
    0x2828448300000000, // 0xd2
 | 
			
		||||
    0x0000000f14282828, // 0xd3
 | 
			
		||||
    0x0000070817181010, // 0xd4
 | 
			
		||||
    0x1018170807000000, // 0xd5
 | 
			
		||||
    0x2828140f00000000, // 0xd6
 | 
			
		||||
    0x2828448344282828, // 0xd7
 | 
			
		||||
    0x1028c700c7281010, // 0xd8
 | 
			
		||||
    0x000000c020101010, // 0xd9
 | 
			
		||||
    0x1010080700000000, // 0xda
 | 
			
		||||
    0xffffffffffffffff, // 0xdb
 | 
			
		||||
    0xffffffff00000000, // 0xdc
 | 
			
		||||
    0xf0f0f0f0f0f0f0f0, // 0xdd
 | 
			
		||||
    0x0f0f0f0f0f0f0f0f, // 0xde
 | 
			
		||||
    0x00000000ffffffff, // 0xdf
 | 
			
		||||
    0x0076888888740200, // 0xe0
 | 
			
		||||
    0x5844444458484830, // 0xe1
 | 
			
		||||
    0x00e04040404242fe, // 0xe2
 | 
			
		||||
    0x00242828a87c0000, // 0xe3
 | 
			
		||||
    0x00fe8240204082fe, // 0xe4
 | 
			
		||||
    0x00384444443e0000, // 0xe5
 | 
			
		||||
    0x405a644444440000, // 0xe6
 | 
			
		||||
    0x0010282020fc0000, // 0xe7
 | 
			
		||||
    0xfe103854543810fe, // 0xe8
 | 
			
		||||
    0x003844aabaaa4438, // 0xe9
 | 
			
		||||
    0x00ee448282824438, // 0xea
 | 
			
		||||
    0x007088887012221c, // 0xeb
 | 
			
		||||
    0x00006c92926c0000, // 0xec
 | 
			
		||||
    0x10107c92924c0000, // 0xed
 | 
			
		||||
    0x0038403040380000, // 0xee
 | 
			
		||||
    0x0082828282824438, // 0xef
 | 
			
		||||
    0x00fe00fe00fe0000, // 0xf0
 | 
			
		||||
    0x007c10107c101000, // 0xf1
 | 
			
		||||
    0x007e006018061860, // 0xf2
 | 
			
		||||
    0x007e000618601806, // 0xf3
 | 
			
		||||
    0x101010101010120c, // 0xf4
 | 
			
		||||
    0x6090101010101010, // 0xf5
 | 
			
		||||
    0x0010007c00100000, // 0xf6
 | 
			
		||||
    0x000c926c92600000, // 0xf7
 | 
			
		||||
    0x0000000030484830, // 0xf8
 | 
			
		||||
    0x0000103810000000, // 0xf9
 | 
			
		||||
    0x0000001000000000, // 0xfa
 | 
			
		||||
    0x10102828a4440202, // 0xfb
 | 
			
		||||
    0x00000000484848b0, // 0xfc
 | 
			
		||||
    0x0000000070201060, // 0xfd
 | 
			
		||||
    0x00007c7c7c7c7c00, // 0xfe
 | 
			
		||||
    0x0000000000000000, // 0xff
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
pub fn load_static() -> BitmapFont {
 | 
			
		||||
    let mut bitmaps =
 | 
			
		||||
        core::array::from_fn(|_| Bitmap::new(TILE_SIZE, TILE_SIZE));
 | 
			
		||||
 | 
			
		||||
    for (char_code, bitmap) in bitmaps.iter_mut().enumerate() {
 | 
			
		||||
        let bits = CP437_FONT_LINEAR[char_code];
 | 
			
		||||
        let mut bytes = bits.to_be_bytes();
 | 
			
		||||
        bytes.reverse();
 | 
			
		||||
        bitmap.data_ref_mut().copy_from_slice(bytes.as_slice());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    BitmapFont::new(bitmaps)
 | 
			
		||||
}
 | 
			
		||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue