Skip to content

Commit

Permalink
add eprintln macro
Browse files Browse the repository at this point in the history
Signed-off-by: Finn Behrens <me@kloenk.de>
  • Loading branch information
kloenk committed Oct 4, 2020
1 parent c17d290 commit 1a6f692
Show file tree
Hide file tree
Showing 4 changed files with 58 additions and 18 deletions.
9 changes: 6 additions & 3 deletions drivers/char/rust_example/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,11 @@ struct RustExample {
impl KernelModule for RustExample {
fn init() -> KernelResult<Self> {
println!("Rust Example (init)");
println!("Am I built-in? {}", !cfg!(MODULE));
println!(
level: kernel::printk::KERN_DEBUG,
"Am I built-in? {}",
!cfg!(MODULE)
);
println!("Parameters:");
println!(" my_bool: {}", my_bool.read());
println!(" my_i32: {}", my_i32.read());
Expand All @@ -45,7 +49,6 @@ impl KernelModule for RustExample {
impl Drop for RustExample {
fn drop(&mut self) {
println!("My message is {}", self.message);
println!("Rust Example (exit)");
eprintln!("Rust Example (exit)");
}
}

7 changes: 7 additions & 0 deletions rust/kernel/build.rs
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,14 @@ const INCLUDED_VARS: &[&str] = &[
"FS_USERNS_MOUNT",
"FS_RENAME_DOES_D_MOVE",
"BINDINGS_GFP_KERNEL",
"KERN_EMERG",
"KERN_ALERT",
"KERN_CRIT",
"KERN_ERR",
"KERN_WARNING",
"KERN_NOTICE",
"KERN_INFO",
"KERN_DEBUG",
"VERIFY_WRITE",
"LINUX_VERSION_CODE",
"SEEK_SET",
Expand Down
7 changes: 1 addition & 6 deletions rust/kernel/src/prelude.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,9 +9,4 @@ pub use alloc::{

pub use module::module;

pub use super::{
println,
KernelResult,
KernelModule,
};

pub use super::{eprintln, println, KernelModule, KernelResult};
53 changes: 44 additions & 9 deletions rust/kernel/src/printk.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,13 +6,18 @@ use core::fmt;
use crate::bindings;
use crate::c_types::c_int;

pub use crate::bindings::{
KERN_ALERT, KERN_CRIT, KERN_DEBUG, KERN_EMERG, KERN_ERR, KERN_INFO, KERN_NOTICE, KERN_WARNING,
};

const LEVEL_LEN: usize = 3;

#[doc(hidden)]
pub fn printk(s: &[u8]) {
pub fn printk(s: &[u8], level: &'static [u8; LEVEL_LEN]) {
// Don't copy the trailing NUL from `KERN_INFO`.
let mut fmt_str = [0; bindings::KERN_INFO.len() - 1 + b"%.*s\0".len()];
fmt_str[..bindings::KERN_INFO.len() - 1]
.copy_from_slice(&bindings::KERN_INFO[..bindings::KERN_INFO.len() - 1]);
fmt_str[bindings::KERN_INFO.len() - 1..].copy_from_slice(b"%.*s\0");
let mut fmt_str = [0; LEVEL_LEN - 1 + b"%.*s\0".len()];
fmt_str[..LEVEL_LEN - 1].copy_from_slice(&level[..LEVEL_LEN - 1]);
fmt_str[LEVEL_LEN - 1..].copy_from_slice(b"%.*s\0");

// TODO: I believe printk never fails
unsafe { bindings::printk(fmt_str.as_ptr() as _, s.len() as c_int, s.as_ptr()) };
Expand Down Expand Up @@ -52,21 +57,51 @@ impl fmt::Write for LogLineWriter {

/// [`println!`] functions the same as it does in `std`, except instead of
/// printing to `stdout`, it writes to the kernel console at the `KERN_INFO`
/// level.
/// level. Level can be overwritten with the argument level.
///
/// [`println!`]: https://doc.rust-lang.org/stable/std/macro.println.html
#[macro_export]
macro_rules! println {
() => ({
$crate::printk::printk("\n".as_bytes());
println!(level: $crate::printk::KERN_INFO);
});
(level: $level:expr) => ({
$crate::printk::printk("\n".as_bytes(), $level);
});
($msg:expr) => ({
println!(level: $crate::printk::KERN_INFO, $msg);
});
(level: $level:expr, $msg:expr) => ({
$crate::printk::printk(concat!($msg, "\n").as_bytes(), $level);
});
(level: $level:expr, $fmt:expr, $($arg:tt)*) => ({
use ::core::fmt;
let mut writer = $crate::printk::LogLineWriter::new();
let _ = fmt::write(&mut writer, format_args!(concat!($fmt, "\n"), $($arg)*)).unwrap();
$crate::printk::printk(writer.as_bytes(), $crate::printk::KERN_INFO);
});
($fmt:expr, $($arg:tt)*) => ({
println!(level: $crate::printk::KERN_INFO, $fmt, $($arg)*);
});
}

/// [`eprintln!`] functions the same as it does in `std`, except instead of
/// printing to `stderr`, it writes to the kernel console at the `KERN_ERR`
/// level.
///
/// [`eprintln!`]: https://doc.rust-lang.org/stable/std/macro.eprintln.html
#[macro_export]
macro_rules! eprintln {
() => ({
$crate::printk::printk("\n".as_bytes(), $crate::printk::KERN_ERR);
});
($fmt:expr) => ({
$crate::printk::printk(concat!($fmt, "\n").as_bytes());
$crate::printk::printk(concat!($fmt, "\n").as_bytes(), $crate::printk::KERN_ERR);
});
($fmt:expr, $($arg:tt)*) => ({
use ::core::fmt;
let mut writer = $crate::printk::LogLineWriter::new();
let _ = fmt::write(&mut writer, format_args!(concat!($fmt, "\n"), $($arg)*)).unwrap();
$crate::printk::printk(writer.as_bytes());
$crate::printk::printk(writer.as_bytes(), $crate::printk::KERN_ERR);
});
}

0 comments on commit 1a6f692

Please sign in to comment.