Go to the documentation of this file.
   57 #include "debug/IdeDisk.hh" 
   67     : 
SimObject(
p), image(
p.image), diskDelay(
p.delay), ideDiskStats(this),
 
   69       dmaReadWaitEvent([
this]{ doDmaRead(); }, 
name()),
 
   70       dmaWriteWaitEvent([
this]{ doDmaWrite(); }, 
name()),
 
   71       dmaPrdReadEvent([
this]{ dmaPrdReadDone(); }, 
name()),
 
   72       dmaReadEvent([
this]{ dmaReadDone(); }, 
name()),
 
   73       dmaWriteEvent([
this]{ dmaWriteDone(); }, 
name())
 
   79     memset(&driveID, 0, 
sizeof(
struct ataparams));
 
   86     uint32_t lba_size = image->size();
 
   87     if (lba_size >= 16383*16*63) {
 
   94         else if (lba_size == 0)
 
   95             panic(
"Bad IDE image size: 0\n");
 
   99         if ((lba_size / sectors) >= 16)
 
  102             heads = (lba_size / sectors);
 
  104         cylinders = lba_size / (heads * sectors);
 
  108     strncpy((
char *)driveID.atap_model, 
"5MI EDD si k",
 
  109             sizeof(driveID.atap_model));
 
  113     driveID.atap_capabilities1 = 0x7;
 
  115     driveID.atap_extensions = 0x6;
 
  117     driveID.atap_cylinders = cylinders;
 
  118     driveID.atap_sectors = sectors;
 
  119     driveID.atap_heads = heads;
 
  122     driveID.atap_curmulti_valid = 0x1;
 
  124     driveID.atap_capacity = lba_size;
 
  126     driveID.atap_dmamode_supp = 0x4;
 
  128     driveID.atap_piomode_supp = 0x3;
 
  130     driveID.atap_udmamode_supp = 0x1f;
 
  132     driveID.atap_hwreset_res = 0x4001;
 
  169     } 
else if (
id == 
DEV1) {
 
  173         panic(
"Invalid device ID: %#x\n", 
id);
 
  209         if (size == 
sizeof(uint16_t)) {
 
  211         } 
else if (size == 
sizeof(uint32_t)) {
 
  216             panic(
"Data read of unsupported size %d.\n", size);
 
  221     assert(size == 
sizeof(uint8_t));
 
  246         panic(
"Invalid IDE command register offset: %#x\n", 
offset);
 
  254     assert(size == 
sizeof(uint8_t));
 
  257         panic(
"Invalid IDE control register offset: %#x\n", 
offset);
 
  265         if (size == 
sizeof(uint16_t)) {
 
  267         } 
else if (size == 
sizeof(uint32_t)) {
 
  272             panic(
"Data write of unsupported size %d.\n", size);
 
  278     assert(size == 
sizeof(uint8_t));
 
  303         panic(
"Invalid IDE command register offset: %#x\n", 
offset);
 
  313         panic(
"Invalid IDE control register offset: %#x\n", 
offset);
 
  343         panic(
"Inconsistent DMA transfer state: dmaState = %d devState = %d\n",
 
  366             "PRD: baseAddr:%#x (%#x) byteCount:%d (%d) eot:%#x sector:%d\n",
 
  394     : statistics::
Group(parent, 
"IdeDisk"),
 
  395       ADD_STAT(dmaReadFullPages, statistics::units::Count::get(),
 
  396                "Number of full page size DMA reads (not PRD)."),
 
  397       ADD_STAT(dmaReadBytes, statistics::units::Byte::get(),
 
  398                "Number of bytes transfered via DMA reads (not PRD)."),
 
  399       ADD_STAT(dmaReadTxs, statistics::units::Count::get(),
 
  400                "Number of DMA read transactions (not PRD)."),
 
  401       ADD_STAT(dmaWriteFullPages, statistics::units::Count::get(),
 
  402                "Number of full page size DMA writes."),
 
  403       ADD_STAT(dmaWriteBytes, statistics::units::Byte::get(),
 
  404                "Number of bytes transfered via DMA writes."),
 
  405       ADD_STAT(dmaWriteTxs, statistics::units::Count::get(),
 
  406                "Number of DMA write transactions.")
 
  452     uint32_t bytesWritten = 0;
 
  477     uint32_t bytesRead = 0;
 
  479     DPRINTF(
IdeDisk, 
"doDmaWrite, diskDelay: %d totalDiskDelay: %d\n",
 
  519         DPRINTF(
IdeDisk, 
"doDmaWrite: not done curPrd byte count %d, eot %#x\n",
 
  527         DPRINTF(
IdeDisk, 
"doDmaWrite: done curPrd byte count %d, eot %#x\n",
 
  540             "doWriteDone: curPrd byte count %d, eot %#x cmd bytes left:%d\n",
 
  562             "Can't read from %s. Only %d of %d read. errno=%d",
 
  572             "Can't write to %s. Only %d of %d written. errno=%d",
 
  584             "Inconsistent DMA state, should be in Dma_Start!");
 
  587             "Inconsistent device state for DMA start!");
 
  602             "Inconsistent DMA state, should be Start or Transfer!");
 
  605             "Inconsistent device state, should be Transfer or Prepare!");
 
  666                 "Attempt to perform CHS access, only supports LBA");
 
  684                 "Attempt to perform CHS access, only supports LBA");
 
  703                 "Attempt to perform CHS access, only supports LBA");
 
  743             "Attempt to post an interrupt with one pending");
 
  888                 panic(
"DEBUG: READING DATA ONE BYTE AT A TIME!\n");
 
  964                 panic(
"DEBUG: WRITING DATA ONE BYTE AT A TIME!\n");
 
 1015                 panic(
"Inconsistent DMA state, should be Dma_Idle\n");
 
 1058             DPRINTF(
IdeDisk, 
"Disk still busy aborting previous DMA command\n");
 
 1108     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.
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
constexpr void replaceBits(T &val, unsigned first, unsigned last, B bit_val)
A convenience function to replace bits first to last of val with bit_val in place.
~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.
IdeDisk * selected() const
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.
bool pendingInterrupt
Interrupt pending.
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)
IdeController::Channel * channel
The channel this disk is connected to.
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
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
void writeDisk(uint32_t sector, uint8_t *data)
void readDisk(uint32_t sector, uint8_t *data)
void paramOut(CheckpointOut &cp, const std::string &name, ExtMachInst const &machInst)
#define UNSERIALIZE_ARRAY(member, size)
Addr addr() const
Return starting address of current chunk.
EventFunctionWrapper dmaPrdReadEvent
void paramIn(CheckpointIn &cp, const std::string &name, ExtMachInst &machInst)
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 Thu Jul 28 2022 13:32:32 for gem5 by  doxygen 1.8.17