Go to the documentation of this file.
56 #include "debug/IdeDisk.hh"
66 :
SimObject(
p), ctrl(NULL), image(
p.image), diskDelay(
p.delay),
70 dmaReadWaitEvent([
this]{ doDmaRead(); },
name()),
72 dmaWriteWaitEvent([
this]{ doDmaWrite(); },
name()),
73 dmaPrdReadEvent([
this]{ dmaPrdReadDone(); },
name()),
74 dmaReadEvent([
this]{ dmaReadDone(); },
name()),
75 dmaWriteEvent([
this]{ dmaWriteDone(); },
name())
81 memset(&driveID, 0,
sizeof(
struct ataparams));
88 uint32_t lba_size = image->size();
89 if (lba_size >= 16383*16*63) {
96 else if (lba_size == 0)
97 panic(
"Bad IDE image size: 0\n");
101 if ((lba_size / sectors) >= 16)
104 heads = (lba_size / sectors);
106 cylinders = lba_size / (heads * sectors);
110 strncpy((
char *)driveID.atap_model,
"5MI EDD si k",
111 sizeof(driveID.atap_model));
115 driveID.atap_capabilities1 = 0x7;
117 driveID.atap_extensions = 0x6;
119 driveID.atap_cylinders = cylinders;
120 driveID.atap_sectors = sectors;
121 driveID.atap_heads = heads;
124 driveID.atap_curmulti_valid = 0x1;
126 driveID.atap_capacity = lba_size;
128 driveID.atap_dmamode_supp = 0x4;
130 driveID.atap_piomode_supp = 0x3;
132 driveID.atap_udmamode_supp = 0x1f;
134 driveID.atap_hwreset_res = 0x4001;
171 }
else if (
id ==
DEV1) {
175 panic(
"Invalid device ID: %#x\n",
id);
202 panic(
"Access to unset controller!\n");
213 if (size ==
sizeof(uint16_t)) {
215 }
else if (size ==
sizeof(uint32_t)) {
220 panic(
"Data read of unsupported size %d.\n", size);
225 assert(size ==
sizeof(uint8_t));
250 panic(
"Invalid IDE command register offset: %#x\n",
offset);
258 assert(size ==
sizeof(uint8_t));
261 panic(
"Invalid IDE control register offset: %#x\n",
offset);
269 if (size ==
sizeof(uint16_t)) {
271 }
else if (size ==
sizeof(uint32_t)) {
276 panic(
"Data write of unsupported size %d.\n", size);
282 assert(size ==
sizeof(uint8_t));
307 panic(
"Invalid IDE command register offset: %#x\n",
offset);
317 panic(
"Invalid IDE control register offset: %#x\n",
offset);
347 panic(
"Inconsistent DMA transfer state: dmaState = %d devState = %d\n",
368 "PRD: baseAddr:%#x (%#x) byteCount:%d (%d) eot:%#x sector:%d\n",
396 : statistics::
Group(parent,
"IdeDisk"),
397 ADD_STAT(dmaReadFullPages, statistics::units::Count::get(),
398 "Number of full page size DMA reads (not PRD)."),
399 ADD_STAT(dmaReadBytes, statistics::units::Byte::get(),
400 "Number of bytes transfered via DMA reads (not PRD)."),
401 ADD_STAT(dmaReadTxs, statistics::units::Count::get(),
402 "Number of DMA read transactions (not PRD)."),
403 ADD_STAT(dmaWriteFullPages, statistics::units::Count::get(),
404 "Number of full page size DMA writes."),
405 ADD_STAT(dmaWriteBytes, statistics::units::Byte::get(),
406 "Number of bytes transfered via DMA writes."),
407 ADD_STAT(dmaWriteTxs, statistics::units::Count::get(),
408 "Number of DMA write transactions.")
454 uint32_t bytesWritten = 0;
479 uint32_t bytesRead = 0;
481 DPRINTF(
IdeDisk,
"doDmaWrite, diskDelay: %d totalDiskDelay: %d\n",
521 DPRINTF(
IdeDisk,
"doDmaWrite: not done curPrd byte count %d, eot %#x\n",
529 DPRINTF(
IdeDisk,
"doDmaWrite: done curPrd byte count %d, eot %#x\n",
541 DPRINTF(
IdeDisk,
"doWriteDone: curPrd byte count %d, eot %#x cmd bytes left:%d\n",
563 panic(
"Can't read from %s. Only %d of %d read. errno=%d\n",
573 panic(
"Can't write to %s. Only %d of %d written. errno=%d\n",
585 panic(
"Inconsistent DMA state, should be in Dma_Start!\n");
588 panic(
"Inconsistent device state for DMA start!\n");
603 panic(
"Inconsistent DMA state, should be Start or Transfer!");
606 panic(
"Inconsistent device state, should be Transfer or Prepare!\n");
656 panic(
"Attempt to perform CHS access, only supports LBA\n");
674 panic(
"Attempt to perform CHS access, only supports LBA\n");
693 panic(
"Attempt to perform CHS access, only supports LBA\n");
732 panic(
"Attempt to post an interrupt with one pending\n");
747 panic(
"Attempt to clear a non-pending interrupt\n");
881 panic(
"DEBUG: READING DATA ONE BYTE AT A TIME!\n");
957 panic(
"DEBUG: WRITING DATA ONE BYTE AT A TIME!\n");
1008 panic(
"Inconsistent DMA state, should be Dma_Idle\n");
1051 DPRINTF(
IdeDisk,
"Disk still busy aborting previous DMA command\n");
1101 assert(eventCount <= 1);
void updateState(DevAction_t action)
Tick curTick()
The universal simulation clock.
Addr pciToDma(Addr pciAddr)
Tick when() const
Get the time that the event is scheduled.
bool intrPending
Interrupt pending.
statistics::Scalar dmaWriteBytes
uint32_t curSector
Current sector in access.
#define UNSERIALIZE_SCALAR(scalar)
DrainState drainState() const
Return the current drain state of an object.
PrdTableEntry curPrd
PRD entry.
bool next()
Advance generator to next chunk.
statistics::Scalar dmaWriteTxs
void reset(int id)
Reset the device state.
Addr complete() const
Number of bytes we have already chunked up.
void unserialize(CheckpointIn &cp) override
Unserialize an object.
#define WDSF_READ_NATIVE_MAX
statistics::Scalar dmaWriteFullPages
~IdeDisk()
Delete the data buffer.
ChunkGenerator * dmaWriteCG
struct ataparams driveID
Drive identification structure for this disk.
uint32_t curPrdAddr
PRD table base address.
void readCommand(const Addr offset, int size, uint8_t *data)
void dmaRead(Addr addr, int size, Event *event, uint8_t *data, uint32_t sid, uint32_t ssid, Tick delay=0)
void schedule(Event &event, Tick when)
Addr chunkBytes
Size of chunks to DMA.
virtual std::streampos write(const uint8_t *data, std::streampos offset)=0
void startDma(const uint32_t &prdTableBase)
IdeDiskStats(statistics::Group *parent)
uint32_t cmdBytesLeft
Number of bytes left in command data transfer.
Addr size() const
Return size in bytes of current chunk.
This class takes an arbitrary memory region (address/length pair) and generates a series of appropria...
#define SERIALIZE_ENUM(scalar)
bool dmaRead
Dma transaction is a read.
DevState_t devState
Device state.
ChunkGenerator * dmaReadCG
bool done() const
Are we done? That is, did the last call to next() advance past the end of the region?
bool nIENBit
Interrupt enable bit.
virtual std::string name() const
EventFunctionWrapper dmaWriteWaitEvent
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
DmaState_t dmaState
Dma state.
uint32_t cmdBytes
Number of bytes in command data transfer.
uint64_t Tick
Tick count type.
statistics::Scalar dmaReadTxs
EventFunctionWrapper dmaReadEvent
void reschedule(Event &event, Tick when, bool always=false)
void dmaWrite(Addr addr, int size, Event *event, uint8_t *data, uint32_t sid, uint32_t ssid, Tick delay=0)
int devID
Device ID (device0=0/device1=1)
#define SERIALIZE_ARRAY(member, size)
void writeCommand(const Addr offset, int size, const uint8_t *data)
Abstract superclass for simulation objects.
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
#define DMA_BACKOFF_PERIOD
const std::string & name()
#define SERIALIZE_SCALAR(scalar)
DiskImage * image
The image that contains the data of this disk.
gem5::IdeDisk::IdeDiskStats ideDiskStats
uint8_t status
Status register.
EventFunctionWrapper dmaReadWaitEvent
void writeDisk(uint32_t sector, uint8_t *data)
void readDisk(uint32_t sector, uint8_t *data)
#define UNSERIALIZE_ARRAY(member, size)
void setDmaComplete(IdeDisk *disk)
bool isDiskSelected(IdeDisk *diskPtr)
See if a disk is selected based on its pointer.
Addr addr() const
Return starting address of current chunk.
EventFunctionWrapper dmaPrdReadEvent
uint32_t drqBytesLeft
Number of bytes left in DRQ block.
statistics::Scalar dmaReadFullPages
void readControl(const Addr offset, int size, uint8_t *data)
#define ATAPI_IDENTIFY_DEVICE
virtual std::streampos size() const =0
#define WDCC_STANDBY_IMMED
virtual std::streampos read(uint8_t *data, std::streampos offset) const =0
std::ostream CheckpointOut
statistics::Scalar dmaReadBytes
IdeController * ctrl
The IDE controller for this disk.
#define UNSERIALIZE_ENUM(scalar)
@ Running
Running normally.
Addr pciToDma(Addr pci_addr) const
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
void serialize(CheckpointOut &cp) const override
Serialize an object.
EventFunctionWrapper dmaTransferEvent
uint8_t * dataBuffer
Data buffer for transfers.
bool dmaAborted
DMA Aborted.
CommandReg_t cmdReg
Command block registers.
int diskDelay
The disk delay in microseconds.
bool scheduled() const
Determine if the current event is scheduled.
void writeControl(const Addr offset, int size, const uint8_t *data)
EventFunctionWrapper dmaWriteEvent
#define panic(...)
This implements a cprintf based panic() function.
Generated on Tue Sep 7 2021 14:53:47 for gem5 by doxygen 1.8.17