-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Signed-off-by: YdrMaster <[email protected]>
- Loading branch information
Showing
6 changed files
with
145 additions
and
105 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,98 @@ | ||
use crate::{tree_on_stack::Node, Str}; | ||
use core::fmt; | ||
|
||
/// 遍历上下文。 | ||
pub struct Context<'a>(Node<'a, Inner<'a>>); | ||
|
||
struct Inner<'a> { | ||
name: Str<'a>, | ||
cells: Cells, | ||
} | ||
|
||
impl Context<'_> { | ||
pub(crate) const ROOT: Self = Context(Node::root(Inner { | ||
name: Str(b""), | ||
cells: Cells::DEFAULT, | ||
})); | ||
|
||
/// 返回路径层数。定义根节点的子节点层数为 0。 | ||
#[inline] | ||
pub fn level(&self) -> usize { | ||
self.0.level() | ||
} | ||
|
||
/// 如果这是根节点的路径则返回 `true`。 | ||
#[inline] | ||
pub fn is_root(&self) -> bool { | ||
self.0.is_root() | ||
} | ||
|
||
/// 返回路径最后一级的节点名。 | ||
#[inline] | ||
pub fn name(&self) -> Str { | ||
self.0.as_ref().name | ||
} | ||
|
||
#[inline] | ||
pub(crate) fn cells(&self) -> Cells { | ||
self.0.as_ref().cells | ||
} | ||
|
||
/// 将路径字符串格式化到 `buf` 中。 | ||
/// | ||
/// 如果返回 `Ok(n)`,表示字符串长度为 `n`(`n` 不大于 `buf.len()`)。 | ||
/// 如果返回 `Err(n)`,表示缓冲区长度无法存放整个字符串,实现保证 `n` 等于 `buf.len()`。 | ||
pub fn fmt_path(&self, buf: &mut [u8]) -> Result<usize, usize> { | ||
self.0.fold(0, |len, inner| match buf.len() - len { | ||
0 => Err(buf.len()), | ||
mut rest => { | ||
let bytes = inner.name.as_bytes(); | ||
buf[len] = b'/'; | ||
rest -= 1; | ||
if bytes.len() > rest { | ||
buf[len + 1..].copy_from_slice(&bytes[..rest]); | ||
Err(buf.len()) | ||
} else { | ||
buf[len + 1..][..bytes.len()].copy_from_slice(bytes); | ||
Ok(len + bytes.len() + 1) | ||
} | ||
} | ||
}) | ||
} | ||
} | ||
|
||
impl<'a> Context<'a> { | ||
#[inline] | ||
pub(crate) fn grow(&'a self, name: Str<'a>, cells: Cells) -> Self { | ||
Self(self.0.grow(Inner { name, cells })) | ||
} | ||
} | ||
|
||
impl fmt::Display for Context<'_> { | ||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||
self.0.fold((), |(), inner| { | ||
'/'.fmt(f)?; | ||
unsafe { inner.name.as_str_unchecked() }.fmt(f) | ||
}) | ||
} | ||
} | ||
|
||
#[derive(Clone, Copy)] | ||
pub(crate) struct Cells { | ||
pub address: u32, | ||
pub size: u32, | ||
pub interrupt: u32, | ||
} | ||
|
||
impl Cells { | ||
pub const DEFAULT: Self = Self { | ||
address: 2, | ||
size: 1, | ||
interrupt: 1, | ||
}; | ||
|
||
#[inline] | ||
pub fn reg_size(&self) -> usize { | ||
(self.address + self.size) as _ | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters