use crate::attribute::NtfsAttributeType;
use crate::error::{NtfsError, Result};
use crate::file::KnownNtfsFileRecordNumber;
use crate::ntfs::Ntfs;
use crate::traits::NtfsReadSeek;
use binread::io::{Read, Seek};
use core::convert::TryInto;
use core::mem;
const UPCASE_CHARACTER_COUNT: usize = 65536;
const UPCASE_TABLE_SIZE: u64 = (UPCASE_CHARACTER_COUNT * mem::size_of::<u16>()) as u64;
#[derive(Clone, Debug)]
pub(crate) struct UpcaseTable {
uppercase_characters: Vec<u16>,
}
impl UpcaseTable {
pub(crate) fn read<T>(ntfs: &Ntfs, fs: &mut T) -> Result<Self>
where
T: Read + Seek,
{
let upcase_file = ntfs.file(fs, KnownNtfsFileRecordNumber::UpCase as u64)?;
let data_item = upcase_file
.data(fs, "")
.ok_or(NtfsError::AttributeNotFound {
position: upcase_file.position(),
ty: NtfsAttributeType::Data,
})??;
let data_attribute = data_item.to_attribute();
if data_attribute.value_length() != UPCASE_TABLE_SIZE {
return Err(NtfsError::InvalidUpcaseTableSize {
expected: UPCASE_TABLE_SIZE,
actual: data_attribute.value_length(),
});
}
let mut data_value = data_attribute.value()?;
let mut data = vec![0u8; UPCASE_TABLE_SIZE as usize];
data_value.read_exact(fs, &mut data)?;
let uppercase_characters = data
.chunks_exact(2)
.map(|two_bytes| u16::from_le_bytes(two_bytes.try_into().unwrap()))
.collect();
Ok(Self {
uppercase_characters,
})
}
pub(crate) fn u16_to_uppercase(&self, character: u16) -> u16 {
self.uppercase_characters[character as usize]
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_upcase_table() {
let mut testfs1 = crate::helpers::tests::testfs1();
let ntfs = Ntfs::new(&mut testfs1).unwrap();
let upcase_table = UpcaseTable::read(&ntfs, &mut testfs1).unwrap();
for (lowercase, uppercase) in (b'a'..=b'z').zip(b'A'..=b'Z') {
assert_eq!(
upcase_table.u16_to_uppercase(lowercase as u16),
uppercase as u16
);
}
}
}