/* * \brief VirtIO MMIO device * \author Piotr Tworek * \date 2019-09-27 */ /* * Copyright (C) 2020 Genode Labs GmbH * * This file is part of the Genode OS framework, which is distributed * under the terms of the GNU Affero General Public License version 3. */ #ifndef _INCLUDE__VIRTIO__MMIO_DEVICE_H_ #define _INCLUDE__VIRTIO__MMIO_DEVICE_H_ #include #include #include #include namespace Virtio { using namespace Genode; class Device; } class Virtio::Device : Genode::Attached_dataspace, Genode::Mmio { public: struct Invalid_device : Genode::Exception { }; enum Status : uint8_t { RESET = 0, ACKNOWLEDGE = 1 << 0, DRIVER = 1 << 1, DRIVER_OK = 1 << 2, FEATURES_OK = 1 << 3, FAILED = 1 << 7, }; enum Access_size : uint8_t { ACCESS_8BIT, ACCESS_16BIT, ACCESS_32BIT, }; private: enum { VIRTIO_MMIO_MAGIC = 0x74726976 }; /** * Some of the registers are actually 8 bits wide, but according to * section 4.2.2.2 of VIRTIO 1.0 spec "The driver MUST use only 32 bit * wide and aligned reads and writes". */ struct Magic : Register<0x000, 32> { }; struct Version : Register<0x004, 32> { }; struct DeviceID : Register<0x008, 32> { }; struct VendorID : Register<0x00C, 32> { }; struct DeviceFeatures : Register<0x010, 32> { }; struct DeviceFeaturesSel : Register<0x014, 32> { }; struct DriverFeatures : Register<0x020, 32> { }; struct DriverFeaturesSel : Register<0x024, 32> { }; struct QueueSel : Register<0x030, 32> { }; struct QueueNumMax : Register<0x034, 32> { }; struct QueueNum : Register<0x038, 32> { }; struct QueueReady : Register<0x044, 32> { }; struct QueueNotify : Register<0x050, 32> { }; struct InterruptStatus : Register<0x060, 32> { }; struct InterruptAck : Register<0x064, 32> { }; struct StatusReg : Register<0x070, 32> { }; struct QueueDescLow : Register<0x080, 32> { }; struct QueueDescHigh : Register<0x084, 32> { }; struct QueueAvailLow : Register<0x090, 32> { }; struct QueueAvailHigh : Register<0x094, 32> { }; struct QueueUsedLow : Register<0x0A0, 32> { }; struct QueueUsedHigh : Register<0x0A4, 32> { }; struct ConfigGeneration : Register<0x0FC, 32> { }; /** * Different views on device configuration space. According to the * VIRTIO 1.0 spec 64 bit wide registers are supposed to be read as * two 32 bit values. */ struct Config_8 : Register_array<0x100, 8, 256, 8> { }; struct Config_16 : Register_array<0x100, 16, 128, 16> { }; struct Config_32 : Register_array<0x100, 32, 64, 32> { }; /* * Noncopyable */ Device(Device const &) = delete; Device &operator = (Device const &) = delete; public: Device(Genode::Env &env, Genode::Io_mem_dataspace_capability io_mem_ds, size_t offset) : Attached_dataspace(env.rm(), io_mem_ds) , Mmio((addr_t)local_addr() + offset) { if (read() != VIRTIO_MMIO_MAGIC) { throw Invalid_device(); } } uint32_t vendor_id() { return read(); } uint32_t device_id() { return read(); } uint8_t get_status() { return read() & 0xff; } bool set_status(uint8_t status) { write(status); return read() == status; } uint32_t get_features(uint32_t selection) { write(selection); return read(); } void set_features(uint32_t selection, uint32_t features) { write(selection); write(features); } uint8_t get_config_generation() { return read() & 0xff; } uint16_t get_max_queue_size(uint16_t queue_index) { write(queue_index); if (read() != 0) { return 0; } return read(); } uint32_t read_config(uint8_t offset, Access_size size) { switch (size) { case ACCESS_8BIT: return read(offset); case ACCESS_16BIT: return read(offset >> 1); case ACCESS_32BIT: return read(offset >> 2); } return 0; } void write_config(uint8_t offset, Access_size size, uint32_t value) { switch (size) { case ACCESS_8BIT: write(value, offset); break; case ACCESS_16BIT: write(value, (offset >> 1)); break; case ACCESS_32BIT: write(value, (offset >> 2)); break; } } bool configure_queue(uint16_t queue_index, Virtio::Queue_description desc) { write(queue_index); if (read() != 0) return false; write(desc.size); uint64_t addr = desc.desc; write((uint32_t)addr); write((uint32_t)(addr >> 32)); addr = desc.avail; write((uint32_t)addr); write((uint32_t)(addr >> 32)); addr = desc.used; write((uint32_t)addr); write((uint32_t)(addr >> 32)); write(1); return read() != 0; } void notify_buffers_available(uint16_t queue_index) { write(queue_index); } uint32_t read_isr() { uint32_t isr = read(); write(isr); return isr; } }; #endif /* _INCLUDE__VIRTIO__MMIO_DEVICE_H_ */