41#include "debug/VIO.hh" 
   42#include "params/VirtIODeviceBase.hh" 
   43#include "params/VirtIODummyDevice.hh" 
   58    *
this = std::forward<VirtDescriptor>(other);
 
 
   69    queue = std::move(rhs.queue);
 
   71    _index = std::move(rhs._index);
 
   72    desc = std::move(rhs.desc);
 
 
   88    memProxy->readBlob(desc_addr, &guest_desc, 
sizeof(guest_desc));
 
   91            "VirtDescriptor(%i): Addr: 0x%x, Len: %i, Flags: 0x%x, " 
 
  102    } 
while ((
desc = 
desc->next()) != NULL && 
desc != 
this);
 
  105        panic(
"Loop in descriptor chain!\n");
 
 
  114    DPRINTF(VIO, 
"Descriptor[%i]: " 
  115            "Addr: 0x%x, Len: %i, Flags: 0x%x, Next: 0x%x\n",
 
  119        auto data = std::make_unique<uint8_t[]>(
desc.len);
 
 
  134    } 
while ((
desc = 
desc->next()) != NULL);
 
 
  150    DPRINTF(VIO, 
"VirtDescriptor(%p, 0x%x, %i)::read: offset: %i, dst: 0x%x, size: %i\n",
 
  154        panic(
"Trying to read from outgoing buffer\n");
 
 
  162    DPRINTF(VIO, 
"VirtDescriptor(%p, 0x%x, %i)::write: offset: %i, src: 0x%x, size: %i\n",
 
  166        panic(
"Trying to write to incoming buffer\n");
 
 
  175    const size_t full_size(
size);
 
  189        panic(
"Failed to read %i bytes from chain of %i bytes @ offset %i\n",
 
 
  198    const size_t full_size(
size);
 
  212        panic(
"Failed to write %i bytes into chain of %i bytes @ offset %i\n",
 
 
  224    } 
while ((
desc = 
desc->next()) != NULL);
 
 
  276                              _size * 
sizeof(uint16_t));
 
  277    const Addr addr_used((addr_avail_end + 
sizeof(uint16_t) +
 
  280    avail.setAddress(addr_avail);
 
  281    used.setAddress(addr_used);
 
 
  288    DPRINTF(VIO, 
"consumeDescriptor: _last_avail: %i, avail.idx: %i (->%i)\n",
 
 
  307    DPRINTF(VIO, 
"produceDescriptor: dscIdx: %i, len: %i, used.idx: %i\n",
 
  313    used.header.index += 1;
 
 
  388    DPRINTF(VIO, 
"onNotify: idx: %i\n", idx);
 
  390        panic(
"Guest tried to notify queue (%i), but only %i " 
  391              "queues registered.\n",
 
 
  400    DPRINTF(VIO, 
"Setting guest features: 0x%x\n", features);
 
  402        panic(
"Guest tried to enable unsupported features:\n" 
  403              "Device features: 0x%x\n" 
  404              "Requested features: 0x%x\n",
 
 
  415    DPRINTF(VIO, 
"ACK: %i, DRIVER: %i, DRIVER_OK: %i, FAILED: %i\n",
 
 
  424    panic(
"Unhandled device config read (offset: 0x%x).\n", cfgOffset);
 
 
  430    panic(
"Unhandled device config write (offset: 0x%x).\n", cfgOffset);
 
 
  436    const unsigned size(pkt->
getSize());
 
  439        panic(
"Config read out of bounds.\n");
 
  442    pkt->
setData(
const_cast<uint8_t *
>(cfg) + cfgOffset);
 
 
  448    const unsigned size(pkt->
getSize());
 
  451        panic(
"Config write out of bounds.\n");
 
  454    pkt->
writeData((uint8_t *)cfg + cfgOffset);
 
 
#define DDUMP(x, data, count)
DPRINTF is a debugging trace facility that allows one to selectively enable tracing statements.
void makeResponse()
Take a request packet and modify it in place to be suitable for returning as a response to that reque...
void setData(const uint8_t *p)
Copy data into the packet from the provided pointer.
void writeData(uint8_t *p) const
Copy data from the packet to the memory at the provided pointer.
This object is a proxy for a port or other object which implements the functional response protocol,...
VirtIO descriptor (chain) wrapper.
VirtDescriptor & operator=(VirtDescriptor &&rhs) noexcept
bool isOutgoing() const
Check if this is a write-only descriptor (outgoing data).
void update()
Populate this descriptor with data from the guest.
~VirtDescriptor() noexcept
bool hasNext() const
Is this descriptor chained to another descriptor?
PortProxy * memProxy
Pointer to memory proxy.
size_t size() const
Retrieve the size of this descriptor.
void dumpChain() const
Dump the contents of a descriptor chain starting at this descriptor.
ByteOrder byteOrder
The byte order the descriptor is stored in.
Index _index
Index in virtqueue.
void chainRead(size_t offset, uint8_t *dst, size_t size) const
Read the contents of a descriptor chain.
void dump() const
Dump the contents of a descriptor.
vring_desc desc
Underlying descriptor.
void read(size_t offset, uint8_t *dst, size_t size) const
Read the contents of a descriptor.
uint16_t Index
Descriptor index in virtqueue.
VirtDescriptor(PortProxy &memProxy, ByteOrder bo, VirtQueue &queue, Index index)
Create a descriptor wrapper.
VirtDescriptor * next() const
Get the pointer to the next descriptor in a chain.
Index index() const
Get the descriptor's index into the virtqueue.
void chainWrite(size_t offset, const uint8_t *src, size_t size)
Write to a descriptor chain.
void write(size_t offset, const uint8_t *src, size_t size)
Write to the contents of a descriptor.
size_t chainSize() const
Retrieve the size of this descriptor chain.
VirtQueue * queue
Pointer to virtqueue owning this descriptor.
bool isIncoming() const
Check if this is a read-only descriptor (incoming data).
void updateChain()
Populate this descriptor chain with data from the guest.
void setDeviceStatus(DeviceStatus status)
Update device status and optionally reset device.
ByteOrder byteOrder
The byte order of the queues, descriptors, etc.
void readConfigBlob(PacketPtr pkt, Addr cfgOffset, const uint8_t *cfg)
Read configuration data from a device structure.
virtual ~VirtIODeviceBase()
const VirtQueue & getCurrentQueue() const
Convenience method to get the currently selected queue.
FeatureBits guestFeatures
Feature set accepted by the guest.
const size_t configSize
Size of the device's configuration space.
VirtIODeviceBase(const Params ¶ms, DeviceId id, size_t config_size, FeatureBits features)
virtual void writeConfig(PacketPtr pkt, Addr cfgOffset)
Write to the configuration space of a device.
uint32_t getQueueAddress() const
Get the host physical address of the currently active queue.
void writeConfigBlob(PacketPtr pkt, Addr cfgOffset, uint8_t *cfg)
Write configuration data to a device structure.
QueueID _queueSelect
Queue select register (set by guest)
const FeatureBits deviceFeatures
Feature set offered by the device.
void unserialize(CheckpointIn &cp) override
Unserialize an object.
std::vector< VirtQueue * > _queues
List of virtual queues supported by this device.
virtual void reset()
Driver-request device reset.
void onNotify(QueueID index)
Driver is requesting service.
void serialize(CheckpointOut &cp) const override
Serialize an object.
virtual void readConfig(PacketPtr pkt, Addr cfgOffset)
Read from the configuration space of a device.
uint16_t DeviceId
Device Type (sometimes known as subsystem ID)
void setQueueAddress(uint32_t address)
Change the host physical address of the currently active queue.
const DeviceId deviceId
Device ID (sometimes known as subsystem ID)
void setGuestFeatures(FeatureBits features)
Set feature bits accepted by the guest driver.
void registerQueue(VirtQueue &queue)
Register a new VirtQueue with the device model.
DeviceStatus _deviceStatus
Status of the device.
static const DeviceId ID_INVALID
VirtIO device ID.
VirtIODummyDevice(const VirtIODummyDeviceParams ¶ms)
Base wrapper around a virtqueue.
void unserialize(CheckpointIn &cp) override
Unserialize an object.
static const Addr ALIGN_BITS
Page size used by VirtIO. It's hard-coded to 4096 bytes in the spec for historical reasons.
void setAddress(Addr address)
Set the base address of this queue.
void produceDescriptor(VirtDescriptor *desc, uint32_t len)
Send a descriptor chain to the guest.
VirtRing< struct vring_used_elem > used
Ring of used (outgoing) descriptors.
std::vector< VirtDescriptor > descriptors
Vector of pre-created descriptors indexed by their index into the queue.
static const Addr ALIGN_SIZE
VirtRing< VirtDescriptor::Index > avail
Ring of available (incoming) descriptors.
virtual void onNotify()
Notify queue of pending events.
uint16_t _last_avail
Offset of last consumed descriptor in the VirtQueue::avail ring.
void reset()
Reset cached state in this queue and in the associated ring buffers.
const uint16_t _size
Queue size in terms of number of descriptors.
void serialize(CheckpointOut &cp) const override
Serialize an object.
PortProxy & memProxy
Guest physical memory proxy.
Addr _address
Base address of the queue.
ByteOrder byteOrder
Byte order in this queue.
VirtDescriptor * consumeDescriptor()
Get an incoming descriptor chain from the queue.
void dump() const
Dump the contents of a queue.
virtual void onNotifyDescriptor(VirtDescriptor *desc)
Notify queue of pending incoming descriptor.
#define panic(...)
This implements a cprintf based panic() function.
void serializeSection(CheckpointOut &cp, const char *name) const
Serialize an object into a new section.
void unserializeSection(CheckpointIn &cp, const char *name)
Unserialize an a child object.
const Params & params() const
SimObject(const Params &p)
Copyright (c) 2024 Arm Limited All rights reserved.
std::ostream CheckpointOut
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
T gtoh(T value, ByteOrder guest_byte_order)
void paramIn(CheckpointIn &cp, const std::string &name, ExtMachInst &machInst)
std::string csprintf(const char *format, const Args &...args)
#define UNSERIALIZE_SCALAR(scalar)
#define SERIALIZE_SCALAR(scalar)