737 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			737 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| use binparse_derive::AllVariants;
 | |
| use binparse_derive::ParseBin;
 | |
| use std::fmt;
 | |
| use std::error;
 | |
| use std::ffi::CString;
 | |
| 
 | |
| /***** BTRFS structures *****/
 | |
| 
 | |
| pub const NODE_SIZE: usize = 0x4000;
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,Copy,AllVariants,PartialEq,Eq,PartialOrd,Ord)]
 | |
| #[repr(u8)]
 | |
| pub enum ItemType {
 | |
| 	Invalid = 0x00,           // invalid
 | |
| 	Inode = 0x01,             // implemented
 | |
| 	Ref = 0x0c,               // implemented
 | |
| 	ExtRef = 0x0d,
 | |
| 	XAttr = 0x18,
 | |
| 	VerityDesc = 0x24,
 | |
| 	VerityMerkle = 0x25,
 | |
| 	Orphan = 0x30,
 | |
| 	DirLog = 0x3c,
 | |
| 	DirLogIndex = 0x48,
 | |
| 	Dir = 0x54,               // implemented (better with len feature; allow multiple?)
 | |
| 	DirIndex = 0x60,          // implemented
 | |
| 	ExtentData = 0x6c,        // implemented
 | |
| 	ExtentCsum = 0x80,
 | |
| 	Root = 0x84,              // implemented
 | |
| 	RootBackRef = 0x90,
 | |
| 	RootRef = 0x9c,
 | |
| 	Extent = 0xa8,            // implemented (with only one version of extra data)
 | |
| 	Metadata = 0xa9,          // implemented (with only one version of extra data)
 | |
| 	TreeBlockRef = 0xb0,
 | |
| 	ExtentDataRef = 0xb2,
 | |
| 	ExtentRefV0 = 0xb4,
 | |
| 	SharedBlockRef = 0xb6,
 | |
| 	SharedDataRef = 0xb8,
 | |
| 	BlockGroup = 0xc0,        // implemented
 | |
| 	FreeSpaceInfo = 0xc6,     // implemented
 | |
| 	FreeSpaceExtent = 0xc7,   // implemented
 | |
| 	FreeSpaceBitmap = 0xc8,
 | |
| 	DevExtent = 0xcc,         // implemented
 | |
| 	Dev = 0xd8,               // implemented
 | |
| 	Chunk = 0xe4,             // implemented
 | |
| 	QGroupStatus = 0xf0,
 | |
| 	QGroupInfo = 0xf2,
 | |
| 	QGroupLimit = 0xf4,
 | |
| 	QGroupRelation = 0xf6,
 | |
| 	Temporary = 0xf8,
 | |
| 	Persistent = 0xf9,        // TODO?
 | |
| 	DevReplace = 0xfa,
 | |
| 	UUIDSubvol = 0xfb,        // implemented
 | |
| 	UUIDReceivedSubvol = 0xfc,
 | |
| 	String = 0xfd,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,Copy,ParseBin,PartialEq,Eq,PartialOrd,Ord)]
 | |
| pub struct Key {
 | |
| 	pub key_id: u64,
 | |
| 	pub key_type: ItemType,
 | |
| 	pub key_offset: u64,
 | |
| }
 | |
| 
 | |
| impl Key {
 | |
| 	pub fn new(key_id: u64, key_type: ItemType, key_offset: u64) -> Key {
 | |
| 		Key { key_id, key_type, key_offset }
 | |
| 	}
 | |
| 
 | |
| 	pub fn id(key_id: u64) -> Key {
 | |
| 		Key { key_id, key_type: ItemType::Invalid, key_offset: 0 }
 | |
| 	}
 | |
| }
 | |
| 
 | |
| pub const ZERO_KEY: Key = Key {key_id: 0, key_type: ItemType::Invalid, key_offset: 0};
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone)]
 | |
