diff options
author | Blue Swirl <blauwirbel@gmail.com> | 2009-11-07 14:13:05 +0000 |
---|---|---|
committer | Blue Swirl <blauwirbel@gmail.com> | 2009-11-07 14:13:05 +0000 |
commit | 4a6435639781214a14f3c6054955e5ef16298d72 (patch) | |
tree | fb4d38641d83dc4bf98b56aa91a0d070db0ae425 /hw/ide/core.c | |
parent | b55a37c981914aa8ecd21b9a2a2fb37f39b917c5 (diff) |
IDE: Fix reset handling
Problem: x86 systems could not survive a few system_resets.
Clear most of IDE state when reset. Implement the missing reset handlers.
Signed-off-by: Blue Swirl <blauwirbel@gmail.com>
Diffstat (limited to 'hw/ide/core.c')
-rw-r--r-- | hw/ide/core.c | 64 |
1 files changed, 60 insertions, 4 deletions
diff --git a/hw/ide/core.c b/hw/ide/core.c index fffcd00280..eafb510b63 100644 --- a/hw/ide/core.c +++ b/hw/ide/core.c @@ -2500,17 +2500,48 @@ static void ide_dummy_transfer_stop(IDEState *s) s->io_buffer[3] = 0xff; } -void ide_reset(IDEState *s) +static void ide_reset(IDEState *s) { - IDEBus *bus = s->bus; - +#ifdef DEBUG_IDE + printf("ide: reset\n"); +#endif if (s->is_cf) s->mult_sectors = 0; else s->mult_sectors = MAX_MULT_SECTORS; - bus->unit = s->unit; + /* ide regs */ + s->feature = 0; + s->error = 0; + s->nsector = 0; + s->sector = 0; + s->lcyl = 0; + s->hcyl = 0; + + /* lba48 */ + s->hob_feature = 0; + s->hob_sector = 0; + s->hob_nsector = 0; + s->hob_lcyl = 0; + s->hob_hcyl = 0; + s->select = 0xa0; s->status = READY_STAT | SEEK_STAT; + + s->lba48 = 0; + + /* ATAPI specific */ + s->sense_key = 0; + s->asc = 0; + s->cdrom_changed = 0; + s->packet_transfer_size = 0; + s->elementary_transfer_size = 0; + s->io_buffer_index = 0; + s->cd_sector_size = 0; + s->atapi_dma = 0; + /* ATA DMA state */ + s->io_buffer_size = 0; + s->req_nb_sectors = 0; + ide_set_signature(s); /* init the transfer handler so that 0xffff is returned on data accesses */ @@ -2519,6 +2550,15 @@ void ide_reset(IDEState *s) s->media_changed = 0; } +void ide_bus_reset(IDEBus *bus) +{ + bus->unit = 0; + bus->cmd = 0; + ide_reset(&bus->ifs[0]); + ide_reset(&bus->ifs[1]); + ide_clear_hob(bus); +} + void ide_init_drive(IDEState *s, DriveInfo *dinfo) { int cylinders, heads, secs; @@ -2704,3 +2744,19 @@ void ide_dma_cancel(BMDMAState *bm) } } +void ide_dma_reset(BMDMAState *bm) +{ +#ifdef DEBUG_IDE + printf("ide: dma_reset\n"); +#endif + ide_dma_cancel(bm); + bm->cmd = 0; + bm->status = 0; + bm->addr = 0; + bm->cur_addr = 0; + bm->cur_prd_last = 0; + bm->cur_prd_addr = 0; + bm->cur_prd_len = 0; + bm->sector_num = 0; + bm->nsector = 0; +} |