From: "Michael S. Tsirkin" <mst@redhat.com>
To: Paolo Bonzini <pbonzini@redhat.com>
Cc: gleb@redhat.com, mtosatti@redhat.com, qemu-devel@nongnu.org,
kvm@vger.kernel.org
Subject: Re: [Qemu-devel] [PATCH 4/4] pci: add pci test device
Date: Wed, 3 Apr 2013 13:33:01 +0300 [thread overview]
Message-ID: <20130403103301.GG18179@redhat.com> (raw)
In-Reply-To: <515C037E.5030706@redhat.com>
On Wed, Apr 03, 2013 at 12:25:02PM +0200, Paolo Bonzini wrote:
>
> > Exactly. I prefer a flexible structure so I can change
> > things host side. For example register multiple
> > datamatches with same address and compare speed
> > of access on first and last one.
> > Similarly for address.
>
> Ok, this makes sense.
>
> >> (BTW, and unrelated to this, please use default-configs/ and
> >> hw/Makefile.objs instead of hw/i386/Makefile.objs to enable the device).
> >
> > Okay but why does pc-test device sit in hw/i386/Makefile.objs ?
>
> Because someone wasn't looking. :) The hw/ reorganization patches I've
> posted fix that.
>
> Paolo
Okay.
Still not sure how to merge this, if it goes in through my tree
and that's the only comment, I'll just fix it silently ...
> >>>
> >>>
> >>>>> Signed-off-by: Michael S. Tsirkin <mst@redhat.com>
> >>>>> ---
> >>>>> hw/i386/Makefile.objs | 2 +-
> >>>>> hw/pci-testdev.c | 306 ++++++++++++++++++++++++++++++++++++++++++++++++++
> >>>>> hw/pci/pci.h | 1 +
> >>>>> 3 files changed, 308 insertions(+), 1 deletion(-)
> >>>>> create mode 100644 hw/pci-testdev.c
> >>>>>
> >>>>> diff --git a/hw/i386/Makefile.objs b/hw/i386/Makefile.objs
> >>>>> index a78c0b2..7497e7a 100644
> >>>>> --- a/hw/i386/Makefile.objs
> >>>>> +++ b/hw/i386/Makefile.objs
> >>>>> @@ -11,7 +11,7 @@ obj-$(CONFIG_XEN_PCI_PASSTHROUGH) += xen-host-pci-device.o
> >>>>> obj-$(CONFIG_XEN_PCI_PASSTHROUGH) += xen_pt.o xen_pt_config_init.o xen_pt_msi.o
> >>>>> obj-y += kvm/
> >>>>> obj-$(CONFIG_SPICE) += qxl.o qxl-logger.o qxl-render.o
> >>>>> -obj-y += pc-testdev.o
> >>>>> +obj-y += pc-testdev.o pci-testdev.o
> >>>>>
> >>>>> obj-y := $(addprefix ../,$(obj-y))
> >>>>>
> >>>>> diff --git a/hw/pci-testdev.c b/hw/pci-testdev.c
> >>>>> new file mode 100644
> >>>>> index 0000000..9486624
> >>>>> --- /dev/null
> >>>>> +++ b/hw/pci-testdev.c
> >>>>> @@ -0,0 +1,306 @@
> >>>>> +#include "hw/hw.h"
> >>>>> +#include "hw/pci/pci.h"
> >>>>> +#include "qemu/event_notifier.h"
> >>>>> +#include "qemu/osdep.h"
> >>>>> +
> >>>>> +typedef struct PCITestDevHdr {
> >>>>> + uint8_t test;
> >>>>> + uint8_t width;
> >>>>> + uint8_t pad0[2];
> >>>>> + uint32_t offset;
> >>>>> + uint8_t data;
> >>>>> + uint8_t pad1[3];
> >>>>> + uint32_t count;
> >>>>> + uint8_t name[];
> >>>>> +} PCITestDevHdr;
> >>>>> +
> >>>>> +typedef struct IOTest {
> >>>>> + MemoryRegion *mr;
> >>>>> + EventNotifier notifier;
> >>>>> + bool hasnotifier;
> >>>>> + unsigned size;
> >>>>> + bool match_data;
> >>>>> + PCITestDevHdr *hdr;
> >>>>> + unsigned bufsize;
> >>>>> +} IOTest;
> >>>>> +
> >>>>> +#define IOTEST_DATAMATCH 0xFA
> >>>>> +#define IOTEST_NOMATCH 0xCE
> >>>>> +
> >>>>> +#define IOTEST_IOSIZE 128
> >>>>> +#define IOTEST_MEMSIZE 2048
> >>>>> +
> >>>>> +static const char *iotest_test[] = {
> >>>>> + "no-eventfd",
> >>>>> + "wildcard-eventfd",
> >>>>> + "datamatch-eventfd"
> >>>>> +};
> >>>>> +
> >>>>> +static const char *iotest_type[] = {
> >>>>> + "mmio",
> >>>>> + "portio"
> >>>>> +};
> >>>>> +
> >>>>> +#define IOTEST_TEST(i) (iotest_test[((i) % ARRAY_SIZE(iotest_test))])
> >>>>> +#define IOTEST_TYPE(i) (iotest_type[((i) / ARRAY_SIZE(iotest_test))])
> >>>>> +#define IOTEST_MAX_TEST (ARRAY_SIZE(iotest_test))
> >>>>> +#define IOTEST_MAX_TYPE (ARRAY_SIZE(iotest_type))
> >>>>> +#define IOTEST_MAX (IOTEST_MAX_TEST * IOTEST_MAX_TYPE)
> >>>>> +
> >>>>> +enum {
> >>>>> + IOTEST_ACCESS_NAME,
> >>>>> + IOTEST_ACCESS_DATA,
> >>>>> + IOTEST_ACCESS_MAX,
> >>>>> +};
> >>>>> +
> >>>>> +#define IOTEST_ACCESS_TYPE uint8_t
> >>>>> +#define IOTEST_ACCESS_WIDTH (sizeof(uint8_t))
> >>>>> +
> >>>>> +typedef struct PCITestDevState {
> >>>>> + PCIDevice dev;
> >>>>> + MemoryRegion mmio;
> >>>>> + MemoryRegion portio;
> >>>>> + IOTest *tests;
> >>>>> + int current;
> >>>>> +} PCITestDevState;
> >>>>> +
> >>>>> +#define IOTEST_IS_MEM(i) (strcmp(IOTEST_TYPE(i), "portio"))
> >>>>> +#define IOTEST_REGION(d, i) (IOTEST_IS_MEM(i) ? &(d)->mmio : &(d)->portio)
> >>>>> +#define IOTEST_SIZE(i) (IOTEST_IS_MEM(i) ? IOTEST_MEMSIZE : IOTEST_IOSIZE)
> >>>>> +#define IOTEST_PCI_BAR(i) (IOTEST_IS_MEM(i) ? PCI_BASE_ADDRESS_SPACE_MEMORY : \
> >>>>> + PCI_BASE_ADDRESS_SPACE_IO)
> >>>>> +
> >>>>> +static int pci_testdev_start(IOTest *test)
> >>>>> +{
> >>>>> + test->hdr->count = 0;
> >>>>> + if (!test->hasnotifier) {
> >>>>> + return 0;
> >>>>> + }
> >>>>> + event_notifier_test_and_clear(&test->notifier);
> >>>>> + memory_region_add_eventfd(test->mr,
> >>>>> + le32_to_cpu(test->hdr->offset),
> >>>>> + test->size,
> >>>>> + test->match_data,
> >>>>> + test->hdr->data,
> >>>>> + &test->notifier);
> >>>>> + return 0;
> >>>>> +}
> >>>>> +
> >>>>> +static void pci_testdev_stop(IOTest *test)
> >>>>> +{
> >>>>> + if (!test->hasnotifier) {
> >>>>> + return;
> >>>>> + }
> >>>>> + memory_region_del_eventfd(test->mr,
> >>>>> + le32_to_cpu(test->hdr->offset),
> >>>>> + test->size,
> >>>>> + test->match_data,
> >>>>> + test->hdr->data,
> >>>>> + &test->notifier);
> >>>>> +}
> >>>>> +
> >>>>> +static void
> >>>>> +pci_testdev_reset(PCITestDevState *d)
> >>>>> +{
> >>>>> + if (d->current == -1) {
> >>>>> + return;
> >>>>> + }
> >>>>> + pci_testdev_stop(&d->tests[d->current]);
> >>>>> + d->current = -1;
> >>>>> +}
> >>>>> +
> >>>>> +static void pci_testdev_inc(IOTest *test, unsigned inc)
> >>>>> +{
> >>>>> + uint32_t c = le32_to_cpu(test->hdr->count);
> >>>>> + test->hdr->count = cpu_to_le32(c + inc);
> >>>>> +}
> >>>>> +
> >>>>> +static void
> >>>>> +pci_testdev_write(void *opaque, hwaddr addr, uint64_t val,
> >>>>> + unsigned size, int type)
> >>>>> +{
> >>>>> + PCITestDevState *d = opaque;
> >>>>> + IOTest *test;
> >>>>> + int t, r;
> >>>>> +
> >>>>> + if (addr == offsetof(PCITestDevHdr, test)) {
> >>>>> + pci_testdev_reset(d);
> >>>>> + if (val >= IOTEST_MAX_TEST) {
> >>>>> + return;
> >>>>> + }
> >>>>> + t = type * IOTEST_MAX_TEST + val;
> >>>>> + r = pci_testdev_start(&d->tests[t]);
> >>>>> + if (r < 0) {
> >>>>> + return;
> >>>>> + }
> >>>>> + d->current = t;
> >>>>> + return;
> >>>>> + }
> >>>>> + if (d->current < 0) {
> >>>>> + return;
> >>>>> + }
> >>>>> + test = &d->tests[d->current];
> >>>>> + if (addr != le32_to_cpu(test->hdr->offset)) {
> >>>>> + return;
> >>>>> + }
> >>>>> + if (test->match_data && test->size != size) {
> >>>>> + return;
> >>>>> + }
> >>>>> + if (test->match_data && val != test->hdr->data) {
> >>>>> + return;
> >>>>> + }
> >>>>> + pci_testdev_inc(test, 1);
> >>>>> +}
> >>>>> +
> >>>>> +static uint64_t
> >>>>> +pci_testdev_read(void *opaque, hwaddr addr, unsigned size)
> >>>>> +{
> >>>>> + PCITestDevState *d = opaque;
> >>>>> + const char *buf;
> >>>>> + IOTest *test;
> >>>>> + if (d->current < 0) {
> >>>>> + return 0;
> >>>>> + }
> >>>>> + test = &d->tests[d->current];
> >>>>> + buf = (const char *)test->hdr;
> >>>>> + if (addr + size >= test->bufsize) {
> >>>>> + return 0;
> >>>>> + }
> >>>>> + if (test->hasnotifier) {
> >>>>> + event_notifier_test_and_clear(&test->notifier);
> >>>>> + }
> >>>>> + return buf[addr];
> >>>>> +}
> >>>>> +
> >>>>> +static void
> >>>>> +pci_testdev_mmio_write(void *opaque, hwaddr addr, uint64_t val,
> >>>>> + unsigned size)
> >>>>> +{
> >>>>> + pci_testdev_write(opaque, addr, val, size, 0);
> >>>>> +}
> >>>>> +
> >>>>> +static void
> >>>>> +pci_testdev_pio_write(void *opaque, hwaddr addr, uint64_t val,
> >>>>> + unsigned size)
> >>>>> +{
> >>>>> + pci_testdev_write(opaque, addr, val, size, 1);
> >>>>> +}
> >>>>> +
> >>>>> +static const MemoryRegionOps pci_testdev_mmio_ops = {
> >>>>> + .read = pci_testdev_read,
> >>>>> + .write = pci_testdev_mmio_write,
> >>>>> + .endianness = DEVICE_LITTLE_ENDIAN,
> >>>>> + .impl = {
> >>>>> + .min_access_size = 1,
> >>>>> + .max_access_size = 1,
> >>>>> + },
> >>>>> +};
> >>>>> +
> >>>>> +static const MemoryRegionOps pci_testdev_pio_ops = {
> >>>>> + .read = pci_testdev_read,
> >>>>> + .write = pci_testdev_pio_write,
> >>>>> + .endianness = DEVICE_LITTLE_ENDIAN,
> >>>>> + .impl = {
> >>>>> + .min_access_size = 1,
> >>>>> + .max_access_size = 1,
> >>>>> + },
> >>>>> +};
> >>>>> +
> >>>>> +static int pci_testdev_init(PCIDevice *pci_dev)
> >>>>> +{
> >>>>> + PCITestDevState *d = DO_UPCAST(PCITestDevState, dev, pci_dev);
> >>>>> + uint8_t *pci_conf;
> >>>>> + char *name;
> >>>>> + int r, i;
> >>>>> +
> >>>>> + pci_conf = d->dev.config;
> >>>>> +
> >>>>> + pci_conf[PCI_INTERRUPT_PIN] = 0; /* no interrupt pin */
> >>>>> +
> >>>>> + memory_region_init_io(&d->mmio, &pci_testdev_mmio_ops, d,
> >>>>> + "pci-testdev-mmio", IOTEST_MEMSIZE * 2);
> >>>>> + memory_region_init_io(&d->portio, &pci_testdev_pio_ops, d,
> >>>>> + "pci-testdev-portio", IOTEST_IOSIZE * 2);
> >>>>> + pci_register_bar(&d->dev, 0, PCI_BASE_ADDRESS_SPACE_MEMORY, &d->mmio);
> >>>>> + pci_register_bar(&d->dev, 1, PCI_BASE_ADDRESS_SPACE_IO, &d->portio);
> >>>>> +
> >>>>> + d->current = -1;
> >>>>> + d->tests = g_malloc0(IOTEST_MAX * sizeof *d->tests);
> >>>>> + for (i = 0; i < IOTEST_MAX; ++i) {
> >>>>> + IOTest *test = &d->tests[i];
> >>>>> + name = g_strdup_printf("%s-%s", IOTEST_TYPE(i), IOTEST_TEST(i));
> >>>>> + test->bufsize = sizeof(PCITestDevHdr) + strlen(name) + 1;
> >>>>> + test->hdr = g_malloc0(test->bufsize);
> >>>>> + memcpy(test->hdr->name, name, strlen(name) + 1);
> >>>>> + g_free(name);
> >>>>> + test->hdr->offset = cpu_to_le32(IOTEST_SIZE(i) + i * IOTEST_ACCESS_WIDTH);
> >>>>> + test->size = IOTEST_ACCESS_WIDTH;
> >>>>> + test->match_data = strcmp(IOTEST_TEST(i), "wildcard-eventfd");
> >>>>> + test->hdr->test = i;
> >>>>> + test->hdr->data = test->match_data ? IOTEST_DATAMATCH : IOTEST_NOMATCH;
> >>>>> + test->hdr->width = IOTEST_ACCESS_WIDTH;
> >>>>> + test->mr = IOTEST_REGION(d, i);
> >>>>> + if (!strcmp(IOTEST_TEST(i), "no-eventfd")) {
> >>>>> + test->hasnotifier = false;
> >>>>> + continue;
> >>>>> + }
> >>>>> + r = event_notifier_init(&test->notifier, 0);
> >>>>> + assert(r >= 0);
> >>>>> + test->hasnotifier = true;
> >>>>> + }
> >>>>> +
> >>>>> + return 0;
> >>>>> +}
> >>>>> +
> >>>>> +static void
> >>>>> +pci_testdev_uninit(PCIDevice *dev)
> >>>>> +{
> >>>>> + PCITestDevState *d = DO_UPCAST(PCITestDevState, dev, dev);
> >>>>> + int i;
> >>>>> +
> >>>>> + pci_testdev_reset(d);
> >>>>> + for (i = 0; i < IOTEST_MAX; ++i) {
> >>>>> + if (d->tests[i].hasnotifier) {
> >>>>> + event_notifier_cleanup(&d->tests[i].notifier);
> >>>>> + }
> >>>>> + g_free(d->tests[i].hdr);
> >>>>> + }
> >>>>> + g_free(d->tests);
> >>>>> + memory_region_destroy(&d->mmio);
> >>>>> + memory_region_destroy(&d->portio);
> >>>>> +}
> >>>>> +
> >>>>> +static void qdev_pci_testdev_reset(DeviceState *dev)
> >>>>> +{
> >>>>> + PCITestDevState *d = DO_UPCAST(PCITestDevState, dev.qdev, dev);
> >>>>> + pci_testdev_reset(d);
> >>>>> +}
> >>>>> +
> >>>>> +static void pci_testdev_class_init(ObjectClass *klass, void *data)
> >>>>> +{
> >>>>> + DeviceClass *dc = DEVICE_CLASS(klass);
> >>>>> + PCIDeviceClass *k = PCI_DEVICE_CLASS(klass);
> >>>>> +
> >>>>> + k->init = pci_testdev_init;
> >>>>> + k->exit = pci_testdev_uninit;
> >>>>> + k->vendor_id = PCI_VENDOR_ID_REDHAT;
> >>>>> + k->device_id = PCI_DEVICE_ID_REDHAT_TEST;
> >>>>> + k->revision = 0x00;
> >>>>> + k->class_id = PCI_CLASS_OTHERS;
> >>>>> + dc->desc = "PCI Test Device";
> >>>>> + dc->reset = qdev_pci_testdev_reset;
> >>>>> +}
> >>>>> +
> >>>>> +static const TypeInfo pci_testdev_info = {
> >>>>> + .name = "pci-testdev",
> >>>>> + .parent = TYPE_PCI_DEVICE,
> >>>>> + .instance_size = sizeof(PCITestDevState),
> >>>>> + .class_init = pci_testdev_class_init,
> >>>>> +};
> >>>>> +
> >>>>> +static void pci_testdev_register_types(void)
> >>>>> +{
> >>>>> + type_register_static(&pci_testdev_info);
> >>>>> +}
> >>>>> +
> >>>>> +type_init(pci_testdev_register_types)
> >>>>> diff --git a/hw/pci/pci.h b/hw/pci/pci.h
> >>>>> index 774369c..d81198c 100644
> >>>>> --- a/hw/pci/pci.h
> >>>>> +++ b/hw/pci/pci.h
> >>>>> @@ -84,6 +84,7 @@
> >>>>> #define PCI_DEVICE_ID_REDHAT_SERIAL 0x0002
> >>>>> #define PCI_DEVICE_ID_REDHAT_SERIAL2 0x0003
> >>>>> #define PCI_DEVICE_ID_REDHAT_SERIAL4 0x0004
> >>>>> +#define PCI_DEVICE_ID_REDHAT_TEST 0x0005
> >>>>> #define PCI_DEVICE_ID_REDHAT_QXL 0x0100
> >>>>>
> >>>>> #define FMT_PCIBUS PRIx64
> >>>>>
next prev parent reply other threads:[~2013-04-03 10:32 UTC|newest]
Thread overview: 28+ messages / expand[flat|nested] mbox.gz Atom feed top
2013-04-03 8:59 [Qemu-devel] [PATCH 0/4] kvm-unittests: add pci PORT IO and MMIO speed tests Michael S. Tsirkin
2013-04-03 8:59 ` [Qemu-devel] [PATCH 1/4] kvm: remove unused APIs Michael S. Tsirkin
2013-04-03 8:59 ` [Qemu-devel] [PATCH 2/4] kvm: support any size for pio eventfd Michael S. Tsirkin
2013-04-03 8:59 ` [Qemu-devel] [PATCH 3/4] kvm: support non datamatch ioeventfd Michael S. Tsirkin
2013-04-03 8:59 ` [Qemu-devel] [PATCH 4/4] pci: add pci test device Michael S. Tsirkin
2013-04-03 9:28 ` Paolo Bonzini
2013-04-03 9:45 ` Michael S. Tsirkin
2013-04-03 9:53 ` Paolo Bonzini
2013-04-03 10:22 ` Michael S. Tsirkin
2013-04-03 10:25 ` Paolo Bonzini
2013-04-03 10:33 ` Michael S. Tsirkin [this message]
2013-04-03 10:34 ` Paolo Bonzini
2013-04-03 10:38 ` Michael S. Tsirkin
2013-04-03 11:48 ` Paolo Bonzini
2013-04-03 12:00 ` Michael S. Tsirkin
2013-04-03 12:05 ` Paolo Bonzini
2013-04-03 14:06 ` Michael S. Tsirkin
2013-04-03 14:08 ` Paolo Bonzini
2013-04-03 14:28 ` Michael S. Tsirkin
2013-04-03 15:46 ` Paolo Bonzini
2013-04-03 17:04 ` Michael S. Tsirkin
2013-04-03 17:10 ` Paolo Bonzini
2013-04-03 14:33 ` Michael S. Tsirkin
2013-04-03 15:09 ` Michael S. Tsirkin
2013-04-03 15:43 ` Paolo Bonzini
2013-04-03 18:39 ` Michael S. Tsirkin
2013-04-03 19:59 ` Paolo Bonzini
-- strict thread matches above, loose matches on Subject: below --
2013-04-18 8:00 [Qemu-devel] [PULL 0/4] " Michael S. Tsirkin
2013-04-18 8:00 ` [Qemu-devel] [PATCH 4/4] " Michael S. Tsirkin
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20130403103301.GG18179@redhat.com \
--to=mst@redhat.com \
--cc=gleb@redhat.com \
--cc=kvm@vger.kernel.org \
--cc=mtosatti@redhat.com \
--cc=pbonzini@redhat.com \
--cc=qemu-devel@nongnu.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).