1f61dddd7SZhenzhong Duan /*
2f61dddd7SZhenzhong Duan * VFIO BASE CONTAINER
3f61dddd7SZhenzhong Duan *
4f61dddd7SZhenzhong Duan * Copyright (C) 2023 Intel Corporation.
5f61dddd7SZhenzhong Duan * Copyright Red Hat, Inc. 2023
6f61dddd7SZhenzhong Duan *
7f61dddd7SZhenzhong Duan * Authors: Yi Liu <yi.l.liu@intel.com>
8f61dddd7SZhenzhong Duan * Eric Auger <eric.auger@redhat.com>
9f61dddd7SZhenzhong Duan *
10f61dddd7SZhenzhong Duan * SPDX-License-Identifier: GPL-2.0-or-later
11f61dddd7SZhenzhong Duan */
12f61dddd7SZhenzhong Duan
13f61dddd7SZhenzhong Duan #ifndef HW_VFIO_VFIO_CONTAINER_BASE_H
14f61dddd7SZhenzhong Duan #define HW_VFIO_VFIO_CONTAINER_BASE_H
15f61dddd7SZhenzhong Duan
16f61dddd7SZhenzhong Duan #include "exec/memory.h"
17f61dddd7SZhenzhong Duan
18f61dddd7SZhenzhong Duan typedef struct VFIODevice VFIODevice;
19fdaa774eSCédric Le Goater typedef struct VFIOIOMMUClass VFIOIOMMUClass;
20f61dddd7SZhenzhong Duan
21f61dddd7SZhenzhong Duan typedef struct {
22f61dddd7SZhenzhong Duan unsigned long *bitmap;
23f61dddd7SZhenzhong Duan hwaddr size;
24f61dddd7SZhenzhong Duan hwaddr pages;
25f61dddd7SZhenzhong Duan } VFIOBitmap;
26f61dddd7SZhenzhong Duan
27e5597063SEric Auger typedef struct VFIOAddressSpace {
28e5597063SEric Auger AddressSpace *as;
29e5597063SEric Auger QLIST_HEAD(, VFIOContainerBase) containers;
30e5597063SEric Auger QLIST_ENTRY(VFIOAddressSpace) list;
31e5597063SEric Auger } VFIOAddressSpace;
32e5597063SEric Auger
33f61dddd7SZhenzhong Duan /*
34f61dddd7SZhenzhong Duan * This is the base object for vfio container backends
35f61dddd7SZhenzhong Duan */
36f61dddd7SZhenzhong Duan typedef struct VFIOContainerBase {
37504d297eSCédric Le Goater Object parent;
38e5597063SEric Auger VFIOAddressSpace *space;
39c7b313d3SEric Auger MemoryListener listener;
40c7b313d3SEric Auger Error *error;
41c7b313d3SEric Auger bool initialized;
424d6b9501SEric Auger uint64_t dirty_pgsizes;
434d6b9501SEric Auger uint64_t max_dirty_bitmap_size;
447ab1cb74SEric Auger unsigned long pgsizes;
457ab1cb74SEric Auger unsigned int dma_max_mappings;
46bb424490SEric Auger bool dirty_pages_supported;
47dddf83abSEric Auger QLIST_HEAD(, VFIOGuestIOMMU) giommu_list;
48dc74a4b0SZhenzhong Duan QLIST_HEAD(, VFIORamDiscardListener) vrdl_list;
49e5597063SEric Auger QLIST_ENTRY(VFIOContainerBase) next;
503e6015d1SZhenzhong Duan QLIST_HEAD(, VFIODevice) device_list;
51f79baf8cSZhenzhong Duan GList *iova_ranges;
520cb51c18SSteve Sistare NotifierWithReturn cpr_reboot_notifier;
53f61dddd7SZhenzhong Duan } VFIOContainerBase;
54f61dddd7SZhenzhong Duan
55dddf83abSEric Auger typedef struct VFIOGuestIOMMU {
56dddf83abSEric Auger VFIOContainerBase *bcontainer;
57dddf83abSEric Auger IOMMUMemoryRegion *iommu_mr;
58dddf83abSEric Auger hwaddr iommu_offset;
59dddf83abSEric Auger IOMMUNotifier n;
60dddf83abSEric Auger QLIST_ENTRY(VFIOGuestIOMMU) giommu_next;
61dddf83abSEric Auger } VFIOGuestIOMMU;
62dddf83abSEric Auger
63dc74a4b0SZhenzhong Duan typedef struct VFIORamDiscardListener {
64dc74a4b0SZhenzhong Duan VFIOContainerBase *bcontainer;
65dc74a4b0SZhenzhong Duan MemoryRegion *mr;
66dc74a4b0SZhenzhong Duan hwaddr offset_within_address_space;
67dc74a4b0SZhenzhong Duan hwaddr size;
68dc74a4b0SZhenzhong Duan uint64_t granularity;
69dc74a4b0SZhenzhong Duan RamDiscardListener listener;
70dc74a4b0SZhenzhong Duan QLIST_ENTRY(VFIORamDiscardListener) next;
71dc74a4b0SZhenzhong Duan } VFIORamDiscardListener;
72dc74a4b0SZhenzhong Duan
73b08501a9SEric Auger int vfio_container_dma_map(VFIOContainerBase *bcontainer,
74b08501a9SEric Auger hwaddr iova, ram_addr_t size,
75b08501a9SEric Auger void *vaddr, bool readonly);
76b08501a9SEric Auger int vfio_container_dma_unmap(VFIOContainerBase *bcontainer,
77b08501a9SEric Auger hwaddr iova, ram_addr_t size,
78b08501a9SEric Auger IOMMUTLBEntry *iotlb);
7933e4c22fSZhenzhong Duan bool vfio_container_add_section_window(VFIOContainerBase *bcontainer,
80233309e8SZhenzhong Duan MemoryRegionSection *section,
81233309e8SZhenzhong Duan Error **errp);
82233309e8SZhenzhong Duan void vfio_container_del_section_window(VFIOContainerBase *bcontainer,
83233309e8SZhenzhong Duan MemoryRegionSection *section);
84bb424490SEric Auger int vfio_container_set_dirty_page_tracking(VFIOContainerBase *bcontainer,
85836bb308SCédric Le Goater bool start, Error **errp);
864517c33cSZhenzhong Duan int vfio_container_query_dirty_bitmap(const VFIOContainerBase *bcontainer,
872da5f9e4SCédric Le Goater VFIOBitmap *vbmap, hwaddr iova, hwaddr size, Error **errp);
88b08501a9SEric Auger
893966bca5SEric Auger GList *vfio_container_get_iova_ranges(const VFIOContainerBase *bcontainer);
903966bca5SEric Auger
91*8fe0ebe1SEric Auger static inline uint64_t
vfio_container_get_page_size_mask(const VFIOContainerBase * bcontainer)92*8fe0ebe1SEric Auger vfio_container_get_page_size_mask(const VFIOContainerBase *bcontainer)
93*8fe0ebe1SEric Auger {
94*8fe0ebe1SEric Auger assert(bcontainer);
95*8fe0ebe1SEric Auger return bcontainer->pgsizes;
96*8fe0ebe1SEric Auger }
97*8fe0ebe1SEric Auger
98fdaa774eSCédric Le Goater #define TYPE_VFIO_IOMMU "vfio-iommu"
999812feefSCédric Le Goater #define TYPE_VFIO_IOMMU_LEGACY TYPE_VFIO_IOMMU "-legacy"
100f221f641SCédric Le Goater #define TYPE_VFIO_IOMMU_SPAPR TYPE_VFIO_IOMMU "-spapr"
101ce5f6d49SCédric Le Goater #define TYPE_VFIO_IOMMU_IOMMUFD TYPE_VFIO_IOMMU "-iommufd"
102fdaa774eSCédric Le Goater
103504d297eSCédric Le Goater OBJECT_DECLARE_TYPE(VFIOContainerBase, VFIOIOMMUClass, VFIO_IOMMU)
104fdaa774eSCédric Le Goater
105fdaa774eSCédric Le Goater struct VFIOIOMMUClass {
106504d297eSCédric Le Goater ObjectClass parent_class;
107fdaa774eSCédric Le Goater
108a7fd91b8SZhenzhong Duan /* Properties */
109a7fd91b8SZhenzhong Duan const char *hiod_typename;
110a7fd91b8SZhenzhong Duan
111f61dddd7SZhenzhong Duan /* basic feature */
11235b25cf4SZhenzhong Duan bool (*setup)(VFIOContainerBase *bcontainer, Error **errp);
1134517c33cSZhenzhong Duan int (*dma_map)(const VFIOContainerBase *bcontainer,
114f61dddd7SZhenzhong Duan hwaddr iova, ram_addr_t size,
115f61dddd7SZhenzhong Duan void *vaddr, bool readonly);
1164517c33cSZhenzhong Duan int (*dma_unmap)(const VFIOContainerBase *bcontainer,
117f61dddd7SZhenzhong Duan hwaddr iova, ram_addr_t size,
118f61dddd7SZhenzhong Duan IOMMUTLBEntry *iotlb);
119b7754835SZhenzhong Duan bool (*attach_device)(const char *name, VFIODevice *vbasedev,
120f61dddd7SZhenzhong Duan AddressSpace *as, Error **errp);
121f61dddd7SZhenzhong Duan void (*detach_device)(VFIODevice *vbasedev);
122836bb308SCédric Le Goater
123f61dddd7SZhenzhong Duan /* migration feature */
124836bb308SCédric Le Goater
125836bb308SCédric Le Goater /**
126836bb308SCédric Le Goater * @set_dirty_page_tracking
127836bb308SCédric Le Goater *
128836bb308SCédric Le Goater * Start or stop dirty pages tracking on VFIO container
129836bb308SCédric Le Goater *
130836bb308SCédric Le Goater * @bcontainer: #VFIOContainerBase on which to de/activate dirty
131836bb308SCédric Le Goater * page tracking
132836bb308SCédric Le Goater * @start: indicates whether to start or stop dirty pages tracking
133836bb308SCédric Le Goater * @errp: pointer to Error*, to store an error if it happens.
134836bb308SCédric Le Goater *
135836bb308SCédric Le Goater * Returns zero to indicate success and negative for error
136836bb308SCédric Le Goater */
1374517c33cSZhenzhong Duan int (*set_dirty_page_tracking)(const VFIOContainerBase *bcontainer,
138836bb308SCédric Le Goater bool start, Error **errp);
1392da5f9e4SCédric Le Goater /**
1402da5f9e4SCédric Le Goater * @query_dirty_bitmap
1412da5f9e4SCédric Le Goater *
1422da5f9e4SCédric Le Goater * Get bitmap of dirty pages from container
1432da5f9e4SCédric Le Goater *
1442da5f9e4SCédric Le Goater * @bcontainer: #VFIOContainerBase from which to get dirty pages
1452da5f9e4SCédric Le Goater * @vbmap: #VFIOBitmap internal bitmap structure
1462da5f9e4SCédric Le Goater * @iova: iova base address
1472da5f9e4SCédric Le Goater * @size: size of iova range
1482da5f9e4SCédric Le Goater * @errp: pointer to Error*, to store an error if it happens.
1492da5f9e4SCédric Le Goater *
1502da5f9e4SCédric Le Goater * Returns zero to indicate success and negative for error
1512da5f9e4SCédric Le Goater */
1524517c33cSZhenzhong Duan int (*query_dirty_bitmap)(const VFIOContainerBase *bcontainer,
1532da5f9e4SCédric Le Goater VFIOBitmap *vbmap, hwaddr iova, hwaddr size, Error **errp);
154c328e7e8SZhenzhong Duan /* PCI specific */
155c328e7e8SZhenzhong Duan int (*pci_hot_reset)(VFIODevice *vbasedev, bool single);
156c328e7e8SZhenzhong Duan
1579b7d38bfSZhenzhong Duan /* SPAPR specific */
15833e4c22fSZhenzhong Duan bool (*add_window)(VFIOContainerBase *bcontainer,
1599b7d38bfSZhenzhong Duan MemoryRegionSection *section,
1609b7d38bfSZhenzhong Duan Error **errp);
1619b7d38bfSZhenzhong Duan void (*del_window)(VFIOContainerBase *bcontainer,
1629b7d38bfSZhenzhong Duan MemoryRegionSection *section);
163001a013eSCédric Le Goater void (*release)(VFIOContainerBase *bcontainer);
164f61dddd7SZhenzhong Duan };
165f61dddd7SZhenzhong Duan #endif /* HW_VFIO_VFIO_CONTAINER_BASE_H */
166