Skip to content

Commit

Permalink
Implement a log buffer to store log messages(Draft initial commit).
Browse files Browse the repository at this point in the history
Signed-off-by: Vasant Karasulli <[email protected]>
  • Loading branch information
vsntk18 committed Jun 22, 2023
1 parent b4ca120 commit 26e9a9f
Show file tree
Hide file tree
Showing 5 changed files with 347 additions and 7 deletions.
2 changes: 2 additions & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,8 @@ pub mod fw_meta;
pub mod io;
pub mod kernel_launch;
pub mod locking;
pub mod log_buffer;
pub mod migrate;
pub mod mm;
pub mod protocols;
pub mod requests;
Expand Down
302 changes: 302 additions & 0 deletions src/log_buffer.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,302 @@
// SPDX-License-Identifier: MIT OR Apache-2.0
//
// Copyright (c) 2022-2023 SUSE LLC
//
//
//

use crate::mm::alloc::allocate_zeroed_page;
extern crate alloc;
use crate::error::SvsmError;
use crate::types::PAGE_SIZE;
use crate::utils::immut_after_init::ImmutAfterInitCell;
use alloc::vec::Vec;
use core::fmt;
use core::fmt::Write;
use core::sync::atomic::{AtomicUsize, Ordering};

pub const BUF_SIZE: usize = PAGE_SIZE;
pub const TAIL_MASK: usize = 0xffffusize;
pub const HEAD_MASK: usize = 0xffffusize << 16;
pub const WRITER_MASK: usize = 0xffffusize << 32;
pub const READER_MASK: usize = 0x7fffusize << 48;
pub const FULL_MASK: usize = 0x1usize << 63;

/* Buffer state encoding:
* bits 0-15 : tail offset
* bits 16-31 : head offset
* bits 32-48 : number of writers writing the buffer
* bits 49-62: number of readers reading the buffer
* bit 63 : set if buffer is full
*/
#[derive(Clone, Copy, Debug)]
struct LogBufferState {
tail: usize,
head: usize,
wc: usize,
rc: usize,
full: usize,
}

impl LogBufferState {
pub fn new(tail: usize, head: usize, wc: usize, rc: usize, full: usize) -> Self {
LogBufferState {
tail,
head,
wc,
rc,
full,
}
}

pub fn compute_state_write(&self, len: usize) -> Self {
let new_head = (self.head + len) % BUF_SIZE;
let place_left = if self.head >= self.tail {
BUF_SIZE - self.head + self.tail
} else {
self.tail - self.head
};
let is_full = if place_left <= len { 1 } else { self.full };
let new_tail = if is_full == 1 { new_head } else { self.tail };

LogBufferState::new(new_tail, new_head, self.wc + 1, self.rc, is_full)
}
}
impl From<usize> for LogBufferState {
fn from(state: usize) -> Self {
LogBufferState {
tail: state & TAIL_MASK,
head: (state & HEAD_MASK) >> 16,
wc: (state & WRITER_MASK) >> 32,
rc: (state & READER_MASK) >> 48,
full: (state & FULL_MASK) >> 63,
}
}
}

impl From<LogBufferState> for usize {
fn from(lb: LogBufferState) -> Self {
let t = lb.tail & TAIL_MASK;
let h = (lb.head << 16) & HEAD_MASK;
let w = (lb.wc << 32) & WRITER_MASK;
let r = (lb.rc << 48) & READER_MASK;
let f = (lb.full << 63) & FULL_MASK;
t | h | w | r | f
}
}
pub struct LogBuffer {
buf: Vec<u8>,
state: AtomicUsize,
}

