1use crate::readers::{Reader, ReaderError};
36use std::collections::HashMap;
37use std::mem::size_of;
38use std::{fmt, str};
39use thiserror::Error;
40use zerocopy::byteorder::little_endian::{U16 as LEU16, U32 as LEU32, U64 as LEU64};
41use zerocopy::{FromBytes, Immutable, KnownLayout, Ref, SplitByteSlice, Unaligned};
42
43pub const FIRST_BG_PADDING: u64 = 1024;
45pub const ROOT_INODE_NUM: u32 = 2;
47pub const SB_MAGIC: u16 = 0xEF53;
49pub const EH_MAGIC: u16 = 0xF30A;
51pub const MIN_EXT4_SIZE: u64 = FIRST_BG_PADDING + size_of::<SuperBlock>() as u64;
53pub const MINIMUM_INODE_SIZE: u64 = 128;
55
56#[derive(KnownLayout, FromBytes, Immutable, Unaligned)]
57#[repr(C)]
58pub struct ExtentHeader {
59 pub eh_magic: LEU16,
61 pub eh_ecount: LEU16,
63 pub eh_max: LEU16,
65 pub eh_depth: LEU16,
68 pub eh_gen: LEU32,
70}
71assert_eq_size!(ExtentHeader, [u8; 12]);
74
75#[derive(KnownLayout, FromBytes, Immutable, Unaligned)]
76#[repr(C)]
77pub struct ExtentIndex {
78 pub ei_blk: LEU32,
80 pub ei_leaf_lo: LEU32,
82 pub ei_leaf_hi: LEU16,
84 pub ei_unused: LEU16,
85}
86assert_eq_size!(ExtentIndex, [u8; 12]);
89
90#[derive(Clone, KnownLayout, FromBytes, Immutable, Unaligned)]
91#[repr(C)]
92pub struct Extent {
93 pub e_blk: LEU32,
95 pub e_len: LEU16,
97 pub e_start_hi: LEU16,
99 pub e_start_lo: LEU32,
101}
102assert_eq_size!(Extent, [u8; 12]);
105
106#[derive(std::fmt::Debug)]
107#[repr(C)]
108pub struct DirEntry2 {
109 pub e2d_ino: LEU32,
111 pub e2d_reclen: LEU16,
113 pub e2d_namlen: u8,
115 pub e2d_type: u8,
117 pub e2d_name: [u8; 255],
119}
120
121#[derive(KnownLayout, FromBytes, Immutable, Unaligned, std::fmt::Debug)]
122#[repr(C)]
123pub struct DirEntryHeader {
124 pub e2d_ino: LEU32,
126 pub e2d_reclen: LEU16,
128 pub e2d_namlen: u8,
130 pub e2d_type: u8,
132}
133assert_eq_size!(DirEntryHeader, [u8; 8]);
136
137#[derive(KnownLayout, FromBytes, Immutable, Unaligned)]
138#[repr(C)]
139pub struct SuperBlock {
140 pub e2fs_icount: LEU32,
142 pub e2fs_bcount: LEU32,
144 pub e2fs_rbcount: LEU32,
146 pub e2fs_fbcount: LEU32,
148 pub e2fs_ficount: LEU32,
150 pub e2fs_first_dblock: LEU32,
152 pub e2fs_log_bsize: LEU32,
154 pub e2fs_log_fsize: LEU32,
156 pub e2fs_bpg: LEU32,
158 pub e2fs_fpg: LEU32,
160 pub e2fs_ipg: LEU32,
162 pub e2fs_mtime: LEU32,
164 pub e2fs_wtime: LEU32,
166 pub e2fs_mnt_count: LEU16,
168 pub e2fs_max_mnt_count: LEU16,
170 pub e2fs_magic: LEU16,
172 pub e2fs_state: LEU16,
174 pub e2fs_beh: LEU16,
176 pub e2fs_minrev: LEU16,
178 pub e2fs_lastfsck: LEU32,
180 pub e2fs_fsckintv: LEU32,
182 pub e2fs_creator: LEU32,
184 pub e2fs_rev: LEU32,
186 pub e2fs_ruid: LEU16,
188 pub e2fs_rgid: LEU16,
190 pub e2fs_first_ino: LEU32,
192 pub e2fs_inode_size: LEU16,
194 pub e2fs_block_group_nr: LEU16,
196 pub e2fs_features_compat: LEU32,
198 pub e2fs_features_incompat: LEU32,
200 pub e2fs_features_rocompat: LEU32,
202 pub e2fs_uuid: [u8; 16],
204 pub e2fs_vname: [u8; 16],
206 pub e2fs_fsmnt: [u8; 64],
208 pub e2fs_algo: LEU32,
210 pub e2fs_prealloc: u8,
212 pub e2fs_dir_prealloc: u8,
214 pub e2fs_reserved_ngdb: LEU16,
216 pub e3fs_journal_uuid: [u8; 16],
218 pub e3fs_journal_inum: LEU32,
220 pub e3fs_journal_dev: LEU32,
222 pub e3fs_last_orphan: LEU32,
224 pub e3fs_hash_seed: [LEU32; 4],
226 pub e3fs_def_hash_version: u8,
228 pub e3fs_jnl_backup_type: u8,
230 pub e3fs_desc_size: LEU16,
232 pub e3fs_default_mount_opts: LEU32,
234 pub e3fs_first_meta_bg: LEU32,
236 pub e3fs_mkfs_time: LEU32,
238 pub e3fs_jnl_blks: [LEU32; 17],
240 pub e4fs_bcount_hi: LEU32,
242 pub e4fs_rbcount_hi: LEU32,
244 pub e4fs_fbcount_hi: LEU32,
246 pub e4fs_min_extra_isize: LEU16,
248 pub e4fs_want_extra_isize: LEU16,
250 pub e4fs_flags: LEU32,
252 pub e4fs_raid_stride: LEU16,
254 pub e4fs_mmpintv: LEU16,
256 pub e4fs_mmpblk: LEU64,
258 pub e4fs_raid_stripe_wid: LEU32,
260 pub e4fs_log_gpf: u8,
262 pub e4fs_chksum_type: u8,
264 pub e4fs_encrypt: u8,
266 pub e4fs_reserved_pad: u8,
267 pub e4fs_kbytes_written: LEU64,
269 pub e4fs_snapinum: LEU32,
271 pub e4fs_snapid: LEU32,
273 pub e4fs_snaprbcount: LEU64,
275 pub e4fs_snaplist: LEU32,
277 pub e4fs_errcount: LEU32,
279 pub e4fs_first_errtime: LEU32,
281 pub e4fs_first_errino: LEU32,
283 pub e4fs_first_errblk: LEU64,
285 pub e4fs_first_errfunc: [u8; 32],
287 pub e4fs_first_errline: LEU32,
289 pub e4fs_last_errtime: LEU32,
291 pub e4fs_last_errino: LEU32,
293 pub e4fs_last_errline: LEU32,
295 pub e4fs_last_errblk: LEU64,
297 pub e4fs_last_errfunc: [u8; 32],
299 pub e4fs_mount_opts: [u8; 64],
301 pub e4fs_usrquota_inum: LEU32,
303 pub e4fs_grpquota_inum: LEU32,
305 pub e4fs_overhead_clusters: LEU32,
307 pub e4fs_backup_bgs: [LEU32; 2],
309 pub e4fs_encrypt_algos: [u8; 4],
311 pub e4fs_encrypt_pw_salt: [u8; 16],
313 pub e4fs_lpf_ino: LEU32,
315 pub e4fs_proj_quota_inum: LEU32,
317 pub e4fs_chksum_seed: LEU32,
319 pub e4fs_reserved: [LEU32; 98],
321 pub e4fs_sbchksum: LEU32,
323}
324assert_eq_size!(SuperBlock, [u8; 1024]);
327
328#[derive(KnownLayout, FromBytes, Immutable, Unaligned)]
329#[repr(C)]
330pub struct BlockGroupDesc32 {
331 pub ext2bgd_b_bitmap: LEU32,
333 pub ext2bgd_i_bitmap: LEU32,
335 pub ext2bgd_i_tables: LEU32,
337 pub ext2bgd_nbfree: LEU16,
339 pub ext2bgd_nifree: LEU16,
341 pub ext2bgd_ndirs: LEU16,
343 pub ext4bgd_flags: LEU16,
345 pub ext4bgd_x_bitmap: LEU32,
347 pub ext4bgd_b_bmap_csum: LEU16,
349 pub ext4bgd_i_bmap_csum: LEU16,
351 pub ext4bgd_i_unused: LEU16,
353 pub ext4bgd_csum: LEU16,
355}
356assert_eq_size!(BlockGroupDesc32, [u8; 32]);
359
360#[derive(KnownLayout, FromBytes, Immutable, Unaligned)]
361#[repr(C)]
362pub struct BlockGroupDesc64 {
363 pub base: BlockGroupDesc32,
364 pub ext4bgd_b_bitmap_hi: LEU32,
365 pub ext4bgd_i_bitmap_hi: LEU32,
366 pub ext4bgd_i_tables_hi: LEU32,
367 pub ext4bgd_nbfree_hi: LEU16,
368 pub ext4bgd_nifree_hi: LEU16,
369 pub ext4bgd_ndirs_hi: LEU16,
370 pub ext4bgd_i_unused_hi: LEU16,
371 pub ext4bgd_x_bitmap_hi: LEU32,
372 pub ext4bgd_b_bmap_csum_hi: LEU16,
373 pub ext4bgd_i_bmap_csum_hi: LEU16,
374 pub ext4bgd_reserved: LEU32,
375}
376assert_eq_size!(BlockGroupDesc64, [u8; 64]);
379
380#[derive(KnownLayout, FromBytes, Immutable)]
381#[repr(C)]
382pub struct ExtentTreeNode<B: SplitByteSlice> {
383 pub header: Ref<B, ExtentHeader>,
384 pub entries: B,
385}
386
387#[derive(KnownLayout, FromBytes, Immutable, Unaligned)]
388#[repr(C)]
389pub struct INode {
390 pub e2di_mode: LEU16,
392 pub e2di_uid: LEU16,
394 pub e2di_size: LEU32,
396 pub e2di_atime: LEU32,
398 pub e2di_ctime: LEU32,
400 pub e2di_mtime: LEU32,
402 pub e2di_dtime: LEU32,
404 pub e2di_gid: LEU16,
406 pub e2di_nlink: LEU16,
408 pub e2di_nblock: LEU32,
410 pub e2di_flags: LEU32,
412 pub e2di_version: [u8; 4],
414 pub e2di_blocks: [u8; 60],
416 pub e2di_gen: LEU32,
418 pub e2di_facl: LEU32,
420 pub e2di_size_high: LEU32,
422 pub e2di_faddr: LEU32,
424 pub e2di_nblock_high: LEU16,
426 pub e2di_facl_high: LEU16,
428 pub e2di_uid_high: LEU16,
430 pub e2di_gid_high: LEU16,
432 pub e2di_chksum_lo: LEU16,
434 pub e2di_lx_reserved: LEU16,
435 e4di_extra_isize: LEU16,
439 e4di_chksum_hi: LEU16,
440 e4di_ctime_extra: LEU32,
441 e4di_mtime_extra: LEU32,
442 e4di_atime_extra: LEU32,
443 e4di_crtime: LEU32,
444 e4di_crtime_extra: LEU32,
445 e4di_version_hi: LEU32,
446 e4di_projid: LEU32,
447}
448assert_eq_size!(INode, [u8; 160]);
451
452#[derive(KnownLayout, FromBytes, Immutable, Unaligned, Debug)]
453#[repr(C)]
454pub struct XattrHeader {
455 pub e_magic: LEU32,
456 pub e_refcount: LEU32,
457 pub e_blocks: LEU32,
458 pub e_hash: LEU32,
459 pub e_checksum: LEU32,
460 e_reserved: [u8; 8],
461}
462
463#[derive(KnownLayout, FromBytes, Immutable, Unaligned, Debug)]
464#[repr(C)]
465pub struct XattrEntryHeader {
466 pub e_name_len: u8,
467 pub e_name_index: u8,
468 pub e_value_offs: LEU16,
469 pub e_value_inum: LEU32,
470 pub e_value_size: LEU32,
471 pub e_hash: LEU32,
472}
473
474#[derive(Debug, PartialEq)]
475pub enum InvalidAddressErrorType {
476 Lower,
477 Upper,
478}
479
480impl fmt::Display for InvalidAddressErrorType {
481 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
482 match *self {
483 InvalidAddressErrorType::Lower => write!(f, "lower"),
484 InvalidAddressErrorType::Upper => write!(f, "upper"),
485 }
486 }
487}
488
489#[derive(Error, Debug, PartialEq)]
490pub enum ParsingError {
491 #[error("Unable to parse Super Block at 0x{:X}", _0)]
492 InvalidSuperBlock(u64),
493 #[error("Invalid Super Block magic number {} should be 0xEF53", _0)]
494 InvalidSuperBlockMagic(u16),
495 #[error("Invalid Super Block inode size {} should be {}", _0, std::mem::size_of::<INode>())]
496 InvalidInodeSize(u16),
497 #[error("Invalid Block Group Descriptor size {}", _0)]
498 InvalidBlockGroupDescSize(u16),
499 #[error("Block number {} out of bounds.", _0)]
500 BlockNumberOutOfBounds(u64),
501 #[error("SuperBlock e2fs_log_bsize value invalid: {}", _0)]
502 BlockSizeInvalid(u32),
503
504 #[error("Unable to parse Block Group Description at 0x{:X}", _0)]
505 InvalidBlockGroupDesc(u64),
506 #[error("Unable to parse INode {}", _0)]
507 InvalidInode(u32),
508
509 #[error("Unable to parse ExtentHeader from INode")]
511 InvalidExtentHeader,
512 #[error("Invalid Extent Header magic number {} should be 0xF30A", _0)]
513 InvalidExtentHeaderMagic(u16),
514 #[error("Unable to parse Extent at 0x{:X}", _0)]
515 InvalidExtent(u64),
516 #[error("Extent has more data {} than expected {}", _0, _1)]
517 ExtentUnexpectedLength(u64, u64),
518
519 #[error("Invalid Directory Entry at 0x{:X}", _0)]
520 InvalidDirEntry2(u64),
521 #[error("Directory Entry has invalid string in name field: {:?}", _0)]
522 DirEntry2NonUtf8(Vec<u8>),
523 #[error("Requested path contains invalid string")]
524 InvalidInputPath,
525 #[error("Non-existent path: {}", _0)]
526 PathNotFound(String),
527 #[error("Entry Type {} unknown", _0)]
528 BadEntryType(u8),
529
530 #[error("Incompatible feature flags (feature_incompat): 0x{:X}", _0)]
532 BannedFeatureIncompat(u32),
533 #[error("Required feature flags (feature_incompat): 0x{:X}", _0)]
534 RequiredFeatureIncompat(u32),
535
536 #[error("{}", _0)]
538 Incompatible(String),
539 #[error("Bad file at {}", _0)]
540 BadFile(String),
541 #[error("Bad directory at {}", _0)]
542 BadDirectory(String),
543
544 #[error("Attempted to access at 0x{:X} when the {} bound is 0x{:X}", _1, _0, _2)]
545 InvalidAddress(InvalidAddressErrorType, u64, u64),
546
547 #[error("Reader failed to read at 0x{:X}", _0)]
548 SourceReadError(u64),
549
550 #[error("Not a file")]
551 NotFile,
552}
553
554impl From<ReaderError> for ParsingError {
555 fn from(err: ReaderError) -> ParsingError {
556 match err {
557 ReaderError::Read(addr) => ParsingError::SourceReadError(addr),
558 ReaderError::OutOfBounds(addr, max) => {
559 ParsingError::InvalidAddress(InvalidAddressErrorType::Upper, addr, max)
560 }
561 }
562 }
563}
564
565#[derive(Debug, PartialEq)]
567#[repr(u8)]
568pub enum EntryType {
569 Unknown = 0x0,
570 RegularFile = 0x1,
571 Directory = 0x2,
572 CharacterDevice = 0x3,
573 BlockDevice = 0x4,
574 FIFO = 0x5,
575 Socket = 0x6,
576 SymLink = 0x7,
577}
578
579impl EntryType {
580 pub fn from_u8(value: u8) -> Result<EntryType, ParsingError> {
581 match value {
582 0x0 => Ok(EntryType::Unknown),
583 0x1 => Ok(EntryType::RegularFile),
584 0x2 => Ok(EntryType::Directory),
585 0x3 => Ok(EntryType::CharacterDevice),
586 0x4 => Ok(EntryType::BlockDevice),
587 0x5 => Ok(EntryType::FIFO),
588 0x6 => Ok(EntryType::Socket),
589 0x7 => Ok(EntryType::SymLink),
590 _ => Err(ParsingError::BadEntryType(value)),
591 }
592 }
593}
594
595#[derive(PartialEq)]
605#[repr(u32)]
606pub enum FeatureIncompat {
607 Compression = 0x1,
608 EntryHasFileType = 0x2,
610
611 HasJournal = 0x4,
618 JournalSeparate = 0x8,
619 MetaBlockGroups = 0x10,
620 Extents = 0x40,
623 Is64Bit = 0x80,
624 MultiMountProtection = 0x100,
625
626 FlexibleBlockGroups = 0x200,
629 ExtendedAttributeINodes = 0x400,
630 ExtendedDirectoryEntry = 0x1000,
631 MetadataChecksum = 0x2000,
633 LargeDirectory = 0x4000,
634 SmallFilesInINode = 0x8000,
635 EncryptedINodes = 0x10000,
636}
637
638pub const REQUIRED_FEATURE_INCOMPAT: u32 =
640 FeatureIncompat::Extents as u32 | FeatureIncompat::EntryHasFileType as u32;
641
642pub const BANNED_FEATURE_INCOMPAT: u32 = FeatureIncompat::Compression as u32
644 | FeatureIncompat::MultiMountProtection as u32
645 | FeatureIncompat::ExtendedAttributeINodes as u32
646 | FeatureIncompat::ExtendedDirectoryEntry as u32
647 | FeatureIncompat::SmallFilesInINode as u32
648 | FeatureIncompat::EncryptedINodes as u32;
649
650pub trait ParseToStruct: FromBytes + KnownLayout + Immutable + Unaligned + Sized {
653 fn from_reader_with_offset(reader: &dyn Reader, offset: u64) -> Result<Self, ParsingError> {
654 if offset < FIRST_BG_PADDING {
655 return Err(ParsingError::InvalidAddress(
656 InvalidAddressErrorType::Lower,
657 offset,
658 FIRST_BG_PADDING,
659 ));
660 }
661 let mut object = Self::new_zeroed();
662 let buffer = unsafe {
667 std::slice::from_raw_parts_mut(&mut object as *mut Self as *mut u8, size_of::<Self>())
668 };
669 reader.read(offset, buffer)?;
670 Ok(object)
671 }
672
673 fn to_struct_ref(data: &[u8], error_type: ParsingError) -> Result<&Self, ParsingError> {
677 Ref::<&[u8], Self>::from_bytes(data).map(|res| Ref::into_ref(res)).map_err(|_| error_type)
678 }
679}
680
681impl<T: FromBytes + KnownLayout + Immutable + Unaligned> ParseToStruct for T {}
683
684impl<B: SplitByteSlice> ExtentTreeNode<B> {
685 pub fn parse(data: B) -> Option<Self> {
690 Ref::<B, ExtentHeader>::from_prefix(data)
691 .ok()
692 .map(|(header, entries)| Self { header, entries })
693 }
694}
695
696impl SuperBlock {
697 pub fn parse(reader: &dyn Reader) -> Result<SuperBlock, ParsingError> {
699 let sb = SuperBlock::from_reader_with_offset(reader, FIRST_BG_PADDING)?;
703 sb.check_magic()?;
704 sb.feature_check()?;
705 sb.check_inode_size()?;
706 sb.check_block_group_descriptor_size()?;
707 Ok(sb)
708 }
709
710 pub fn check_magic(&self) -> Result<(), ParsingError> {
711 if self.e2fs_magic.get() == SB_MAGIC {
712 Ok(())
713 } else {
714 Err(ParsingError::InvalidSuperBlockMagic(self.e2fs_magic.get()))
715 }
716 }
717
718 pub fn check_inode_size(&self) -> Result<(), ParsingError> {
719 let inode_size: u64 = self.e2fs_inode_size.into();
720 if inode_size < MINIMUM_INODE_SIZE {
721 Err(ParsingError::InvalidInodeSize(self.e2fs_inode_size.get()))
722 } else {
723 Ok(())
724 }
725 }
726
727 fn check_block_group_descriptor_size(&self) -> Result<(), ParsingError> {
728 let desc_size: usize = self.e3fs_desc_size.into();
729 let is_64bit = self.is_64bit();
730 if desc_size == 0
732 || is_64bit && desc_size == size_of::<BlockGroupDesc64>()
733 || !is_64bit && desc_size == size_of::<BlockGroupDesc32>()
734 {
735 Ok(())
736 } else {
737 Err(ParsingError::InvalidBlockGroupDescSize(self.e3fs_desc_size.get()))
738 }
739 }
740
741 pub fn block_size(&self) -> Result<u64, ParsingError> {
746 let bs = 2u64
747 .checked_pow(self.e2fs_log_bsize.get() + 10)
748 .ok_or_else(|| ParsingError::BlockSizeInvalid(self.e2fs_log_bsize.get()))?;
749 if bs == 1024 || bs == 2048 || bs == 4096 || bs == 65536 {
750 Ok(bs)
751 } else {
752 Err(ParsingError::BlockSizeInvalid(self.e2fs_log_bsize.get()))
753 }
754 }
755
756 pub fn block_group_descriptor_size(&self) -> usize {
758 if self.is_64bit() { size_of::<BlockGroupDesc64>() } else { size_of::<BlockGroupDesc32>() }
761 }
762
763 pub fn is_64bit(&self) -> bool {
765 self.e2fs_features_incompat.get() & FeatureIncompat::Is64Bit as u32 != 0
766 }
767
768 fn feature_check(&self) -> Result<(), ParsingError> {
769 let banned = self.e2fs_features_incompat.get() & BANNED_FEATURE_INCOMPAT;
770 if banned > 0 {
771 return Err(ParsingError::BannedFeatureIncompat(banned));
772 }
773 let required = self.e2fs_features_incompat.get() & REQUIRED_FEATURE_INCOMPAT;
774 if required != REQUIRED_FEATURE_INCOMPAT {
775 return Err(ParsingError::RequiredFeatureIncompat(
776 required ^ REQUIRED_FEATURE_INCOMPAT,
777 ));
778 }
779 Ok(())
780 }
781}
782
783impl INode {
784 pub fn extent_tree_node(&self) -> Result<ExtentTreeNode<&[u8]>, ParsingError> {
787 let eh = ExtentTreeNode::<&[u8]>::parse(
788 &self.e2di_blocks,
791 )
792 .ok_or(ParsingError::InvalidExtentHeader)?;
793 eh.header.check_magic()?;
794 Ok(eh)
795 }
796
797 pub fn size(&self) -> u64 {
799 (self.e2di_size_high.get() as u64) << 32 | self.e2di_size.get() as u64
800 }
801
802 pub fn facl(&self) -> u64 {
803 (self.e2di_facl_high.get() as u64) << 32 | self.e2di_facl.get() as u64
804 }
805
806 fn check_inode_size(sb: &SuperBlock) -> Result<(), ParsingError> {
807 let inode_size: usize = sb.e2fs_inode_size.into();
808 if inode_size < std::mem::size_of::<INode>() {
809 Err(ParsingError::Incompatible("Inode size too small.".to_string()))
810 } else {
811 Ok(())
812 }
813 }
814
815 pub fn e4di_extra_isize(&self, sb: &SuperBlock) -> Result<LEU16, ParsingError> {
816 INode::check_inode_size(sb)?;
817 Ok(self.e4di_extra_isize)
818 }
819
820 pub fn e4di_chksum_hi(&self, sb: &SuperBlock) -> Result<LEU16, ParsingError> {
821 INode::check_inode_size(sb)?;
822 Ok(self.e4di_chksum_hi)
823 }
824
825 pub fn e4di_ctime_extra(&self, sb: &SuperBlock) -> Result<LEU32, ParsingError> {
826 INode::check_inode_size(sb)?;
827 Ok(self.e4di_ctime_extra)
828 }
829
830 pub fn e4di_mtime_extra(&self, sb: &SuperBlock) -> Result<LEU32, ParsingError> {
831 INode::check_inode_size(sb)?;
832 Ok(self.e4di_mtime_extra)
833 }
834
835 pub fn e4di_atime_extra(&self, sb: &SuperBlock) -> Result<LEU32, ParsingError> {
836 INode::check_inode_size(sb)?;
837 Ok(self.e4di_atime_extra)
838 }
839
840 pub fn e4di_crtime(&self, sb: &SuperBlock) -> Result<LEU32, ParsingError> {
841 INode::check_inode_size(sb)?;
842 Ok(self.e4di_crtime)
843 }
844
845 pub fn e4di_crtime_extra(&self, sb: &SuperBlock) -> Result<LEU32, ParsingError> {
846 INode::check_inode_size(sb)?;
847 Ok(self.e4di_crtime_extra)
848 }
849
850 pub fn e4di_version_hi(&self, sb: &SuperBlock) -> Result<LEU32, ParsingError> {
851 INode::check_inode_size(sb)?;
852 Ok(self.e4di_version_hi)
853 }
854
855 pub fn e4di_projid(&self, sb: &SuperBlock) -> Result<LEU32, ParsingError> {
856 INode::check_inode_size(sb)?;
857 Ok(self.e4di_projid)
858 }
859}
860
861impl ExtentHeader {
862 pub fn check_magic(&self) -> Result<(), ParsingError> {
863 if self.eh_magic.get() == EH_MAGIC {
864 Ok(())
865 } else {
866 Err(ParsingError::InvalidExtentHeaderMagic(self.eh_magic.get()))
867 }
868 }
869}
870
871impl DirEntry2 {
872 pub fn name(&self) -> Result<&str, ParsingError> {
874 str::from_utf8(&self.e2d_name[0..self.e2d_namlen as usize]).map_err(|_| {
875 ParsingError::DirEntry2NonUtf8(self.e2d_name[0..self.e2d_namlen as usize].to_vec())
876 })
877 }
878
879 pub fn name_bytes(&self) -> &[u8] {
880 &self.e2d_name[0..self.e2d_namlen as usize]
881 }
882
883 pub fn as_hash_map(
888 entries: Vec<DirEntry2>,
889 ) -> Result<HashMap<String, DirEntry2>, ParsingError> {
890 let mut entry_map: HashMap<String, DirEntry2> = HashMap::with_capacity(entries.len());
891
892 for entry in entries {
893 entry_map.insert(entry.name()?.to_string(), entry);
894 }
895 Ok(entry_map)
896 }
897}
898
899impl Extent {
900 pub fn target_block_num(&self) -> u64 {
902 (self.e_start_hi.get() as u64) << 32 | self.e_start_lo.get() as u64
903 }
904}
905
906impl ExtentIndex {
907 pub fn target_block_num(&self) -> u64 {
909 (self.ei_leaf_hi.get() as u64) << 32 | self.ei_leaf_lo.get() as u64
910 }
911}
912
913#[cfg(test)]
914mod test {
915 use super::{
916 EH_MAGIC, Extent, ExtentHeader, ExtentIndex, FIRST_BG_PADDING, FeatureIncompat, LEU16,
917 LEU32, LEU64, ParseToStruct, REQUIRED_FEATURE_INCOMPAT, SB_MAGIC, SuperBlock,
918 };
919 use crate::readers::VecReader;
920 use std::fs;
921
922 impl Default for SuperBlock {
923 fn default() -> SuperBlock {
924 SuperBlock {
925 e2fs_icount: LEU32::new(0),
926 e2fs_bcount: LEU32::new(0),
927 e2fs_rbcount: LEU32::new(0),
928 e2fs_fbcount: LEU32::new(0),
929 e2fs_ficount: LEU32::new(0),
930 e2fs_first_dblock: LEU32::new(0),
931 e2fs_log_bsize: LEU32::new(0),
932 e2fs_log_fsize: LEU32::new(0),
933 e2fs_bpg: LEU32::new(0),
934 e2fs_fpg: LEU32::new(0),
935 e2fs_ipg: LEU32::new(0),
936 e2fs_mtime: LEU32::new(0),
937 e2fs_wtime: LEU32::new(0),
938 e2fs_mnt_count: LEU16::new(0),
939 e2fs_max_mnt_count: LEU16::new(0),
940 e2fs_magic: LEU16::new(0),
941 e2fs_state: LEU16::new(0),
942 e2fs_beh: LEU16::new(0),
943 e2fs_minrev: LEU16::new(0),
944 e2fs_lastfsck: LEU32::new(0),
945 e2fs_fsckintv: LEU32::new(0),
946 e2fs_creator: LEU32::new(0),
947 e2fs_rev: LEU32::new(0),
948 e2fs_ruid: LEU16::new(0),
949 e2fs_rgid: LEU16::new(0),
950 e2fs_first_ino: LEU32::new(0),
951 e2fs_inode_size: LEU16::new(0),
952 e2fs_block_group_nr: LEU16::new(0),
953 e2fs_features_compat: LEU32::new(0),
954 e2fs_features_incompat: LEU32::new(0),
955 e2fs_features_rocompat: LEU32::new(0),
956 e2fs_uuid: [0; 16],
957 e2fs_vname: [0; 16],
958 e2fs_fsmnt: [0; 64],
959 e2fs_algo: LEU32::new(0),
960 e2fs_prealloc: 0,
961 e2fs_dir_prealloc: 0,
962 e2fs_reserved_ngdb: LEU16::new(0),
963 e3fs_journal_uuid: [0; 16],
964 e3fs_journal_inum: LEU32::new(0),
965 e3fs_journal_dev: LEU32::new(0),
966 e3fs_last_orphan: LEU32::new(0),
967 e3fs_hash_seed: [LEU32::new(0); 4],
968 e3fs_def_hash_version: 0,
969 e3fs_jnl_backup_type: 0,
970 e3fs_desc_size: LEU16::new(0),
971 e3fs_default_mount_opts: LEU32::new(0),
972 e3fs_first_meta_bg: LEU32::new(0),
973 e3fs_mkfs_time: LEU32::new(0),
974 e3fs_jnl_blks: [LEU32::new(0); 17],
975 e4fs_bcount_hi: LEU32::new(0),
976 e4fs_rbcount_hi: LEU32::new(0),
977 e4fs_fbcount_hi: LEU32::new(0),
978 e4fs_min_extra_isize: LEU16::new(0),
979 e4fs_want_extra_isize: LEU16::new(0),
980 e4fs_flags: LEU32::new(0),
981 e4fs_raid_stride: LEU16::new(0),
982 e4fs_mmpintv: LEU16::new(0),
983 e4fs_mmpblk: LEU64::new(0),
984 e4fs_raid_stripe_wid: LEU32::new(0),
985 e4fs_log_gpf: 0,
986 e4fs_chksum_type: 0,
987 e4fs_encrypt: 0,
988 e4fs_reserved_pad: 0,
989 e4fs_kbytes_written: LEU64::new(0),
990 e4fs_snapinum: LEU32::new(0),
991 e4fs_snapid: LEU32::new(0),
992 e4fs_snaprbcount: LEU64::new(0),
993 e4fs_snaplist: LEU32::new(0),
994 e4fs_errcount: LEU32::new(0),
995 e4fs_first_errtime: LEU32::new(0),
996 e4fs_first_errino: LEU32::new(0),
997 e4fs_first_errblk: LEU64::new(0),
998 e4fs_first_errfunc: [0; 32],
999 e4fs_first_errline: LEU32::new(0),
1000 e4fs_last_errtime: LEU32::new(0),
1001 e4fs_last_errino: LEU32::new(0),
1002 e4fs_last_errline: LEU32::new(0),
1003 e4fs_last_errblk: LEU64::new(0),
1004 e4fs_last_errfunc: [0; 32],
1005 e4fs_mount_opts: [0; 64],
1006 e4fs_usrquota_inum: LEU32::new(0),
1007 e4fs_grpquota_inum: LEU32::new(0),
1008 e4fs_overhead_clusters: LEU32::new(0),
1009 e4fs_backup_bgs: [LEU32::new(0); 2],
1010 e4fs_encrypt_algos: [0; 4],
1011 e4fs_encrypt_pw_salt: [0; 16],
1012 e4fs_lpf_ino: LEU32::new(0),
1013 e4fs_proj_quota_inum: LEU32::new(0),
1014 e4fs_chksum_seed: LEU32::new(0),
1015 e4fs_reserved: [LEU32::new(0); 98],
1016 e4fs_sbchksum: LEU32::new(0),
1017 }
1018 }
1019 }
1020
1021 #[fuchsia::test]
1030 fn parse_superblock() {
1031 let data = fs::read("/pkg/data/1file.img").expect("Unable to read file");
1032 let reader = VecReader::new(data);
1033 let sb = SuperBlock::parse(&reader).expect("Parsed Super Block");
1034 assert_eq!(sb.block_size().unwrap(), FIRST_BG_PADDING);
1036
1037 assert!(sb.check_magic().is_ok());
1039
1040 let mut sb = SuperBlock::default();
1041 assert!(sb.check_magic().is_err());
1042
1043 sb.e2fs_magic = LEU16::new(SB_MAGIC);
1044 assert!(sb.check_magic().is_ok());
1045
1046 sb.e2fs_log_bsize = LEU32::new(0); assert!(sb.block_size().is_ok());
1049 sb.e2fs_log_bsize = LEU32::new(1); assert!(sb.block_size().is_ok());
1051 sb.e2fs_log_bsize = LEU32::new(2); assert!(sb.block_size().is_ok());
1053 sb.e2fs_log_bsize = LEU32::new(6); assert!(sb.block_size().is_ok());
1055
1056 sb.e2fs_log_bsize = LEU32::new(3);
1058 assert!(sb.block_size().is_err());
1059 sb.e2fs_log_bsize = LEU32::new(5);
1060 assert!(sb.block_size().is_err());
1061 sb.e2fs_log_bsize = LEU32::new(7);
1062 assert!(sb.block_size().is_err());
1063 sb.e2fs_log_bsize = LEU32::new(20);
1065 assert!(sb.block_size().is_err());
1066 }
1067
1068 #[fuchsia::test]
1070 fn parse_to_struct_from_reader_with_offset() {
1071 let data = fs::read("/pkg/data/1file.img").expect("Unable to read file");
1072 let reader = VecReader::new(data);
1073 let sb = SuperBlock::from_reader_with_offset(&reader, FIRST_BG_PADDING)
1074 .expect("Parsed Super Block");
1075 assert!(sb.check_magic().is_ok());
1076 }
1077
1078 #[fuchsia::test]
1080 fn incompatible_feature_flags() {
1081 let data = fs::read("/pkg/data/1file.img").expect("Unable to read file");
1082 let reader = VecReader::new(data);
1083 let sb = SuperBlock::parse(&reader).expect("Parsed Super Block");
1084 assert_eq!(sb.e2fs_magic.get(), SB_MAGIC);
1085 assert!(sb.feature_check().is_ok());
1086
1087 let mut sb = SuperBlock::default();
1088 match sb.feature_check() {
1089 Ok(_) => assert!(false, "Feature flags should be incorrect."),
1090 Err(e) => assert_eq!(
1091 format!("{}", e),
1092 format!(
1093 "Required feature flags (feature_incompat): 0x{:X}",
1094 REQUIRED_FEATURE_INCOMPAT
1095 )
1096 ),
1097 }
1098
1099 sb.e2fs_features_incompat = LEU32::new(REQUIRED_FEATURE_INCOMPAT | 0xF00000);
1101 assert!(sb.feature_check().is_ok());
1102
1103 sb.e2fs_features_incompat = LEU32::new(FeatureIncompat::Extents as u32);
1105 match sb.feature_check() {
1106 Ok(_) => assert!(false, "Feature flags should be incorrect."),
1107 Err(e) => assert_eq!(
1108 format!("{}", e),
1109 format!(
1110 "Required feature flags (feature_incompat): 0x{:X}",
1111 REQUIRED_FEATURE_INCOMPAT ^ FeatureIncompat::Extents as u32
1112 )
1113 ),
1114 }
1115
1116 sb.e2fs_features_incompat = LEU32::new(FeatureIncompat::Compression as u32);
1118 match sb.feature_check() {
1119 Ok(_) => assert!(false, "Feature flags should be incorrect."),
1120 Err(e) => assert_eq!(
1121 format!("{}", e),
1122 format!(
1123 "Incompatible feature flags (feature_incompat): 0x{:X}",
1124 FeatureIncompat::Compression as u32
1125 )
1126 ),
1127 }
1128 }
1129
1130 #[fuchsia::test]
1132 fn extent_target_block_num() {
1133 let e = Extent {
1134 e_blk: LEU32::new(0),
1135 e_len: LEU16::new(0),
1136 e_start_hi: LEU16::new(0x4444),
1137 e_start_lo: LEU32::new(0x6666_8888),
1138 };
1139 assert_eq!(e.target_block_num(), 0x4444_6666_8888);
1140 }
1141
1142 #[fuchsia::test]
1144 fn extent_index_target_block_num() {
1145 let e = ExtentIndex {
1146 ei_blk: LEU32::new(0),
1147 ei_leaf_lo: LEU32::new(0x6666_8888),
1148 ei_leaf_hi: LEU16::new(0x4444),
1149 ei_unused: LEU16::new(0),
1150 };
1151 assert_eq!(e.target_block_num(), 0x4444_6666_8888);
1152 }
1153
1154 #[fuchsia::test]
1156 fn extent_header_check_magic() {
1157 let e = ExtentHeader {
1158 eh_magic: LEU16::new(EH_MAGIC),
1159 eh_ecount: LEU16::new(0),
1160 eh_max: LEU16::new(0),
1161 eh_depth: LEU16::new(0),
1162 eh_gen: LEU32::new(0),
1163 };
1164 assert!(e.check_magic().is_ok());
1165
1166 let e = ExtentHeader {
1167 eh_magic: LEU16::new(0x1234),
1168 eh_ecount: LEU16::new(0),
1169 eh_max: LEU16::new(0),
1170 eh_depth: LEU16::new(0),
1171 eh_gen: LEU32::new(0),
1172 };
1173 assert!(e.check_magic().is_err());
1174 }
1175}