From 9352090965dd053641a73011f7ce8e0786407353 Mon Sep 17 00:00:00 2001 From: Rien Maertens Date: Sat, 12 Dec 2020 22:45:47 +0100 Subject: [PATCH] Add restream Rust executable --- .cargo/config | 14 ++++++ .gitignore | 1 + Cargo.toml | 9 ++++ src/main.rs | 133 ++++++++++++++++++++++++++++++++++++++++++++++++++ 4 files changed, 157 insertions(+) create mode 100644 .cargo/config create mode 100644 .gitignore create mode 100644 Cargo.toml create mode 100644 src/main.rs diff --git a/.cargo/config b/.cargo/config new file mode 100644 index 0000000..2b2e52a --- /dev/null +++ b/.cargo/config @@ -0,0 +1,14 @@ +[target.armv7-unknown-linux-gnueabihf] +linker = "/usr/local/oecore-x86_64/sysroots/x86_64-oesdk-linux/usr/bin/arm-oe-linux-gnueabi/arm-oe-linux-gnueabi-gcc" +rustflags = [ + "-C", "link-arg=-march=armv7-a", + "-C", "link-arg=-marm", + "-C", "link-arg=-mfpu=neon", + "-C", "link-arg=-mfloat-abi=hard", + "-C", "link-arg=-mcpu=cortex-a9", + "-C", "link-arg=--sysroot=/usr/local/oecore-x86_64/sysroots/cortexa9hf-neon-oe-linux-gnueabi", +] + +[build] +# Set the default --target flag +target = "armv7-unknown-linux-gnueabihf" diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..ea8c4bf --- /dev/null +++ b/.gitignore @@ -0,0 +1 @@ +/target diff --git a/Cargo.toml b/Cargo.toml new file mode 100644 index 0000000..c297c6d --- /dev/null +++ b/Cargo.toml @@ -0,0 +1,9 @@ +[package] +name = "restream" +version = "0.1.0" +authors = ["Rien Maertens "] +edition = "2018" + +[dependencies] +anyhow = "1.0" +lz-fear = "0.1" diff --git a/src/main.rs b/src/main.rs new file mode 100644 index 0000000..4eaa9d8 --- /dev/null +++ b/src/main.rs @@ -0,0 +1,133 @@ +#[macro_use] +extern crate anyhow; +extern crate lz_fear; + +use anyhow::{Context, Result}; +use lz_fear::CompressionSettings; + +use std::default::Default; +use std::fs::File; +use std::io::{BufRead, BufReader, Read, Seek, SeekFrom}; +use std::process::Command; + +fn main() -> Result<()> { + let version = remarkable_version()?; + let streamer = if version == "reMarkable 1.0\n" { + let width = 1408; + let height = 1872; + let bytes_per_pixel = 2; + ReStreamer::init("/dev/fb0", 0, width, height, bytes_per_pixel)? + } else if version == "reMarkable 2.0\n" { + let width = 1404; + let height = 1872; + let bytes_per_pixel = 1; + + let pid = xochitl_pid()?; + let offset = fb_start(pid)?; + let mem = format!("/proc/{}/mem", pid); + ReStreamer::init(&mem, offset, width, height, bytes_per_pixel)? + } else { + Err(anyhow!( + "Unknown reMarkable version: {}\nPlease open a feature request to support your device.", + version + ))? + }; + + let lz4: CompressionSettings = Default::default(); + lz4.compress(streamer, std::io::stdout().lock()) + .context("Error while compressing framebuffer stream") +} + +fn remarkable_version() -> Result { + let content = std::fs::read("/sys/devices/soc0/machine") + .context("Failed to read /sys/devices/soc0/machine")?; + Ok(String::from_utf8(content)?) +} + +fn xochitl_pid() -> Result { + let output = Command::new("/bin/pidof") + .args(&["xochitl"]) + .output() + .context("Failed to run `/bin/pidof xochitl`")?; + if output.status.success() { + let pid = &output.stdout; + std::str::from_utf8(pid)? + .parse() + .with_context(|| format!("Failed to parse xochitl's pid: {:?}", pid)) + } else { + Err(anyhow!( + "Could not find pid of xochitl, is xochitl running?" + )) + } +} + +fn fb_start(pid: usize) -> Result { + let file = File::open(format!("/proc/{}/maps", &pid))?; + let line = BufReader::new(file) + .lines() + .skip_while(|line| matches!(line, Ok(l) if l.ends_with("/dev/fb0"))) + //.skip(1) + .next() + .with_context(|| format!("No line containing /dev/fb0 in /proc/{}/maps file", pid))? + .with_context(|| format!("Error reading file /proc/{}/maps", pid))?; + + let addr = line + .split("-") + .next() + .with_context(|| format!("Error parsing line in /proc/{}/maps", pid))?; + + usize::from_str_radix(addr, 16).context("Error parsing framebuffer address") +} + +pub struct ReStreamer { + file: File, + start: u64, + cursor: usize, + size: usize, +} + +impl ReStreamer { + pub fn init( + path: &str, + offset: usize, + width: usize, + height: usize, + bytes_per_pixel: usize, + ) -> Result { + let start = offset as u64; + let size = width * height * bytes_per_pixel; + let cursor = 0; + let file = File::open(path)?; + let mut streamer = ReStreamer { + file, + start: start, + cursor, + size, + }; + streamer.next_frame()?; + Ok(streamer) + } + + pub fn next_frame(&mut self) -> std::io::Result<()> { + self.file.seek(SeekFrom::Start(self.start))?; + self.cursor = 0; + Ok(()) + } +} + +impl Read for ReStreamer { + fn read(&mut self, buf: &mut [u8]) -> std::io::Result { + let requested = buf.len(); + let bytes_read = if self.cursor + requested < self.size { + self.file.read(buf)? + } else { + let rest = self.size - self.cursor; + self.file.read(&mut buf[0..rest])? + }; + self.cursor += bytes_read; + if self.cursor == self.size { + self.next_frame()?; + } + Ok(bytes_read) + } +}