Compare commits

..

No commits in common. "625e561ba359b1fe217a076308ba709d7874db28" and "4eed089a061b24b35bfcc7e615a312877d9b8997" have entirely different histories.

9 changed files with 57 additions and 96 deletions

18
Cargo.lock generated
View file

@ -760,6 +760,17 @@ dependencies = [
"hashbrown", "hashbrown",
] ]
[[package]]
name = "inherent"
version = "1.0.12"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "6c38228f24186d9cc68c729accb4d413be9eaed6ad07ff79e0270d9e56f3de13"
dependencies = [
"proc-macro2",
"quote",
"syn",
]
[[package]] [[package]]
name = "is_terminal_polyfill" name = "is_terminal_polyfill"
version = "1.70.1" version = "1.70.1"
@ -1510,13 +1521,14 @@ dependencies = [
[[package]] [[package]]
name = "servicepoint" name = "servicepoint"
version = "0.16.0" version = "0.15.2"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9b04582e916474f1bc1605cad3773262c425d9062b487e49a0df59662f2cca8d" checksum = "7d1e88713031e003dc3ee708dbb282e36714eee466a12d311d0e2e24c61c7118"
dependencies = [ dependencies = [
"bitvec", "bitvec",
"bzip2", "bzip2",
"flate2", "flate2",
"inherent",
"log", "log",
"once_cell", "once_cell",
"rust-lzma", "rust-lzma",
@ -1526,7 +1538,7 @@ dependencies = [
[[package]] [[package]]
name = "servicepoint-simulator" name = "servicepoint-simulator"
version = "0.2.4" version = "0.2.3"
dependencies = [ dependencies = [
"clap", "clap",
"env_logger", "env_logger",

View file

@ -1,6 +1,6 @@
[package] [package]
name = "servicepoint-simulator" name = "servicepoint-simulator"
version = "0.2.4" version = "0.2.3"
edition = "2021" edition = "2021"
publish = true publish = true
license = "GPL-3.0-or-later" license = "GPL-3.0-or-later"
@ -19,6 +19,9 @@ env_logger = "0.11"
clap = { version = "4.5", features = ["derive"] } clap = { version = "4.5", features = ["derive"] }
thiserror = "2.0" thiserror = "2.0"
# package parsing
servicepoint = { features = ["all_compressions"], version = "0.15.2" }
# font rendering # font rendering
font-kit = "0.14.2" font-kit = "0.14.2"
# 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...
@ -29,10 +32,6 @@ winit = "0.30"
# for drawing pixels onto the surface of the window # for drawing pixels onto the surface of the window
softbuffer = "0.4.6" softbuffer = "0.4.6"
[dependencies.servicepoint]
version = "0.16.0"
features = ["all_compressions"]
[profile.release] [profile.release]
lto = true # Enable link-time optimization lto = true # Enable link-time optimization
codegen-units = 1 # Reduce number of codegen units to increase optimizations codegen-units = 1 # Reduce number of codegen units to increase optimizations

View file

@ -42,24 +42,14 @@ Make sure to run a release build, because a debug build _way_ slower.
Usage: servicepoint-simulator [OPTIONS] Usage: servicepoint-simulator [OPTIONS]
Options: Options:
--bind <BIND> --bind <BIND> address and port to bind to [default: 0.0.0.0:2342]
address and port to bind to [default: 0.0.0.0:2342] -f, --font <FONT> The name of the font family to use. This defaults to the system monospace font.
-f, --font <FONT> -s, --spacers add spacers between tile rows to simulate gaps in real display
The name of the font family to use. This defaults to the system monospace font. -r, --red Use the red color channel
-s, --spacers -g, --green Use the green color channel
add spacers between tile rows to simulate gaps in real display -b, --blue Use the blue color channel
-r, --red -v, --verbose Set default log level lower. You can also change this via the RUST_LOG environment variable.
Use the red color channel -h, --help Print help
-g, --green
Use the green color channel
-b, --blue
Use the blue color channel
-v, --verbose
Set default log level lower. You can also change this via the RUST_LOG environment variable.
--experimental-null-char-handling
When receiving a null byte as a char in the CharGridCommand, do not overwrite any pixels instead of clearing all pixels.
-h, --help
Print help
``` ```
See [env_logger](https://docs.rs/env_logger/latest/env_logger/) to configure logging. See [env_logger](https://docs.rs/env_logger/latest/env_logger/) to configure logging.

View file

@ -75,7 +75,6 @@
NIX_LD_LIBRARY_PATH = LD_LIBRARY_PATH; NIX_LD_LIBRARY_PATH = LD_LIBRARY_PATH;
NIX_LD = pkgs.stdenv.cc.bintools.dynamicLinker; NIX_LD = pkgs.stdenv.cc.bintools.dynamicLinker;
RUST_SRC_PATH = "${pkgs.rust.packages.stable.rustPlatform.rustLibSrc}"; RUST_SRC_PATH = "${pkgs.rust.packages.stable.rustPlatform.rustLibSrc}";
RUST_BACKTRACE = "1";
}; };
} }
); );

View file

@ -22,11 +22,6 @@ pub struct Cli {
help = "Set default log level lower. You can also change this via the RUST_LOG environment variable." help = "Set default log level lower. You can also change this via the RUST_LOG environment variable."
)] )]
pub verbose: bool, pub verbose: bool,
#[arg(
long,
help = "When receiving a null byte as a char in the CharGridCommand, do not overwrite any pixels instead of clearing all pixels."
)]
pub experimental_null_char_handling: bool,
} }
#[derive(Parser, Debug)] #[derive(Parser, Debug)]

