Compare commits

...

2 commits

Author SHA1 Message Date
Vinzenz Schroeter ad5f1e8abe update dependencies, initial UTF-8 support
Some checks failed
Rust / build (push) Has been cancelled
2025-01-16 22:28:08 +01:00
Vinzenz Schroeter 398cb8c165 update flake, use correct nix dependencies instead of non-default fontkit features 2025-01-09 22:50:56 +01:00
10 changed files with 833 additions and 682 deletions

639
Cargo.lock generated

File diff suppressed because it is too large Load diff

View file

@ -11,12 +11,14 @@ log = "0.4"
env_logger = "0.11" env_logger = "0.11"
clap = { version = "4.5", features = ["derive"] } clap = { version = "4.5", features = ["derive"] }
# for drawing pixels onto the surface of the window # for drawing pixels onto the surface of the window
pixels = "0.14" pixels = "0.15"
# I should not need this as a direct dependency, but then I cannot spell the types needed to use font-kit... # 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" pathfinder_geometry = "0.5.1"
font-kit = "0.14.2"
thiserror = "2.0"
[dependencies.servicepoint] [dependencies.servicepoint]
version = "0.12.0" version = "0.13.0"
features = ["all_compressions"] features = ["all_compressions"]
[dependencies.winit] [dependencies.winit]
@ -24,11 +26,3 @@ version = "0.30"
features = ["rwh_05"] features = ["rwh_05"]
default-features = true 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": { "locked": {
"lastModified": 1721727458, "lastModified": 1736429655,
"narHash": "sha256-r/xppY958gmZ4oTfLiHN0ZGuQ+RSTijDblVgVLFi1mw=", "narHash": "sha256-BwMekRuVlSB9C0QgwKMICiJ5EVbLGjfe4qyueyNQyGI=",
"owner": "nix-community", "owner": "nix-community",
"repo": "naersk", "repo": "naersk",
"rev": "3fb418eaf352498f6b6c30592e3beb63df42ef11", "rev": "0621e47bd95542b8e1ce2ee2d65d6a1f887a13ce",
"type": "github" "type": "github"
}, },
"original": { "original": {
@ -37,16 +37,16 @@
}, },
"nixpkgs": { "nixpkgs": {
"locked": { "locked": {
"lastModified": 1732749044, "lastModified": 1736200483,
"narHash": "sha256-T38FQOg0BV5M8FN1712fovzNakSOENEYs+CSkg31C9Y=", "narHash": "sha256-JO+lFN2HsCwSLMUWXHeOad6QUxOuwe9UOAF/iSl1J4I=",
"owner": "nixos", "owner": "nixos",
"repo": "nixpkgs", "repo": "nixpkgs",
"rev": "0c5b4ecbed5b155b705336aa96d878e55acd8685", "rev": "3f0a8ac25fb674611b98089ca3a5dd6480175751",
"type": "github" "type": "github"
}, },
"original": { "original": {
"owner": "nixos", "owner": "nixos",
"ref": "nixos-24.05", "ref": "nixos-24.11",
"repo": "nixpkgs", "repo": "nixpkgs",
"type": "github" "type": "github"
} }

View file

@ -2,7 +2,7 @@
description = "Flake for servicepoint-simulator"; description = "Flake for servicepoint-simulator";
inputs = { inputs = {
nixpkgs.url = "github:nixos/nixpkgs/nixos-24.05"; nixpkgs.url = "github:nixos/nixpkgs/nixos-24.11";
nix-filter.url = "github:numtide/nix-filter"; nix-filter.url = "github:numtide/nix-filter";
naersk = { naersk = {
url = "github:nix-community/naersk"; url = "github:nix-community/naersk";
@ -70,13 +70,25 @@
with pkgs; with pkgs;
[ [
xe xe
lzma xz
] ]
++ (lib.optionals pkgs.stdenv.isLinux ( ++ lib.optionals pkgs.stdenv.isLinux (
with pkgs; with pkgs;
[ [
libxkbcommon # gpu
libGL libGL
vulkan-headers
vulkan-loader
vulkan-tools vulkan-tools-lunarg
vulkan-extension-layer
vulkan-validation-layers
# keyboard
libxkbcommon
# font loading
fontconfig
freetype
# WINIT_UNIX_BACKEND=wayland # WINIT_UNIX_BACKEND=wayland
wayland wayland
@ -88,15 +100,15 @@
xorg.libX11 xorg.libX11
xorg.libX11.dev xorg.libX11.dev
] ]
)) )
++ (lib.optionals pkgs.stdenv.isDarwin ( ++ lib.optionals pkgs.stdenv.isDarwin (
with pkgs.darwin.apple_sdk.frameworks; with pkgs.darwin.apple_sdk.frameworks;
[ [
Carbon Carbon
QuartzCore QuartzCore
AppKit AppKit
] ]
)); );
postInstall = '' postInstall = ''
wrapProgram $out/bin/servicepoint-simulator \ wrapProgram $out/bin/servicepoint-simulator \
@ -127,7 +139,7 @@
{ {
default = pkgs.mkShell rec { default = pkgs.mkShell rec {
inputsFrom = [ self.packages.${system}.default ]; inputsFrom = [ self.packages.${system}.default ];
packages = [ rust-toolchain ]; packages = [ rust-toolchain pkgs.gdb ];
LD_LIBRARY_PATH = "${pkgs.lib.makeLibraryPath (builtins.concatMap (d: d.buildInputs) inputsFrom)}"; LD_LIBRARY_PATH = "${pkgs.lib.makeLibraryPath (builtins.concatMap (d: d.buildInputs) inputsFrom)}";
RUST_SRC_PATH = "${pkgs.rust.packages.stable.rustPlatform.rustLibSrc}"; RUST_SRC_PATH = "${pkgs.rust.packages.stable.rustPlatform.rustLibSrc}";
}; };

View file

@ -1,16 +1,17 @@
use std::sync::{RwLock, RwLockWriteGuard};
use log::{debug, error, info, trace, warn}; use log::{debug, error, info, trace, warn};
use servicepoint::{ use servicepoint::{
Bitmap, BrightnessGrid, Command, Cp437Grid, Grid, Origin, Tiles, Bitmap, BrightnessGrid, CharGrid, Command, Cp437Grid, Grid, Origin, Tiles,
PIXEL_COUNT, PIXEL_WIDTH, TILE_SIZE, PIXEL_COUNT, PIXEL_WIDTH, TILE_SIZE,
}; };
use std::sync::{RwLock, RwLockWriteGuard};
use crate::font::BitmapFont; use crate::font::Cp437Font;
use crate::font_renderer::FontRenderer8x8;
pub(crate) fn execute_command( pub(crate) fn execute_command(
command: Command, command: Command,
font: &BitmapFont, cp436_font: &Cp437Font,
utf8_font: &FontRenderer8x8,
display_ref: &RwLock<Bitmap>, display_ref: &RwLock<Bitmap>,
luma_ref: &RwLock<BrightnessGrid>, luma_ref: &RwLock<BrightnessGrid>,
) -> bool { ) -> bool {
@ -30,7 +31,7 @@ pub(crate) fn execute_command(
} }
Command::Cp437Data(origin, grid) => { Command::Cp437Data(origin, grid) => {
let mut display = display_ref.write().unwrap(); let mut display = display_ref.write().unwrap();
print_cp437_data(origin, &grid, font, &mut display); print_cp437_data(origin, &grid, cp436_font, &mut display);
} }
#[allow(deprecated)] #[allow(deprecated)]
Command::BitmapLegacy => { Command::BitmapLegacy => {
@ -99,6 +100,10 @@ pub(crate) fn execute_command(
Command::FadeOut => { Command::FadeOut => {
error!("command not implemented: {command:?}") 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 true
@ -116,7 +121,7 @@ fn check_bitmap_valid(offset: u16, payload_len: usize) -> bool {
fn print_cp437_data( fn print_cp437_data(
origin: Origin<Tiles>, origin: Origin<Tiles>,
grid: &Cp437Grid, grid: &Cp437Grid,
font: &BitmapFont, font: &Cp437Font,
display: &mut RwLockWriteGuard<Bitmap>, display: &mut RwLockWriteGuard<Bitmap>,
) { ) {
let Origin { x, y, .. } = origin; let Origin { x, y, .. } = origin;
@ -145,6 +150,33 @@ 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( fn print_pixel_grid(
offset_x: usize, offset_x: usize,
offset_y: usize, offset_y: usize,

View file

@ -1,73 +1,297 @@
use crate::static_font; use servicepoint::{Bitmap, DataRef, TILE_SIZE};
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; const CHAR_COUNT: usize = u8::MAX as usize + 1;
pub struct BitmapFont { pub struct Cp437Font {
bitmaps: [Bitmap; CHAR_COUNT], bitmaps: [Bitmap; CHAR_COUNT],
} }
impl BitmapFont { impl Cp437Font {
pub fn new(bitmaps: [Bitmap; CHAR_COUNT]) -> Self { pub fn new(bitmaps: [Bitmap; CHAR_COUNT]) -> Self {
Self { bitmaps } 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 { pub fn get_bitmap(&self, char_code: u8) -> &Bitmap {
&self.bitmaps[char_code as usize] &self.bitmaps[char_code as usize]
} }
} }
impl Default for BitmapFont { impl Default for Cp437Font {
fn default() -> Self { fn default() -> Self {
static_font::load_static() 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)
}

94
src/font_renderer.rs Normal file
View file

@ -0,0 +1,94 @@
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>, display: &'t RwLock<Bitmap>,
luma: &'t RwLock<BrightnessGrid>, luma: &'t RwLock<BrightnessGrid>,
window: Option<Window>, window: Option<Window>,
pixels: Option<Pixels>,
stop_udp_tx: Sender<()>, stop_udp_tx: Sender<()>,
cli: &'t Cli, cli: &'t Cli,
logical_size: LogicalSize<u16>,
} }
const SPACER_HEIGHT: usize = 4; const SPACER_HEIGHT: usize = 4;
@ -40,18 +40,42 @@ impl<'t> App<'t> {
stop_udp_tx: Sender<()>, stop_udp_tx: Sender<()>,
cli: &'t Cli, cli: &'t Cli,
) -> Self { ) -> 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 { App {
display, display,
luma, luma,
stop_udp_tx, stop_udp_tx,
pixels: None,
window: None, window: None,
cli, cli,
logical_size,
} }
} }
fn draw(&mut self) { fn draw(&mut self) {
let pixels = self.pixels.as_mut().unwrap(); 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 mut frame = pixels.frame_mut().chunks_exact_mut(4); let mut frame = pixels.frame_mut().chunks_exact_mut(4);
let display = self.display.read().unwrap(); let display = self.display.read().unwrap();
@ -95,35 +119,13 @@ impl<'t> App<'t> {
impl ApplicationHandler<AppEvents> for App<'_> { impl ApplicationHandler<AppEvents> for App<'_> {
fn resumed(&mut self, event_loop: &ActiveEventLoop) { 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() let attributes = Window::default_attributes()
.with_title("servicepoint-simulator") .with_title("servicepoint-simulator")
.with_inner_size(size) .with_inner_size(self.logical_size)
.with_transparent(false); .with_transparent(false);
let window = event_loop.create_window(attributes).unwrap(); let window = event_loop.create_window(attributes).unwrap();
self.window = Some(window); 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) { fn user_event(&mut self, event_loop: &ActiveEventLoop, event: AppEvents) {

View file

@ -11,13 +11,14 @@ use servicepoint::*;
use winit::event_loop::{ControlFlow, EventLoop}; use winit::event_loop::{ControlFlow, EventLoop};
use crate::execute_command::execute_command; use crate::execute_command::execute_command;
use crate::font::BitmapFont; use crate::font::Cp437Font;
use crate::font_renderer::FontRenderer8x8;
use crate::gui::{App, AppEvents}; use crate::gui::{App, AppEvents};
mod execute_command; mod execute_command;
mod font; mod font;
mod font_renderer;
mod gui; mod gui;
mod static_font;
#[derive(Parser, Debug)] #[derive(Parser, Debug)]
struct Cli { struct Cli {
@ -56,14 +57,14 @@ fn main() {
luma.fill(Brightness::MAX); luma.fill(Brightness::MAX);
let luma = RwLock::new(luma); let luma = RwLock::new(luma);
run(&display, &luma, socket, BitmapFont::default(), &cli); run(&display, &luma, socket, Cp437Font::default(), &cli);
} }
fn run( fn run(
display_ref: &RwLock<Bitmap>, display_ref: &RwLock<Bitmap>,
luma_ref: &RwLock<BrightnessGrid>, luma_ref: &RwLock<BrightnessGrid>,
socket: UdpSocket, socket: UdpSocket,
font: BitmapFont, cp437_font: Cp437Font,
cli: &Cli, cli: &Cli,
) { ) {
let (stop_udp_tx, stop_udp_rx) = mpsc::channel(); let (stop_udp_tx, stop_udp_rx) = mpsc::channel();
@ -80,6 +81,7 @@ fn run(
std::thread::scope(move |scope| { std::thread::scope(move |scope| {
let udp_thread = scope.spawn(move || { let udp_thread = scope.spawn(move || {
let mut buf = [0; 8985]; let mut buf = [0; 8985];
let utf8_font = FontRenderer8x8::default();
while stop_udp_rx.try_recv().is_err() { while stop_udp_rx.try_recv().is_err() {
let (amount, _) = match socket.recv_from(&mut buf) { let (amount, _) = match socket.recv_from(&mut buf) {
@ -98,8 +100,8 @@ fn run(
); );
} }
let package = match servicepoint::packet::Packet::try_from(&buf[..amount]) { let package = match servicepoint::Packet::try_from(&buf[..amount]) {
Err(_) => { Err(_) => {
warn!("could not load packet with length {amount} into header"); warn!("could not load packet with length {amount} into header");
continue; continue;
} }
@ -114,7 +116,7 @@ fn run(
Ok(val) => val, Ok(val) => val,
}; };
if !execute_command(command, &font, display_ref, luma_ref) { if !execute_command(command, &cp437_font, &utf8_font, display_ref, luma_ref) {
// hard reset // hard reset
event_proxy event_proxy
.send_event(AppEvents::UdpThreadClosed) .send_event(AppEvents::UdpThreadClosed)

View file

@ -1,276 +0,0 @@
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)
}