ahcidisk.rs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581
  1. use super::{_port, hba::HbaCmdTable, virt_2_phys};
  2. use crate::driver::base::block::block_device::{BlockDevice, BlockId};
  3. use crate::driver::base::block::disk_info::Partition;
  4. use crate::driver::base::block::SeekFrom;
  5. use crate::driver::base::device::bus::Bus;
  6. use crate::driver::base::device::driver::Driver;
  7. use crate::driver::base::device::{Device, DeviceType, IdTable};
  8. use crate::driver::base::kobject::{KObjType, KObject, KObjectState};
  9. use crate::driver::base::kset::KSet;
  10. use crate::driver::disk::ahci::HBA_PxIS_TFES;
  11. use crate::filesystem::kernfs::KernFSInode;
  12. use crate::filesystem::mbr::MbrDiskPartionTable;
  13. use crate::kdebug;
  14. use crate::libs::rwlock::{RwLockReadGuard, RwLockWriteGuard};
  15. use crate::libs::{spinlock::SpinLock, vec_cursor::VecCursor};
  16. use crate::mm::{phys_2_virt, verify_area, VirtAddr};
  17. use crate::syscall::SystemError;
  18. use crate::{
  19. driver::disk::ahci::hba::{
  20. FisRegH2D, FisType, HbaCmdHeader, ATA_CMD_READ_DMA_EXT, ATA_CMD_WRITE_DMA_EXT,
  21. ATA_DEV_BUSY, ATA_DEV_DRQ,
  22. },
  23. kerror,
  24. };
  25. use alloc::sync::Weak;
  26. use alloc::{string::String, sync::Arc, vec::Vec};
  27. use core::fmt::Debug;
  28. use core::sync::atomic::{compiler_fence, Ordering};
  29. use core::{mem::size_of, ptr::write_bytes};
  30. /// @brief: 只支持MBR分区格式的磁盘结构体
  31. pub struct AhciDisk {
  32. pub name: String,
  33. pub flags: u16, // 磁盘的状态flags
  34. pub partitions: Vec<Arc<Partition>>, // 磁盘分区数组
  35. // port: &'static mut HbaPort, // 控制硬盘的端口
  36. pub ctrl_num: u8,
  37. pub port_num: u8,
  38. /// 指向LockAhciDisk的弱引用
  39. self_ref: Weak<LockedAhciDisk>,
  40. }
  41. /// @brief: 带锁的AhciDisk
  42. #[derive(Debug)]
  43. pub struct LockedAhciDisk(pub SpinLock<AhciDisk>);
  44. /// 函数实现
  45. impl Debug for AhciDisk {
  46. fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
  47. write!(
  48. f,
  49. "{{ name: {}, flags: {}, part_s: {:?} }}",
  50. self.name, self.flags, self.partitions
  51. )?;
  52. return Ok(());
  53. }
  54. }
  55. impl AhciDisk {
  56. fn read_at(
  57. &self,
  58. lba_id_start: BlockId, // 起始lba编号
  59. count: usize, // 读取lba的数量
  60. buf: &mut [u8],
  61. ) -> Result<usize, SystemError> {
  62. assert!((buf.len() & 511) == 0);
  63. compiler_fence(core::sync::atomic::Ordering::SeqCst);
  64. let check_length = ((count - 1) >> 4) + 1; // prdt length
  65. if count * 512 > buf.len() || check_length > 8 as usize {
  66. kerror!("ahci read: e2big");
  67. // 不可能的操作
  68. return Err(SystemError::E2BIG);
  69. } else if count == 0 {
  70. return Ok(0);
  71. }
  72. let port = _port(self.ctrl_num, self.port_num);
  73. volatile_write!(port.is, u32::MAX); // Clear pending interrupt bits
  74. let slot = port.find_cmdslot().unwrap_or(u32::MAX);
  75. if slot == u32::MAX {
  76. return Err(SystemError::EIO);
  77. }
  78. #[allow(unused_unsafe)]
  79. let cmdheader: &mut HbaCmdHeader = unsafe {
  80. (phys_2_virt(
  81. volatile_read!(port.clb) as usize
  82. + slot as usize * size_of::<HbaCmdHeader>() as usize,
  83. ) as *mut HbaCmdHeader)
  84. .as_mut()
  85. .unwrap()
  86. };
  87. cmdheader.cfl = (size_of::<FisRegH2D>() / size_of::<u32>()) as u8;
  88. volatile_set_bit!(cmdheader.cfl, 1 << 6, false); // Read/Write bit : Read from device
  89. volatile_write!(cmdheader.prdtl, check_length as u16); // PRDT entries count
  90. // 设置数据存放地址
  91. let mut buf_ptr = buf as *mut [u8] as *mut usize as usize;
  92. // 由于目前的内存管理机制无法把用户空间的内存地址转换为物理地址,所以只能先把数据拷贝到内核空间
  93. // TODO:在内存管理重构后,可以直接使用用户空间的内存地址
  94. let user_buf = verify_area(VirtAddr::new(buf_ptr as usize), buf.len()).is_ok();
  95. let mut kbuf = if user_buf {
  96. let mut x: Vec<u8> = Vec::new();
  97. x.resize(buf.len(), 0);
  98. Some(x)
  99. } else {
  100. None
  101. };
  102. if kbuf.is_some() {
  103. buf_ptr = kbuf.as_mut().unwrap().as_mut_ptr() as usize;
  104. }
  105. #[allow(unused_unsafe)]
  106. let cmdtbl = unsafe {
  107. (phys_2_virt(volatile_read!(cmdheader.ctba) as usize) as *mut HbaCmdTable)
  108. .as_mut()
  109. .unwrap() // 必须使用 as_mut ,得到的才是原来的变量
  110. };
  111. let mut tmp_count = count;
  112. unsafe {
  113. // 清空整个table的旧数据
  114. write_bytes(cmdtbl, 0, 1);
  115. }
  116. // kdebug!("cmdheader.prdtl={}", volatile_read!(cmdheader.prdtl));
  117. // 8K bytes (16 sectors) per PRDT
  118. for i in 0..((volatile_read!(cmdheader.prdtl) - 1) as usize) {
  119. volatile_write!(cmdtbl.prdt_entry[i].dba, virt_2_phys(buf_ptr) as u64);
  120. cmdtbl.prdt_entry[i].dbc = 8 * 1024 - 1;
  121. volatile_set_bit!(cmdtbl.prdt_entry[i].dbc, 1 << 31, true); // 允许中断 prdt_entry.i
  122. buf_ptr += 8 * 1024;
  123. tmp_count -= 16;
  124. }
  125. // Last entry
  126. let las = (volatile_read!(cmdheader.prdtl) - 1) as usize;
  127. volatile_write!(cmdtbl.prdt_entry[las].dba, virt_2_phys(buf_ptr) as u64);
  128. cmdtbl.prdt_entry[las].dbc = ((tmp_count << 9) - 1) as u32; // 数据长度
  129. volatile_set_bit!(cmdtbl.prdt_entry[las].dbc, 1 << 31, true); // 允许中断
  130. // 设置命令
  131. let cmdfis = unsafe {
  132. ((&mut cmdtbl.cfis) as *mut [u8] as *mut usize as *mut FisRegH2D)
  133. .as_mut()
  134. .unwrap()
  135. };
  136. volatile_write!(cmdfis.fis_type, FisType::RegH2D as u8);
  137. volatile_set_bit!(cmdfis.pm, 1 << 7, true); // command_bit set
  138. volatile_write!(cmdfis.command, ATA_CMD_READ_DMA_EXT);
  139. volatile_write!(cmdfis.lba0, (lba_id_start & 0xFF) as u8);
  140. volatile_write!(cmdfis.lba1, ((lba_id_start >> 8) & 0xFF) as u8);
  141. volatile_write!(cmdfis.lba2, ((lba_id_start >> 16) & 0xFF) as u8);
  142. volatile_write!(cmdfis.lba3, ((lba_id_start >> 24) & 0xFF) as u8);
  143. volatile_write!(cmdfis.lba4, ((lba_id_start >> 32) & 0xFF) as u8);
  144. volatile_write!(cmdfis.lba5, ((lba_id_start >> 40) & 0xFF) as u8);
  145. volatile_write!(cmdfis.countl, (count & 0xFF) as u8);
  146. volatile_write!(cmdfis.counth, ((count >> 8) & 0xFF) as u8);
  147. volatile_write!(cmdfis.device, 1 << 6); // LBA Mode
  148. // 等待之前的操作完成
  149. let mut spin_count = 0;
  150. const SPIN_LIMIT: u32 = 10000;
  151. while (volatile_read!(port.tfd) as u8 & (ATA_DEV_BUSY | ATA_DEV_DRQ)) > 0
  152. && spin_count < SPIN_LIMIT
  153. {
  154. spin_count += 1;
  155. }
  156. if spin_count == SPIN_LIMIT {
  157. kerror!("Port is hung");
  158. return Err(SystemError::EIO);
  159. }
  160. volatile_set_bit!(port.ci, 1 << slot, true); // Issue command
  161. // kdebug!("To wait ahci read complete.");
  162. // 等待操作完成
  163. loop {
  164. if (volatile_read!(port.ci) & (1 << slot)) == 0 {
  165. break;
  166. }
  167. if (volatile_read!(port.is) & HBA_PxIS_TFES) > 0 {
  168. kerror!("Read disk error");
  169. return Err(SystemError::EIO);
  170. }
  171. }
  172. if kbuf.is_some() {
  173. buf.copy_from_slice(kbuf.as_ref().unwrap());
  174. }
  175. compiler_fence(core::sync::atomic::Ordering::SeqCst);
  176. // successfully read
  177. return Ok(count * 512);
  178. }
  179. fn write_at(
  180. &self,
  181. lba_id_start: BlockId,
  182. count: usize,
  183. buf: &[u8],
  184. ) -> Result<usize, SystemError> {
  185. assert!((buf.len() & 511) == 0);
  186. compiler_fence(core::sync::atomic::Ordering::SeqCst);
  187. let check_length = ((count - 1) >> 4) + 1; // prdt length
  188. if count * 512 > buf.len() || check_length > 8 as usize {
  189. // 不可能的操作
  190. return Err(SystemError::E2BIG);
  191. } else if count == 0 {
  192. return Ok(0);
  193. }
  194. let port = _port(self.ctrl_num, self.port_num);
  195. volatile_write!(port.is, u32::MAX); // Clear pending interrupt bits
  196. let slot = port.find_cmdslot().unwrap_or(u32::MAX);
  197. if slot == u32::MAX {
  198. return Err(SystemError::EIO);
  199. }
  200. compiler_fence(core::sync::atomic::Ordering::SeqCst);
  201. #[allow(unused_unsafe)]
  202. let cmdheader: &mut HbaCmdHeader = unsafe {
  203. (phys_2_virt(
  204. volatile_read!(port.clb) as usize
  205. + slot as usize * size_of::<HbaCmdHeader>() as usize,
  206. ) as *mut HbaCmdHeader)
  207. .as_mut()
  208. .unwrap()
  209. };
  210. compiler_fence(core::sync::atomic::Ordering::SeqCst);
  211. volatile_write_bit!(
  212. cmdheader.cfl,
  213. (1 << 5) - 1 as u8,
  214. (size_of::<FisRegH2D>() / size_of::<u32>()) as u8
  215. ); // Command FIS size
  216. volatile_set_bit!(cmdheader.cfl, 7 << 5, true); // (p,c,w)都设置为1, Read/Write bit : Write from device
  217. volatile_write!(cmdheader.prdtl, check_length as u16); // PRDT entries count
  218. // 设置数据存放地址
  219. compiler_fence(core::sync::atomic::Ordering::SeqCst);
  220. let mut buf_ptr = buf as *const [u8] as *mut usize as usize;
  221. // 由于目前的内存管理机制无法把用户空间的内存地址转换为物理地址,所以只能先把数据拷贝到内核空间
  222. // TODO:在内存管理重构后,可以直接使用用户空间的内存地址
  223. let user_buf = verify_area(VirtAddr::new(buf_ptr as usize), buf.len()).is_ok();
  224. let mut kbuf = if user_buf {
  225. let mut x: Vec<u8> = Vec::with_capacity(buf.len());
  226. x.resize(buf.len(), 0);
  227. x.copy_from_slice(buf);
  228. Some(x)
  229. } else {
  230. None
  231. };
  232. if kbuf.is_some() {
  233. buf_ptr = kbuf.as_mut().unwrap().as_mut_ptr() as usize;
  234. }
  235. #[allow(unused_unsafe)]
  236. let cmdtbl = unsafe {
  237. (phys_2_virt(volatile_read!(cmdheader.ctba) as usize) as *mut HbaCmdTable)
  238. .as_mut()
  239. .unwrap()
  240. };
  241. let mut tmp_count = count;
  242. compiler_fence(core::sync::atomic::Ordering::SeqCst);
  243. unsafe {
  244. // 清空整个table的旧数据
  245. write_bytes(cmdtbl, 0, 1);
  246. }
  247. // 8K bytes (16 sectors) per PRDT
  248. for i in 0..((volatile_read!(cmdheader.prdtl) - 1) as usize) {
  249. volatile_write!(cmdtbl.prdt_entry[i].dba, virt_2_phys(buf_ptr) as u64);
  250. volatile_write_bit!(cmdtbl.prdt_entry[i].dbc, (1 << 22) - 1, 8 * 1024 - 1); // 数据长度
  251. volatile_set_bit!(cmdtbl.prdt_entry[i].dbc, 1 << 31, true); // 允许中断
  252. buf_ptr += 8 * 1024;
  253. tmp_count -= 16;
  254. }
  255. // Last entry
  256. let las = (volatile_read!(cmdheader.prdtl) - 1) as usize;
  257. volatile_write!(cmdtbl.prdt_entry[las].dba, virt_2_phys(buf_ptr) as u64);
  258. volatile_set_bit!(cmdtbl.prdt_entry[las].dbc, 1 << 31, true); // 允许中断
  259. volatile_write_bit!(
  260. cmdtbl.prdt_entry[las].dbc,
  261. (1 << 22) - 1,
  262. ((tmp_count << 9) - 1) as u32
  263. ); // 数据长度
  264. // 设置命令
  265. let cmdfis = unsafe {
  266. ((&mut cmdtbl.cfis) as *mut [u8] as *mut usize as *mut FisRegH2D)
  267. .as_mut()
  268. .unwrap()
  269. };
  270. volatile_write!(cmdfis.fis_type, FisType::RegH2D as u8);
  271. volatile_set_bit!(cmdfis.pm, 1 << 7, true); // command_bit set
  272. volatile_write!(cmdfis.command, ATA_CMD_WRITE_DMA_EXT);
  273. volatile_write!(cmdfis.lba0, (lba_id_start & 0xFF) as u8);
  274. volatile_write!(cmdfis.lba1, ((lba_id_start >> 8) & 0xFF) as u8);
  275. volatile_write!(cmdfis.lba2, ((lba_id_start >> 16) & 0xFF) as u8);
  276. volatile_write!(cmdfis.lba3, ((lba_id_start >> 24) & 0xFF) as u8);
  277. volatile_write!(cmdfis.lba4, ((lba_id_start >> 32) & 0xFF) as u8);
  278. volatile_write!(cmdfis.lba5, ((lba_id_start >> 40) & 0xFF) as u8);
  279. volatile_write!(cmdfis.countl, (count & 0xFF) as u8);
  280. volatile_write!(cmdfis.counth, ((count >> 8) & 0xFF) as u8);
  281. volatile_write!(cmdfis.device, 1 << 6); // LBA Mode
  282. volatile_set_bit!(port.ci, 1 << slot, true); // Issue command
  283. // 等待操作完成
  284. loop {
  285. if (volatile_read!(port.ci) & (1 << slot)) == 0 {
  286. break;
  287. }
  288. if (volatile_read!(port.is) & HBA_PxIS_TFES) > 0 {
  289. kerror!("Write disk error");
  290. return Err(SystemError::EIO);
  291. }
  292. }
  293. compiler_fence(core::sync::atomic::Ordering::SeqCst);
  294. // successfully read
  295. return Ok(count * 512);
  296. }
  297. fn sync(&self) -> Result<(), SystemError> {
  298. // 由于目前没有block cache, 因此sync返回成功即可
  299. return Ok(());
  300. }
  301. }
  302. impl LockedAhciDisk {
  303. pub fn new(
  304. name: String,
  305. flags: u16,
  306. ctrl_num: u8,
  307. port_num: u8,
  308. ) -> Result<Arc<LockedAhciDisk>, SystemError> {
  309. // 构建磁盘结构体
  310. let result: Arc<LockedAhciDisk> = Arc::new(LockedAhciDisk(SpinLock::new(AhciDisk {
  311. name,
  312. flags,
  313. partitions: Default::default(),
  314. ctrl_num,
  315. port_num,
  316. self_ref: Weak::default(),
  317. })));
  318. let table: MbrDiskPartionTable = result.read_mbr_table()?;
  319. // 求出有多少可用分区
  320. for i in 0..4 {
  321. compiler_fence(Ordering::SeqCst);
  322. if table.dpte[i].part_type != 0 {
  323. let w = Arc::downgrade(&result);
  324. result.0.lock().partitions.push(Partition::new(
  325. table.dpte[i].starting_sector() as u64,
  326. table.dpte[i].starting_lba as u64,
  327. table.dpte[i].total_sectors as u64,
  328. w,
  329. i as u16,
  330. ));
  331. }
  332. }
  333. result.0.lock().self_ref = Arc::downgrade(&result);
  334. return Ok(result);
  335. }
  336. /// @brief: 从磁盘中读取 MBR 分区表结构体 TODO: Cursor
  337. pub fn read_mbr_table(&self) -> Result<MbrDiskPartionTable, SystemError> {
  338. let mut table: MbrDiskPartionTable = Default::default();
  339. // 数据缓冲区
  340. let mut buf: Vec<u8> = Vec::new();
  341. buf.resize(size_of::<MbrDiskPartionTable>(), 0);
  342. self.read_at(0, 1, &mut buf)?;
  343. // 创建 Cursor 用于按字节读取
  344. let mut cursor = VecCursor::new(buf);
  345. cursor.seek(SeekFrom::SeekCurrent(446))?;
  346. for i in 0..4 {
  347. kdebug!("infomation of partition {}:\n", i);
  348. table.dpte[i].flags = cursor.read_u8()?;
  349. table.dpte[i].starting_head = cursor.read_u8()?;
  350. table.dpte[i].starting_sector_cylinder = cursor.read_u16()?;
  351. table.dpte[i].part_type = cursor.read_u8()?;
  352. table.dpte[i].ending_head = cursor.read_u8()?;
  353. table.dpte[i].ending_sector_cylingder = cursor.read_u16()?;
  354. table.dpte[i].starting_lba = cursor.read_u32()?;
  355. table.dpte[i].total_sectors = cursor.read_u32()?;
  356. kdebug!("dpte[i] = {:?}", table.dpte[i]);
  357. }
  358. table.bs_trailsig = cursor.read_u16()?;
  359. // kdebug!("bs_trailsig = {}", unsafe {
  360. // read_unaligned(addr_of!(table.bs_trailsig))
  361. // });
  362. return Ok(table);
  363. }
  364. }
  365. impl KObject for LockedAhciDisk {
  366. fn as_any_ref(&self) -> &dyn core::any::Any {
  367. self
  368. }
  369. fn inode(&self) -> Option<Arc<KernFSInode>> {
  370. todo!()
  371. }
  372. fn kobj_type(&self) -> Option<&'static dyn KObjType> {
  373. todo!()
  374. }
  375. fn kset(&self) -> Option<Arc<KSet>> {
  376. todo!()
  377. }
  378. fn parent(&self) -> Option<Weak<dyn KObject>> {
  379. todo!()
  380. }
  381. fn set_inode(&self, _inode: Option<Arc<KernFSInode>>) {
  382. todo!()
  383. }
  384. fn kobj_state(&self) -> RwLockReadGuard<KObjectState> {
  385. todo!()
  386. }
  387. fn kobj_state_mut(&self) -> RwLockWriteGuard<KObjectState> {
  388. todo!()
  389. }
  390. fn set_kobj_state(&self, _state: KObjectState) {
  391. todo!()
  392. }
  393. fn name(&self) -> alloc::string::String {
  394. todo!()
  395. }
  396. fn set_name(&self, _name: alloc::string::String) {
  397. todo!()
  398. }
  399. fn set_kset(&self, _kset: Option<Arc<KSet>>) {
  400. todo!()
  401. }
  402. fn set_parent(&self, _parent: Option<Weak<dyn KObject>>) {
  403. todo!()
  404. }
  405. fn set_kobj_type(&self, _ktype: Option<&'static dyn KObjType>) {
  406. todo!()
  407. }
  408. }
  409. impl Device for LockedAhciDisk {
  410. fn dev_type(&self) -> DeviceType {
  411. return DeviceType::Block;
  412. }
  413. fn id_table(&self) -> IdTable {
  414. todo!()
  415. }
  416. fn bus(&self) -> Option<Arc<dyn Bus>> {
  417. todo!("LockedAhciDisk::bus()")
  418. }
  419. fn set_bus(&self, _bus: Option<Arc<dyn Bus>>) {
  420. todo!("LockedAhciDisk::set_bus()")
  421. }
  422. fn driver(&self) -> Option<Arc<dyn Driver>> {
  423. todo!("LockedAhciDisk::driver()")
  424. }
  425. fn is_dead(&self) -> bool {
  426. false
  427. }
  428. fn set_driver(&self, _driver: Option<Weak<dyn Driver>>) {
  429. todo!("LockedAhciDisk::set_driver()")
  430. }
  431. fn can_match(&self) -> bool {
  432. todo!()
  433. }
  434. fn set_can_match(&self, _can_match: bool) {
  435. todo!()
  436. }
  437. fn state_synced(&self) -> bool {
  438. todo!()
  439. }
  440. }
  441. impl BlockDevice for LockedAhciDisk {
  442. #[inline]
  443. fn as_any_ref(&self) -> &dyn core::any::Any {
  444. self
  445. }
  446. #[inline]
  447. fn blk_size_log2(&self) -> u8 {
  448. 9
  449. }
  450. fn sync(&self) -> Result<(), SystemError> {
  451. return self.0.lock().sync();
  452. }
  453. #[inline]
  454. fn device(&self) -> Arc<dyn Device> {
  455. return self.0.lock().self_ref.upgrade().unwrap();
  456. }
  457. fn block_size(&self) -> usize {
  458. todo!()
  459. }
  460. fn partitions(&self) -> Vec<Arc<Partition>> {
  461. return self.0.lock().partitions.clone();
  462. }
  463. #[inline]
  464. fn read_at(
  465. &self,
  466. lba_id_start: BlockId, // 起始lba编号
  467. count: usize, // 读取lba的数量
  468. buf: &mut [u8],
  469. ) -> Result<usize, SystemError> {
  470. self.0.lock().read_at(lba_id_start, count, buf)
  471. }
  472. #[inline]
  473. fn write_at(
  474. &self,
  475. lba_id_start: BlockId,
  476. count: usize,
  477. buf: &[u8],
  478. ) -> Result<usize, SystemError> {
  479. self.0.lock().write_at(lba_id_start, count, buf)
  480. }
  481. }