提交 aaeda4a3 编写于 作者: J John Snow

ide: unify io_buffer_offset increments

IDEState's io_buffer_offset was originally added to keep track of offsets
in AHCI rather exclusively, but it was added to IDEState instead of an
AHCI-specific structure.

AHCI fakes all PIO transfers using DMA and a scatter-gather list. When
the core or atapi layers invoke HBA-specific mechanisms for transfers,
they do not always know that it is being backed by DMA or a sglist, so
this offset is not always updated by the HBA code everywhere.

If we modify it in dma_buf_commit, however, any HBA that needs to use
this offset to manage operating on only part of a sglist will have
access to it.

This will fix ATAPI PIO transfers performed through the AHCI HBA,
which were previously not modifying this value appropriately.

This will fix ATAPI PIO transfers larger than one sector.
Reported-by: NHannes Reinecke <hare@suse.de>
Signed-off-by: NJohn Snow <jsnow@redhat.com>
Reviewed-by: NPaolo Bonzini <pbonzini@redhat.com>
Tested-by: NLaszlo Ersek <lersek@redhat.com>
Message-id: 1440546331-29087-2-git-send-email-jsnow@redhat.com
CC: qemu-stable@nongnu.org
上级 16a1b6e9
...@@ -49,7 +49,6 @@ static void ahci_reset_port(AHCIState *s, int port); ...@@ -49,7 +49,6 @@ 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, int32_t limit); static int ahci_dma_prepare_buf(IDEDMA *dma, int32_t limit);
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);
static void ahci_unmap_clb_address(AHCIDevice *ad); static void ahci_unmap_clb_address(AHCIDevice *ad);
...@@ -1289,7 +1288,7 @@ out: ...@@ -1289,7 +1288,7 @@ out:
s->data_ptr = s->data_end; s->data_ptr = s->data_end;
/* Update number of transferred bytes, destroy sglist */ /* Update number of transferred bytes, destroy sglist */
ahci_commit_buf(dma, size); dma_buf_commit(s, size);
s->end_transfer_func(s); s->end_transfer_func(s);
...@@ -1331,9 +1330,8 @@ static void ahci_restart(IDEDMA *dma) ...@@ -1331,9 +1330,8 @@ static void ahci_restart(IDEDMA *dma)
} }
/** /**
* Called in DMA R/W chains to read the PRDT, utilizing ahci_populate_sglist. * Called in DMA and PIO R/W chains to read the PRDT.
* Not currently invoked by PIO R/W chains, * Not shared with NCQ pathways.
* which invoke ahci_populate_sglist via ahci_start_transfer.
*/ */
static int32_t ahci_dma_prepare_buf(IDEDMA *dma, int32_t limit) static int32_t ahci_dma_prepare_buf(IDEDMA *dma, int32_t limit)
{ {
...@@ -1352,21 +1350,16 @@ static int32_t ahci_dma_prepare_buf(IDEDMA *dma, int32_t limit) ...@@ -1352,21 +1350,16 @@ static int32_t ahci_dma_prepare_buf(IDEDMA *dma, int32_t limit)
} }
/** /**
* Destroys the scatter-gather list, * Updates the command header with a bytes-read value.
* and updates the command header with a bytes-read value. * Called via dma_buf_commit, for both DMA and PIO paths.
* called explicitly via ahci_dma_rw_buf (ATAPI DMA), * sglist destruction is handled within dma_buf_commit.
* and ahci_start_transfer (PIO R/W),
* and called via callback from ide_dma_cb for DMA R/W paths.
*/ */
static void ahci_commit_buf(IDEDMA *dma, uint32_t tx_bytes) static void ahci_commit_buf(IDEDMA *dma, uint32_t tx_bytes)
{ {
AHCIDevice *ad = DO_UPCAST(AHCIDevice, dma, dma); AHCIDevice *ad = DO_UPCAST(AHCIDevice, dma, dma);
IDEState *s = &ad->port.ifs[0];
tx_bytes += le32_to_cpu(ad->cur_cmd->status); tx_bytes += le32_to_cpu(ad->cur_cmd->status);
ad->cur_cmd->status = cpu_to_le32(tx_bytes); ad->cur_cmd->status = cpu_to_le32(tx_bytes);
qemu_sglist_destroy(&s->sg);
} }
static int ahci_dma_rw_buf(IDEDMA *dma, int is_write) static int ahci_dma_rw_buf(IDEDMA *dma, int is_write)
...@@ -1387,10 +1380,9 @@ static int ahci_dma_rw_buf(IDEDMA *dma, int is_write) ...@@ -1387,10 +1380,9 @@ static int ahci_dma_rw_buf(IDEDMA *dma, int is_write)
} }
/* free sglist, update byte count */ /* free sglist, update byte count */
ahci_commit_buf(dma, l); dma_buf_commit(s, l);
s->io_buffer_index += l; s->io_buffer_index += l;
s->io_buffer_offset += l;
DPRINTF(ad->port_no, "len=%#x\n", l); DPRINTF(ad->port_no, "len=%#x\n", l);
......
...@@ -591,7 +591,6 @@ static void ide_sector_read_cb(void *opaque, int ret) ...@@ -591,7 +591,6 @@ static void ide_sector_read_cb(void *opaque, int ret)
s->nsector -= n; s->nsector -= n;
/* Allow the guest to read the io_buffer */ /* Allow the guest to read the io_buffer */
ide_transfer_start(s, s->io_buffer, n * BDRV_SECTOR_SIZE, ide_sector_read); ide_transfer_start(s, s->io_buffer, n * BDRV_SECTOR_SIZE, ide_sector_read);
s->io_buffer_offset += 512 * n;
ide_set_irq(s->bus); ide_set_irq(s->bus);
} }
...@@ -635,11 +634,12 @@ static void ide_sector_read(IDEState *s) ...@@ -635,11 +634,12 @@ static void ide_sector_read(IDEState *s)
ide_sector_read_cb, s); ide_sector_read_cb, s);
} }
static void dma_buf_commit(IDEState *s, uint32_t tx_bytes) void dma_buf_commit(IDEState *s, uint32_t tx_bytes)
{ {
if (s->bus->dma->ops->commit_buf) { if (s->bus->dma->ops->commit_buf) {
s->bus->dma->ops->commit_buf(s->bus->dma, tx_bytes); s->bus->dma->ops->commit_buf(s->bus->dma, tx_bytes);
} }
s->io_buffer_offset += tx_bytes;
qemu_sglist_destroy(&s->sg); qemu_sglist_destroy(&s->sg);
} }
...@@ -842,7 +842,6 @@ static void ide_sector_write_cb(void *opaque, int ret) ...@@ -842,7 +842,6 @@ static void ide_sector_write_cb(void *opaque, int ret)
n = s->req_nb_sectors; n = s->req_nb_sectors;
} }
s->nsector -= n; s->nsector -= n;
s->io_buffer_offset += 512 * n;
ide_set_sector(s, ide_get_sector(s) + n); ide_set_sector(s, ide_get_sector(s) + n);
if (s->nsector == 0) { if (s->nsector == 0) {
......
...@@ -536,6 +536,7 @@ int64_t ide_get_sector(IDEState *s); ...@@ -536,6 +536,7 @@ int64_t ide_get_sector(IDEState *s);
void ide_set_sector(IDEState *s, int64_t sector_num); void ide_set_sector(IDEState *s, int64_t sector_num);
void ide_start_dma(IDEState *s, BlockCompletionFunc *cb); void ide_start_dma(IDEState *s, BlockCompletionFunc *cb);
void dma_buf_commit(IDEState *s, uint32_t tx_bytes);
void ide_dma_error(IDEState *s); void ide_dma_error(IDEState *s);
void ide_atapi_cmd_ok(IDEState *s); void ide_atapi_cmd_ok(IDEState *s);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册