mirror of https://github.com/xemu-project/xemu.git
qtest: add libqos including PCI support
This includes basic PCI support for the PC platform. Enough abstraction should be present to support non-PC platforms too. Signed-off-by: Anthony Liguori <aliguori@us.ibm.com> Message-id: 1366123521-4330-3-git-send-email-aliguori@us.ibm.com
This commit is contained in:
parent
8a8fd63734
commit
c4efe1cada
|
@ -4511,7 +4511,7 @@ if [ "$pixman" = "internal" ]; then
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# build tree in object directory in case the source is not in the current directory
|
# build tree in object directory in case the source is not in the current directory
|
||||||
DIRS="tests tests/tcg tests/tcg/cris tests/tcg/lm32"
|
DIRS="tests tests/tcg tests/tcg/cris tests/tcg/lm32 tests/libqos"
|
||||||
DIRS="$DIRS pc-bios/optionrom pc-bios/spapr-rtas"
|
DIRS="$DIRS pc-bios/optionrom pc-bios/spapr-rtas"
|
||||||
DIRS="$DIRS roms/seabios roms/vgabios"
|
DIRS="$DIRS roms/seabios roms/vgabios"
|
||||||
DIRS="$DIRS qapi-generated"
|
DIRS="$DIRS qapi-generated"
|
||||||
|
|
|
@ -77,6 +77,7 @@ test-obj-y = tests/check-qint.o tests/check-qstring.o tests/check-qdict.o \
|
||||||
test-qapi-obj-y = tests/test-qapi-visit.o tests/test-qapi-types.o
|
test-qapi-obj-y = tests/test-qapi-visit.o tests/test-qapi-types.o
|
||||||
|
|
||||||
$(test-obj-y): QEMU_INCLUDES += -Itests
|
$(test-obj-y): QEMU_INCLUDES += -Itests
|
||||||
|
QEMU_CFLAGS += -I$(SRC_PATH)/tests
|
||||||
|
|
||||||
tests/test-x86-cpuid.o: QEMU_INCLUDES += -I$(SRC_PATH)/target-i386
|
tests/test-x86-cpuid.o: QEMU_INCLUDES += -I$(SRC_PATH)/target-i386
|
||||||
|
|
||||||
|
@ -105,7 +106,6 @@ tests/test-qmp-commands.h tests/test-qmp-marshal.c :\
|
||||||
$(SRC_PATH)/qapi-schema-test.json $(SRC_PATH)/scripts/qapi-commands.py
|
$(SRC_PATH)/qapi-schema-test.json $(SRC_PATH)/scripts/qapi-commands.py
|
||||||
$(call quiet-command,$(PYTHON) $(SRC_PATH)/scripts/qapi-commands.py $(gen-out-type) -o tests -p "test-" < $<, " GEN $@")
|
$(call quiet-command,$(PYTHON) $(SRC_PATH)/scripts/qapi-commands.py $(gen-out-type) -o tests -p "test-" < $<, " GEN $@")
|
||||||
|
|
||||||
|
|
||||||
tests/test-string-output-visitor$(EXESUF): tests/test-string-output-visitor.o $(test-qapi-obj-y) libqemuutil.a libqemustub.a
|
tests/test-string-output-visitor$(EXESUF): tests/test-string-output-visitor.o $(test-qapi-obj-y) libqemuutil.a libqemustub.a
|
||||||
tests/test-string-input-visitor$(EXESUF): tests/test-string-input-visitor.o $(test-qapi-obj-y) libqemuutil.a libqemustub.a
|
tests/test-string-input-visitor$(EXESUF): tests/test-string-input-visitor.o $(test-qapi-obj-y) libqemuutil.a libqemustub.a
|
||||||
tests/test-qmp-output-visitor$(EXESUF): tests/test-qmp-output-visitor.o $(test-qapi-obj-y) libqemuutil.a libqemustub.a
|
tests/test-qmp-output-visitor$(EXESUF): tests/test-qmp-output-visitor.o $(test-qapi-obj-y) libqemuutil.a libqemustub.a
|
||||||
|
@ -116,6 +116,9 @@ tests/test-visitor-serialization$(EXESUF): tests/test-visitor-serialization.o $(
|
||||||
|
|
||||||
tests/test-mul64$(EXESUF): tests/test-mul64.o libqemuutil.a
|
tests/test-mul64$(EXESUF): tests/test-mul64.o libqemuutil.a
|
||||||
|
|
||||||
|
libqos-obj-y = tests/libqos/pci.o
|
||||||
|
libqos-pc-obj-y = $(libqos-obj-y) tests/libqos/pci-pc.o
|
||||||
|
|
||||||
tests/rtc-test$(EXESUF): tests/rtc-test.o
|
tests/rtc-test$(EXESUF): tests/rtc-test.o
|
||||||
tests/m48t59-test$(EXESUF): tests/m48t59-test.o
|
tests/m48t59-test$(EXESUF): tests/m48t59-test.o
|
||||||
tests/fdc-test$(EXESUF): tests/fdc-test.o
|
tests/fdc-test$(EXESUF): tests/fdc-test.o
|
||||||
|
|
|
@ -0,0 +1,239 @@
|
||||||
|
/*
|
||||||
|
* libqos PCI bindings for PC
|
||||||
|
*
|
||||||
|
* Copyright IBM, Corp. 2012-2013
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anthony Liguori <aliguori@us.ibm.com>
|
||||||
|
*
|
||||||
|
* This work is licensed under the terms of the GNU GPL, version 2 or later.
|
||||||
|
* See the COPYING file in the top-level directory.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "libqtest.h"
|
||||||
|
#include "libqos/pci-pc.h"
|
||||||
|
|
||||||
|
#include "hw/pci/pci_regs.h"
|
||||||
|
|
||||||
|
#include "qemu-common.h"
|
||||||
|
#include "qemu/host-utils.h"
|
||||||
|
|
||||||
|
#include <glib.h>
|
||||||
|
|
||||||
|
typedef struct QPCIBusPC
|
||||||
|
{
|
||||||
|
QPCIBus bus;
|
||||||
|
|
||||||
|
uint32_t pci_hole_start;
|
||||||
|
uint32_t pci_hole_size;
|
||||||
|
uint32_t pci_hole_alloc;
|
||||||
|
|
||||||
|
uint16_t pci_iohole_start;
|
||||||
|
uint16_t pci_iohole_size;
|
||||||
|
uint16_t pci_iohole_alloc;
|
||||||
|
} QPCIBusPC;
|
||||||
|
|
||||||
|
static uint8_t qpci_pc_io_readb(QPCIBus *bus, void *addr)
|
||||||
|
{
|
||||||
|
uintptr_t port = (uintptr_t)addr;
|
||||||
|
uint8_t value;
|
||||||
|
|
||||||
|
if (port < 0x10000) {
|
||||||
|
value = inb(port);
|
||||||
|
} else {
|
||||||
|
memread(port, &value, sizeof(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint16_t qpci_pc_io_readw(QPCIBus *bus, void *addr)
|
||||||
|
{
|
||||||
|
uintptr_t port = (uintptr_t)addr;
|
||||||
|
uint16_t value;
|
||||||
|
|
||||||
|
if (port < 0x10000) {
|
||||||
|
value = inw(port);
|
||||||
|
} else {
|
||||||
|
memread(port, &value, sizeof(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint32_t qpci_pc_io_readl(QPCIBus *bus, void *addr)
|
||||||
|
{
|
||||||
|
uintptr_t port = (uintptr_t)addr;
|
||||||
|
uint32_t value;
|
||||||
|
|
||||||
|
if (port < 0x10000) {
|
||||||
|
value = inl(port);
|
||||||
|
} else {
|
||||||
|
memread(port, &value, sizeof(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void qpci_pc_io_writeb(QPCIBus *bus, void *addr, uint8_t value)
|
||||||
|
{
|
||||||
|
uintptr_t port = (uintptr_t)addr;
|
||||||
|
|
||||||
|
if (port < 0x10000) {
|
||||||
|
outb(port, value);
|
||||||
|
} else {
|
||||||
|
memwrite(port, &value, sizeof(value));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void qpci_pc_io_writew(QPCIBus *bus, void *addr, uint16_t value)
|
||||||
|
{
|
||||||
|
uintptr_t port = (uintptr_t)addr;
|
||||||
|
|
||||||
|
if (port < 0x10000) {
|
||||||
|
outw(port, value);
|
||||||
|
} else {
|
||||||
|
memwrite(port, &value, sizeof(value));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void qpci_pc_io_writel(QPCIBus *bus, void *addr, uint32_t value)
|
||||||
|
{
|
||||||
|
uintptr_t port = (uintptr_t)addr;
|
||||||
|
|
||||||
|
if (port < 0x10000) {
|
||||||
|
outl(port, value);
|
||||||
|
} else {
|
||||||
|
memwrite(port, &value, sizeof(value));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint8_t qpci_pc_config_readb(QPCIBus *bus, int devfn, uint8_t offset)
|
||||||
|
{
|
||||||
|
outl(0xcf8, (1 << 31) | (devfn << 8) | offset);
|
||||||
|
return inb(0xcfc);
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint16_t qpci_pc_config_readw(QPCIBus *bus, int devfn, uint8_t offset)
|
||||||
|
{
|
||||||
|
outl(0xcf8, (1 << 31) | (devfn << 8) | offset);
|
||||||
|
return inw(0xcfc);
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint32_t qpci_pc_config_readl(QPCIBus *bus, int devfn, uint8_t offset)
|
||||||
|
{
|
||||||
|
outl(0xcf8, (1 << 31) | (devfn << 8) | offset);
|
||||||
|
return inl(0xcfc);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void qpci_pc_config_writeb(QPCIBus *bus, int devfn, uint8_t offset, uint8_t value)
|
||||||
|
{
|
||||||
|
outl(0xcf8, (1 << 31) | (devfn << 8) | offset);
|
||||||
|
outb(0xcfc, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void qpci_pc_config_writew(QPCIBus *bus, int devfn, uint8_t offset, uint16_t value)
|
||||||
|
{
|
||||||
|
outl(0xcf8, (1 << 31) | (devfn << 8) | offset);
|
||||||
|
outw(0xcfc, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void qpci_pc_config_writel(QPCIBus *bus, int devfn, uint8_t offset, uint32_t value)
|
||||||
|
{
|
||||||
|
outl(0xcf8, (1 << 31) | (devfn << 8) | offset);
|
||||||
|
outl(0xcfc, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void *qpci_pc_iomap(QPCIBus *bus, QPCIDevice *dev, int barno)
|
||||||
|
{
|
||||||
|
QPCIBusPC *s = container_of(bus, QPCIBusPC, bus);
|
||||||
|
static const int bar_reg_map[] = {
|
||||||
|
PCI_BASE_ADDRESS_0, PCI_BASE_ADDRESS_1, PCI_BASE_ADDRESS_2,
|
||||||
|
PCI_BASE_ADDRESS_3, PCI_BASE_ADDRESS_4, PCI_BASE_ADDRESS_5,
|
||||||
|
};
|
||||||
|
int bar_reg;
|
||||||
|
uint32_t addr;
|
||||||
|
uint64_t size;
|
||||||
|
uint32_t io_type;
|
||||||
|
|
||||||
|
g_assert(barno >= 0 && barno <= 5);
|
||||||
|
bar_reg = bar_reg_map[barno];
|
||||||
|
|
||||||
|
qpci_config_writel(dev, bar_reg, 0xFFFFFFFF);
|
||||||
|
addr = qpci_config_readl(dev, bar_reg);
|
||||||
|
|
||||||
|
io_type = addr & PCI_BASE_ADDRESS_SPACE;
|
||||||
|
if (io_type == PCI_BASE_ADDRESS_SPACE_IO) {
|
||||||
|
addr &= PCI_BASE_ADDRESS_IO_MASK;
|
||||||
|
} else {
|
||||||
|
addr &= PCI_BASE_ADDRESS_MEM_MASK;
|
||||||
|
}
|
||||||
|
|
||||||
|
size = (1ULL << ctzl(addr));
|
||||||
|
if (size == 0) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (io_type == PCI_BASE_ADDRESS_SPACE_IO) {
|
||||||
|
uint16_t loc;
|
||||||
|
|
||||||
|
g_assert((s->pci_iohole_alloc + size) <= s->pci_iohole_size);
|
||||||
|
loc = s->pci_iohole_start + s->pci_iohole_alloc;
|
||||||
|
s->pci_iohole_alloc += size;
|
||||||
|
|
||||||
|
qpci_config_writel(dev, bar_reg, loc | PCI_BASE_ADDRESS_SPACE_IO);
|
||||||
|
|
||||||
|
return (void *)(intptr_t)loc;
|
||||||
|
} else {
|
||||||
|
uint64_t loc;
|
||||||
|
|
||||||
|
g_assert((s->pci_hole_alloc + size) <= s->pci_hole_size);
|
||||||
|
loc = s->pci_hole_start + s->pci_hole_alloc;
|
||||||
|
s->pci_hole_alloc += size;
|
||||||
|
|
||||||
|
qpci_config_writel(dev, bar_reg, loc);
|
||||||
|
|
||||||
|
return (void *)(intptr_t)loc;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void qpci_pc_iounmap(QPCIBus *bus, void *data)
|
||||||
|
{
|
||||||
|
/* FIXME */
|
||||||
|
}
|
||||||
|
|
||||||
|
QPCIBus *qpci_init_pc(void)
|
||||||
|
{
|
||||||
|
QPCIBusPC *ret;
|
||||||
|
|
||||||
|
ret = g_malloc(sizeof(*ret));
|
||||||
|
|
||||||
|
ret->bus.io_readb = qpci_pc_io_readb;
|
||||||
|
ret->bus.io_readw = qpci_pc_io_readw;
|
||||||
|
ret->bus.io_readl = qpci_pc_io_readl;
|
||||||
|
|
||||||
|
ret->bus.io_writeb = qpci_pc_io_writeb;
|
||||||
|
ret->bus.io_writew = qpci_pc_io_writew;
|
||||||
|
ret->bus.io_writel = qpci_pc_io_writel;
|
||||||
|
|
||||||
|
ret->bus.config_readb = qpci_pc_config_readb;
|
||||||
|
ret->bus.config_readw = qpci_pc_config_readw;
|
||||||
|
ret->bus.config_readl = qpci_pc_config_readl;
|
||||||
|
|
||||||
|
ret->bus.config_writeb = qpci_pc_config_writeb;
|
||||||
|
ret->bus.config_writew = qpci_pc_config_writew;
|
||||||
|
ret->bus.config_writel = qpci_pc_config_writel;
|
||||||
|
|
||||||
|
ret->bus.iomap = qpci_pc_iomap;
|
||||||
|
ret->bus.iounmap = qpci_pc_iounmap;
|
||||||
|
|
||||||
|
ret->pci_hole_start = 0xE0000000;
|
||||||
|
ret->pci_hole_size = 0x20000000;
|
||||||
|
ret->pci_hole_alloc = 0;
|
||||||
|
|
||||||
|
ret->pci_iohole_start = 0xc000;
|
||||||
|
ret->pci_iohole_size = 0x4000;
|
||||||
|
ret->pci_iohole_alloc = 0;
|
||||||
|
|
||||||
|
return &ret->bus;
|
||||||
|
}
|
|
@ -0,0 +1,20 @@
|
||||||
|
/*
|
||||||
|
* libqos PCI bindings for PC
|
||||||
|
*
|
||||||
|
* Copyright IBM, Corp. 2012-2013
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anthony Liguori <aliguori@us.ibm.com>
|
||||||
|
*
|
||||||
|
* This work is licensed under the terms of the GNU GPL, version 2 or later.
|
||||||
|
* See the COPYING file in the top-level directory.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef LIBQOS_PCI_PC_H
|
||||||
|
#define LIBQOS_PCI_PC_H
|
||||||
|
|
||||||
|
#include "libqos/pci.h"
|
||||||
|
|
||||||
|
QPCIBus *qpci_init_pc(void);
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,151 @@
|
||||||
|
/*
|
||||||
|
* libqos PCI bindings
|
||||||
|
*
|
||||||
|
* Copyright IBM, Corp. 2012-2013
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anthony Liguori <aliguori@us.ibm.com>
|
||||||
|
*
|
||||||
|
* This work is licensed under the terms of the GNU GPL, version 2 or later.
|
||||||
|
* See the COPYING file in the top-level directory.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "libqos/pci.h"
|
||||||
|
|
||||||
|
#include "hw/pci/pci_regs.h"
|
||||||
|
#include <glib.h>
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
void qpci_device_foreach(QPCIBus *bus, int vendor_id, int device_id,
|
||||||
|
void (*func)(QPCIDevice *dev, int devfn, void *data),
|
||||||
|
void *data)
|
||||||
|
{
|
||||||
|
int slot;
|
||||||
|
|
||||||
|
for (slot = 0; slot < 32; slot++) {
|
||||||
|
int fn;
|
||||||
|
|
||||||
|
for (fn = 0; fn < 8; fn++) {
|
||||||
|
QPCIDevice *dev;
|
||||||
|
|
||||||
|
dev = qpci_device_find(bus, QPCI_DEVFN(slot, fn));
|
||||||
|
if (!dev) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (vendor_id != -1 &&
|
||||||
|
qpci_config_readw(dev, PCI_VENDOR_ID) != vendor_id) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (device_id != -1 &&
|
||||||
|
qpci_config_readw(dev, PCI_DEVICE_ID) != device_id) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
func(dev, QPCI_DEVFN(slot, fn), data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QPCIDevice *qpci_device_find(QPCIBus *bus, int devfn)
|
||||||
|
{
|
||||||
|
QPCIDevice *dev;
|
||||||
|
|
||||||
|
dev = g_malloc0(sizeof(*dev));
|
||||||
|
dev->bus = bus;
|
||||||
|
dev->devfn = devfn;
|
||||||
|
|
||||||
|
if (qpci_config_readw(dev, PCI_VENDOR_ID) == 0xFFFF) {
|
||||||
|
g_free(dev);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return dev;
|
||||||
|
}
|
||||||
|
|
||||||
|
void qpci_device_enable(QPCIDevice *dev)
|
||||||
|
{
|
||||||
|
uint16_t cmd;
|
||||||
|
|
||||||
|
/* FIXME -- does this need to be a bus callout? */
|
||||||
|
cmd = qpci_config_readw(dev, PCI_COMMAND);
|
||||||
|
cmd |= PCI_COMMAND_IO | PCI_COMMAND_MEMORY;
|
||||||
|
qpci_config_writew(dev, PCI_COMMAND, cmd);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t qpci_config_readb(QPCIDevice *dev, uint8_t offset)
|
||||||
|
{
|
||||||
|
return dev->bus->config_readb(dev->bus, dev->devfn, offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16_t qpci_config_readw(QPCIDevice *dev, uint8_t offset)
|
||||||
|
{
|
||||||
|
return dev->bus->config_readw(dev->bus, dev->devfn, offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t qpci_config_readl(QPCIDevice *dev, uint8_t offset)
|
||||||
|
{
|
||||||
|
return dev->bus->config_readl(dev->bus, dev->devfn, offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void qpci_config_writeb(QPCIDevice *dev, uint8_t offset, uint8_t value)
|
||||||
|
{
|
||||||
|
dev->bus->config_writeb(dev->bus, dev->devfn, offset, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
void qpci_config_writew(QPCIDevice *dev, uint8_t offset, uint16_t value)
|
||||||
|
{
|
||||||
|
dev->bus->config_writew(dev->bus, dev->devfn, offset, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
void qpci_config_writel(QPCIDevice *dev, uint8_t offset, uint32_t value)
|
||||||
|
{
|
||||||
|
dev->bus->config_writew(dev->bus, dev->devfn, offset, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
uint8_t qpci_io_readb(QPCIDevice *dev, void *data)
|
||||||
|
{
|
||||||
|
return dev->bus->io_readb(dev->bus, data);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16_t qpci_io_readw(QPCIDevice *dev, void *data)
|
||||||
|
{
|
||||||
|
return dev->bus->io_readw(dev->bus, data);
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t qpci_io_readl(QPCIDevice *dev, void *data)
|
||||||
|
{
|
||||||
|
return dev->bus->io_readl(dev->bus, data);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void qpci_io_writeb(QPCIDevice *dev, void *data, uint8_t value)
|
||||||
|
{
|
||||||
|
dev->bus->io_writeb(dev->bus, data, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
void qpci_io_writew(QPCIDevice *dev, void *data, uint16_t value)
|
||||||
|
{
|
||||||
|
dev->bus->io_writew(dev->bus, data, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
void qpci_io_writel(QPCIDevice *dev, void *data, uint32_t value)
|
||||||
|
{
|
||||||
|
dev->bus->io_writel(dev->bus, data, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
void *qpci_iomap(QPCIDevice *dev, int barno)
|
||||||
|
{
|
||||||
|
return dev->bus->iomap(dev->bus, dev, barno);
|
||||||
|
}
|
||||||
|
|
||||||
|
void qpci_iounmap(QPCIDevice *dev, void *data)
|
||||||
|
{
|
||||||
|
dev->bus->iounmap(dev->bus, data);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,80 @@
|
||||||
|
/*
|
||||||
|
* libqos PCI bindings
|
||||||
|
*
|
||||||
|
* Copyright IBM, Corp. 2012-2013
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Anthony Liguori <aliguori@us.ibm.com>
|
||||||
|
*
|
||||||
|
* This work is licensed under the terms of the GNU GPL, version 2 or later.
|
||||||
|
* See the COPYING file in the top-level directory.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef LIBQOS_PCI_H
|
||||||
|
#define LIBQOS_PCI_H
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
#define QPCI_DEVFN(dev, fn) (((dev) << 3) | (fn))
|
||||||
|
|
||||||
|
typedef struct QPCIDevice QPCIDevice;
|
||||||
|
typedef struct QPCIBus QPCIBus;
|
||||||
|
|
||||||
|
struct QPCIBus
|
||||||
|
{
|
||||||
|
uint8_t (*io_readb)(QPCIBus *bus, void *addr);
|
||||||
|
uint16_t (*io_readw)(QPCIBus *bus, void *addr);
|
||||||
|
uint32_t (*io_readl)(QPCIBus *bus, void *addr);
|
||||||
|
|
||||||
|
void (*io_writeb)(QPCIBus *bus, void *addr, uint8_t value);
|
||||||
|
void (*io_writew)(QPCIBus *bus, void *addr, uint16_t value);
|
||||||
|
void (*io_writel)(QPCIBus *bus, void *addr, uint32_t value);
|
||||||
|
|
||||||
|
uint8_t (*config_readb)(QPCIBus *bus, int devfn, uint8_t offset);
|
||||||
|
uint16_t (*config_readw)(QPCIBus *bus, int devfn, uint8_t offset);
|
||||||
|
uint32_t (*config_readl)(QPCIBus *bus, int devfn, uint8_t offset);
|
||||||
|
|
||||||
|
void (*config_writeb)(QPCIBus *bus, int devfn,
|
||||||
|
uint8_t offset, uint8_t value);
|
||||||
|
void (*config_writew)(QPCIBus *bus, int devfn,
|
||||||
|
uint8_t offset, uint16_t value);
|
||||||
|
void (*config_writel)(QPCIBus *bus, int devfn,
|
||||||
|
uint8_t offset, uint32_t value);
|
||||||
|
|
||||||
|
void *(*iomap)(QPCIBus *bus, QPCIDevice *dev, int barno);
|
||||||
|
void (*iounmap)(QPCIBus *bus, void *data);
|
||||||
|
};
|
||||||
|
|
||||||
|
struct QPCIDevice
|
||||||
|
{
|
||||||
|
QPCIBus *bus;
|
||||||
|
int devfn;
|
||||||
|
};
|
||||||
|
|
||||||
|
void qpci_device_foreach(QPCIBus *bus, int vendor_id, int device_id,
|
||||||
|
void (*func)(QPCIDevice *dev, int devfn, void *data),
|
||||||
|
void *data);
|
||||||
|
QPCIDevice *qpci_device_find(QPCIBus *bus, int devfn);
|
||||||
|
|
||||||
|
void qpci_device_enable(QPCIDevice *dev);
|
||||||
|
|
||||||
|
uint8_t qpci_config_readb(QPCIDevice *dev, uint8_t offset);
|
||||||
|
uint16_t qpci_config_readw(QPCIDevice *dev, uint8_t offset);
|
||||||
|
uint32_t qpci_config_readl(QPCIDevice *dev, uint8_t offset);
|
||||||
|
|
||||||
|
void qpci_config_writeb(QPCIDevice *dev, uint8_t offset, uint8_t value);
|
||||||
|
void qpci_config_writew(QPCIDevice *dev, uint8_t offset, uint16_t value);
|
||||||
|
void qpci_config_writel(QPCIDevice *dev, uint8_t offset, uint32_t value);
|
||||||
|
|
||||||
|
uint8_t qpci_io_readb(QPCIDevice *dev, void *data);
|
||||||
|
uint16_t qpci_io_readw(QPCIDevice *dev, void *data);
|
||||||
|
uint32_t qpci_io_readl(QPCIDevice *dev, void *data);
|
||||||
|
|
||||||
|
void qpci_io_writeb(QPCIDevice *dev, void *data, uint8_t value);
|
||||||
|
void qpci_io_writew(QPCIDevice *dev, void *data, uint16_t value);
|
||||||
|
void qpci_io_writel(QPCIDevice *dev, void *data, uint32_t value);
|
||||||
|
|
||||||
|
void *qpci_iomap(QPCIDevice *dev, int barno);
|
||||||
|
void qpci_iounmap(QPCIDevice *dev, void *data);
|
||||||
|
|
||||||
|
#endif
|
Loading…
Reference in New Issue