aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorThanos Makatos <thanos.makatos@nutanix.com>2020-12-04 07:14:59 -0500
committerThanos Makatos <tmakatos@gmail.com>2020-12-07 11:53:52 +0000
commit03181093d2ff01daee166252996cc6c0a01db483 (patch)
tree66330c205ca094b90a85d91377bc260585d3e2c6
parent56ad046f32b46b24d0e743dc41b400d937eb2c9f (diff)
downloadlibvfio-user-03181093d2ff01daee166252996cc6c0a01db483.zip
libvfio-user-03181093d2ff01daee166252996cc6c0a01db483.tar.gz
libvfio-user-03181093d2ff01daee166252996cc6c0a01db483.tar.bz2
add lspci sample
Signed-off-by: Thanos Makatos <thanos.makatos@nutanix.com>
-rw-r--r--README.md45
-rw-r--r--samples/CMakeLists.txt3
-rw-r--r--samples/lspci.c78
3 files changed, 120 insertions, 6 deletions
diff --git a/README.md b/README.md
index c8293c4..7be9c7a 100644
--- a/README.md
+++ b/README.md
@@ -92,12 +92,45 @@ Finally build your program and link with `libvfio-user.so`.
Example
=======
-The [samples directory](./samples/) contains a client/server implementation. The
-server implements a device that can be programmed to trigger interrupts (INTx)
-to the client. This is done by writing the desired time in seconds since Epoch.
-The server then trigger an eventfd-based IRQ and then a message-based one (in
-order to demonstrate how it's done when passing of file descriptors isn't
-possible/desirable).
+The [samples directory](./samples/) contains various libvfio-user samples.
+
+
+lspci
+-----
+
+[lspci](./samples/lspci.c) implements an example of how to dump the PCI header
+of a libvfio-user device and examine it with lspci(8):
+
+ # lspci -vv -F <(build/dbg/samples/lspci)
+ 00:00.0 Non-VGA unclassified device: Device 0000:0000
+ Control: I/O- Mem- BusMaster- SpecCycle- MemWINV- VGASnoop- ParErr- Stepping- SERR- FastB2B- DisINTx-
+ Status: Cap+ 66MHz- UDF- FastB2B- ParErr- DEVSEL=fast >TAbort- <TAbort- <MAbort- >SERR- <PERR- INTx-
+ Region 0: I/O ports at <unassigned> [disabled]
+ Region 1: I/O ports at <unassigned> [disabled]
+ Region 2: I/O ports at <unassigned> [disabled]
+ Region 3: I/O ports at <unassigned> [disabled]
+ Region 4: I/O ports at <unassigned> [disabled]
+ Region 5: I/O ports at <unassigned> [disabled]
+ Capabilities: [40] Power Management version 0
+ Flags: PMEClk- DSI- D1- D2- AuxCurrent=0mA PME(D0-,D1-,D2-,D3hot-,D3cold-)
+ Status: D0 NoSoftRst+ PME-Enable- DSel=0 DScale=0 PME-
+
+The above sample implements a very simple PCI device that supports the Power
+Management PCI capability. The sample can be trivially modified to change the
+PCI configuration space header and add more PCI capabilities.
+
+
+Client/Server Implementation
+----------------------------
+
+[Client](./samples/client.c)/[server](./samples/server.c) implements a basic
+client/server model where basic tasks are performed.
+
+The server implements a device that can be programmed to trigger interrupts
+(INTx) to the client. This is done by writing the desired time in seconds since
+Epoch. The server then trigger an eventfd-based IRQ and then a message-based
+one (in order to demonstrate how it's done when passing of file descriptors
+isn't possible/desirable).
The client excercises all commands in the vfio-user protocol, and then proceeds
to perform live migration. The client spawns the destination server (this would
diff --git a/samples/CMakeLists.txt b/samples/CMakeLists.txt
index 3395c6c..6f1173b 100644
--- a/samples/CMakeLists.txt
+++ b/samples/CMakeLists.txt
@@ -40,3 +40,6 @@ target_link_libraries(null vfio-user pthread)
add_executable(gpio-pci-idio-16 gpio-pci-idio-16.c)
target_link_libraries(gpio-pci-idio-16 vfio-user)
+
+add_executable(lspci lspci.c)
+target_link_libraries(lspci vfio-user)
diff --git a/samples/lspci.c b/samples/lspci.c
new file mode 100644
index 0000000..674164b
--- /dev/null
+++ b/samples/lspci.c
@@ -0,0 +1,78 @@
+/*
+ * Copyright (c) 2019 Nutanix Inc. All rights reserved.
+ *
+ * Authors: Thanos Makatos <thanos@nutanix.com>
+ * Swapnil Ingle <swapnil.ingle@nutanix.com>
+ * Felipe Franciosi <felipe@nutanix.com>
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * * Neither the name of Nutanix nor the names of its contributors may be
+ * used to endorse or promote products derived from this software without
+ * specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+ * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+ * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
+ * DAMAGE.
+ *
+ */
+
+#include <stdio.h>
+#include <err.h>
+#include <stdlib.h>
+#include <assert.h>
+
+#include "libvfio-user.h"
+
+int main(void)
+{
+ int i, j;
+ char *buf;
+ const int bytes_per_line = 0x10;
+ vfu_pci_hdr_id_t id = { 0 };
+ vfu_pci_hdr_ss_t ss = { 0 };
+ vfu_pci_hdr_cc_t cc = { 0 };
+ vfu_cap_t pm = {.pm = {.hdr.id = PCI_CAP_ID_PM, .pmcs.nsfrst = 0x1}};
+ vfu_cap_t *caps[1] = {&pm};
+ vfu_ctx_t *vfu_ctx = vfu_create_ctx(VFU_TRANS_SOCK, "",
+ LIBVFIO_USER_FLAG_ATTACH_NB, NULL,
+ VFU_DEV_TYPE_PCI);
+ if (vfu_ctx == NULL) {
+ err(EXIT_FAILURE, "failed to create libvfio-user context");
+ }
+ if (vfu_pci_setup_config_hdr(vfu_ctx, id, ss, cc, VFU_PCI_TYPE_CONVENTIONAL, 0) < 0) {
+ err(EXIT_FAILURE, "failed to setup PCI configuration space header");
+ }
+ if (vfu_pci_setup_caps(vfu_ctx, caps, 1) < 0) {
+ err(EXIT_FAILURE, "failed to setup PCI capabilities");
+ }
+ if (vfu_realize_ctx(vfu_ctx) < 0) {
+ err(EXIT_FAILURE, "failed to realize device");
+ }
+ buf = (char*)vfu_pci_get_config_space(vfu_ctx);;
+ printf("00:00.0 bogus PCI device\n");
+ for (i = 0; i < PCI_CFG_SPACE_SIZE / bytes_per_line; i++) {
+ printf("%02x:", i * bytes_per_line);
+ for (j = 0; j < bytes_per_line; j++) {
+ printf(" %02x", buf[i * bytes_per_line + j] & 0xff);
+ }
+ printf("\n");
+ }
+
+ return 0;
+}
+
+/* ex: set tabstop=4 shiftwidth=4 softtabstop=4 expandtab: */