impl LogBuffer {
pub const fn new() -> LogBuffer {
LogBuffer {
buf: Vec::new(),
state: AtomicUsize::new(0),
}
}

pub fn init(&mut self, buf_addr: *mut u8) {
self.buf = unsafe { Vec::from_raw_parts(buf_addr, BUF_SIZE, BUF_SIZE) };
}

pub fn write_log(&mut self, s: &str) {
let mut head;
let len = s.len();

loop {
let pos = self.state.load(Ordering::Acquire);
let st = LogBufferState::from(pos);
head = st.head;
/* wait if there are readers */
if st.rc > 0 {
core::hint::spin_loop();
continue;
}
let st_new = st.compute_state_write(len);
let new_pos = usize::from(st_new);
if self
.state
.compare_exchange(pos, new_pos, Ordering::Acquire, Ordering::Relaxed)
.is_ok()
{
break;
}
core::hint::spin_loop();
}

for b in s.bytes() {
self.buf[head] = b;
head = (head + 1) % BUF_SIZE;
}

/* Decrement writer count */
loop {
let pos = self.state.load(Ordering::Acquire);
let st = LogBufferState::from(pos);
let st_new = LogBufferState::new(st.tail, st.head, st.wc - 1, st.rc, st.full);
let new_pos = usize::from(st_new);
if self
.state
.compare_exchange(pos, new_pos, Ordering::Acquire, Ordering::Relaxed)
.is_ok()
{
break;
}
core::hint::spin_loop();
}
}

pub fn read_log(&mut self) -> Result<Vec<u8>, SvsmError> {
let ret: Result<Vec<u8>, SvsmError>;
let mut st;
loop {
let pos = self.state.load(Ordering::Acquire);
st = LogBufferState::from(pos);
/* wait if there are writers */
if st.wc > 0 {
core::hint::spin_loop();
continue;
}

let new_tail = st.head;
let st_new = LogBufferState::new(new_tail, st.head, st.wc, st.rc + 1, st.full);
let new_pos = usize::from(st_new);
if self
.state
.compare_exchange(pos, new_pos, Ordering::Acquire, Ordering::Relaxed)
.is_ok()
{
break;
}
core::hint::spin_loop();
}

if st.head == st.tail && st.full == 0 {
/* Buffer is empty */
ret = Ok(Vec::new());
} else if st.head > st.tail && st.full == 0 {
let vec = self.buf[st.tail..st.head].to_vec();
ret = Ok(vec);
} else {
let mut vec = self.buf[st.tail..].to_vec();
vec.extend_from_slice(&self.buf[..st.head]);
ret = Ok(vec);
}

/* clear the buffer-full status */
let is_full: usize = 0;

loop {
let pos = self.state.load(Ordering::Acquire);
let st = LogBufferState::from(pos);
let st_new = LogBufferState::new(st.tail, st.head, st.wc, st.rc - 1, is_full);
let new_pos = usize::from(st_new);
if self
.state
.compare_exchange(pos, new_pos, Ordering::Acquire, Ordering::Relaxed)
.is_ok()
{
break;
}
core::hint::spin_loop();
}
ret
}
}

impl fmt::Write for LogBuffer {
fn write_str(&mut self, s: &str) -> fmt::Result {
self.write_log(s);
Ok(())
}
}

#[derive(Clone, Copy)]
struct BufferLogger {
component: &'static str,
}

impl BufferLogger {
fn new(component: &'static str) -> BufferLogger {
BufferLogger { component }
}
}

impl log::Log for BufferLogger {
fn enabled(&self, _metadata: &log::Metadata) -> bool {
true
}

fn log(&self, record: &log::Record) {
let component: &'static str = &self.component;
// Log format/detail depends on the level.
match record.metadata().level() {
log::Level::Error | log::Level::Warn => {
unsafe {
write!(
&mut LB,
"[{}] {}: {}\n",
component,
record.metadata().level().as_str(),
record.args()
)
.expect("write error")
};
}

log::Level::Info => {
unsafe {
write!(&mut LB, "[{}] {}\n", component, record.args()).expect("write error")
};
}

log::Level::Debug | log::Level::Trace => {
unsafe {
write!(
&mut LB,
"[{}/{}] {} {}\n",
component,
record.metadata().target(),
record.metadata().level().as_str(),
record.args()
)
.expect("write error")
};
}
}
}

fn flush(&self) {}
}