| pub enum Value {
 | |
| 	Extent(ExtentItem),
 | |
| 	BlockGroup(BlockGroupItem),
 | |
| 	Inode(InodeItem),
 | |
| 	Chunk(ChunkItem),
 | |
| 	Root(RootItem),
 | |
| 	Dir(DirItem),
 | |
| 	DirIndex(DirItem),
 | |
| 	FreeSpaceInfo(FreeSpaceInfoItem),
 | |
| 	FreeSpaceExtent,
 | |
| 	UUIDSubvol(UUIDSubvolItem),
 | |
| 	Dev(DevItem),
 | |
| 	DevExtent(DevExtentItem),
 | |
| 	ExtentData(ExtentDataItem),
 | |
| 	Ref(RefItem),
 | |
| 	Unknown(Vec<u8>),
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone)]
 | |
| pub struct Item {
 | |
| 	pub key: Key,
 | |
| 	pub value: Value,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Clone,ParseBin)]
 | |
| pub struct Checksum([u8; 32]);
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Clone,ParseBin)]
 | |
| pub struct UUID([u8; 16]);
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,ParseBin)]
 | |
| pub struct Superblock {
 | |
| 	pub csum: Checksum,
 | |
| 	pub fsid: UUID,
 | |
| 	pub bytenr: u64,
 | |
| 	pub flags: u64,
 | |
| 	pub magic: u64,
 | |
| 	pub generation: u64,
 | |
| 	pub root: u64,
 | |
| 	pub chunk_root: u64,
 | |
| 	pub log_root: u64,
 | |
| 	#[skip_bytes = 8]
 | |
| 	pub total_bytes: u64,
 | |
| 	pub bytes_used: u64,
 | |
| 
 | |
| 	pub root_dir_objectid: u64,
 | |
| 	pub num_devices: u64,
 | |
| 	pub sectorsize: u32,
 | |
| 	pub nodesize: u32,
 | |
| 	#[skip_bytes = 4]
 | |
| 	pub stripesize: u32,
 | |
| 	pub sys_chunk_array_size: u32,
 | |
| 	pub chunk_root_generation: u64,
 | |
| 	pub compat_flags: u64,
 | |
| 	pub compat_ro_flags: u64,
 | |
| 	pub incompat_flags: u64,
 | |
| 	pub csum_type: u16,
 | |
| 	pub root_level: u8,
 | |
| 	pub chunk_root_level: u8,
 | |
| 	pub log_root_level: u8,
 | |
| 
 | |
| 	pub dev_item: DevItem,
 | |
| 
 | |
| 	pub label: [u8; 0x100],
 | |
| 
 | |
| 	pub cache_generation: u64,
 | |
| 	pub uuid_tree_generation: u64,
 | |
| 
 | |
| 	pub metadata_uuid: UUID,
 | |
| 	pub nr_global_roots: u64,
 | |
| 
 | |
| 	#[skip_bytes = 216]
 | |
| 	pub sys_chunk_array: [u8; 0x800],
 | |
| 
 | |
| 	// next up would be the root backups, but we don't read them for now
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct NodeHeader {
 | |
| 	pub csum: Checksum,
 | |
| 	pub fs_uid: UUID,
 | |
| 	pub bytenr: u64,
 | |
| 	pub flags: u64,
 | |
| 	pub chunk_tree_uid: UUID,
 | |
| 	pub generation: u64,
 | |
| 	pub owner: u64,
 | |
| 	pub nritems: u32,
 | |
| 	pub level: u8,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone)]
 | |
| pub struct Leaf {
 | |
| 	pub header: NodeHeader,
 | |
| 	pub items: Vec<Item>,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct KeyPointer {
 | |
| 	pub key: Key,
 | |
| 	pub ptr: u64,
 | |
| 	pub generation: u64,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone)]
 | |
| pub struct InteriorNode {
 | |
| 	pub header: NodeHeader,
 | |
| 	pub children: Vec<KeyPointer>,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone)]
 | |
