#define IDDEV_OFFSERIALNUM 10
#define IDDEV_OFFMODELNUM 27
#define IDDEV_OFFADDSUPPORT 69
+#define IDDEV_OFFA48SUPPORT 83
#define IDDEV_OFFALIGN 209
#define IDDEV_OFFLPP 106
#define IDDEV_OFFCAPABILITIES 49
-static uint32_t cdb_size[] = { SCSI_CDB12, SCSI_CDB16, 0, 0 };
+static u32_t cdb_size[] = { SCSI_CDB12, SCSI_CDB16, 0, 0 };
void
-ahci_parse_dev_info(struct hba_device* dev_info, uint16_t* data)
+ahci_parse_dev_info(struct hba_device* dev_info, u16_t* data)
{
- dev_info->max_lba = *((uint32_t*)(data + IDDEV_OFFMAXLBA));
- dev_info->block_size = *((uint32_t*)(data + IDDEV_OFFLSECSIZE));
+ dev_info->max_lba = *((u32_t*)(data + IDDEV_OFFMAXLBA));
+ dev_info->block_size = *((u32_t*)(data + IDDEV_OFFLSECSIZE));
dev_info->cbd_size = cdb_size[(*data & 0x3)];
- dev_info->wwn = *(uint64_t*)(data + IDDEV_OFFWWN);
+ dev_info->wwn = *(u64_t*)(data + IDDEV_OFFWWN);
dev_info->block_per_sec = 1 << (*(data + IDDEV_OFFLPP) & 0xf);
dev_info->alignment_offset = *(data + IDDEV_OFFALIGN) & 0x3fff;
- dev_info->capabilities = *((uint32_t*)(data + IDDEV_OFFCAPABILITIES));
+ dev_info->capabilities = *((u32_t*)(data + IDDEV_OFFCAPABILITIES));
if (!dev_info->block_size) {
dev_info->block_size = 512;
}
- if ((*(data + IDDEV_OFFADDSUPPORT) & 0x8)) {
- dev_info->max_lba = *((uint64_t*)(data + IDDEV_OFFMAXLBA_EXT));
+ if ((*(data + IDDEV_OFFADDSUPPORT) & 0x8) &&
+ (*(data + IDDEV_OFFA48SUPPORT) & 0x400)) {
+ dev_info->max_lba = *((lba_t*)(data + IDDEV_OFFMAXLBA_EXT));
dev_info->flags |= HBA_DEV_FEXTLBA;
}
}
void
-ahci_parsestr(char* str, uint16_t* reg_start, int size_word)
+ahci_parsestr(char* str, u16_t* reg_start, int size_word)
{
int j = 0;
for (int i = 0; i < size_word; i++, j += 2) {
- uint16_t reg = *(reg_start + i);
+ u16_t reg = *(reg_start + i);
str[j] = (char)(reg >> 8);
str[j + 1] = (char)(reg & 0xff);
}
hba_clear_reg(port->regs[HBA_RPxIS]);
return retries < MAX_RETRY;
+}
+
+void
+ahci_post(struct hba_port* port, struct hba_cmd_state* state, int slot)
+{
+ int bitmask = 1 << slot;
+
+ // 确保端口是空闲的
+ wait_until(!(port->regs[HBA_RPxTFD] & (HBA_PxTFD_BSY | HBA_PxTFD_DRQ)));
+
+ hba_clear_reg(port->regs[HBA_RPxIS]);
+
+ port->cmdctx.issued[slot] = state;
+ port->cmdctx.tracked_ci |= bitmask;
+ port->regs[HBA_RPxCI] |= bitmask;
}
\ No newline at end of file