pub static mut LB: LogBuffer = LogBuffer::new();
static BUFFER_LOGGER: ImmutAfterInitCell<BufferLogger> = ImmutAfterInitCell::uninit();

pub fn init_log_buffer() {
let buf_addr = allocate_zeroed_page()
.expect("Failed to allocate buffer page")
.as_mut_ptr::<u8>();
unsafe { LB.init(buf_addr) };
}

pub fn migrate_log_buffer(log_buf: &LogBuffer) {
init_log_buffer();
let val = log_buf.state.load(Ordering::Acquire);
unsafe { LB.state.store(val, Ordering::Release) };
unsafe { LB.buf = log_buf.buf.clone() };
}

pub fn install_buffer_logger(component: &'static str) -> Result<(), ()> {
BUFFER_LOGGER
.init(&BufferLogger::new(component))
.expect("already initialized the logger");
if let Err(_) = log::set_logger(&*BUFFER_LOGGER) {
return Err(());
}

// Log levels are to be configured via the log's library feature configuration.
log::set_max_level(log::LevelFilter::Trace);
Ok(())
}
24 changes: 24 additions & 0 deletions src/migrate.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
// SPDX-License-Identifier: MIT OR Apache-2.0
//
// Copyright (c) 2022-2023 SUSE LLC
//
//
use crate::address::VirtAddr;
use crate::log_buffer::LogBuffer;

// struct containing information that
// is migrated from stage2 to svsm kernel
#[repr(C)]
pub struct MigrateInfo {
pub bitmap_addr: VirtAddr,
pub log_buf: &'static LogBuffer,
}

impl MigrateInfo {
pub fn new(vb: VirtAddr, lb: &'static LogBuffer) -> Self {
MigrateInfo {
bitmap_addr: vb,
log_buf: lb,
}
}
}
11 changes: 9 additions & 2 deletions src/stage2.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,8 @@ use svsm::cpu::percpu::{this_cpu_mut, PerCpu};
use svsm::elf;
use svsm::fw_cfg::FwCfg;
use svsm::kernel_launch::KernelLaunchInfo;
use svsm::log_buffer::{init_log_buffer, install_buffer_logger, LB};
use svsm::migrate::MigrateInfo;
use svsm::mm::alloc::{memory_info, print_memory_info, root_mem_init};
use svsm::mm::init_kernel_mapping_info;
use svsm::mm::pagetable::{
Expand Down Expand Up @@ -83,7 +85,7 @@ static mut CONSOLE_SERIAL: SerialPort = SerialPort {
};

fn setup_env() {
install_console_logger("Stage2");
//install_console_logger("Stage2");
init_kernel_mapping_info(
VirtAddr::null(),
VirtAddr::from(640 * 1024usize),
Expand All @@ -103,6 +105,10 @@ fn setup_env() {
WRITER.lock().set(&mut CONSOLE_SERIAL);
}
init_console();
init_log_buffer();
if let Err(_) = install_buffer_logger("Stage2") {
panic!("log buffer installation error")
}

// Console is fully working now and any unsupported configuration can be
// properly reported.
Expand Down Expand Up @@ -286,6 +292,7 @@ pub extern "C" fn stage2_main(launch_info: &Stage1LaunchInfo) {

let kernel_entry = kernel_elf.get_entry(kernel_vaddr_alloc_base);
let valid_bitmap = valid_bitmap_addr();
let migrate_info = unsafe { MigrateInfo::new(VirtAddr::from(valid_bitmap.bits()), &LB) };

// Shut down the GHCB
shutdown_percpu();
Expand All @@ -294,7 +301,7 @@ pub extern "C" fn stage2_main(launch_info: &Stage1LaunchInfo) {
asm!("jmp *%rax",
in("rax") kernel_entry,
in("r8") &launch_info,
in("r9") valid_bitmap.bits(),
in("r9") &migrate_info,
options(att_syntax))
};

Expand Down
Loading

0 comments on commit 26e9a9f

Please sign in to comment.