ahcidisk.rs 18 KB

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