SATA 链路的初始化与速度协商机制主要集中在内核框架层 (libata-sata.c与libata-core.c) 和平台特定的硬件控制驱动 (如 ahci_xgene.c)。该流程包括初始探测、物理协议协商 (PHY OOB 控制) 与特定硬件工作区的修正。
1. 初始化控制阶段 (sata_link_init_spd)
在识别或热插拔设备前,控制器需要先确立初始通讯速度上限和链路状态:
- 读取现有效应值 :通过
sata_scr_read读取SCR_CONTROL(SControl, 偏移量 0x2c) 寄存器获取当前的 PHY 通讯速度 (bits [7:4])。 - 强制限制匹配 :根据系统设定 (
sata_spd_limit) 保留限制位,避免部分终端在过高频率下无法建立通信,通过ata_force_link_limits将软性与硬性速度限额同步。
2. 核心:硬件复位与链路协商 (sata_link_hardreset)
整个 SATA 物理层连接和速度协商主要通过 COMRESET 触发,这也是 sata_link_hardreset() 负责的核心所在。
int sata_link_hardreset(struct ata_link *link, const unsigned long *timing, unsigned long deadline, bool *online, int (*check_ready)...);
2.1 修改目标速率与触发 COMRESET
-
判定降频 : 调用
sata_set_spd_needed决定是否需要变更目标限制速率。如果需要变更,则重写 SControl。 -
发送 COMRESET : 通过改写
SControl中控制 DET (Device Detection) 位为0x1,主动向从设备发送物理 OOB 信号 (COMRESET)。scontrol = (scontrol & 0x0f0) | 0x301; // 发送 COMRESET
sata_scr_write_flush(link, SCR_CONTROL, scontrol);
-
保持延时 : 根据 SATA 规范并为 AHCI 留出安全裕度,会延时至少
1ms保持复位信号。
2.2 解除复位并等待 PHY 握手 (sata_link_resume)
- 解除复位,将
SControl的 DET 设置清零 (0x300)。 - PHY 脱抖协商 (
sata_link_debounce) : 在解除复位后,主控与设备的物理层开始互相发送COMINIT和COMWAKE,从而进行模拟包协商(Speed negotiation 和 Dword Sync)。驱动会在这段时间里以轮询的姿态读取SCR_STATUS(SStatus, 偏移 0x28) 的检测状态,直到表示连接稳定的标志固定(通常不为 1 时被视为锁定状态)。
2.3 链路与设备就绪 (ata_wait_ready)
当 PHY 层面完成互锁以后(即成功建立指定世代如 Gen3 6.0Gbps 的连接),内核会调用 ata_wait_ready() 收接收备发来的 D2H (Device to Host) Register FIS。接收成功即代表盘不仅物理在线,同时寄存器状态已经通过自检准备就绪,可以安全发后续的 IDENTIFY 命令。
3. 硬件特异性纠正:X-Gene SoC 的协商降级
在平台具体的控制器上(如 ahci_xgene.c 中),标准的流程可能会面临硬件缺陷 (Erratas),因此有定制重写的 xgene_ahci_do_hardreset :
特定环境中的 PHY 差异会引起严重的 OOB 协商失序。X-Gene PHY 在长运算周期内进行 COMRESET 会存在随机的字位对齐失常。
在自定义的硬复位实现处可以明显看到针对协商错误的恢复代码:
- 多级尝试 : 因为速率协商由于环境噪音可能失败,驱动进行了
MAX_LINK_DOWN_RETRY次重试硬复位。 - 硬件检测位补偿 : 监控
SCR_ERROR(SError 偏移量 0x30)。如果有异常的SERR_DISPARITY或10B_8B_ERR触发,这表示在速度协商对齐期间接收端发生了差异错误。硬件缺陷要求复位之后人工清空此类错误标记并可能需要单独复位内部接收器,确保后期的NCQ/PIO命令不被中断。