Compare commits

..

No commits in common. "ad5f1e8abe31b875d32eebed900a363c31615181" and "24d1c54299bb9795ce11f605e345ee812abc57ac" have entirely different histories.

10 changed files with 684 additions and 835 deletions

637
Cargo.lock generated

File diff suppressed because it is too large Load diff

View file

@ -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

View file

@ -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"
}

View file

@ -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}";
};

View file

@ -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,

View file

@ -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)
}

View file

@ -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('?'))
}
}

View file

@ -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) {

View file

@ -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,8 +98,8 @@ fn run(
);
}
let package = match servicepoint::Packet::try_from(&buf[..amount]) {
Err(_) => {
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
View 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)
}