From 5b4b95fc25d92615fe2555cac0d53c425e5201b0 Mon Sep 17 00:00:00 2001 From: Romain Porte Date: Sat, 8 Feb 2020 14:28:16 +0100 Subject: add width and height support (hack) --- Cargo.toml | 1 + examples/width_height.rs | 12 ++++++++++++ src/lib.rs | 27 +++++++++++++++++++++++++++ src/of_node.rs | 24 ++++++++++++++++++++++++ 4 files changed, 64 insertions(+) create mode 100644 examples/width_height.rs create mode 100644 src/of_node.rs diff --git a/Cargo.toml b/Cargo.toml index 2c6276c..3c7837b 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -7,3 +7,4 @@ edition = "2018" # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [dependencies] +byteorder = "1.*" diff --git a/examples/width_height.rs b/examples/width_height.rs new file mode 100644 index 0000000..77bb5da --- /dev/null +++ b/examples/width_height.rs @@ -0,0 +1,12 @@ +use charlcd::Screen; +use std::io::Write; + +fn main() -> std::io::Result<()> { + let mut screen = Screen::default()?; + + screen.clear()?; + screen.write(format!("width: {}\n", screen.width()?).as_bytes())?; + screen.write(format!("height: {}\n", screen.height()?).as_bytes())?; + + Ok(()) +} diff --git a/src/lib.rs b/src/lib.rs index f849712..0e32ee9 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -1,4 +1,5 @@ mod codes; +mod of_node; use std::fs::{File, OpenOptions}; use std::path::Path; @@ -198,4 +199,30 @@ impl FileScreen { pub fn default() -> std::io::Result { Screen::from_dev_path(&Path::new(DEFAULT_SCREEN_DEV_PATH)) } + + /// Get the width of the screen, in number of characters it can display. + /// + /// **Important note:** The implementation behind this function is + /// currently a hack that will go find the value in the `auxdisplay` + /// platform device tree node in + /// `/sys/devices/platform/auxdisplay/of_node/*`. This is because the + /// `charlcd` driver does not export the screen width nor height to + /// userspace (could be using `ioctl` or `read` syscalls). + /// + pub fn width(&self) -> std::io::Result { + of_node::display_width_chars() + } + + /// Get the height of the screen, in number of characters it can display. + /// + /// **Important note:** The implementation behind this function is + /// currently a hack that will go find the value in the `auxdisplay` + /// platform device tree node in + /// `/sys/devices/platform/auxdisplay/of_node/*`. This is because the + /// `charlcd` driver does not export the screen width nor height to + /// userspace (could be using `ioctl` or `read` syscalls). + /// + pub fn height(&self) -> std::io::Result { + of_node::display_height_chars() + } } diff --git a/src/of_node.rs b/src/of_node.rs new file mode 100644 index 0000000..5dff2a0 --- /dev/null +++ b/src/of_node.rs @@ -0,0 +1,24 @@ +// This module is a hack to read additional information in the DT nodes in the +// kernel to retrieve data that is currently not exposed in the charlcd.c +// driver. + +use std::fs::File; +use std::io::Result; + +use byteorder::{BigEndian, ReadBytesExt}; + +fn read_u32_node(node_name: &str) -> Result { + let mut f = File::open(format!( + "/sys/devices/platform/auxdisplay/of_node/{}", + node_name + ))?; + Ok(f.read_u32::()?) +} + +pub fn display_height_chars() -> Result { + read_u32_node("display-height-chars") +} + +pub fn display_width_chars() -> Result { + read_u32_node("display-width-chars") +} -- cgit v1.2.3