| pub enum Node {
 | |
| 	Interior(InteriorNode),
 | |
| 	Leaf(Leaf),
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct BlockGroupItem {
 | |
| 	pub used: u64,
 | |
| 	pub chunk_objectid: u64,
 | |
| 	pub flags: u64,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone)]
 | |
| pub struct ExtentItem {
 | |
| 	pub refs: u64,
 | |
| 	pub generation: u64,
 | |
| 	pub flags: u64,
 | |
| 	//	pub data: Vec<u8>,
 | |
| 
 | |
| 	// this is only correct if flags == 2, fix later!
 | |
| 	pub block_refs: Vec<(ItemType, u64)>,
 | |
| //	pub tree_block_key_type: ItemType,
 | |
| //	pub tree_block_key_id: u64,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct Time {
 | |
| 	pub sec: u64,
 | |
| 	pub nsec: u32,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct InodeItem {
 | |
| 	pub generation: u64,
 | |
| 	pub transid: u64,
 | |
| 	pub size: u64,
 | |
| 	pub nbytes: u64,
 | |
| 	pub block_group: u64,
 | |
| 	pub nlink: u32,
 | |
| 	pub uid: u32,
 | |
| 	pub gid: u32,
 | |
| 	pub mode: u32,
 | |
| 	pub rdev: u64,
 | |
| 	pub flags: u64,
 | |
| 	pub sequence: u64,
 | |
| 	#[skip_bytes = 32]
 | |
| 	pub atime: Time,
 | |
| 	pub ctime: Time,
 | |
| 	pub mtime: Time,
 | |
| 	pub otime: Time,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct ChunkItem {
 | |
| 	pub size: u64,
 | |
| 	pub root: u64,
 | |
| 	pub stripelen: u64,
 | |
| 	pub chunktype: u64,
 | |
| 	pub align: u32,
 | |
| 	pub width: u32,
 | |
| 	pub sectorsize: u32,
 | |
| 	pub nrstripes: u16,
 | |
| 	pub substripes: u16,
 | |
| 
 | |
| 	#[len = "nrstripes"]
 | |
| 	pub stripes: Vec<ChunkItemStripe>,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct ChunkItemStripe {
 | |
| 	pub devid: u64,
 | |
| 	pub offset: u64,
 | |
| 	pub devuuid: UUID,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct RootItem {
 | |
| 	pub inode: InodeItem,
 | |
| 	pub generation: u64,
 | |
| 	pub root_dirid: u64,
 | |
| 	pub bytenr: u64,
 | |
| 	pub byte_limit: u64,
 | |
| 	pub bytes_used: u64,
 | |
| 	pub last_snapshot: u64,
 | |
| 	pub flags: u64,
 | |
| 	pub refs: u32,
 | |
| 	pub drop_progress: Key,
 | |
| 	pub drop_level: u8,
 | |
| 	pub level: u8,
 | |
| 
 | |
| 	pub generation_v2: u64,
 | |
| 	pub uuid: UUID,
 | |
| 	pub parent_uuid: UUID,
 | |
| 	pub received_uuid: UUID,
 | |
| 	pub ctransid: u64,
 | |
| 	pub otransid: u64,
 | |
| 	pub stransid: u64,
 | |
| 	pub rtransid: u64,
 | |
| 	pub ctime: Time,
 | |
| 	pub otime: Time,
 | |
| 	pub stime: Time,
 | |
| 	pub rtime: Time,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct DirItem {
 | |
| 	location: Key,
 | |
| 	transid: u64,
 | |
| 	data_len: u16,
 | |
| 	name_len: u16,
 | |
| 	dir_type: u8,
 | |
| 
 | |
| //	#[len = "name_len"]
 | |
| 	name: CString,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct FreeSpaceInfoItem {
 | |
| 	extent_count: u32,
 | |
| 	flags: u32,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct UUIDSubvolItem {
 | |
| 	subvol_id: u64,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct DevItem {
 | |
| 	devid: u64,
 | |
| 	total_bytes: u64,
 | |
| 	bytes_used: u64,
 | |
| 	io_align: u32,
 | |
| 	io_width: u32,
 | |
| 	sector_size: u32,
 | |
| 	dev_type: u64,
 | |
| 	generation: u64,
 | |
| 	start_offset: u64,
 | |
| 	dev_group: u32,
 | |
| 	seek_speed: u8,
 | |
| 	bandwidth: u8,
 | |
| 	uuid: UUID,
 | |
| 	fsid: UUID,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct DevExtentItem {
 | |
| 	chunk_tree: u64,
 | |
| 	chunk_objectid: u64,
 | |
| 	chunk_offset: u64,
 | |
| 	length: u64,
 | |
| 	chunk_tree_uuid: UUID,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone)]
 | |
| pub struct ExtentDataItem {
 | |
| 	header: ExtentDataHeader,
 | |
| 	data: ExtentDataBody,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone)]
 | |
| pub enum ExtentDataBody {
 | |
| 	Inline(Vec<u8>),
 | |
| 	External(ExternalExtent),
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct ExternalExtent {
 | |
| 	disk_bytenr: u64,
 | |
| 	disk_num_bytes: u64,
 | |
| 	offset: u64,
 | |
| 	num_bytes: u64,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct ExtentDataHeader {
 | |
| 	generation: u64,
 | |
| 	ram_bytes: u64,
 | |
| 	compression: u8,
 | |
| 	encryption: u8,
 | |
| 	other_encoding: u16,
 | |
| 	extent_type: u8,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[derive(Debug,Clone,ParseBin)]
 | |
| pub struct RefItem {
 | |
| 	index: u64,
 | |
| 	name_len: u16,
 | |
| 
 | |
| 	//	#[len = "name_len"]
 | |
| 	name: Vec<u8>,
 | |
| }
 | |
| 
 | |
| #[allow(unused)]
 | |
| #[repr(u64)]
 | |
| #[derive(Clone,Copy,Debug)]
 | |
| pub enum TreeID {
 | |
| 	Root = 1,
 | |
| 	Extent = 2,
 | |
| 	Chunk = 3,
 | |
| 	Dev = 4,
 | |
| 	FS = 5,
 | |
| 	RootDir = 6,
 | |
| 	CSum = 7,
 | |
| 	Quota = 8,
 | |
| 	UUID = 9,
 | |
| 	FreeSpace = 10,
 | |
| 	BlockGroup = 11,
 | |
| }
 | |
| 
 | |
| impl From<TreeID> for u64 {
 | |
| 	fn from(value: TreeID) -> u64 {
 | |
| 		value as u64
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /***** trait for parsing, and implementations for basic types *****/
 | |
| // most of the more complex types will be parsed using derive macros
 | |
| 
 | |
| #[derive(Debug)]
 | |
| pub struct ParseError(String);
 | |
| 
 | |
| impl error::Error for ParseError {}
 | |
| 
 | |
| impl fmt::Display for ParseError {
 | |
| 	fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
 | |
| 		write!(f, "{}", &self.0)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl From<String> for ParseError {
 | |
| 	fn from(value: String) -> ParseError {
 | |
| 		ParseError(value)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl From<&str> for ParseError {
 | |
| 	fn from(value: &str) -> ParseError {
 | |
| 		ParseError::from(String::from(value))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| pub trait ParseBin where Self: Sized {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(Self, usize), ParseError>;
 | |
| 
 | |
| 	fn parse(bytes: &[u8]) -> Result<Self, ParseError> {
 | |
| 		Self::parse_len(bytes).map(|x|x.0)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl ParseBin for u8 {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(Self, usize), ParseError> {
 | |
| 		if bytes.len() < 1 {
 | |
| 			err!("not enough data")
 | |
| 		} else {
 | |
| 			Ok((bytes[0], 1))
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl ParseBin for u16 {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(Self, usize), ParseError> {
 | |
| 		if bytes.len() < 2 {
 | |
| 			err!("not enough data")
 | |
| 		} else {
 | |
| 			let result = u16::from_le_bytes(bytes[0..2].try_into().unwrap());
 | |
| 			Ok((result, 2))
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl ParseBin for u32 {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(Self, usize), ParseError> {
 | |
| 		if bytes.len() < 4 {
 | |
| 			err!("not enough data")
 | |
| 		} else {
 | |
| 			let result = u32::from_le_bytes(bytes[0..4].try_into().unwrap());
 | |
| 			Ok((result, 4))
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl ParseBin for u64 {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(Self, usize), ParseError> {
 | |
| 		if bytes.len() < 8 {
 | |
| 			err!("not enough data")
 | |
| 		} else {
 | |
| 			let result = u64::from_le_bytes(bytes[0..8].try_into().unwrap());
 | |
| 			Ok((result, 8))
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl<const N: usize> ParseBin for [u8; N] {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<([u8; N], usize), ParseError> {
 | |
| 		if bytes.len() < N {
 | |
| 			err!("not enough data")
 | |
| 		} else {
 | |
| 			Ok((bytes[0..N].try_into().unwrap(), N))
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // we use Vec<u8> for "unknown extra data", so just eat up everything
 | |
| impl ParseBin for Vec<u8> {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(Self, usize), ParseError> {
 | |
| 		Ok((Vec::from(bytes), bytes.len()))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl ParseBin for CString {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(Self, usize), ParseError> {
 | |
| 		let mut chars = Vec::from(bytes);
 | |
| 		chars.push(0);
 | |
| 		Ok((CString::from_vec_with_nul(chars).unwrap(), bytes.len()))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /***** parsing ItemType and Leaf *****/
 | |
| 
 | |
| impl From<ItemType> for u8 {
 | |
| 	fn from(value: ItemType) -> u8 {
 | |
| 		value as u8
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl From<u8> for ItemType {
 | |
| 	fn from(value: u8) -> ItemType {
 | |
| 		let variants = ItemType::all_variants();
 | |
| 		match variants.binary_search_by_key(&value, |x|u8::from(*x)) {
 | |
| 			Ok(idx) => variants[idx],
 | |
| 			Err(_) => ItemType::Invalid,
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl ParseBin for ItemType {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(ItemType, usize), ParseError> {
 | |
| 		u8::parse(bytes).map(|x|(ItemType::from(x),1))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl ParseBin for Node {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(Node, usize), ParseError> {
 | |
| 		if bytes.len() < 0x65 {
 | |
| 			return err!("Not enough data to parse node header");
 | |
| 		}
 | |
| 
 | |
| 		let header = NodeHeader::parse(&bytes[0..0x65])?;
 | |
| 
 | |
| 		if header.level > 0 {
 | |
| 			// interior node
 | |
| 			let mut children = Vec::new();
 | |
| 			let num = header.nritems as usize;
 | |
| 
 | |
| 			for i in 0 .. num {
 | |
| 				children.push(KeyPointer::parse(&bytes[0x65 + i*0x21 .. 0x86 + i*0x21])?);
 | |
| 			}
 | |
| 
 | |
| 			Ok((Node::Interior(InteriorNode { header, children }), NODE_SIZE))
 | |
| 		} else {
 | |
| 			// leaf node
 | |
| 			let mut items = Vec::new();
 | |
| 
 | |
| 			for i in 0..header.nritems as usize {
 | |
| 				let key =    Key::parse(&bytes[i*0x19 + 0x65        .. i*0x19 + 0x65 + 0x11])?;
 | |
| 				let offset = u32::parse(&bytes[i*0x19 + 0x65 + 0x11 .. i*0x19 + 0x65 + 0x15])?;
 | |
| 				let size =   u32::parse(&bytes[i*0x19 + 0x65 + 0x15 .. i*0x19 + 0x65 + 0x19])?;
 | |
| 
 | |
| 				let data_slice = &bytes[0x65 + offset as usize ..
 | |
| 										0x65 + offset as usize + size as usize];
 | |
| 
 | |
| 				let value = match key.key_type {
 | |
| 					ItemType::BlockGroup =>
 | |
| 						Value::BlockGroup(BlockGroupItem::parse(data_slice)?),
 | |
| 					ItemType::Metadata => {
 | |
| 						let item = ExtentItem::parse(data_slice)?;
 | |
| 						if item.flags != 2 || item.refs > 1 {
 | |
| 							println!("Metadata item with refs = {}, flags = {}, data = {:x?}", item.refs, item.flags, &data_slice[0x18..]);
 | |
| 						}
 | |
| 						Value::Extent(item)
 | |
| 					},
 | |
| 					ItemType::Extent =>
 | |
| 						Value::Extent(ExtentItem::parse(data_slice)?),
 | |
| 					ItemType::Inode =>
 | |
| 						Value::Inode(InodeItem::parse(data_slice)?),
 | |
| 					ItemType::Root =>
 | |
| 						Value::Root(RootItem::parse(data_slice)?),
 | |
| 					ItemType::Dir =>
 | |
| 						Value::Dir(DirItem::parse(data_slice)?),
 | |
| 					ItemType::DirIndex =>
 | |
| 						Value::DirIndex(DirItem::parse(data_slice)?),
 | |
| 					ItemType::Chunk =>
 | |
| 						Value::Chunk(ChunkItem::parse(data_slice)?),
 | |
| 					ItemType::FreeSpaceInfo =>
 | |
| 						Value::FreeSpaceInfo(FreeSpaceInfoItem::parse(data_slice)?),
 | |
| 					ItemType::FreeSpaceExtent =>
 | |
| 						Value::FreeSpaceExtent,
 | |
| 					ItemType::UUIDSubvol =>
 | |
| 						Value::UUIDSubvol(UUIDSubvolItem::parse(data_slice)?),
 | |
| 					ItemType::Dev =>
 | |
| 						Value::Dev(DevItem::parse(data_slice)?),
 | |
| 					ItemType::DevExtent =>
 | |
| 						Value::DevExtent(DevExtentItem::parse(data_slice)?),
 | |
| 					ItemType::ExtentData =>
 | |
| 						Value::ExtentData(ExtentDataItem::parse(data_slice)?),
 | |
| 					ItemType::Ref =>
 | |
| 						Value::Ref(RefItem::parse(data_slice)?),
 | |
| 					_ =>
 | |
| 						Value::Unknown(Vec::from(data_slice)),
 | |
| 				};
 | |
| 
 | |
| 				items.push(Item { key, value });
 | |
| 			}
 | |
| 
 | |
| 			Ok((Node::Leaf(Leaf { header, items }), NODE_SIZE))
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl ParseBin for InteriorNode {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(Self, usize), ParseError> {
 | |
| 		match Node::parse_len(bytes)? {
 | |
| 			(Node::Interior(int_node), len) => Ok((int_node, len)),
 | |
| 			_ => err!("Expected interior node, found leaf"),
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl ParseBin for Leaf {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(Self, usize), ParseError> {
 | |
| 		match Node::parse_len(bytes)? {
 | |
| 			(Node::Leaf(leaf_node), len) => Ok((leaf_node, len)),
 | |
| 			_ => err!("Expected leaf, found interior node"),
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // ExtentDataItem needs a custom implementation because it can have inline or external data
 | |
| impl ParseBin for ExtentDataItem {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(Self, usize), ParseError> {
 | |
| 		let (header, header_size) = ExtentDataHeader::parse_len(bytes)?;
 | |
| 		if header.extent_type == 1 { // external extent
 | |
| 			let (body, body_size) = ExternalExtent::parse_len(&bytes[header_size..])?;
 | |
| 			return Ok((ExtentDataItem { header: header, data: ExtentDataBody::External(body)},
 | |
| 					   header_size + body_size))
 | |
| 		} else { // inline extent
 | |
| 			let data_slice = &bytes[header_size..];
 | |
| 			return Ok((ExtentDataItem {
 | |
| 				header: header,
 | |
| 				data: ExtentDataBody::Inline(Vec::from(data_slice))
 | |
| 			}, header_size + data_slice.len()))
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl ParseBin for ExtentItem {
 | |
| 	fn parse_len(bytes: &[u8]) -> Result<(Self, usize), ParseError> {
 | |
| 		let refs = u64::parse(bytes)?;
 | |
| 		let generation = u64::parse(&bytes[0x08..])?;
 | |
| 		let flags = u64::parse(&bytes[0x10..])?;
 | |
| 
 | |
| 		let mut block_refs = Vec::new();
 | |
| 
 | |
| 		if flags & 0x03 == 0x02 {
 | |
| 			for i in 0 .. refs as usize {
 | |
| 				let key_type = ItemType::parse(&bytes[0x18 + i*0x09 .. ])?;
 | |
| 				let key_id = u64::parse(&bytes[0x19 + i*0x09 .. ])?;
 | |
| 				block_refs.push((key_type, key_id));
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		Ok((ExtentItem { refs, generation, flags, block_refs }, 0x18 + refs as usize * 0x09))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /***** prettier debug output for UUIDs and checksums *****/
 | |
| 
 | |
| impl fmt::Debug for UUID {
 | |
| 	fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
 | |
| 		let UUID([x0, x1, x2,  x3,  x4,  x5,  x6,  x7,
 | |
| 				  x8, x9, x10, x11, x12, x13, x14, x15]) = self;
 | |
| 		write!(f, "{:02x}{:02x}{:02x}{:02x}-{:02x}{:02x}-{:02x}{:02x}-\
 | |
| 				   {:02x}{:02x}-{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}",
 | |
| 			   x0, x1, x2,  x3,  x4,  x5,  x6,  x7,
 | |
| 			   x8, x9, x10, x11, x12, x13, x14, x15)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| impl fmt::Debug for Checksum {
 | |
| 	fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
 | |
| 		let Checksum(
 | |
| 			[x0,  x1,  x2,  x3,  x4,  x5,  x6,  x7,
 | |
| 			 x8,  x9,  x10, x11, x12, x13, x14, x15,
 | |
| 			 x16, x17, x18, x19, x20, x21, x22, x23,
 | |
| 			 x24, x25, x26, x27, x28, x29, x30, x31]) = self;
 | |
| 		write!(f, "{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}\
 | |
| 				   {:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}\
 | |
| 				   {:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}\
 | |
| 				   {:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}",
 | |
| 			   x0,  x1,  x2,  x3,  x4,  x5,  x6,  x7,
 | |
| 			   x8,  x9,  x10, x11, x12, x13, x14, x15,
 | |
| 			   x16, x17, x18, x19, x20, x21, x22, x23,
 | |
| 			   x24, x25, x26, x27, x28, x29, x30, x31)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| macro_rules! key {
 | |
| 	($arg1:expr) => {
 | |
| 		btrfs_structs::Key { key_id: $arg1, key_type: btrfs_structs::ItemType::Invalid, key_offset: 0 }
 | |
| 	};
 | |
| 	($arg1:expr, $arg2:expr) => {
 | |
| 		btrfs_structs::Key { key_id: $arg1, key_type: $arg2, key_offset: 0 }
 | |
| 	};
 | |
| 	($arg1:expr, $arg2:expr, $arg3:expr) => {
 | |
| 		btrfs_structs::Key { key_id: $arg1, key_type: $arg2, key_offset: $arg3 }
 | |
| 	};
 | |
| }
 | |
| 
 | |
| pub(crate) use key;
 |