diff options
author | Thanos Makatos <thanos.makatos@nutanix.com> | 2020-12-04 07:14:59 -0500 |
---|---|---|
committer | Thanos Makatos <tmakatos@gmail.com> | 2020-12-07 11:53:52 +0000 |
commit | 03181093d2ff01daee166252996cc6c0a01db483 (patch) | |
tree | 66330c205ca094b90a85d91377bc260585d3e2c6 | |
parent | 56ad046f32b46b24d0e743dc41b400d937eb2c9f (diff) | |
download | libvfio-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.md | 45 | ||||
-rw-r--r-- | samples/CMakeLists.txt | 3 | ||||
-rw-r--r-- | samples/lspci.c | 78 |
3 files changed, 120 insertions, 6 deletions
@@ -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: */ |