-----BEGIN PGP SIGNATURE-----
Version: GnuPG v1 iQIcBAABAgAGBQJVl3fQAAoJEH3vgQaq/DkOgNkP/0gkVzBeIhuJHEWJasX12zS4 VaXAKNc/haSvnT4nOOw2VfvyXvbawUgN1izBKhqrGu+Qdj9kZXMuQN/H8V7kLQeK s9/c9FmiI4On9uheol294Ig9Q8R/NUc9Z69st0usEdAIlGOhzQlItwzQQecDWmd6 LIb0pVAId4jrW+bW5XbMEzCSKQkD5ETKAZ8cMh1+U60b8Oi0xZsY/q+SPhj/PHrX vH98oOVT0rgQ9ZJPpSmxa2SpeeMINkROeer19rJY9tbcK4W52i4yt7c0NxhICYfP bB75Zis0TUR9csuuW2801bejxhNhwDka5vur6q14h4wEawtPMV07zhmxTVyKFk89 Y1Jzt7ys2awotHy6/XrHU4eUBwpd1BAgZ8P5M0HzKdxH6+Xxg269zc/QmG3FqCVo HKJ7e/vcmImYJZG1GhJrP4PUfXrLOciBzYhmaxwGNQLXrM1YUE6F/PvgIJyRGRdB vXrFUifxHylgQUAdlHHAqT4LlfJZVjTVBVntGqoZgH7RiyK4lSmyUkdYRtlG+Fcm SCCjRcCAh1nzSp7XI2QziNl/ezavOI4lpAv64ubXU3VGxudGgcq+WsYF4iEOdbNJ W04QVCjR9PzvLabRLYzTtWTywYT79cg4xh/Lq+EyXAoWFRM2/CYliHJk4VSlk3Us v8a6sWB9qc3dq6f6uUTX =nQZA -----END PGP SIGNATURE----- Merge remote-tracking branch 'remotes/jnsnow/tags/ide-pull-request' into staging # gpg: Signature made Sat Jul 4 07:06:08 2015 BST using RSA key ID AAFC390E # gpg: Good signature from "John Snow (John Huston) <jsnow@redhat.com>" # gpg: WARNING: This key is not certified with sufficiently trusted signatures! # gpg: It is not certain that the signature belongs to the owner. # Primary key fingerprint: FAEB 9711 A12C F475 812F 18F2 88A9 064D 1835 61EB # Subkey fingerprint: F9B7 ABDB BCAC DF95 BE76 CBD0 7DEF 8106 AAFC 390E * remotes/jnsnow/tags/ide-pull-request: (35 commits) ahci: fix sdb fis semantics qtest/ahci: halted ncq migration test ahci: Do not map cmd_fis to generate response ahci: ncq migration ahci: add get_cmd_header helper ahci: add cmd header to ncq transfer state qtest/ahci: halted NCQ test ahci: correct ncq sector count ahci: correct types in NCQTransferState ahci: add rwerror=stop support for ncq ahci: factor ncq_finish out of ncq_cb ahci: refactor process_ncq_command ahci: assert is_ncq for process_ncq ahci: stash ncq command ide: add limit to .prepare_buf() qtest/ahci: ncq migration test qtest/ahci: simple ncq data test libqos/ahci: Force all NCQ commands to be LBA48 libqos/ahci: set the NCQ tag on command_commit libqos/ahci: adjust expected NCQ interrupts ... Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
commit
f50a1640fb
439
hw/ide/ahci.c
439
hw/ide/ahci.c
@ -45,11 +45,11 @@ do { \
|
|||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
static void check_cmd(AHCIState *s, int port);
|
static void check_cmd(AHCIState *s, int port);
|
||||||
static int handle_cmd(AHCIState *s,int port,int slot);
|
static int handle_cmd(AHCIState *s, int port, uint8_t slot);
|
||||||
static void ahci_reset_port(AHCIState *s, int port);
|
static void ahci_reset_port(AHCIState *s, int port);
|
||||||
static void ahci_write_fis_d2h(AHCIDevice *ad, uint8_t *cmd_fis);
|
static void ahci_write_fis_d2h(AHCIDevice *ad, uint8_t *cmd_fis);
|
||||||
static void ahci_init_d2h(AHCIDevice *ad);
|
static void ahci_init_d2h(AHCIDevice *ad);
|
||||||
static int ahci_dma_prepare_buf(IDEDMA *dma, int is_write);
|
static int ahci_dma_prepare_buf(IDEDMA *dma, int32_t limit);
|
||||||
static void ahci_commit_buf(IDEDMA *dma, uint32_t tx_bytes);
|
static void ahci_commit_buf(IDEDMA *dma, uint32_t tx_bytes);
|
||||||
static bool ahci_map_clb_address(AHCIDevice *ad);
|
static bool ahci_map_clb_address(AHCIDevice *ad);
|
||||||
static bool ahci_map_fis_address(AHCIDevice *ad);
|
static bool ahci_map_fis_address(AHCIDevice *ad);
|
||||||
@ -106,8 +106,6 @@ static uint32_t ahci_port_read(AHCIState *s, int port, int offset)
|
|||||||
val = pr->scr_err;
|
val = pr->scr_err;
|
||||||
break;
|
break;
|
||||||
case PORT_SCR_ACT:
|
case PORT_SCR_ACT:
|
||||||
pr->scr_act &= ~s->dev[port].finished;
|
|
||||||
s->dev[port].finished = 0;
|
|
||||||
val = pr->scr_act;
|
val = pr->scr_act;
|
||||||
break;
|
break;
|
||||||
case PORT_CMD_ISSUE:
|
case PORT_CMD_ISSUE:
|
||||||
@ -331,8 +329,7 @@ static void ahci_port_write(AHCIState *s, int port, int offset, uint32_t val)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static uint64_t ahci_mem_read(void *opaque, hwaddr addr,
|
static uint64_t ahci_mem_read_32(void *opaque, hwaddr addr)
|
||||||
unsigned size)
|
|
||||||
{
|
{
|
||||||
AHCIState *s = opaque;
|
AHCIState *s = opaque;
|
||||||
uint32_t val = 0;
|
uint32_t val = 0;
|
||||||
@ -368,6 +365,30 @@ static uint64_t ahci_mem_read(void *opaque, hwaddr addr,
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AHCI 1.3 section 3 ("HBA Memory Registers")
|
||||||
|
* Support unaligned 8/16/32 bit reads, and 64 bit aligned reads.
|
||||||
|
* Caller is responsible for masking unwanted higher order bytes.
|
||||||
|
*/
|
||||||
|
static uint64_t ahci_mem_read(void *opaque, hwaddr addr, unsigned size)
|
||||||
|
{
|
||||||
|
hwaddr aligned = addr & ~0x3;
|
||||||
|
int ofst = addr - aligned;
|
||||||
|
uint64_t lo = ahci_mem_read_32(opaque, aligned);
|
||||||
|
uint64_t hi;
|
||||||
|
|
||||||
|
/* if < 8 byte read does not cross 4 byte boundary */
|
||||||
|
if (ofst + size <= 4) {
|
||||||
|
return lo >> (ofst * 8);
|
||||||
|
}
|
||||||
|
g_assert_cmpint(size, >, 1);
|
||||||
|
|
||||||
|
/* If the 64bit read is unaligned, we will produce undefined
|
||||||
|
* results. AHCI does not support unaligned 64bit reads. */
|
||||||
|
hi = ahci_mem_read_32(opaque, aligned + 4);
|
||||||
|
return (hi << 32 | lo) >> (ofst * 8);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static void ahci_mem_write(void *opaque, hwaddr addr,
|
static void ahci_mem_write(void *opaque, hwaddr addr,
|
||||||
uint64_t val, unsigned size)
|
uint64_t val, unsigned size)
|
||||||
@ -483,7 +504,7 @@ static void ahci_reg_init(AHCIState *s)
|
|||||||
static void check_cmd(AHCIState *s, int port)
|
static void check_cmd(AHCIState *s, int port)
|
||||||
{
|
{
|
||||||
AHCIPortRegs *pr = &s->dev[port].port_regs;
|
AHCIPortRegs *pr = &s->dev[port].port_regs;
|
||||||
int slot;
|
uint8_t slot;
|
||||||
|
|
||||||
if ((pr->cmd & PORT_CMD_START) && pr->cmd_issue) {
|
if ((pr->cmd & PORT_CMD_START) && pr->cmd_issue) {
|
||||||
for (slot = 0; (slot < 32) && pr->cmd_issue; slot++) {
|
for (slot = 0; (slot < 32) && pr->cmd_issue; slot++) {
|
||||||
@ -558,6 +579,7 @@ static void ahci_reset_port(AHCIState *s, int port)
|
|||||||
/* reset ncq queue */
|
/* reset ncq queue */
|
||||||
for (i = 0; i < AHCI_MAX_CMDS; i++) {
|
for (i = 0; i < AHCI_MAX_CMDS; i++) {
|
||||||
NCQTransferState *ncq_tfs = &s->dev[port].ncq_tfs[i];
|
NCQTransferState *ncq_tfs = &s->dev[port].ncq_tfs[i];
|
||||||
|
ncq_tfs->halt = false;
|
||||||
if (!ncq_tfs->used) {
|
if (!ncq_tfs->used) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -642,14 +664,14 @@ static void ahci_unmap_clb_address(AHCIDevice *ad)
|
|||||||
ad->lst = NULL;
|
ad->lst = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ahci_write_fis_sdb(AHCIState *s, int port, uint32_t finished)
|
static void ahci_write_fis_sdb(AHCIState *s, NCQTransferState *ncq_tfs)
|
||||||
{
|
{
|
||||||
AHCIDevice *ad = &s->dev[port];
|
AHCIDevice *ad = ncq_tfs->drive;
|
||||||
AHCIPortRegs *pr = &ad->port_regs;
|
AHCIPortRegs *pr = &ad->port_regs;
|
||||||
IDEState *ide_state;
|
IDEState *ide_state;
|
||||||
SDBFIS *sdb_fis;
|
SDBFIS *sdb_fis;
|
||||||
|
|
||||||
if (!s->dev[port].res_fis ||
|
if (!ad->res_fis ||
|
||||||
!(pr->cmd & PORT_CMD_FIS_RX)) {
|
!(pr->cmd & PORT_CMD_FIS_RX)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -659,53 +681,35 @@ static void ahci_write_fis_sdb(AHCIState *s, int port, uint32_t finished)
|
|||||||
|
|
||||||
sdb_fis->type = SATA_FIS_TYPE_SDB;
|
sdb_fis->type = SATA_FIS_TYPE_SDB;
|
||||||
/* Interrupt pending & Notification bit */
|
/* Interrupt pending & Notification bit */
|
||||||
sdb_fis->flags = (ad->hba->control_regs.irqstatus ? (1 << 6) : 0);
|
sdb_fis->flags = 0x40; /* Interrupt bit, always 1 for NCQ */
|
||||||
sdb_fis->status = ide_state->status & 0x77;
|
sdb_fis->status = ide_state->status & 0x77;
|
||||||
sdb_fis->error = ide_state->error;
|
sdb_fis->error = ide_state->error;
|
||||||
/* update SAct field in SDB_FIS */
|
/* update SAct field in SDB_FIS */
|
||||||
s->dev[port].finished |= finished;
|
|
||||||
sdb_fis->payload = cpu_to_le32(ad->finished);
|
sdb_fis->payload = cpu_to_le32(ad->finished);
|
||||||
|
|
||||||
/* Update shadow registers (except BSY 0x80 and DRQ 0x08) */
|
/* Update shadow registers (except BSY 0x80 and DRQ 0x08) */
|
||||||
pr->tfdata = (ad->port.ifs[0].error << 8) |
|
pr->tfdata = (ad->port.ifs[0].error << 8) |
|
||||||
(ad->port.ifs[0].status & 0x77) |
|
(ad->port.ifs[0].status & 0x77) |
|
||||||
(pr->tfdata & 0x88);
|
(pr->tfdata & 0x88);
|
||||||
|
pr->scr_act &= ~ad->finished;
|
||||||
|
ad->finished = 0;
|
||||||
|
|
||||||
ahci_trigger_irq(s, ad, PORT_IRQ_SDB_FIS);
|
/* Trigger IRQ if interrupt bit is set (which currently, it always is) */
|
||||||
|
if (sdb_fis->flags & 0x40) {
|
||||||
|
ahci_trigger_irq(s, ad, PORT_IRQ_SDB_FIS);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ahci_write_fis_pio(AHCIDevice *ad, uint16_t len)
|
static void ahci_write_fis_pio(AHCIDevice *ad, uint16_t len)
|
||||||
{
|
{
|
||||||
AHCIPortRegs *pr = &ad->port_regs;
|
AHCIPortRegs *pr = &ad->port_regs;
|
||||||
uint8_t *pio_fis, *cmd_fis;
|
uint8_t *pio_fis;
|
||||||
uint64_t tbl_addr;
|
|
||||||
dma_addr_t cmd_len = 0x80;
|
|
||||||
IDEState *s = &ad->port.ifs[0];
|
IDEState *s = &ad->port.ifs[0];
|
||||||
|
|
||||||
if (!ad->res_fis || !(pr->cmd & PORT_CMD_FIS_RX)) {
|
if (!ad->res_fis || !(pr->cmd & PORT_CMD_FIS_RX)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* map cmd_fis */
|
|
||||||
tbl_addr = le64_to_cpu(ad->cur_cmd->tbl_addr);
|
|
||||||
cmd_fis = dma_memory_map(ad->hba->as, tbl_addr, &cmd_len,
|
|
||||||
DMA_DIRECTION_TO_DEVICE);
|
|
||||||
|
|
||||||
if (cmd_fis == NULL) {
|
|
||||||
DPRINTF(ad->port_no, "dma_memory_map failed in ahci_write_fis_pio");
|
|
||||||
ahci_trigger_irq(ad->hba, ad, PORT_IRQ_HBUS_ERR);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cmd_len != 0x80) {
|
|
||||||
DPRINTF(ad->port_no,
|
|
||||||
"dma_memory_map mapped too few bytes in ahci_write_fis_pio");
|
|
||||||
dma_memory_unmap(ad->hba->as, cmd_fis, cmd_len,
|
|
||||||
DMA_DIRECTION_TO_DEVICE, cmd_len);
|
|
||||||
ahci_trigger_irq(ad->hba, ad, PORT_IRQ_HBUS_ERR);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
pio_fis = &ad->res_fis[RES_FIS_PSFIS];
|
pio_fis = &ad->res_fis[RES_FIS_PSFIS];
|
||||||
|
|
||||||
pio_fis[0] = SATA_FIS_TYPE_PIO_SETUP;
|
pio_fis[0] = SATA_FIS_TYPE_PIO_SETUP;
|
||||||
@ -721,8 +725,8 @@ static void ahci_write_fis_pio(AHCIDevice *ad, uint16_t len)
|
|||||||
pio_fis[9] = s->hob_lcyl;
|
pio_fis[9] = s->hob_lcyl;
|
||||||
pio_fis[10] = s->hob_hcyl;
|
pio_fis[10] = s->hob_hcyl;
|
||||||
pio_fis[11] = 0;
|
pio_fis[11] = 0;
|
||||||
pio_fis[12] = cmd_fis[12];
|
pio_fis[12] = s->nsector & 0xFF;
|
||||||
pio_fis[13] = cmd_fis[13];
|
pio_fis[13] = (s->nsector >> 8) & 0xFF;
|
||||||
pio_fis[14] = 0;
|
pio_fis[14] = 0;
|
||||||
pio_fis[15] = s->status;
|
pio_fis[15] = s->status;
|
||||||
pio_fis[16] = len & 255;
|
pio_fis[16] = len & 255;
|
||||||
@ -739,9 +743,6 @@ static void ahci_write_fis_pio(AHCIDevice *ad, uint16_t len)
|
|||||||
}
|
}
|
||||||
|
|
||||||
ahci_trigger_irq(ad->hba, ad, PORT_IRQ_PIOS_FIS);
|
ahci_trigger_irq(ad->hba, ad, PORT_IRQ_PIOS_FIS);
|
||||||
|
|
||||||
dma_memory_unmap(ad->hba->as, cmd_fis, cmd_len,
|
|
||||||
DMA_DIRECTION_TO_DEVICE, cmd_len);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ahci_write_fis_d2h(AHCIDevice *ad, uint8_t *cmd_fis)
|
static void ahci_write_fis_d2h(AHCIDevice *ad, uint8_t *cmd_fis)
|
||||||
@ -749,22 +750,12 @@ static void ahci_write_fis_d2h(AHCIDevice *ad, uint8_t *cmd_fis)
|
|||||||
AHCIPortRegs *pr = &ad->port_regs;
|
AHCIPortRegs *pr = &ad->port_regs;
|
||||||
uint8_t *d2h_fis;
|
uint8_t *d2h_fis;
|
||||||
int i;
|
int i;
|
||||||
dma_addr_t cmd_len = 0x80;
|
|
||||||
int cmd_mapped = 0;
|
|
||||||
IDEState *s = &ad->port.ifs[0];
|
IDEState *s = &ad->port.ifs[0];
|
||||||
|
|
||||||
if (!ad->res_fis || !(pr->cmd & PORT_CMD_FIS_RX)) {
|
if (!ad->res_fis || !(pr->cmd & PORT_CMD_FIS_RX)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!cmd_fis) {
|
|
||||||
/* map cmd_fis */
|
|
||||||
uint64_t tbl_addr = le64_to_cpu(ad->cur_cmd->tbl_addr);
|
|
||||||
cmd_fis = dma_memory_map(ad->hba->as, tbl_addr, &cmd_len,
|
|
||||||
DMA_DIRECTION_TO_DEVICE);
|
|
||||||
cmd_mapped = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
d2h_fis = &ad->res_fis[RES_FIS_RFIS];
|
d2h_fis = &ad->res_fis[RES_FIS_RFIS];
|
||||||
|
|
||||||
d2h_fis[0] = SATA_FIS_TYPE_REGISTER_D2H;
|
d2h_fis[0] = SATA_FIS_TYPE_REGISTER_D2H;
|
||||||
@ -780,8 +771,8 @@ static void ahci_write_fis_d2h(AHCIDevice *ad, uint8_t *cmd_fis)
|
|||||||
d2h_fis[9] = s->hob_lcyl;
|
d2h_fis[9] = s->hob_lcyl;
|
||||||
d2h_fis[10] = s->hob_hcyl;
|
d2h_fis[10] = s->hob_hcyl;
|
||||||
d2h_fis[11] = 0;
|
d2h_fis[11] = 0;
|
||||||
d2h_fis[12] = cmd_fis[12];
|
d2h_fis[12] = s->nsector & 0xFF;
|
||||||
d2h_fis[13] = cmd_fis[13];
|
d2h_fis[13] = (s->nsector >> 8) & 0xFF;
|
||||||
for (i = 14; i < 20; i++) {
|
for (i = 14; i < 20; i++) {
|
||||||
d2h_fis[i] = 0;
|
d2h_fis[i] = 0;
|
||||||
}
|
}
|
||||||
@ -795,26 +786,22 @@ static void ahci_write_fis_d2h(AHCIDevice *ad, uint8_t *cmd_fis)
|
|||||||
}
|
}
|
||||||
|
|
||||||
ahci_trigger_irq(ad->hba, ad, PORT_IRQ_D2H_REG_FIS);
|
ahci_trigger_irq(ad->hba, ad, PORT_IRQ_D2H_REG_FIS);
|
||||||
|
|
||||||
if (cmd_mapped) {
|
|
||||||
dma_memory_unmap(ad->hba->as, cmd_fis, cmd_len,
|
|
||||||
DMA_DIRECTION_TO_DEVICE, cmd_len);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int prdt_tbl_entry_size(const AHCI_SG *tbl)
|
static int prdt_tbl_entry_size(const AHCI_SG *tbl)
|
||||||
{
|
{
|
||||||
|
/* flags_size is zero-based */
|
||||||
return (le32_to_cpu(tbl->flags_size) & AHCI_PRDT_SIZE_MASK) + 1;
|
return (le32_to_cpu(tbl->flags_size) & AHCI_PRDT_SIZE_MASK) + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ahci_populate_sglist(AHCIDevice *ad, QEMUSGList *sglist,
|
static int ahci_populate_sglist(AHCIDevice *ad, QEMUSGList *sglist,
|
||||||
int32_t offset)
|
AHCICmdHdr *cmd, int64_t limit, int32_t offset)
|
||||||
{
|
{
|
||||||
AHCICmdHdr *cmd = ad->cur_cmd;
|
uint16_t opts = le16_to_cpu(cmd->opts);
|
||||||
uint32_t opts = le32_to_cpu(cmd->opts);
|
uint16_t prdtl = le16_to_cpu(cmd->prdtl);
|
||||||
uint64_t prdt_addr = le64_to_cpu(cmd->tbl_addr) + 0x80;
|
uint64_t cfis_addr = le64_to_cpu(cmd->tbl_addr);
|
||||||
int sglist_alloc_hint = opts >> AHCI_CMD_HDR_PRDT_LEN;
|
uint64_t prdt_addr = cfis_addr + 0x80;
|
||||||
dma_addr_t prdt_len = (sglist_alloc_hint * sizeof(AHCI_SG));
|
dma_addr_t prdt_len = (prdtl * sizeof(AHCI_SG));
|
||||||
dma_addr_t real_prdt_len = prdt_len;
|
dma_addr_t real_prdt_len = prdt_len;
|
||||||
uint8_t *prdt;
|
uint8_t *prdt;
|
||||||
int i;
|
int i;
|
||||||
@ -834,7 +821,7 @@ static int ahci_populate_sglist(AHCIDevice *ad, QEMUSGList *sglist,
|
|||||||
* request for sector sizes up to 32K.
|
* request for sector sizes up to 32K.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
if (!sglist_alloc_hint) {
|
if (!prdtl) {
|
||||||
DPRINTF(ad->port_no, "no sg list given by guest: 0x%08x\n", opts);
|
DPRINTF(ad->port_no, "no sg list given by guest: 0x%08x\n", opts);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@ -853,13 +840,12 @@ static int ahci_populate_sglist(AHCIDevice *ad, QEMUSGList *sglist,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Get entries in the PRDT, init a qemu sglist accordingly */
|
/* Get entries in the PRDT, init a qemu sglist accordingly */
|
||||||
if (sglist_alloc_hint > 0) {
|
if (prdtl > 0) {
|
||||||
AHCI_SG *tbl = (AHCI_SG *)prdt;
|
AHCI_SG *tbl = (AHCI_SG *)prdt;
|
||||||
sum = 0;
|
sum = 0;
|
||||||
for (i = 0; i < sglist_alloc_hint; i++) {
|
for (i = 0; i < prdtl; i++) {
|
||||||
/* flags_size is zero-based */
|
|
||||||
tbl_entry_size = prdt_tbl_entry_size(&tbl[i]);
|
tbl_entry_size = prdt_tbl_entry_size(&tbl[i]);
|
||||||
if (offset <= (sum + tbl_entry_size)) {
|
if (offset < (sum + tbl_entry_size)) {
|
||||||
off_idx = i;
|
off_idx = i;
|
||||||
off_pos = offset - sum;
|
off_pos = offset - sum;
|
||||||
break;
|
break;
|
||||||
@ -874,15 +860,16 @@ static int ahci_populate_sglist(AHCIDevice *ad, QEMUSGList *sglist,
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
qemu_sglist_init(sglist, qbus->parent, (sglist_alloc_hint - off_idx),
|
qemu_sglist_init(sglist, qbus->parent, (prdtl - off_idx),
|
||||||
ad->hba->as);
|
ad->hba->as);
|
||||||
qemu_sglist_add(sglist, le64_to_cpu(tbl[off_idx].addr) + off_pos,
|
qemu_sglist_add(sglist, le64_to_cpu(tbl[off_idx].addr) + off_pos,
|
||||||
prdt_tbl_entry_size(&tbl[off_idx]) - off_pos);
|
MIN(prdt_tbl_entry_size(&tbl[off_idx]) - off_pos,
|
||||||
|
limit));
|
||||||
|
|
||||||
for (i = off_idx + 1; i < sglist_alloc_hint; i++) {
|
for (i = off_idx + 1; i < prdtl && sglist->size < limit; i++) {
|
||||||
/* flags_size is zero-based */
|
|
||||||
qemu_sglist_add(sglist, le64_to_cpu(tbl[i].addr),
|
qemu_sglist_add(sglist, le64_to_cpu(tbl[i].addr),
|
||||||
prdt_tbl_entry_size(&tbl[i]));
|
MIN(prdt_tbl_entry_size(&tbl[i]),
|
||||||
|
limit - sglist->size));
|
||||||
if (sglist->size > INT32_MAX) {
|
if (sglist->size > INT32_MAX) {
|
||||||
error_report("AHCI Physical Region Descriptor Table describes "
|
error_report("AHCI Physical Region Descriptor Table describes "
|
||||||
"more than 2 GiB.\n");
|
"more than 2 GiB.\n");
|
||||||
@ -899,28 +886,25 @@ out:
|
|||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ncq_cb(void *opaque, int ret)
|
static void ncq_err(NCQTransferState *ncq_tfs)
|
||||||
{
|
{
|
||||||
NCQTransferState *ncq_tfs = (NCQTransferState *)opaque;
|
|
||||||
IDEState *ide_state = &ncq_tfs->drive->port.ifs[0];
|
IDEState *ide_state = &ncq_tfs->drive->port.ifs[0];
|
||||||
|
|
||||||
if (ret == -ECANCELED) {
|
ide_state->error = ABRT_ERR;
|
||||||
return;
|
ide_state->status = READY_STAT | ERR_STAT;
|
||||||
}
|
ncq_tfs->drive->port_regs.scr_err |= (1 << ncq_tfs->tag);
|
||||||
/* Clear bit for this tag in SActive */
|
}
|
||||||
ncq_tfs->drive->port_regs.scr_act &= ~(1 << ncq_tfs->tag);
|
|
||||||
|
|
||||||
if (ret < 0) {
|
static void ncq_finish(NCQTransferState *ncq_tfs)
|
||||||
/* error */
|
{
|
||||||
ide_state->error = ABRT_ERR;
|
/* If we didn't error out, set our finished bit. Errored commands
|
||||||
ide_state->status = READY_STAT | ERR_STAT;
|
* do not get a bit set for the SDB FIS ACT register, nor do they
|
||||||
ncq_tfs->drive->port_regs.scr_err |= (1 << ncq_tfs->tag);
|
* clear the outstanding bit in scr_act (PxSACT). */
|
||||||
} else {
|
if (!(ncq_tfs->drive->port_regs.scr_err & (1 << ncq_tfs->tag))) {
|
||||||
ide_state->status = READY_STAT | SEEK_STAT;
|
ncq_tfs->drive->finished |= (1 << ncq_tfs->tag);
|
||||||
}
|
}
|
||||||
|
|
||||||
ahci_write_fis_sdb(ncq_tfs->drive->hba, ncq_tfs->drive->port_no,
|
ahci_write_fis_sdb(ncq_tfs->drive->hba, ncq_tfs);
|
||||||
(1 << ncq_tfs->tag));
|
|
||||||
|
|
||||||
DPRINTF(ncq_tfs->drive->port_no, "NCQ transfer tag %d finished\n",
|
DPRINTF(ncq_tfs->drive->port_no, "NCQ transfer tag %d finished\n",
|
||||||
ncq_tfs->tag);
|
ncq_tfs->tag);
|
||||||
@ -931,6 +915,35 @@ static void ncq_cb(void *opaque, int ret)
|
|||||||
ncq_tfs->used = 0;
|
ncq_tfs->used = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void ncq_cb(void *opaque, int ret)
|
||||||
|
{
|
||||||
|
NCQTransferState *ncq_tfs = (NCQTransferState *)opaque;
|
||||||
|
IDEState *ide_state = &ncq_tfs->drive->port.ifs[0];
|
||||||
|
|
||||||
|
if (ret == -ECANCELED) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ret < 0) {
|
||||||
|
bool is_read = ncq_tfs->cmd == READ_FPDMA_QUEUED;
|
||||||
|
BlockErrorAction action = blk_get_error_action(ide_state->blk,
|
||||||
|
is_read, -ret);
|
||||||
|
if (action == BLOCK_ERROR_ACTION_STOP) {
|
||||||
|
ncq_tfs->halt = true;
|
||||||
|
ide_state->bus->error_status = IDE_RETRY_HBA;
|
||||||
|
} else if (action == BLOCK_ERROR_ACTION_REPORT) {
|
||||||
|
ncq_err(ncq_tfs);
|
||||||
|
}
|
||||||
|
blk_error_action(ide_state->blk, action, is_read, -ret);
|
||||||
|
} else {
|
||||||
|
ide_state->status = READY_STAT | SEEK_STAT;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!ncq_tfs->halt) {
|
||||||
|
ncq_finish(ncq_tfs);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static int is_ncq(uint8_t ata_cmd)
|
static int is_ncq(uint8_t ata_cmd)
|
||||||
{
|
{
|
||||||
/* Based on SATA 3.2 section 13.6.3.2 */
|
/* Based on SATA 3.2 section 13.6.3.2 */
|
||||||
@ -946,13 +959,60 @@ static int is_ncq(uint8_t ata_cmd)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void process_ncq_command(AHCIState *s, int port, uint8_t *cmd_fis,
|
static void execute_ncq_command(NCQTransferState *ncq_tfs)
|
||||||
int slot)
|
|
||||||
{
|
{
|
||||||
|
AHCIDevice *ad = ncq_tfs->drive;
|
||||||
|
IDEState *ide_state = &ad->port.ifs[0];
|
||||||
|
int port = ad->port_no;
|
||||||
|
|
||||||
|
g_assert(is_ncq(ncq_tfs->cmd));
|
||||||
|
ncq_tfs->halt = false;
|
||||||
|
|
||||||
|
switch (ncq_tfs->cmd) {
|
||||||
|
case READ_FPDMA_QUEUED:
|
||||||
|
DPRINTF(port, "NCQ reading %d sectors from LBA %"PRId64", tag %d\n",
|
||||||
|
ncq_tfs->sector_count, ncq_tfs->lba, ncq_tfs->tag);
|
||||||
|
|
||||||
|
DPRINTF(port, "tag %d aio read %"PRId64"\n",
|
||||||
|
ncq_tfs->tag, ncq_tfs->lba);
|
||||||
|
|
||||||
|
dma_acct_start(ide_state->blk, &ncq_tfs->acct,
|
||||||
|
&ncq_tfs->sglist, BLOCK_ACCT_READ);
|
||||||
|
ncq_tfs->aiocb = dma_blk_read(ide_state->blk, &ncq_tfs->sglist,
|
||||||
|
ncq_tfs->lba, ncq_cb, ncq_tfs);
|
||||||
|
break;
|
||||||
|
case WRITE_FPDMA_QUEUED:
|
||||||
|
DPRINTF(port, "NCQ writing %d sectors to LBA %"PRId64", tag %d\n",
|
||||||
|
ncq_tfs->sector_count, ncq_tfs->lba, ncq_tfs->tag);
|
||||||
|
|
||||||
|
DPRINTF(port, "tag %d aio write %"PRId64"\n",
|
||||||
|
ncq_tfs->tag, ncq_tfs->lba);
|
||||||
|
|
||||||
|
dma_acct_start(ide_state->blk, &ncq_tfs->acct,
|
||||||
|
&ncq_tfs->sglist, BLOCK_ACCT_WRITE);
|
||||||
|
ncq_tfs->aiocb = dma_blk_write(ide_state->blk, &ncq_tfs->sglist,
|
||||||
|
ncq_tfs->lba, ncq_cb, ncq_tfs);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
DPRINTF(port, "error: unsupported NCQ command (0x%02x) received\n",
|
||||||
|
ncq_tfs->cmd);
|
||||||
|
qemu_sglist_destroy(&ncq_tfs->sglist);
|
||||||
|
ncq_err(ncq_tfs);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void process_ncq_command(AHCIState *s, int port, uint8_t *cmd_fis,
|
||||||
|
uint8_t slot)
|
||||||
|
{
|
||||||
|
AHCIDevice *ad = &s->dev[port];
|
||||||
|
IDEState *ide_state = &ad->port.ifs[0];
|
||||||
NCQFrame *ncq_fis = (NCQFrame*)cmd_fis;
|
NCQFrame *ncq_fis = (NCQFrame*)cmd_fis;
|
||||||
uint8_t tag = ncq_fis->tag >> 3;
|
uint8_t tag = ncq_fis->tag >> 3;
|
||||||
NCQTransferState *ncq_tfs = &s->dev[port].ncq_tfs[tag];
|
NCQTransferState *ncq_tfs = &ad->ncq_tfs[tag];
|
||||||
|
size_t size;
|
||||||
|
|
||||||
|
g_assert(is_ncq(ncq_fis->command));
|
||||||
if (ncq_tfs->used) {
|
if (ncq_tfs->used) {
|
||||||
/* error - already in use */
|
/* error - already in use */
|
||||||
fprintf(stderr, "%s: tag %d already used\n", __FUNCTION__, tag);
|
fprintf(stderr, "%s: tag %d already used\n", __FUNCTION__, tag);
|
||||||
@ -960,75 +1020,82 @@ static void process_ncq_command(AHCIState *s, int port, uint8_t *cmd_fis,
|
|||||||
}
|
}
|
||||||
|
|
||||||
ncq_tfs->used = 1;
|
ncq_tfs->used = 1;
|
||||||
ncq_tfs->drive = &s->dev[port];
|
ncq_tfs->drive = ad;
|
||||||
ncq_tfs->slot = slot;
|
ncq_tfs->slot = slot;
|
||||||
|
ncq_tfs->cmdh = &((AHCICmdHdr *)ad->lst)[slot];
|
||||||
|
ncq_tfs->cmd = ncq_fis->command;
|
||||||
ncq_tfs->lba = ((uint64_t)ncq_fis->lba5 << 40) |
|
ncq_tfs->lba = ((uint64_t)ncq_fis->lba5 << 40) |
|
||||||
((uint64_t)ncq_fis->lba4 << 32) |
|
((uint64_t)ncq_fis->lba4 << 32) |
|
||||||
((uint64_t)ncq_fis->lba3 << 24) |
|
((uint64_t)ncq_fis->lba3 << 24) |
|
||||||
((uint64_t)ncq_fis->lba2 << 16) |
|
((uint64_t)ncq_fis->lba2 << 16) |
|
||||||
((uint64_t)ncq_fis->lba1 << 8) |
|
((uint64_t)ncq_fis->lba1 << 8) |
|
||||||
(uint64_t)ncq_fis->lba0;
|
(uint64_t)ncq_fis->lba0;
|
||||||
|
ncq_tfs->tag = tag;
|
||||||
|
|
||||||
/* Note: We calculate the sector count, but don't currently rely on it.
|
/* Sanity-check the NCQ packet */
|
||||||
* The total size of the DMA buffer tells us the transfer size instead. */
|
if (tag != slot) {
|
||||||
ncq_tfs->sector_count = ((uint16_t)ncq_fis->sector_count_high << 8) |
|
DPRINTF(port, "Warn: NCQ slot (%d) did not match the given tag (%d)\n",
|
||||||
ncq_fis->sector_count_low;
|
slot, tag);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ncq_fis->aux0 || ncq_fis->aux1 || ncq_fis->aux2 || ncq_fis->aux3) {
|
||||||
|
DPRINTF(port, "Warn: Attempt to use NCQ auxiliary fields.\n");
|
||||||
|
}
|
||||||
|
if (ncq_fis->prio || ncq_fis->icc) {
|
||||||
|
DPRINTF(port, "Warn: Unsupported attempt to use PRIO/ICC fields\n");
|
||||||
|
}
|
||||||
|
if (ncq_fis->fua & NCQ_FIS_FUA_MASK) {
|
||||||
|
DPRINTF(port, "Warn: Unsupported attempt to use Force Unit Access\n");
|
||||||
|
}
|
||||||
|
if (ncq_fis->tag & NCQ_FIS_RARC_MASK) {
|
||||||
|
DPRINTF(port, "Warn: Unsupported attempt to use Rebuild Assist\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
ncq_tfs->sector_count = ((ncq_fis->sector_count_high << 8) |
|
||||||
|
ncq_fis->sector_count_low);
|
||||||
|
if (!ncq_tfs->sector_count) {
|
||||||
|
ncq_tfs->sector_count = 0x10000;
|
||||||
|
}
|
||||||
|
size = ncq_tfs->sector_count * 512;
|
||||||
|
ahci_populate_sglist(ad, &ncq_tfs->sglist, ncq_tfs->cmdh, size, 0);
|
||||||
|
|
||||||
|
if (ncq_tfs->sglist.size < size) {
|
||||||
|
error_report("ahci: PRDT length for NCQ command (0x%zx) "
|
||||||
|
"is smaller than the requested size (0x%zx)",
|
||||||
|
ncq_tfs->sglist.size, size);
|
||||||
|
qemu_sglist_destroy(&ncq_tfs->sglist);
|
||||||
|
ncq_err(ncq_tfs);
|
||||||
|
ahci_trigger_irq(ad->hba, ad, PORT_IRQ_OVERFLOW);
|
||||||
|
return;
|
||||||
|
} else if (ncq_tfs->sglist.size != size) {
|
||||||
|
DPRINTF(port, "Warn: PRDTL (0x%zx)"
|
||||||
|
" does not match requested size (0x%zx)",
|
||||||
|
ncq_tfs->sglist.size, size);
|
||||||
|
}
|
||||||
|
|
||||||
DPRINTF(port, "NCQ transfer LBA from %"PRId64" to %"PRId64", "
|
DPRINTF(port, "NCQ transfer LBA from %"PRId64" to %"PRId64", "
|
||||||
"drive max %"PRId64"\n",
|
"drive max %"PRId64"\n",
|
||||||
ncq_tfs->lba, ncq_tfs->lba + ncq_tfs->sector_count - 2,
|
ncq_tfs->lba, ncq_tfs->lba + ncq_tfs->sector_count - 1,
|
||||||
s->dev[port].port.ifs[0].nb_sectors - 1);
|
ide_state->nb_sectors - 1);
|
||||||
|
|
||||||
ahci_populate_sglist(&s->dev[port], &ncq_tfs->sglist, 0);
|
execute_ncq_command(ncq_tfs);
|
||||||
ncq_tfs->tag = tag;
|
}
|
||||||
|
|
||||||
switch(ncq_fis->command) {
|
static AHCICmdHdr *get_cmd_header(AHCIState *s, uint8_t port, uint8_t slot)
|
||||||
case READ_FPDMA_QUEUED:
|
{
|
||||||
DPRINTF(port, "NCQ reading %d sectors from LBA %"PRId64", "
|
if (port >= s->ports || slot >= AHCI_MAX_CMDS) {
|
||||||
"tag %d\n",
|
return NULL;
|
||||||
ncq_tfs->sector_count-1, ncq_tfs->lba, ncq_tfs->tag);
|
|
||||||
|
|
||||||
DPRINTF(port, "tag %d aio read %"PRId64"\n",
|
|
||||||
ncq_tfs->tag, ncq_tfs->lba);
|
|
||||||
|
|
||||||
dma_acct_start(ncq_tfs->drive->port.ifs[0].blk, &ncq_tfs->acct,
|
|
||||||
&ncq_tfs->sglist, BLOCK_ACCT_READ);
|
|
||||||
ncq_tfs->aiocb = dma_blk_read(ncq_tfs->drive->port.ifs[0].blk,
|
|
||||||
&ncq_tfs->sglist, ncq_tfs->lba,
|
|
||||||
ncq_cb, ncq_tfs);
|
|
||||||
break;
|
|
||||||
case WRITE_FPDMA_QUEUED:
|
|
||||||
DPRINTF(port, "NCQ writing %d sectors to LBA %"PRId64", tag %d\n",
|
|
||||||
ncq_tfs->sector_count-1, ncq_tfs->lba, ncq_tfs->tag);
|
|
||||||
|
|
||||||
DPRINTF(port, "tag %d aio write %"PRId64"\n",
|
|
||||||
ncq_tfs->tag, ncq_tfs->lba);
|
|
||||||
|
|
||||||
dma_acct_start(ncq_tfs->drive->port.ifs[0].blk, &ncq_tfs->acct,
|
|
||||||
&ncq_tfs->sglist, BLOCK_ACCT_WRITE);
|
|
||||||
ncq_tfs->aiocb = dma_blk_write(ncq_tfs->drive->port.ifs[0].blk,
|
|
||||||
&ncq_tfs->sglist, ncq_tfs->lba,
|
|
||||||
ncq_cb, ncq_tfs);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
if (is_ncq(cmd_fis[2])) {
|
|
||||||
DPRINTF(port,
|
|
||||||
"error: unsupported NCQ command (0x%02x) received\n",
|
|
||||||
cmd_fis[2]);
|
|
||||||
} else {
|
|
||||||
DPRINTF(port,
|
|
||||||
"error: tried to process non-NCQ command as NCQ\n");
|
|
||||||
}
|
|
||||||
qemu_sglist_destroy(&ncq_tfs->sglist);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return s->dev[port].lst ? &((AHCICmdHdr *)s->dev[port].lst)[slot] : NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void handle_reg_h2d_fis(AHCIState *s, int port,
|
static void handle_reg_h2d_fis(AHCIState *s, int port,
|
||||||
int slot, uint8_t *cmd_fis)
|
uint8_t slot, uint8_t *cmd_fis)
|
||||||
{
|
{
|
||||||
IDEState *ide_state = &s->dev[port].port.ifs[0];
|
IDEState *ide_state = &s->dev[port].port.ifs[0];
|
||||||
AHCICmdHdr *cmd = s->dev[port].cur_cmd;
|
AHCICmdHdr *cmd = get_cmd_header(s, port, slot);
|
||||||
uint32_t opts = le32_to_cpu(cmd->opts);
|
uint16_t opts = le16_to_cpu(cmd->opts);
|
||||||
|
|
||||||
if (cmd_fis[1] & 0x0F) {
|
if (cmd_fis[1] & 0x0F) {
|
||||||
DPRINTF(port, "Port Multiplier not supported."
|
DPRINTF(port, "Port Multiplier not supported."
|
||||||
@ -1108,7 +1175,7 @@ static void handle_reg_h2d_fis(AHCIState *s, int port,
|
|||||||
ide_exec_cmd(&s->dev[port].port, cmd_fis[2]);
|
ide_exec_cmd(&s->dev[port].port, cmd_fis[2]);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int handle_cmd(AHCIState *s, int port, int slot)
|
static int handle_cmd(AHCIState *s, int port, uint8_t slot)
|
||||||
{
|
{
|
||||||
IDEState *ide_state;
|
IDEState *ide_state;
|
||||||
uint64_t tbl_addr;
|
uint64_t tbl_addr;
|
||||||
@ -1126,7 +1193,7 @@ static int handle_cmd(AHCIState *s, int port, int slot)
|
|||||||
DPRINTF(port, "error: lst not given but cmd handled");
|
DPRINTF(port, "error: lst not given but cmd handled");
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
cmd = &((AHCICmdHdr *)s->dev[port].lst)[slot];
|
cmd = get_cmd_header(s, port, slot);
|
||||||
/* remember current slot handle for later */
|
/* remember current slot handle for later */
|
||||||
s->dev[port].cur_cmd = cmd;
|
s->dev[port].cur_cmd = cmd;
|
||||||
|
|
||||||
@ -1185,7 +1252,7 @@ static void ahci_start_transfer(IDEDMA *dma)
|
|||||||
IDEState *s = &ad->port.ifs[0];
|
IDEState *s = &ad->port.ifs[0];
|
||||||
uint32_t size = (uint32_t)(s->data_end - s->data_ptr);
|
uint32_t size = (uint32_t)(s->data_end - s->data_ptr);
|
||||||
/* write == ram -> device */
|
/* write == ram -> device */
|
||||||
uint32_t opts = le32_to_cpu(ad->cur_cmd->opts);
|
uint16_t opts = le16_to_cpu(ad->cur_cmd->opts);
|
||||||
int is_write = opts & AHCI_CMD_WRITE;
|
int is_write = opts & AHCI_CMD_WRITE;
|
||||||
int is_atapi = opts & AHCI_CMD_ATAPI;
|
int is_atapi = opts & AHCI_CMD_ATAPI;
|
||||||
int has_sglist = 0;
|
int has_sglist = 0;
|
||||||
@ -1197,7 +1264,7 @@ static void ahci_start_transfer(IDEDMA *dma)
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ahci_dma_prepare_buf(dma, is_write)) {
|
if (ahci_dma_prepare_buf(dma, size)) {
|
||||||
has_sglist = 1;
|
has_sglist = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1242,17 +1309,35 @@ static void ahci_restart_dma(IDEDMA *dma)
|
|||||||
/* Nothing to do, ahci_start_dma already resets s->io_buffer_offset. */
|
/* Nothing to do, ahci_start_dma already resets s->io_buffer_offset. */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* IDE/PIO restarts are handled by the core layer, but NCQ commands
|
||||||
|
* need an extra kick from the AHCI HBA.
|
||||||
|
*/
|
||||||
|
static void ahci_restart(IDEDMA *dma)
|
||||||
|
{
|
||||||
|
AHCIDevice *ad = DO_UPCAST(AHCIDevice, dma, dma);
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < AHCI_MAX_CMDS; i++) {
|
||||||
|
NCQTransferState *ncq_tfs = &ad->ncq_tfs[i];
|
||||||
|
if (ncq_tfs->halt) {
|
||||||
|
execute_ncq_command(ncq_tfs);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Called in DMA R/W chains to read the PRDT, utilizing ahci_populate_sglist.
|
* Called in DMA R/W chains to read the PRDT, utilizing ahci_populate_sglist.
|
||||||
* Not currently invoked by PIO R/W chains,
|
* Not currently invoked by PIO R/W chains,
|
||||||
* which invoke ahci_populate_sglist via ahci_start_transfer.
|
* which invoke ahci_populate_sglist via ahci_start_transfer.
|
||||||
*/
|
*/
|
||||||
static int32_t ahci_dma_prepare_buf(IDEDMA *dma, int is_write)
|
static int32_t ahci_dma_prepare_buf(IDEDMA *dma, int32_t limit)
|
||||||
{
|
{
|
||||||
AHCIDevice *ad = DO_UPCAST(AHCIDevice, dma, dma);
|
AHCIDevice *ad = DO_UPCAST(AHCIDevice, dma, dma);
|
||||||
IDEState *s = &ad->port.ifs[0];
|
IDEState *s = &ad->port.ifs[0];
|
||||||
|
|
||||||
if (ahci_populate_sglist(ad, &s->sg, s->io_buffer_offset) == -1) {
|
if (ahci_populate_sglist(ad, &s->sg, ad->cur_cmd,
|
||||||
|
limit, s->io_buffer_offset) == -1) {
|
||||||
DPRINTF(ad->port_no, "ahci_dma_prepare_buf failed.\n");
|
DPRINTF(ad->port_no, "ahci_dma_prepare_buf failed.\n");
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@ -1287,7 +1372,7 @@ static int ahci_dma_rw_buf(IDEDMA *dma, int is_write)
|
|||||||
uint8_t *p = s->io_buffer + s->io_buffer_index;
|
uint8_t *p = s->io_buffer + s->io_buffer_index;
|
||||||
int l = s->io_buffer_size - s->io_buffer_index;
|
int l = s->io_buffer_size - s->io_buffer_index;
|
||||||
|
|
||||||
if (ahci_populate_sglist(ad, &s->sg, s->io_buffer_offset)) {
|
if (ahci_populate_sglist(ad, &s->sg, ad->cur_cmd, l, s->io_buffer_offset)) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1330,6 +1415,7 @@ static void ahci_irq_set(void *opaque, int n, int level)
|
|||||||
|
|
||||||
static const IDEDMAOps ahci_dma_ops = {
|
static const IDEDMAOps ahci_dma_ops = {
|
||||||
.start_dma = ahci_start_dma,
|
.start_dma = ahci_start_dma,
|
||||||
|
.restart = ahci_restart,
|
||||||
.restart_dma = ahci_restart_dma,
|
.restart_dma = ahci_restart_dma,
|
||||||
.start_transfer = ahci_start_transfer,
|
.start_transfer = ahci_start_transfer,
|
||||||
.prepare_buf = ahci_dma_prepare_buf,
|
.prepare_buf = ahci_dma_prepare_buf,
|
||||||
@ -1400,6 +1486,21 @@ void ahci_reset(AHCIState *s)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const VMStateDescription vmstate_ncq_tfs = {
|
||||||
|
.name = "ncq state",
|
||||||
|
.version_id = 1,
|
||||||
|
.fields = (VMStateField[]) {
|
||||||
|
VMSTATE_UINT32(sector_count, NCQTransferState),
|
||||||
|
VMSTATE_UINT64(lba, NCQTransferState),
|
||||||
|
VMSTATE_UINT8(tag, NCQTransferState),
|
||||||
|
VMSTATE_UINT8(cmd, NCQTransferState),
|
||||||
|
VMSTATE_UINT8(slot, NCQTransferState),
|
||||||
|
VMSTATE_BOOL(used, NCQTransferState),
|
||||||
|
VMSTATE_BOOL(halt, NCQTransferState),
|
||||||
|
VMSTATE_END_OF_LIST()
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
static const VMStateDescription vmstate_ahci_device = {
|
static const VMStateDescription vmstate_ahci_device = {
|
||||||
.name = "ahci port",
|
.name = "ahci port",
|
||||||
.version_id = 1,
|
.version_id = 1,
|
||||||
@ -1425,14 +1526,17 @@ static const VMStateDescription vmstate_ahci_device = {
|
|||||||
VMSTATE_BOOL(done_atapi_packet, AHCIDevice),
|
VMSTATE_BOOL(done_atapi_packet, AHCIDevice),
|
||||||
VMSTATE_INT32(busy_slot, AHCIDevice),
|
VMSTATE_INT32(busy_slot, AHCIDevice),
|
||||||
VMSTATE_BOOL(init_d2h_sent, AHCIDevice),
|
VMSTATE_BOOL(init_d2h_sent, AHCIDevice),
|
||||||
|
VMSTATE_STRUCT_ARRAY(ncq_tfs, AHCIDevice, AHCI_MAX_CMDS,
|
||||||
|
1, vmstate_ncq_tfs, NCQTransferState),
|
||||||
VMSTATE_END_OF_LIST()
|
VMSTATE_END_OF_LIST()
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
static int ahci_state_post_load(void *opaque, int version_id)
|
static int ahci_state_post_load(void *opaque, int version_id)
|
||||||
{
|
{
|
||||||
int i;
|
int i, j;
|
||||||
struct AHCIDevice *ad;
|
struct AHCIDevice *ad;
|
||||||
|
NCQTransferState *ncq_tfs;
|
||||||
AHCIState *s = opaque;
|
AHCIState *s = opaque;
|
||||||
|
|
||||||
for (i = 0; i < s->ports; i++) {
|
for (i = 0; i < s->ports; i++) {
|
||||||
@ -1444,6 +1548,37 @@ static int ahci_state_post_load(void *opaque, int version_id)
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (j = 0; j < AHCI_MAX_CMDS; j++) {
|
||||||
|
ncq_tfs = &ad->ncq_tfs[j];
|
||||||
|
ncq_tfs->drive = ad;
|
||||||
|
|
||||||
|
if (ncq_tfs->used != ncq_tfs->halt) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (!ncq_tfs->halt) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (!is_ncq(ncq_tfs->cmd)) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (ncq_tfs->slot != ncq_tfs->tag) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
/* If ncq_tfs->halt is justly set, the engine should be engaged,
|
||||||
|
* and the command list buffer should be mapped. */
|
||||||
|
ncq_tfs->cmdh = get_cmd_header(s, i, ncq_tfs->slot);
|
||||||
|
if (!ncq_tfs->cmdh) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
ahci_populate_sglist(ncq_tfs->drive, &ncq_tfs->sglist,
|
||||||
|
ncq_tfs->cmdh, ncq_tfs->sector_count * 512,
|
||||||
|
0);
|
||||||
|
if (ncq_tfs->sector_count != ncq_tfs->sglist.size >> 9) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If an error is present, ad->busy_slot will be valid and not -1.
|
* If an error is present, ad->busy_slot will be valid and not -1.
|
||||||
* In this case, an operation is waiting to resume and will re-check
|
* In this case, an operation is waiting to resume and will re-check
|
||||||
@ -1460,7 +1595,7 @@ static int ahci_state_post_load(void *opaque, int version_id)
|
|||||||
if (ad->busy_slot < 0 || ad->busy_slot >= AHCI_MAX_CMDS) {
|
if (ad->busy_slot < 0 || ad->busy_slot >= AHCI_MAX_CMDS) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
ad->cur_cmd = &((AHCICmdHdr *)ad->lst)[ad->busy_slot];
|
ad->cur_cmd = get_cmd_header(s, i, ad->busy_slot);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -195,6 +195,9 @@
|
|||||||
#define RECEIVE_FPDMA_QUEUED 0x65
|
#define RECEIVE_FPDMA_QUEUED 0x65
|
||||||
#define SEND_FPDMA_QUEUED 0x64
|
#define SEND_FPDMA_QUEUED 0x64
|
||||||
|
|
||||||
|
#define NCQ_FIS_FUA_MASK 0x80
|
||||||
|
#define NCQ_FIS_RARC_MASK 0x01
|
||||||
|
|
||||||
#define RES_FIS_DSFIS 0x00
|
#define RES_FIS_DSFIS 0x00
|
||||||
#define RES_FIS_PSFIS 0x20
|
#define RES_FIS_PSFIS 0x20
|
||||||
#define RES_FIS_RFIS 0x40
|
#define RES_FIS_RFIS 0x40
|
||||||
@ -233,7 +236,8 @@ typedef struct AHCIPortRegs {
|
|||||||
} AHCIPortRegs;
|
} AHCIPortRegs;
|
||||||
|
|
||||||
typedef struct AHCICmdHdr {
|
typedef struct AHCICmdHdr {
|
||||||
uint32_t opts;
|
uint16_t opts;
|
||||||
|
uint16_t prdtl;
|
||||||
uint32_t status;
|
uint32_t status;
|
||||||
uint64_t tbl_addr;
|
uint64_t tbl_addr;
|
||||||
uint32_t reserved[4];
|
uint32_t reserved[4];
|
||||||
@ -250,13 +254,16 @@ typedef struct AHCIDevice AHCIDevice;
|
|||||||
typedef struct NCQTransferState {
|
typedef struct NCQTransferState {
|
||||||
AHCIDevice *drive;
|
AHCIDevice *drive;
|
||||||
BlockAIOCB *aiocb;
|
BlockAIOCB *aiocb;
|
||||||
|
AHCICmdHdr *cmdh;
|
||||||
QEMUSGList sglist;
|
QEMUSGList sglist;
|
||||||
BlockAcctCookie acct;
|
BlockAcctCookie acct;
|
||||||
uint16_t sector_count;
|
uint32_t sector_count;
|
||||||
uint64_t lba;
|
uint64_t lba;
|
||||||
uint8_t tag;
|
uint8_t tag;
|
||||||
int slot;
|
uint8_t cmd;
|
||||||
int used;
|
uint8_t slot;
|
||||||
|
bool used;
|
||||||
|
bool halt;
|
||||||
} NCQTransferState;
|
} NCQTransferState;
|
||||||
|
|
||||||
struct AHCIDevice {
|
struct AHCIDevice {
|
||||||
@ -312,27 +319,39 @@ extern const VMStateDescription vmstate_ahci;
|
|||||||
.offset = vmstate_offset_value(_state, _field, AHCIState), \
|
.offset = vmstate_offset_value(_state, _field, AHCIState), \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* NCQFrame is the same as a Register H2D FIS (described in SATA 3.2),
|
||||||
|
* but some fields have been re-mapped and re-purposed, as seen in
|
||||||
|
* SATA 3.2 section 13.6.4.1 ("READ FPDMA QUEUED")
|
||||||
|
*
|
||||||
|
* cmd_fis[3], feature 7:0, becomes sector count 7:0.
|
||||||
|
* cmd_fis[7], device 7:0, uses bit 7 as the Force Unit Access bit.
|
||||||
|
* cmd_fis[11], feature 15:8, becomes sector count 15:8.
|
||||||
|
* cmd_fis[12], count 7:0, becomes the NCQ TAG (7:3) and RARC bit (0)
|
||||||
|
* cmd_fis[13], count 15:8, becomes the priority value (7:6)
|
||||||
|
* bytes 16-19 become an le32 "auxiliary" field.
|
||||||
|
*/
|
||||||
typedef struct NCQFrame {
|
typedef struct NCQFrame {
|
||||||
uint8_t fis_type;
|
uint8_t fis_type;
|
||||||
uint8_t c;
|
uint8_t c;
|
||||||
uint8_t command;
|
uint8_t command;
|
||||||
uint8_t sector_count_low;
|
uint8_t sector_count_low; /* (feature 7:0) */
|
||||||
uint8_t lba0;
|
uint8_t lba0;
|
||||||
uint8_t lba1;
|
uint8_t lba1;
|
||||||
uint8_t lba2;
|
uint8_t lba2;
|
||||||
uint8_t fua;
|
uint8_t fua; /* (device 7:0) */
|
||||||
uint8_t lba3;
|
uint8_t lba3;
|
||||||
uint8_t lba4;
|
uint8_t lba4;
|
||||||
uint8_t lba5;
|
uint8_t lba5;
|
||||||
uint8_t sector_count_high;
|
uint8_t sector_count_high; /* (feature 15:8) */
|
||||||
uint8_t tag;
|
uint8_t tag; /* (count 0:7) */
|
||||||
uint8_t reserved5;
|
uint8_t prio; /* (count 15:8) */
|
||||||
uint8_t reserved6;
|
uint8_t icc;
|
||||||
uint8_t control;
|
uint8_t control;
|
||||||
uint8_t reserved7;
|
uint8_t aux0;
|
||||||
uint8_t reserved8;
|
uint8_t aux1;
|
||||||
uint8_t reserved9;
|
uint8_t aux2;
|
||||||
uint8_t reserved10;
|
uint8_t aux3;
|
||||||
} QEMU_PACKED NCQFrame;
|
} QEMU_PACKED NCQFrame;
|
||||||
|
|
||||||
typedef struct SDBFIS {
|
typedef struct SDBFIS {
|
||||||
|
@ -716,8 +716,8 @@ static void ide_dma_cb(void *opaque, int ret)
|
|||||||
|
|
||||||
sector_num = ide_get_sector(s);
|
sector_num = ide_get_sector(s);
|
||||||
if (n > 0) {
|
if (n > 0) {
|
||||||
assert(s->io_buffer_size == s->sg.size);
|
assert(n * 512 == s->sg.size);
|
||||||
dma_buf_commit(s, s->io_buffer_size);
|
dma_buf_commit(s, s->sg.size);
|
||||||
sector_num += n;
|
sector_num += n;
|
||||||
ide_set_sector(s, sector_num);
|
ide_set_sector(s, sector_num);
|
||||||
s->nsector -= n;
|
s->nsector -= n;
|
||||||
@ -734,7 +734,7 @@ static void ide_dma_cb(void *opaque, int ret)
|
|||||||
n = s->nsector;
|
n = s->nsector;
|
||||||
s->io_buffer_index = 0;
|
s->io_buffer_index = 0;
|
||||||
s->io_buffer_size = n * 512;
|
s->io_buffer_size = n * 512;
|
||||||
if (s->bus->dma->ops->prepare_buf(s->bus->dma, ide_cmd_is_read(s)) < 512) {
|
if (s->bus->dma->ops->prepare_buf(s->bus->dma, s->io_buffer_size) < 512) {
|
||||||
/* The PRDs were too short. Reset the Active bit, but don't raise an
|
/* The PRDs were too short. Reset the Active bit, but don't raise an
|
||||||
* interrupt. */
|
* interrupt. */
|
||||||
s->status = READY_STAT | SEEK_STAT;
|
s->status = READY_STAT | SEEK_STAT;
|
||||||
@ -2326,7 +2326,7 @@ static void ide_nop(IDEDMA *dma)
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
static int32_t ide_nop_int32(IDEDMA *dma, int x)
|
static int32_t ide_nop_int32(IDEDMA *dma, int32_t l)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -2371,6 +2371,13 @@ static void ide_restart_bh(void *opaque)
|
|||||||
* called function can set a new error status. */
|
* called function can set a new error status. */
|
||||||
bus->error_status = 0;
|
bus->error_status = 0;
|
||||||
|
|
||||||
|
/* The HBA has generically asked to be kicked on retry */
|
||||||
|
if (error_status & IDE_RETRY_HBA) {
|
||||||
|
if (s->bus->dma->ops->restart) {
|
||||||
|
s->bus->dma->ops->restart(s->bus->dma);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (error_status & IDE_RETRY_DMA) {
|
if (error_status & IDE_RETRY_DMA) {
|
||||||
if (error_status & IDE_RETRY_TRIM) {
|
if (error_status & IDE_RETRY_TRIM) {
|
||||||
ide_restart_dma(s, IDE_DMA_TRIM);
|
ide_restart_dma(s, IDE_DMA_TRIM);
|
||||||
|
@ -324,7 +324,7 @@ typedef void EndTransferFunc(IDEState *);
|
|||||||
typedef void DMAStartFunc(IDEDMA *, IDEState *, BlockCompletionFunc *);
|
typedef void DMAStartFunc(IDEDMA *, IDEState *, BlockCompletionFunc *);
|
||||||
typedef void DMAVoidFunc(IDEDMA *);
|
typedef void DMAVoidFunc(IDEDMA *);
|
||||||
typedef int DMAIntFunc(IDEDMA *, int);
|
typedef int DMAIntFunc(IDEDMA *, int);
|
||||||
typedef int32_t DMAInt32Func(IDEDMA *, int);
|
typedef int32_t DMAInt32Func(IDEDMA *, int32_t len);
|
||||||
typedef void DMAu32Func(IDEDMA *, uint32_t);
|
typedef void DMAu32Func(IDEDMA *, uint32_t);
|
||||||
typedef void DMAStopFunc(IDEDMA *, bool);
|
typedef void DMAStopFunc(IDEDMA *, bool);
|
||||||
typedef void DMARestartFunc(void *, int, RunState);
|
typedef void DMARestartFunc(void *, int, RunState);
|
||||||
@ -436,6 +436,7 @@ struct IDEDMAOps {
|
|||||||
DMAInt32Func *prepare_buf;
|
DMAInt32Func *prepare_buf;
|
||||||
DMAu32Func *commit_buf;
|
DMAu32Func *commit_buf;
|
||||||
DMAIntFunc *rw_buf;
|
DMAIntFunc *rw_buf;
|
||||||
|
DMAVoidFunc *restart;
|
||||||
DMAVoidFunc *restart_dma;
|
DMAVoidFunc *restart_dma;
|
||||||
DMAStopFunc *set_inactive;
|
DMAStopFunc *set_inactive;
|
||||||
DMAVoidFunc *cmd_done;
|
DMAVoidFunc *cmd_done;
|
||||||
@ -499,6 +500,7 @@ struct IDEDevice {
|
|||||||
#define IDE_RETRY_READ 0x20
|
#define IDE_RETRY_READ 0x20
|
||||||
#define IDE_RETRY_FLUSH 0x40
|
#define IDE_RETRY_FLUSH 0x40
|
||||||
#define IDE_RETRY_TRIM 0x80
|
#define IDE_RETRY_TRIM 0x80
|
||||||
|
#define IDE_RETRY_HBA 0x100
|
||||||
|
|
||||||
static inline IDEState *idebus_active_if(IDEBus *bus)
|
static inline IDEState *idebus_active_if(IDEBus *bus)
|
||||||
{
|
{
|
||||||
|
@ -499,7 +499,7 @@ static int ide_nop_int(IDEDMA *dma, int x)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int32_t ide_nop_int32(IDEDMA *dma, int x)
|
static int32_t ide_nop_int32(IDEDMA *dma, int32_t l)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
21
hw/ide/pci.c
21
hw/ide/pci.c
@ -53,10 +53,14 @@ static void bmdma_start_dma(IDEDMA *dma, IDEState *s,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return the number of bytes successfully prepared.
|
* Prepare an sglist based on available PRDs.
|
||||||
* -1 on error.
|
* @limit: How many bytes to prepare total.
|
||||||
|
*
|
||||||
|
* Returns the number of bytes prepared, -1 on error.
|
||||||
|
* IDEState.io_buffer_size will contain the number of bytes described
|
||||||
|
* by the PRDs, whether or not we added them to the sglist.
|
||||||
*/
|
*/
|
||||||
static int32_t bmdma_prepare_buf(IDEDMA *dma, int is_write)
|
static int32_t bmdma_prepare_buf(IDEDMA *dma, int32_t limit)
|
||||||
{
|
{
|
||||||
BMDMAState *bm = DO_UPCAST(BMDMAState, dma, dma);
|
BMDMAState *bm = DO_UPCAST(BMDMAState, dma, dma);
|
||||||
IDEState *s = bmdma_active_if(bm);
|
IDEState *s = bmdma_active_if(bm);
|
||||||
@ -75,7 +79,7 @@ static int32_t bmdma_prepare_buf(IDEDMA *dma, int is_write)
|
|||||||
/* end of table (with a fail safe of one page) */
|
/* end of table (with a fail safe of one page) */
|
||||||
if (bm->cur_prd_last ||
|
if (bm->cur_prd_last ||
|
||||||
(bm->cur_addr - bm->addr) >= BMDMA_PAGE_SIZE) {
|
(bm->cur_addr - bm->addr) >= BMDMA_PAGE_SIZE) {
|
||||||
return s->io_buffer_size;
|
return s->sg.size;
|
||||||
}
|
}
|
||||||
pci_dma_read(pci_dev, bm->cur_addr, &prd, 8);
|
pci_dma_read(pci_dev, bm->cur_addr, &prd, 8);
|
||||||
bm->cur_addr += 8;
|
bm->cur_addr += 8;
|
||||||
@ -90,7 +94,14 @@ static int32_t bmdma_prepare_buf(IDEDMA *dma, int is_write)
|
|||||||
}
|
}
|
||||||
l = bm->cur_prd_len;
|
l = bm->cur_prd_len;
|
||||||
if (l > 0) {
|
if (l > 0) {
|
||||||
qemu_sglist_add(&s->sg, bm->cur_prd_addr, l);
|
uint64_t sg_len;
|
||||||
|
|
||||||
|
/* Don't add extra bytes to the SGList; consume any remaining
|
||||||
|
* PRDs from the guest, but ignore them. */
|
||||||
|
sg_len = MIN(limit - s->sg.size, bm->cur_prd_len);
|
||||||
|
if (sg_len) {
|
||||||
|
qemu_sglist_add(&s->sg, bm->cur_prd_addr, sg_len);
|
||||||
|
}
|
||||||
|
|
||||||
/* Note: We limit the max transfer to be 2GiB.
|
/* Note: We limit the max transfer to be 2GiB.
|
||||||
* This should accommodate the largest ATA transaction
|
* This should accommodate the largest ATA transaction
|
||||||
|
@ -228,6 +228,8 @@ static AHCIQState *ahci_boot_and_enable(const char *cli, ...)
|
|||||||
{
|
{
|
||||||
AHCIQState *ahci;
|
AHCIQState *ahci;
|
||||||
va_list ap;
|
va_list ap;
|
||||||
|
uint16_t buff[256];
|
||||||
|
uint8_t port;
|
||||||
|
|
||||||
if (cli) {
|
if (cli) {
|
||||||
va_start(ap, cli);
|
va_start(ap, cli);
|
||||||
@ -239,6 +241,10 @@ static AHCIQState *ahci_boot_and_enable(const char *cli, ...)
|
|||||||
|
|
||||||
ahci_pci_enable(ahci);
|
ahci_pci_enable(ahci);
|
||||||
ahci_hba_enable(ahci);
|
ahci_hba_enable(ahci);
|
||||||
|
/* Initialize test device */
|
||||||
|
port = ahci_port_select(ahci);
|
||||||
|
ahci_port_clear(ahci, port);
|
||||||
|
ahci_io(ahci, port, CMD_IDENTIFY, &buff, sizeof(buff), 0);
|
||||||
|
|
||||||
return ahci;
|
return ahci;
|
||||||
}
|
}
|
||||||
@ -890,21 +896,23 @@ static void ahci_test_io_rw_simple(AHCIQState *ahci, unsigned bufsize,
|
|||||||
g_free(rx);
|
g_free(rx);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ahci_test_nondata(AHCIQState *ahci, uint8_t ide_cmd)
|
static uint8_t ahci_test_nondata(AHCIQState *ahci, uint8_t ide_cmd)
|
||||||
{
|
{
|
||||||
uint8_t px;
|
uint8_t port;
|
||||||
AHCICommand *cmd;
|
AHCICommand *cmd;
|
||||||
|
|
||||||
/* Sanitize */
|
/* Sanitize */
|
||||||
px = ahci_port_select(ahci);
|
port = ahci_port_select(ahci);
|
||||||
ahci_port_clear(ahci, px);
|
ahci_port_clear(ahci, port);
|
||||||
|
|
||||||
/* Issue Command */
|
/* Issue Command */
|
||||||
cmd = ahci_command_create(ide_cmd);
|
cmd = ahci_command_create(ide_cmd);
|
||||||
ahci_command_commit(ahci, cmd, px);
|
ahci_command_commit(ahci, cmd, port);
|
||||||
ahci_command_issue(ahci, cmd);
|
ahci_command_issue(ahci, cmd);
|
||||||
ahci_command_verify(ahci, cmd);
|
ahci_command_verify(ahci, cmd);
|
||||||
ahci_command_free(cmd);
|
ahci_command_free(cmd);
|
||||||
|
|
||||||
|
return port;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ahci_test_flush(AHCIQState *ahci)
|
static void ahci_test_flush(AHCIQState *ahci)
|
||||||
@ -912,6 +920,33 @@ static void ahci_test_flush(AHCIQState *ahci)
|
|||||||
ahci_test_nondata(ahci, CMD_FLUSH_CACHE);
|
ahci_test_nondata(ahci, CMD_FLUSH_CACHE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void ahci_test_max(AHCIQState *ahci)
|
||||||
|
{
|
||||||
|
RegD2HFIS *d2h = g_malloc0(0x20);
|
||||||
|
uint64_t nsect;
|
||||||
|
uint8_t port;
|
||||||
|
uint8_t cmd;
|
||||||
|
uint64_t config_sect = TEST_IMAGE_SECTORS - 1;
|
||||||
|
|
||||||
|
if (config_sect > 0xFFFFFF) {
|
||||||
|
cmd = CMD_READ_MAX_EXT;
|
||||||
|
} else {
|
||||||
|
cmd = CMD_READ_MAX;
|
||||||
|
}
|
||||||
|
|
||||||
|
port = ahci_test_nondata(ahci, cmd);
|
||||||
|
memread(ahci->port[port].fb + 0x40, d2h, 0x20);
|
||||||
|
nsect = (uint64_t)d2h->lba_hi[2] << 40 |
|
||||||
|
(uint64_t)d2h->lba_hi[1] << 32 |
|
||||||
|
(uint64_t)d2h->lba_hi[0] << 24 |
|
||||||
|
(uint64_t)d2h->lba_lo[2] << 16 |
|
||||||
|
(uint64_t)d2h->lba_lo[1] << 8 |
|
||||||
|
(uint64_t)d2h->lba_lo[0];
|
||||||
|
|
||||||
|
g_assert_cmphex(nsect, ==, config_sect);
|
||||||
|
g_free(d2h);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/******************************************************************************/
|
/******************************************************************************/
|
||||||
/* Test Interfaces */
|
/* Test Interfaces */
|
||||||
@ -1111,9 +1146,9 @@ static void test_migrate_sanity(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* DMA Migration test: Write a pattern, migrate, then read.
|
* Simple migration test: Write a pattern, migrate, then read.
|
||||||
*/
|
*/
|
||||||
static void test_migrate_dma(void)
|
static void ahci_migrate_simple(uint8_t cmd_read, uint8_t cmd_write)
|
||||||
{
|
{
|
||||||
AHCIQState *src, *dst;
|
AHCIQState *src, *dst;
|
||||||
uint8_t px;
|
uint8_t px;
|
||||||
@ -1141,9 +1176,9 @@ static void test_migrate_dma(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Write, migrate, then read. */
|
/* Write, migrate, then read. */
|
||||||
ahci_io(src, px, CMD_WRITE_DMA, tx, bufsize, 0);
|
ahci_io(src, px, cmd_write, tx, bufsize, 0);
|
||||||
ahci_migrate(src, dst, uri);
|
ahci_migrate(src, dst, uri);
|
||||||
ahci_io(dst, px, CMD_READ_DMA, rx, bufsize, 0);
|
ahci_io(dst, px, cmd_read, rx, bufsize, 0);
|
||||||
|
|
||||||
/* Verify pattern */
|
/* Verify pattern */
|
||||||
g_assert_cmphex(memcmp(tx, rx, bufsize), ==, 0);
|
g_assert_cmphex(memcmp(tx, rx, bufsize), ==, 0);
|
||||||
@ -1154,14 +1189,24 @@ static void test_migrate_dma(void)
|
|||||||
g_free(tx);
|
g_free(tx);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void test_migrate_dma(void)
|
||||||
|
{
|
||||||
|
ahci_migrate_simple(CMD_READ_DMA, CMD_WRITE_DMA);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void test_migrate_ncq(void)
|
||||||
|
{
|
||||||
|
ahci_migrate_simple(READ_FPDMA_QUEUED, WRITE_FPDMA_QUEUED);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* DMA Error Test
|
* Halted IO Error Test
|
||||||
*
|
*
|
||||||
* Simulate an error on first write, Try to write a pattern,
|
* Simulate an error on first write, Try to write a pattern,
|
||||||
* Confirm the VM has stopped, resume the VM, verify command
|
* Confirm the VM has stopped, resume the VM, verify command
|
||||||
* has completed, then read back the data and verify.
|
* has completed, then read back the data and verify.
|
||||||
*/
|
*/
|
||||||
static void test_halted_dma(void)
|
static void ahci_halted_io_test(uint8_t cmd_read, uint8_t cmd_write)
|
||||||
{
|
{
|
||||||
AHCIQState *ahci;
|
AHCIQState *ahci;
|
||||||
uint8_t port;
|
uint8_t port;
|
||||||
@ -1196,7 +1241,7 @@ static void test_halted_dma(void)
|
|||||||
memwrite(ptr, tx, bufsize);
|
memwrite(ptr, tx, bufsize);
|
||||||
|
|
||||||
/* Attempt to write (and fail) */
|
/* Attempt to write (and fail) */
|
||||||
cmd = ahci_guest_io_halt(ahci, port, CMD_WRITE_DMA,
|
cmd = ahci_guest_io_halt(ahci, port, cmd_write,
|
||||||
ptr, bufsize, 0);
|
ptr, bufsize, 0);
|
||||||
|
|
||||||
/* Attempt to resume the command */
|
/* Attempt to resume the command */
|
||||||
@ -1204,7 +1249,7 @@ static void test_halted_dma(void)
|
|||||||
ahci_free(ahci, ptr);
|
ahci_free(ahci, ptr);
|
||||||
|
|
||||||
/* Read back and verify */
|
/* Read back and verify */
|
||||||
ahci_io(ahci, port, CMD_READ_DMA, rx, bufsize, 0);
|
ahci_io(ahci, port, cmd_read, rx, bufsize, 0);
|
||||||
g_assert_cmphex(memcmp(tx, rx, bufsize), ==, 0);
|
g_assert_cmphex(memcmp(tx, rx, bufsize), ==, 0);
|
||||||
|
|
||||||
/* Cleanup and go home */
|
/* Cleanup and go home */
|
||||||
@ -1213,14 +1258,24 @@ static void test_halted_dma(void)
|
|||||||
g_free(tx);
|
g_free(tx);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void test_halted_dma(void)
|
||||||
|
{
|
||||||
|
ahci_halted_io_test(CMD_READ_DMA, CMD_WRITE_DMA);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void test_halted_ncq(void)
|
||||||
|
{
|
||||||
|
ahci_halted_io_test(READ_FPDMA_QUEUED, WRITE_FPDMA_QUEUED);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* DMA Error Migration Test
|
* IO Error Migration Test
|
||||||
*
|
*
|
||||||
* Simulate an error on first write, Try to write a pattern,
|
* Simulate an error on first write, Try to write a pattern,
|
||||||
* Confirm the VM has stopped, migrate, resume the VM,
|
* Confirm the VM has stopped, migrate, resume the VM,
|
||||||
* verify command has completed, then read back the data and verify.
|
* verify command has completed, then read back the data and verify.
|
||||||
*/
|
*/
|
||||||
static void test_migrate_halted_dma(void)
|
static void ahci_migrate_halted_io(uint8_t cmd_read, uint8_t cmd_write)
|
||||||
{
|
{
|
||||||
AHCIQState *src, *dst;
|
AHCIQState *src, *dst;
|
||||||
uint8_t port;
|
uint8_t port;
|
||||||
@ -1266,14 +1321,14 @@ static void test_migrate_halted_dma(void)
|
|||||||
memwrite(ptr, tx, bufsize);
|
memwrite(ptr, tx, bufsize);
|
||||||
|
|
||||||
/* Write, trigger the VM to stop, migrate, then resume. */
|
/* Write, trigger the VM to stop, migrate, then resume. */
|
||||||
cmd = ahci_guest_io_halt(src, port, CMD_WRITE_DMA,
|
cmd = ahci_guest_io_halt(src, port, cmd_write,
|
||||||
ptr, bufsize, 0);
|
ptr, bufsize, 0);
|
||||||
ahci_migrate(src, dst, uri);
|
ahci_migrate(src, dst, uri);
|
||||||
ahci_guest_io_resume(dst, cmd);
|
ahci_guest_io_resume(dst, cmd);
|
||||||
ahci_free(dst, ptr);
|
ahci_free(dst, ptr);
|
||||||
|
|
||||||
/* Read back */
|
/* Read back */
|
||||||
ahci_io(dst, port, CMD_READ_DMA, rx, bufsize, 0);
|
ahci_io(dst, port, cmd_read, rx, bufsize, 0);
|
||||||
|
|
||||||
/* Verify TX and RX are identical */
|
/* Verify TX and RX are identical */
|
||||||
g_assert_cmphex(memcmp(tx, rx, bufsize), ==, 0);
|
g_assert_cmphex(memcmp(tx, rx, bufsize), ==, 0);
|
||||||
@ -1285,6 +1340,16 @@ static void test_migrate_halted_dma(void)
|
|||||||
g_free(tx);
|
g_free(tx);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void test_migrate_halted_dma(void)
|
||||||
|
{
|
||||||
|
ahci_migrate_halted_io(CMD_READ_DMA, CMD_WRITE_DMA);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void test_migrate_halted_ncq(void)
|
||||||
|
{
|
||||||
|
ahci_migrate_halted_io(READ_FPDMA_QUEUED, WRITE_FPDMA_QUEUED);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Migration test: Try to flush, migrate, then resume.
|
* Migration test: Try to flush, migrate, then resume.
|
||||||
*/
|
*/
|
||||||
@ -1334,6 +1399,49 @@ static void test_flush_migrate(void)
|
|||||||
ahci_shutdown(dst);
|
ahci_shutdown(dst);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void test_max(void)
|
||||||
|
{
|
||||||
|
AHCIQState *ahci;
|
||||||
|
|
||||||
|
ahci = ahci_boot_and_enable(NULL);
|
||||||
|
ahci_test_max(ahci);
|
||||||
|
ahci_shutdown(ahci);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void test_reset(void)
|
||||||
|
{
|
||||||
|
AHCIQState *ahci;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
ahci = ahci_boot(NULL);
|
||||||
|
ahci_test_pci_spec(ahci);
|
||||||
|
ahci_pci_enable(ahci);
|
||||||
|
|
||||||
|
for (i = 0; i < 2; i++) {
|
||||||
|
ahci_test_hba_spec(ahci);
|
||||||
|
ahci_hba_enable(ahci);
|
||||||
|
ahci_test_identify(ahci);
|
||||||
|
ahci_test_io_rw_simple(ahci, 4096, 0,
|
||||||
|
CMD_READ_DMA_EXT,
|
||||||
|
CMD_WRITE_DMA_EXT);
|
||||||
|
ahci_set(ahci, AHCI_GHC, AHCI_GHC_HR);
|
||||||
|
ahci_clean_mem(ahci);
|
||||||
|
}
|
||||||
|
|
||||||
|
ahci_shutdown(ahci);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void test_ncq_simple(void)
|
||||||
|
{
|
||||||
|
AHCIQState *ahci;
|
||||||
|
|
||||||
|
ahci = ahci_boot_and_enable(NULL);
|
||||||
|
ahci_test_io_rw_simple(ahci, 4096, 0,
|
||||||
|
READ_FPDMA_QUEUED,
|
||||||
|
WRITE_FPDMA_QUEUED);
|
||||||
|
ahci_shutdown(ahci);
|
||||||
|
}
|
||||||
|
|
||||||
/******************************************************************************/
|
/******************************************************************************/
|
||||||
/* AHCI I/O Test Matrix Definitions */
|
/* AHCI I/O Test Matrix Definitions */
|
||||||
|
|
||||||
@ -1584,6 +1692,14 @@ int main(int argc, char **argv)
|
|||||||
qtest_add_func("/ahci/io/dma/lba28/retry", test_halted_dma);
|
qtest_add_func("/ahci/io/dma/lba28/retry", test_halted_dma);
|
||||||
qtest_add_func("/ahci/migrate/dma/halted", test_migrate_halted_dma);
|
qtest_add_func("/ahci/migrate/dma/halted", test_migrate_halted_dma);
|
||||||
|
|
||||||
|
qtest_add_func("/ahci/max", test_max);
|
||||||
|
qtest_add_func("/ahci/reset", test_reset);
|
||||||
|
|
||||||
|
qtest_add_func("/ahci/io/ncq/simple", test_ncq_simple);
|
||||||
|
qtest_add_func("/ahci/migrate/ncq/simple", test_migrate_ncq);
|
||||||
|
qtest_add_func("/ahci/io/ncq/retry", test_halted_ncq);
|
||||||
|
qtest_add_func("/ahci/migrate/ncq/halted", test_migrate_halted_ncq);
|
||||||
|
|
||||||
ret = g_test_run();
|
ret = g_test_run();
|
||||||
|
|
||||||
/* Cleanup */
|
/* Cleanup */
|
||||||
|
@ -50,27 +50,47 @@ typedef struct AHCICommandProp {
|
|||||||
} AHCICommandProp;
|
} AHCICommandProp;
|
||||||
|
|
||||||
AHCICommandProp ahci_command_properties[] = {
|
AHCICommandProp ahci_command_properties[] = {
|
||||||
{ .cmd = CMD_READ_PIO, .data = true, .pio = true,
|
{ .cmd = CMD_READ_PIO, .data = true, .pio = true,
|
||||||
.lba28 = true, .read = true },
|
.lba28 = true, .read = true },
|
||||||
{ .cmd = CMD_WRITE_PIO, .data = true, .pio = true,
|
{ .cmd = CMD_WRITE_PIO, .data = true, .pio = true,
|
||||||
.lba28 = true, .write = true },
|
.lba28 = true, .write = true },
|
||||||
{ .cmd = CMD_READ_PIO_EXT, .data = true, .pio = true,
|
{ .cmd = CMD_READ_PIO_EXT, .data = true, .pio = true,
|
||||||
.lba48 = true, .read = true },
|
.lba48 = true, .read = true },
|
||||||
{ .cmd = CMD_WRITE_PIO_EXT, .data = true, .pio = true,
|
{ .cmd = CMD_WRITE_PIO_EXT, .data = true, .pio = true,
|
||||||
.lba48 = true, .write = true },
|
.lba48 = true, .write = true },
|
||||||
{ .cmd = CMD_READ_DMA, .data = true, .dma = true,
|
{ .cmd = CMD_READ_DMA, .data = true, .dma = true,
|
||||||
.lba28 = true, .read = true },
|
.lba28 = true, .read = true },
|
||||||
{ .cmd = CMD_WRITE_DMA, .data = true, .dma = true,
|
{ .cmd = CMD_WRITE_DMA, .data = true, .dma = true,
|
||||||
.lba28 = true, .write = true },
|
.lba28 = true, .write = true },
|
||||||
{ .cmd = CMD_READ_DMA_EXT, .data = true, .dma = true,
|
{ .cmd = CMD_READ_DMA_EXT, .data = true, .dma = true,
|
||||||
.lba48 = true, .read = true },
|
.lba48 = true, .read = true },
|
||||||
{ .cmd = CMD_WRITE_DMA_EXT, .data = true, .dma = true,
|
{ .cmd = CMD_WRITE_DMA_EXT, .data = true, .dma = true,
|
||||||
.lba48 = true, .write = true },
|
.lba48 = true, .write = true },
|
||||||
{ .cmd = CMD_IDENTIFY, .data = true, .pio = true,
|
{ .cmd = CMD_IDENTIFY, .data = true, .pio = true,
|
||||||
.size = 512, .read = true },
|
.size = 512, .read = true },
|
||||||
{ .cmd = CMD_READ_MAX, .lba28 = true },
|
{ .cmd = READ_FPDMA_QUEUED, .data = true, .dma = true,
|
||||||
{ .cmd = CMD_READ_MAX_EXT, .lba48 = true },
|
.lba48 = true, .read = true, .ncq = true },
|
||||||
{ .cmd = CMD_FLUSH_CACHE, .data = false }
|
{ .cmd = WRITE_FPDMA_QUEUED, .data = true, .dma = true,
|
||||||
|
.lba48 = true, .write = true, .ncq = true },
|
||||||
|
{ .cmd = CMD_READ_MAX, .lba28 = true },
|
||||||
|
{ .cmd = CMD_READ_MAX_EXT, .lba48 = true },
|
||||||
|
{ .cmd = CMD_FLUSH_CACHE, .data = false }
|
||||||
|
};
|
||||||
|
|
||||||
|
struct AHCICommand {
|
||||||
|
/* Test Management Data */
|
||||||
|
uint8_t name;
|
||||||
|
uint8_t port;
|
||||||
|
uint8_t slot;
|
||||||
|
uint32_t interrupts;
|
||||||
|
uint64_t xbytes;
|
||||||
|
uint32_t prd_size;
|
||||||
|
uint64_t buffer;
|
||||||
|
AHCICommandProp *props;
|
||||||
|
/* Data to be transferred to the guest */
|
||||||
|
AHCICommandHeader header;
|
||||||
|
RegH2DFIS fis;
|
||||||
|
void *atapi_cmd;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -138,12 +158,14 @@ void ahci_clean_mem(AHCIQState *ahci)
|
|||||||
for (port = 0; port < 32; ++port) {
|
for (port = 0; port < 32; ++port) {
|
||||||
if (ahci->port[port].fb) {
|
if (ahci->port[port].fb) {
|
||||||
ahci_free(ahci, ahci->port[port].fb);
|
ahci_free(ahci, ahci->port[port].fb);
|
||||||
|
ahci->port[port].fb = 0;
|
||||||
}
|
}
|
||||||
if (ahci->port[port].clb) {
|
if (ahci->port[port].clb) {
|
||||||
for (slot = 0; slot < 32; slot++) {
|
for (slot = 0; slot < 32; slot++) {
|
||||||
ahci_destroy_command(ahci, port, slot);
|
ahci_destroy_command(ahci, port, slot);
|
||||||
}
|
}
|
||||||
ahci_free(ahci, ahci->port[port].clb);
|
ahci_free(ahci, ahci->port[port].clb);
|
||||||
|
ahci->port[port].clb = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -252,7 +274,7 @@ void ahci_hba_enable(AHCIQState *ahci)
|
|||||||
/* Allocate Memory for the Command List Buffer & FIS Buffer */
|
/* Allocate Memory for the Command List Buffer & FIS Buffer */
|
||||||
/* PxCLB space ... 0x20 per command, as in 4.2.2 p 36 */
|
/* PxCLB space ... 0x20 per command, as in 4.2.2 p 36 */
|
||||||
ahci->port[i].clb = ahci_alloc(ahci, num_cmd_slots * 0x20);
|
ahci->port[i].clb = ahci_alloc(ahci, num_cmd_slots * 0x20);
|
||||||
qmemset(ahci->port[i].clb, 0x00, 0x100);
|
qmemset(ahci->port[i].clb, 0x00, num_cmd_slots * 0x20);
|
||||||
g_test_message("CLB: 0x%08" PRIx64, ahci->port[i].clb);
|
g_test_message("CLB: 0x%08" PRIx64, ahci->port[i].clb);
|
||||||
ahci_px_wreg(ahci, i, AHCI_PX_CLB, ahci->port[i].clb);
|
ahci_px_wreg(ahci, i, AHCI_PX_CLB, ahci->port[i].clb);
|
||||||
g_assert_cmphex(ahci->port[i].clb, ==,
|
g_assert_cmphex(ahci->port[i].clb, ==,
|
||||||
@ -460,13 +482,15 @@ void ahci_port_check_pio_sanity(AHCIQState *ahci, uint8_t port,
|
|||||||
g_free(pio);
|
g_free(pio);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ahci_port_check_cmd_sanity(AHCIQState *ahci, uint8_t port,
|
void ahci_port_check_cmd_sanity(AHCIQState *ahci, AHCICommand *cmd)
|
||||||
uint8_t slot, size_t buffsize)
|
|
||||||
{
|
{
|
||||||
AHCICommandHeader cmd;
|
AHCICommandHeader cmdh;
|
||||||
|
|
||||||
ahci_get_command_header(ahci, port, slot, &cmd);
|
ahci_get_command_header(ahci, cmd->port, cmd->slot, &cmdh);
|
||||||
g_assert_cmphex(buffsize, ==, cmd.prdbc);
|
/* Physical Region Descriptor Byte Count is not required to work for NCQ. */
|
||||||
|
if (!cmd->props->ncq) {
|
||||||
|
g_assert_cmphex(cmd->xbytes, ==, cmdh.prdbc);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get the command in #slot of port #port. */
|
/* Get the command in #slot of port #port. */
|
||||||
@ -549,7 +573,7 @@ unsigned ahci_pick_cmd(AHCIQState *ahci, uint8_t port)
|
|||||||
if (reg & (1 << j)) {
|
if (reg & (1 << j)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
ahci_destroy_command(ahci, port, i);
|
ahci_destroy_command(ahci, port, j);
|
||||||
ahci->port[port].next = (j + 1) % 32;
|
ahci->port[port].next = (j + 1) % 32;
|
||||||
return j;
|
return j;
|
||||||
}
|
}
|
||||||
@ -610,22 +634,6 @@ void ahci_guest_io(AHCIQState *ahci, uint8_t port, uint8_t ide_cmd,
|
|||||||
ahci_command_free(cmd);
|
ahci_command_free(cmd);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct AHCICommand {
|
|
||||||
/* Test Management Data */
|
|
||||||
uint8_t name;
|
|
||||||
uint8_t port;
|
|
||||||
uint8_t slot;
|
|
||||||
uint32_t interrupts;
|
|
||||||
uint64_t xbytes;
|
|
||||||
uint32_t prd_size;
|
|
||||||
uint64_t buffer;
|
|
||||||
AHCICommandProp *props;
|
|
||||||
/* Data to be transferred to the guest */
|
|
||||||
AHCICommandHeader header;
|
|
||||||
RegH2DFIS fis;
|
|
||||||
void *atapi_cmd;
|
|
||||||
};
|
|
||||||
|
|
||||||
static AHCICommandProp *ahci_command_find(uint8_t command_name)
|
static AHCICommandProp *ahci_command_find(uint8_t command_name)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
@ -691,19 +699,34 @@ static void command_header_init(AHCICommand *cmd)
|
|||||||
static void command_table_init(AHCICommand *cmd)
|
static void command_table_init(AHCICommand *cmd)
|
||||||
{
|
{
|
||||||
RegH2DFIS *fis = &(cmd->fis);
|
RegH2DFIS *fis = &(cmd->fis);
|
||||||
|
uint16_t sect_count = (cmd->xbytes / AHCI_SECTOR_SIZE);
|
||||||
|
|
||||||
fis->fis_type = REG_H2D_FIS;
|
fis->fis_type = REG_H2D_FIS;
|
||||||
fis->flags = REG_H2D_FIS_CMD; /* "Command" bit */
|
fis->flags = REG_H2D_FIS_CMD; /* "Command" bit */
|
||||||
fis->command = cmd->name;
|
fis->command = cmd->name;
|
||||||
cmd->fis.feature_low = 0x00;
|
|
||||||
cmd->fis.feature_high = 0x00;
|
if (cmd->props->ncq) {
|
||||||
if (cmd->props->lba28 || cmd->props->lba48) {
|
NCQFIS *ncqfis = (NCQFIS *)fis;
|
||||||
cmd->fis.device = ATA_DEVICE_LBA;
|
/* NCQ is weird and re-uses FIS frames for unrelated data.
|
||||||
|
* See SATA 3.2, 13.6.4.1 READ FPDMA QUEUED for an example. */
|
||||||
|
ncqfis->sector_low = sect_count & 0xFF;
|
||||||
|
ncqfis->sector_hi = (sect_count >> 8) & 0xFF;
|
||||||
|
ncqfis->device = NCQ_DEVICE_MAGIC;
|
||||||
|
/* Force Unit Access is bit 7 in the device register */
|
||||||
|
ncqfis->tag = 0; /* bits 3-7 are the NCQ tag */
|
||||||
|
ncqfis->prio = 0; /* bits 6,7 are a prio tag */
|
||||||
|
/* RARC bit is bit 0 of TAG field */
|
||||||
|
} else {
|
||||||
|
fis->feature_low = 0x00;
|
||||||
|
fis->feature_high = 0x00;
|
||||||
|
if (cmd->props->lba28 || cmd->props->lba48) {
|
||||||
|
fis->device = ATA_DEVICE_LBA;
|
||||||
|
}
|
||||||
|
fis->count = (cmd->xbytes / AHCI_SECTOR_SIZE);
|
||||||
}
|
}
|
||||||
cmd->fis.count = (cmd->xbytes / AHCI_SECTOR_SIZE);
|
fis->icc = 0x00;
|
||||||
cmd->fis.icc = 0x00;
|
fis->control = 0x00;
|
||||||
cmd->fis.control = 0x00;
|
memset(fis->aux, 0x00, ARRAY_SIZE(fis->aux));
|
||||||
memset(cmd->fis.aux, 0x00, ARRAY_SIZE(cmd->fis.aux));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
AHCICommand *ahci_command_create(uint8_t command_name)
|
AHCICommand *ahci_command_create(uint8_t command_name)
|
||||||
@ -717,6 +740,7 @@ AHCICommand *ahci_command_create(uint8_t command_name)
|
|||||||
g_assert(!(props->lba28 && props->lba48));
|
g_assert(!(props->lba28 && props->lba48));
|
||||||
g_assert(!(props->read && props->write));
|
g_assert(!(props->read && props->write));
|
||||||
g_assert(!props->size || props->data);
|
g_assert(!props->size || props->data);
|
||||||
|
g_assert(!props->ncq || (props->ncq && props->lba48));
|
||||||
|
|
||||||
/* Defaults and book-keeping */
|
/* Defaults and book-keeping */
|
||||||
cmd->props = props;
|
cmd->props = props;
|
||||||
@ -725,12 +749,15 @@ AHCICommand *ahci_command_create(uint8_t command_name)
|
|||||||
cmd->prd_size = 4096;
|
cmd->prd_size = 4096;
|
||||||
cmd->buffer = 0xabad1dea;
|
cmd->buffer = 0xabad1dea;
|
||||||
|
|
||||||
cmd->interrupts = AHCI_PX_IS_DHRS;
|
if (!cmd->props->ncq) {
|
||||||
|
cmd->interrupts = AHCI_PX_IS_DHRS;
|
||||||
|
}
|
||||||
/* BUG: We expect the DPS interrupt for data commands */
|
/* BUG: We expect the DPS interrupt for data commands */
|
||||||
/* cmd->interrupts |= props->data ? AHCI_PX_IS_DPS : 0; */
|
/* cmd->interrupts |= props->data ? AHCI_PX_IS_DPS : 0; */
|
||||||
/* BUG: We expect the DMA Setup interrupt for DMA commands */
|
/* BUG: We expect the DMA Setup interrupt for DMA commands */
|
||||||
/* cmd->interrupts |= props->dma ? AHCI_PX_IS_DSS : 0; */
|
/* cmd->interrupts |= props->dma ? AHCI_PX_IS_DSS : 0; */
|
||||||
cmd->interrupts |= props->pio ? AHCI_PX_IS_PSS : 0;
|
cmd->interrupts |= props->pio ? AHCI_PX_IS_PSS : 0;
|
||||||
|
cmd->interrupts |= props->ncq ? AHCI_PX_IS_SDBS : 0;
|
||||||
|
|
||||||
command_header_init(cmd);
|
command_header_init(cmd);
|
||||||
command_table_init(cmd);
|
command_table_init(cmd);
|
||||||
@ -758,7 +785,7 @@ void ahci_command_set_offset(AHCICommand *cmd, uint64_t lba_sect)
|
|||||||
RegH2DFIS *fis = &(cmd->fis);
|
RegH2DFIS *fis = &(cmd->fis);
|
||||||
if (cmd->props->lba28) {
|
if (cmd->props->lba28) {
|
||||||
g_assert_cmphex(lba_sect, <=, 0xFFFFFFF);
|
g_assert_cmphex(lba_sect, <=, 0xFFFFFFF);
|
||||||
} else if (cmd->props->lba48) {
|
} else if (cmd->props->lba48 || cmd->props->ncq) {
|
||||||
g_assert_cmphex(lba_sect, <=, 0xFFFFFFFFFFFF);
|
g_assert_cmphex(lba_sect, <=, 0xFFFFFFFFFFFF);
|
||||||
} else {
|
} else {
|
||||||
/* Can't set offset if we don't know the format. */
|
/* Can't set offset if we don't know the format. */
|
||||||
@ -785,6 +812,8 @@ void ahci_command_set_buffer(AHCICommand *cmd, uint64_t buffer)
|
|||||||
void ahci_command_set_sizes(AHCICommand *cmd, uint64_t xbytes,
|
void ahci_command_set_sizes(AHCICommand *cmd, uint64_t xbytes,
|
||||||
unsigned prd_size)
|
unsigned prd_size)
|
||||||
{
|
{
|
||||||
|
uint16_t sect_count;
|
||||||
|
|
||||||
/* Each PRD can describe up to 4MiB, and must not be odd. */
|
/* Each PRD can describe up to 4MiB, and must not be odd. */
|
||||||
g_assert_cmphex(prd_size, <=, 4096 * 1024);
|
g_assert_cmphex(prd_size, <=, 4096 * 1024);
|
||||||
g_assert_cmphex(prd_size & 0x01, ==, 0x00);
|
g_assert_cmphex(prd_size & 0x01, ==, 0x00);
|
||||||
@ -792,7 +821,15 @@ void ahci_command_set_sizes(AHCICommand *cmd, uint64_t xbytes,
|
|||||||
cmd->prd_size = prd_size;
|
cmd->prd_size = prd_size;
|
||||||
}
|
}
|
||||||
cmd->xbytes = xbytes;
|
cmd->xbytes = xbytes;
|
||||||
cmd->fis.count = (cmd->xbytes / AHCI_SECTOR_SIZE);
|
sect_count = (cmd->xbytes / AHCI_SECTOR_SIZE);
|
||||||
|
|
||||||
|
if (cmd->props->ncq) {
|
||||||
|
NCQFIS *nfis = (NCQFIS *)&(cmd->fis);
|
||||||
|
nfis->sector_low = sect_count & 0xFF;
|
||||||
|
nfis->sector_hi = (sect_count >> 8) & 0xFF;
|
||||||
|
} else {
|
||||||
|
cmd->fis.count = sect_count;
|
||||||
|
}
|
||||||
cmd->header.prdtl = size_to_prdtl(cmd->xbytes, cmd->prd_size);
|
cmd->header.prdtl = size_to_prdtl(cmd->xbytes, cmd->prd_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -824,6 +861,11 @@ void ahci_command_commit(AHCIQState *ahci, AHCICommand *cmd, uint8_t port)
|
|||||||
cmd->port = port;
|
cmd->port = port;
|
||||||
cmd->slot = ahci_pick_cmd(ahci, port);
|
cmd->slot = ahci_pick_cmd(ahci, port);
|
||||||
|
|
||||||
|
if (cmd->props->ncq) {
|
||||||
|
NCQFIS *nfis = (NCQFIS *)&cmd->fis;
|
||||||
|
nfis->tag = (cmd->slot << 3) & 0xFC;
|
||||||
|
}
|
||||||
|
|
||||||
/* Create a buffer for the command table */
|
/* Create a buffer for the command table */
|
||||||
prdtl = size_to_prdtl(cmd->xbytes, cmd->prd_size);
|
prdtl = size_to_prdtl(cmd->xbytes, cmd->prd_size);
|
||||||
table_size = CMD_TBL_SIZ(prdtl);
|
table_size = CMD_TBL_SIZ(prdtl);
|
||||||
@ -878,11 +920,15 @@ void ahci_command_wait(AHCIQState *ahci, AHCICommand *cmd)
|
|||||||
/* We can't rely on STS_BSY until the command has started processing.
|
/* We can't rely on STS_BSY until the command has started processing.
|
||||||
* Therefore, we also use the Command Issue bit as indication of
|
* Therefore, we also use the Command Issue bit as indication of
|
||||||
* a command in-flight. */
|
* a command in-flight. */
|
||||||
while (BITSET(ahci_px_rreg(ahci, cmd->port, AHCI_PX_TFD),
|
|
||||||
AHCI_PX_TFD_STS_BSY) ||
|
#define RSET(REG, MASK) (BITSET(ahci_px_rreg(ahci, cmd->port, (REG)), (MASK)))
|
||||||
BITSET(ahci_px_rreg(ahci, cmd->port, AHCI_PX_CI), (1 << cmd->slot))) {
|
|
||||||
|
while (RSET(AHCI_PX_TFD, AHCI_PX_TFD_STS_BSY) ||
|
||||||
|
RSET(AHCI_PX_CI, 1 << cmd->slot) ||
|
||||||
|
(cmd->props->ncq && RSET(AHCI_PX_SACT, 1 << cmd->slot))) {
|
||||||
usleep(50);
|
usleep(50);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ahci_command_issue(AHCIQState *ahci, AHCICommand *cmd)
|
void ahci_command_issue(AHCIQState *ahci, AHCICommand *cmd)
|
||||||
@ -899,8 +945,10 @@ void ahci_command_verify(AHCIQState *ahci, AHCICommand *cmd)
|
|||||||
ahci_port_check_error(ahci, port);
|
ahci_port_check_error(ahci, port);
|
||||||
ahci_port_check_interrupts(ahci, port, cmd->interrupts);
|
ahci_port_check_interrupts(ahci, port, cmd->interrupts);
|
||||||
ahci_port_check_nonbusy(ahci, port, slot);
|
ahci_port_check_nonbusy(ahci, port, slot);
|
||||||
ahci_port_check_cmd_sanity(ahci, port, slot, cmd->xbytes);
|
ahci_port_check_cmd_sanity(ahci, cmd);
|
||||||
ahci_port_check_d2h_sanity(ahci, port, slot);
|
if (cmd->interrupts & AHCI_PX_IS_DHRS) {
|
||||||
|
ahci_port_check_d2h_sanity(ahci, port, slot);
|
||||||
|
}
|
||||||
if (cmd->props->pio) {
|
if (cmd->props->pio) {
|
||||||
ahci_port_check_pio_sanity(ahci, port, slot, cmd->xbytes);
|
ahci_port_check_pio_sanity(ahci, port, slot, cmd->xbytes);
|
||||||
}
|
}
|
||||||
|
@ -263,20 +263,23 @@ enum {
|
|||||||
/* ATA Commands */
|
/* ATA Commands */
|
||||||
enum {
|
enum {
|
||||||
/* DMA */
|
/* DMA */
|
||||||
CMD_READ_DMA = 0xC8,
|
CMD_READ_DMA = 0xC8,
|
||||||
CMD_READ_DMA_EXT = 0x25,
|
CMD_READ_DMA_EXT = 0x25,
|
||||||
CMD_WRITE_DMA = 0xCA,
|
CMD_WRITE_DMA = 0xCA,
|
||||||
CMD_WRITE_DMA_EXT = 0x35,
|
CMD_WRITE_DMA_EXT = 0x35,
|
||||||
/* PIO */
|
/* PIO */
|
||||||
CMD_READ_PIO = 0x20,
|
CMD_READ_PIO = 0x20,
|
||||||
CMD_READ_PIO_EXT = 0x24,
|
CMD_READ_PIO_EXT = 0x24,
|
||||||
CMD_WRITE_PIO = 0x30,
|
CMD_WRITE_PIO = 0x30,
|
||||||
CMD_WRITE_PIO_EXT = 0x34,
|
CMD_WRITE_PIO_EXT = 0x34,
|
||||||
/* Misc */
|
/* Misc */
|
||||||
CMD_READ_MAX = 0xF8,
|
CMD_READ_MAX = 0xF8,
|
||||||
CMD_READ_MAX_EXT = 0x27,
|
CMD_READ_MAX_EXT = 0x27,
|
||||||
CMD_FLUSH_CACHE = 0xE7,
|
CMD_FLUSH_CACHE = 0xE7,
|
||||||
CMD_IDENTIFY = 0xEC
|
CMD_IDENTIFY = 0xEC,
|
||||||
|
/* NCQ */
|
||||||
|
READ_FPDMA_QUEUED = 0x60,
|
||||||
|
WRITE_FPDMA_QUEUED = 0x61,
|
||||||
};
|
};
|
||||||
|
|
||||||
/* AHCI Command Header Flags & Masks*/
|
/* AHCI Command Header Flags & Masks*/
|
||||||
@ -291,8 +294,9 @@ enum {
|
|||||||
#define CMDH_PMP (0xF000)
|
#define CMDH_PMP (0xF000)
|
||||||
|
|
||||||
/* ATA device register masks */
|
/* ATA device register masks */
|
||||||
#define ATA_DEVICE_MAGIC 0xA0
|
#define ATA_DEVICE_MAGIC 0xA0 /* used in ata1-3 */
|
||||||
#define ATA_DEVICE_LBA 0x40
|
#define ATA_DEVICE_LBA 0x40
|
||||||
|
#define NCQ_DEVICE_MAGIC 0x40 /* for ncq device registers */
|
||||||
#define ATA_DEVICE_DRIVE 0x10
|
#define ATA_DEVICE_DRIVE 0x10
|
||||||
#define ATA_DEVICE_HEAD 0x0F
|
#define ATA_DEVICE_HEAD 0x0F
|
||||||
|
|
||||||
@ -396,6 +400,32 @@ typedef struct RegH2DFIS {
|
|||||||
uint8_t aux[4];
|
uint8_t aux[4];
|
||||||
} __attribute__((__packed__)) RegH2DFIS;
|
} __attribute__((__packed__)) RegH2DFIS;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register host-to-device FIS structure, for NCQ commands.
|
||||||
|
* Actually just a RegH2DFIS, but with fields repurposed.
|
||||||
|
* Repurposed fields are annotated below.
|
||||||
|
*/
|
||||||
|
typedef struct NCQFIS {
|
||||||
|
/* DW0 */
|
||||||
|
uint8_t fis_type;
|
||||||
|
uint8_t flags;
|
||||||
|
uint8_t command;
|
||||||
|
uint8_t sector_low; /* H2D: Feature 7:0 */
|
||||||
|
/* DW1 */
|
||||||
|
uint8_t lba_lo[3];
|
||||||
|
uint8_t device;
|
||||||
|
/* DW2 */
|
||||||
|
uint8_t lba_hi[3];
|
||||||
|
uint8_t sector_hi; /* H2D: Feature 15:8 */
|
||||||
|
/* DW3 */
|
||||||
|
uint8_t tag; /* H2D: Count 0:7 */
|
||||||
|
uint8_t prio; /* H2D: Count 15:8 */
|
||||||
|
uint8_t icc;
|
||||||
|
uint8_t control;
|
||||||
|
/* DW4 */
|
||||||
|
uint8_t aux[4];
|
||||||
|
} __attribute__((__packed__)) NCQFIS;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Command List entry structure.
|
* Command List entry structure.
|
||||||
* The command list contains between 1-32 of these structures.
|
* The command list contains between 1-32 of these structures.
|
||||||
@ -512,8 +542,7 @@ void ahci_port_check_nonbusy(AHCIQState *ahci, uint8_t port, uint8_t slot);
|
|||||||
void ahci_port_check_d2h_sanity(AHCIQState *ahci, uint8_t port, uint8_t slot);
|
void ahci_port_check_d2h_sanity(AHCIQState *ahci, uint8_t port, uint8_t slot);
|
||||||
void ahci_port_check_pio_sanity(AHCIQState *ahci, uint8_t port,
|
void ahci_port_check_pio_sanity(AHCIQState *ahci, uint8_t port,
|
||||||
uint8_t slot, size_t buffsize);
|
uint8_t slot, size_t buffsize);
|
||||||
void ahci_port_check_cmd_sanity(AHCIQState *ahci, uint8_t port,
|
void ahci_port_check_cmd_sanity(AHCIQState *ahci, AHCICommand *cmd);
|
||||||
uint8_t slot, size_t buffsize);
|
|
||||||
void ahci_get_command_header(AHCIQState *ahci, uint8_t port,
|
void ahci_get_command_header(AHCIQState *ahci, uint8_t port,
|
||||||
uint8_t slot, AHCICommandHeader *cmd);
|
uint8_t slot, AHCICommandHeader *cmd);
|
||||||
void ahci_set_command_header(AHCIQState *ahci, uint8_t port,
|
void ahci_set_command_header(AHCIQState *ahci, uint8_t port,
|
||||||
|
Loading…
Reference in New Issue
Block a user