提交 cd369c46 编写于 作者: C Christoph Hellwig 提交者: Kevin Wolf

ide: factor dma handling helpers

Factor the DMA I/O path that is duplicated between read and write
commands, into common helpers using the s->is_read flag added for
the macio ATA controller.
Signed-off-by: NChristoph Hellwig <hch@lst.de>
Signed-off-by: NKevin Wolf <kwolf@redhat.com>
上级 8b6b2afc
...@@ -487,16 +487,18 @@ static int ide_handle_rw_error(IDEState *s, int error, int op) ...@@ -487,16 +487,18 @@ static int ide_handle_rw_error(IDEState *s, int error, int op)
return 1; return 1;
} }
void ide_read_dma_cb(void *opaque, int ret) void ide_dma_cb(void *opaque, int ret)
{ {
IDEState *s = opaque; IDEState *s = opaque;
int n; int n;
int64_t sector_num; int64_t sector_num;
if (ret < 0) { if (ret < 0) {
if (ide_handle_rw_error(s, -ret, int op = BM_STATUS_DMA_RETRY;
BM_STATUS_DMA_RETRY | BM_STATUS_RETRY_READ))
{ if (s->is_read)
op |= BM_STATUS_RETRY_READ;
if (ide_handle_rw_error(s, -ret, op)) {
return; return;
} }
} }
...@@ -504,7 +506,7 @@ void ide_read_dma_cb(void *opaque, int ret) ...@@ -504,7 +506,7 @@ void ide_read_dma_cb(void *opaque, int ret)
n = s->io_buffer_size >> 9; n = s->io_buffer_size >> 9;
sector_num = ide_get_sector(s); sector_num = ide_get_sector(s);
if (n > 0) { if (n > 0) {
dma_buf_commit(s, 1); dma_buf_commit(s, s->is_read);
sector_num += n; sector_num += n;
ide_set_sector(s, sector_num); ide_set_sector(s, sector_num);
s->nsector -= n; s->nsector -= n;
...@@ -514,32 +516,44 @@ void ide_read_dma_cb(void *opaque, int ret) ...@@ -514,32 +516,44 @@ void ide_read_dma_cb(void *opaque, int ret)
if (s->nsector == 0) { if (s->nsector == 0) {
s->status = READY_STAT | SEEK_STAT; s->status = READY_STAT | SEEK_STAT;
ide_set_irq(s->bus); ide_set_irq(s->bus);
eot: goto eot;
s->bus->dma->ops->add_status(s->bus->dma, BM_STATUS_INT);
ide_set_inactive(s);
return;
} }
/* launch next transfer */ /* launch next transfer */
n = s->nsector; n = s->nsector;
s->io_buffer_index = 0; if (s->is_read)
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, 1) == 0) if (s->bus->dma->ops->prepare_buf(s->bus->dma, s->is_read) == 0)
goto eot; goto eot;
#ifdef DEBUG_AIO #ifdef DEBUG_AIO
printf("aio_read: sector_num=%" PRId64 " n=%d\n", sector_num, n); printf("ide_dma_cb: sector_num=%" PRId64 " n=%d, is_read=%d\n",
sector_num, n, s->is_read);
#endif #endif
s->bus->dma->aiocb = dma_bdrv_read(s->bs, &s->sg, sector_num, ide_read_dma_cb, s);
ide_dma_submit_check(s, ide_read_dma_cb); if (s->is_read) {
s->bus->dma->aiocb = dma_bdrv_read(s->bs, &s->sg, sector_num,
ide_dma_cb, s);
} else {
s->bus->dma->aiocb = dma_bdrv_write(s->bs, &s->sg, sector_num,
ide_dma_cb, s);
}
ide_dma_submit_check(s, ide_dma_cb);
return;
eot:
s->bus->dma->ops->add_status(s->bus->dma, BM_STATUS_INT);
ide_set_inactive(s);
} }
static void ide_sector_read_dma(IDEState *s) static void ide_sector_start_dma(IDEState *s, int is_read)
{ {
s->status = READY_STAT | SEEK_STAT | DRQ_STAT | BUSY_STAT; s->status = READY_STAT | SEEK_STAT | DRQ_STAT | BUSY_STAT;
s->io_buffer_index = 0; s->io_buffer_index = 0;
s->io_buffer_size = 0; s->io_buffer_size = 0;
s->is_read = 1; s->is_read = is_read;
s->bus->dma->ops->start_dma(s->bus->dma, s, ide_read_dma_cb); s->bus->dma->ops->start_dma(s->bus->dma, s, ide_dma_cb);
} }
static void ide_sector_write_timer_cb(void *opaque) static void ide_sector_write_timer_cb(void *opaque)
...@@ -594,57 +608,6 @@ void ide_sector_write(IDEState *s) ...@@ -594,57 +608,6 @@ void ide_sector_write(IDEState *s)
} }
} }
void ide_write_dma_cb(void *opaque, int ret)
{
IDEState *s = opaque;
int n;
int64_t sector_num;
if (ret < 0) {
if (ide_handle_rw_error(s, -ret, BM_STATUS_DMA_RETRY))
return;
}
n = s->io_buffer_size >> 9;
sector_num = ide_get_sector(s);
if (n > 0) {
dma_buf_commit(s, 0);
sector_num += n;
ide_set_sector(s, sector_num);
s->nsector -= n;
}
/* end of transfer ? */
if (s->nsector == 0) {
s->status = READY_STAT | SEEK_STAT;
ide_set_irq(s->bus);
eot:
s->bus->dma->ops->add_status(s->bus->dma, BM_STATUS_INT);
ide_set_inactive(s);
return;
}
n = s->nsector;
s->io_buffer_size = n * 512;
/* launch next transfer */
if (s->bus->dma->ops->prepare_buf(s->bus->dma, 0) == 0)
goto eot;
#ifdef DEBUG_AIO
printf("aio_write: sector_num=%" PRId64 " n=%d\n", sector_num, n);
#endif
s->bus->dma->aiocb = dma_bdrv_write(s->bs, &s->sg, sector_num, ide_write_dma_cb, s);
ide_dma_submit_check(s, ide_write_dma_cb);
}
static void ide_sector_write_dma(IDEState *s)
{
s->status = READY_STAT | SEEK_STAT | DRQ_STAT | BUSY_STAT;
s->io_buffer_index = 0;
s->io_buffer_size = 0;
s->is_read = 0;
s->bus->dma->ops->start_dma(s->bus->dma, s, ide_write_dma_cb);
}
void ide_atapi_cmd_ok(IDEState *s) void ide_atapi_cmd_ok(IDEState *s)
{ {
s->error = 0; s->error = 0;
...@@ -1858,7 +1821,7 @@ void ide_exec_cmd(IDEBus *bus, uint32_t val) ...@@ -1858,7 +1821,7 @@ void ide_exec_cmd(IDEBus *bus, uint32_t val)
if (!s->bs) if (!s->bs)
goto abort_cmd; goto abort_cmd;
ide_cmd_lba48_transform(s, lba48); ide_cmd_lba48_transform(s, lba48);
ide_sector_read_dma(s); ide_sector_start_dma(s, 1);
break; break;
case WIN_WRITEDMA_EXT: case WIN_WRITEDMA_EXT:
lba48 = 1; lba48 = 1;
...@@ -1867,7 +1830,7 @@ void ide_exec_cmd(IDEBus *bus, uint32_t val) ...@@ -1867,7 +1830,7 @@ void ide_exec_cmd(IDEBus *bus, uint32_t val)
if (!s->bs) if (!s->bs)
goto abort_cmd; goto abort_cmd;
ide_cmd_lba48_transform(s, lba48); ide_cmd_lba48_transform(s, lba48);
ide_sector_write_dma(s); ide_sector_start_dma(s, 0);
s->media_changed = 1; s->media_changed = 1;
break; break;
case WIN_READ_NATIVE_MAX_EXT: case WIN_READ_NATIVE_MAX_EXT:
......
...@@ -439,7 +439,6 @@ struct IDEState { ...@@ -439,7 +439,6 @@ struct IDEState {
uint32_t mdata_size; uint32_t mdata_size;
uint8_t *mdata_storage; uint8_t *mdata_storage;
int media_changed; int media_changed;
/* for pmac */
int is_read; int is_read;
/* SMART */ /* SMART */
uint8_t smart_enabled; uint8_t smart_enabled;
...@@ -560,8 +559,7 @@ void ide_init2_with_non_qdev_drives(IDEBus *bus, DriveInfo *hd0, ...@@ -560,8 +559,7 @@ void ide_init2_with_non_qdev_drives(IDEBus *bus, DriveInfo *hd0,
void ide_init_ioport(IDEBus *bus, int iobase, int iobase2); void ide_init_ioport(IDEBus *bus, int iobase, int iobase2);
void ide_exec_cmd(IDEBus *bus, uint32_t val); void ide_exec_cmd(IDEBus *bus, uint32_t val);
void ide_read_dma_cb(void *opaque, int ret); void ide_dma_cb(void *opaque, int ret);
void ide_write_dma_cb(void *opaque, int ret);
void ide_sector_write(IDEState *s); void ide_sector_write(IDEState *s);
void ide_sector_read(IDEState *s); void ide_sector_read(IDEState *s);
void ide_flush_cache(IDEState *s); void ide_flush_cache(IDEState *s);
......
...@@ -178,14 +178,9 @@ static void bmdma_restart_dma(BMDMAState *bm, int is_read) ...@@ -178,14 +178,9 @@ static void bmdma_restart_dma(BMDMAState *bm, int is_read)
s->io_buffer_index = 0; s->io_buffer_index = 0;
s->io_buffer_size = 0; s->io_buffer_size = 0;
s->nsector = bm->nsector; s->nsector = bm->nsector;
s->is_read = is_read;
bm->cur_addr = bm->addr; bm->cur_addr = bm->addr;
bm->dma_cb = ide_dma_cb;
if (is_read) {
bm->dma_cb = ide_read_dma_cb;
} else {
bm->dma_cb = ide_write_dma_cb;
}
bmdma_start_dma(&bm->dma, s, bm->dma_cb); bmdma_start_dma(&bm->dma, s, bm->dma_cb);
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册
新手
引导
客服 返回
顶部