View file

@ -5,11 +5,10 @@ use crate::{
}; };
use log::{debug, error, info, trace, warn}; use log::{debug, error, info, trace, warn};
use servicepoint::{ use servicepoint::{
BinaryOperation, BitVecCommand, Bitmap, BitmapCommand, BrightnessGrid, BinaryOperation, BitVecCommand, Bitmap, BitmapCommand, GlobalBrightnessCommand,
BrightnessGridCommand, CharGridCommand, ClearCommand, CompressionCode, BrightnessGrid, BrightnessGridCommand, CharGridCommand, ClearCommand,
Cp437GridCommand, FadeOutCommand, GlobalBrightnessCommand, Grid, GridMut, CompressionCode, Cp437GridCommand, FadeOutCommand, Grid, HardResetCommand,
HardResetCommand, Origin, TypedCommand, PIXEL_COUNT, PIXEL_WIDTH, Origin, TypedCommand, PIXEL_COUNT, PIXEL_WIDTH, TILE_SIZE,
TILE_SIZE,
}; };
use std::{ use std::{
ops::{BitAnd, BitOr, BitXor}, ops::{BitAnd, BitOr, BitXor},
@ -22,7 +21,6 @@ pub struct CommandExecutionContext<'t> {
luma: &'t RwLock<BrightnessGrid>, luma: &'t RwLock<BrightnessGrid>,
cp437_font: Cp437Font, cp437_font: Cp437Font,
font_renderer: FontRenderer8x8, font_renderer: FontRenderer8x8,
experimental_null_char_handling: bool,
} }
#[must_use] #[must_use]
@ -48,11 +46,11 @@ impl CommandExecute for BitmapCommand {
fn execute(&self, context: &CommandExecutionContext) -> ExecutionResult { fn execute(&self, context: &CommandExecutionContext) -> ExecutionResult {
let Self { let Self {
origin: origin:
Origin { Origin {
x: offset_x, x: offset_x,
y: offset_y, y: offset_y,
.. ..
}, },
bitmap: pixels, bitmap: pixels,
.. ..
} = self; } = self;
@ -145,7 +143,7 @@ impl CommandExecute for Cp437GridCommand {
bitmap: context.cp437_font[char_code].clone(), bitmap: context.cp437_font[char_code].clone(),
compression: CompressionCode::default(), compression: CompressionCode::default(),
} }
.execute(context); .execute(context);
match execute_result { match execute_result {
Success => {} Success => {}
Failure => { Failure => {
@ -194,30 +192,16 @@ impl CommandExecute for CharGridCommand {
for char_y in 0usize..grid.height() { for char_y in 0usize..grid.height() {
for char_x in 0usize..grid.width() { for char_x in 0usize..grid.width() {
let char = grid.get(char_x, char_y); let char = grid.get(char_x, char_y);
let mut bitmap_window = {
let pixel_x = (char_x + x) * TILE_SIZE;
let pixel_y = (char_y + y) * TILE_SIZE;
display
.window_mut(
pixel_x..pixel_x + TILE_SIZE,
pixel_y..pixel_y + TILE_SIZE,
)
.unwrap()
};
if char == '\0' {
if context.experimental_null_char_handling {
trace!("skipping {char:?}");
} else {
bitmap_window.fill(false);
}
continue;
}
trace!("drawing {char}"); trace!("drawing {char}");
if let Err(e) =
context.font_renderer.render(char, &mut bitmap_window) let tile_x = char_x + x;
{ let tile_y = char_y + y;
if let Err(e) = context.font_renderer.render(
char,
&mut display,
Origin::new(tile_x * TILE_SIZE, tile_y * TILE_SIZE),
) {
error!( error!(
"stopping drawing text because char draw failed: {e}" "stopping drawing text because char draw failed: {e}"
); );
@ -267,14 +251,12 @@ impl<'t> CommandExecutionContext<'t> {
display: &'t RwLock<Bitmap>, display: &'t RwLock<Bitmap>,
luma: &'t RwLock<BrightnessGrid>, luma: &'t RwLock<BrightnessGrid>,
font_renderer: FontRenderer8x8, font_renderer: FontRenderer8x8,
experimental_null_char_handling: bool,
) -> Self { ) -> Self {
CommandExecutionContext { CommandExecutionContext {
display, display,
luma, luma,
font_renderer, font_renderer,
cp437_font: Cp437Font::default(), cp437_font: Cp437Font::default(),
experimental_null_char_handling,
} }
} }
} }

View file

@ -12,11 +12,8 @@ use pathfinder_geometry::{
transform2d::Transform2F, transform2d::Transform2F,
vector::{vec2f, vec2i}, vector::{vec2f, vec2i},
}; };
use servicepoint::{Bitmap, GridMut, WindowMut, TILE_SIZE}; use servicepoint::{Bitmap, Grid, Origin, Pixels, TILE_SIZE};
use std::{ use std::sync::{Mutex, MutexGuard};
collections::HashMap,
sync::{Mutex, MutexGuard},
};
#[derive(Debug)] #[derive(Debug)]
struct SendFont(Font); struct SendFont(Font);
@ -35,7 +32,6 @@ pub struct FontRenderer8x8 {
font: SendFont, font: SendFont,
canvas: Mutex<Canvas>, canvas: Mutex<Canvas>,
fallback_char: Option<u32>, fallback_char: Option<u32>,
cache: Mutex<HashMap<char, Bitmap>>,
} }
#[derive(Debug, thiserror::Error)] #[derive(Debug, thiserror::Error)]
@ -60,7 +56,6 @@ impl FontRenderer8x8 {
font: SendFont(font), font: SendFont(font),
fallback_char, fallback_char,
canvas: Mutex::new(canvas), canvas: Mutex::new(canvas),
cache: Mutex::new(HashMap::new()),
} }
} }
@ -79,13 +74,9 @@ impl FontRenderer8x8 {
pub fn render( pub fn render(
&self, &self,
char: char, char: char,
target: &mut WindowMut<bool, Bitmap>, bitmap: &mut Bitmap,
offset: Origin<Pixels>,
) -> Result<(), RenderError> { ) -> Result<(), RenderError> {
let cache = &mut *self.cache.lock().unwrap();
if let Some(drawn_char) = cache.get(&char) {
target.deref_assign(drawn_char);
}
let glyph_id = self.get_glyph(char)?; let glyph_id = self.get_glyph(char)?;
let mut canvas = self.canvas.lock().unwrap(); let mut canvas = self.canvas.lock().unwrap();
@ -100,21 +91,20 @@ impl FontRenderer8x8 {
RasterizationOptions::Bilevel, RasterizationOptions::Bilevel,
)?; )?;
let mut bitmap = Bitmap::new(TILE_SIZE, TILE_SIZE).unwrap(); Self::copy_to_bitmap(canvas, bitmap, offset)
Self::copy_to_bitmap(canvas, &mut bitmap)?;
target.deref_assign(&bitmap);
cache.insert(char, bitmap);
Ok(())
} }
fn copy_to_bitmap( fn copy_to_bitmap(
canvas: MutexGuard<Canvas>, canvas: MutexGuard<Canvas>,
bitmap: &mut Bitmap, bitmap: &mut Bitmap,
offset: Origin<Pixels>,
) -> Result<(), RenderError> { ) -> Result<(), RenderError> {
for y in 0..TILE_SIZE { for y in 0..TILE_SIZE {
for x in 0..TILE_SIZE { for x in 0..TILE_SIZE {
let canvas_val = canvas.pixels[x + y * TILE_SIZE] != 0; let canvas_val = canvas.pixels[x + y * TILE_SIZE] != 0;
if !bitmap.set_optional(x, y, canvas_val) { let bitmap_x = offset.x + x;
let bitmap_y = offset.y + y;
if !bitmap.set_optional(bitmap_x, bitmap_y, canvas_val) {
return Err(OutOfBounds(x, y)); return Err(OutOfBounds(x, y));
} }
} }

View file

@ -24,7 +24,6 @@ const PIXEL_HEIGHT_WITH_SPACERS: usize =
PIXEL_HEIGHT + NUM_SPACERS * SPACER_HEIGHT; PIXEL_HEIGHT + NUM_SPACERS * SPACER_HEIGHT;
const OFF_COLOR: u32 = u32::from_ne_bytes([0u8, 0, 0, 0]); const OFF_COLOR: u32 = u32::from_ne_bytes([0u8, 0, 0, 0]);
const SPACER_COLOR: u32 = u32::from_ne_bytes([100u8, 100, 100, 0]);
#[derive(Debug)] #[derive(Debug)]
pub enum AppEvents { pub enum AppEvents {
@ -62,7 +61,7 @@ impl<'t> Gui<'t> {
if self.options.spacers && tile_y != 0 { if self.options.spacers && tile_y != 0 {
// cannot just frame.skip(PIXEL_WIDTH as usize * SPACER_HEIGHT as usize) because of typing // cannot just frame.skip(PIXEL_WIDTH as usize * SPACER_HEIGHT as usize) because of typing
for _ in 0..PIXEL_WIDTH * SPACER_HEIGHT { for _ in 0..PIXEL_WIDTH * SPACER_HEIGHT {
*frame.next().unwrap() = SPACER_COLOR; frame.next().unwrap();
} }
} }

View file

@ -39,12 +39,7 @@ fn main() {
.font .font
.map(FontRenderer8x8::from_name) .map(FontRenderer8x8::from_name)
.unwrap_or_else(FontRenderer8x8::default); .unwrap_or_else(FontRenderer8x8::default);
let context = CommandExecutionContext::new( let context = CommandExecutionContext::new(&display, &luma, font_renderer);
&display,
&luma,
font_renderer,
cli.experimental_null_char_handling,
);
let mut udp_server = UdpServer::new( let mut udp_server = UdpServer::new(
cli.bind, cli.bind,
stop_udp_rx, stop_udp_rx,