VFIO updates 2018-03-13
- Display support for vGPUs (Gerd Hoffmann) - Enable new kernel support for mmaps overlapping MSI-X vector table, disable MSI-X emulation on POWER (Alexey Kardashevskiy) -----BEGIN PGP SIGNATURE----- Version: GnuPG v2.0.14 (GNU/Linux) iQIcBAABAgAGBQJaqCshAAoJECObm247sIsivFQP/3Y0e0mW4fbf1Z0uRiduZ3Bh bAKD52n8Id3alAmelo1P7OUS8LP0vdw9lLAJz9m3VB+Nvye0XewFn0/9SVLnL022 aCBIj9cGg57hraTd6PUfsfG8T1HryDeQqE/PRIYEaqf9htBZ51VM2EY3+n2O7Bcy TVDnac0IOaWb8FrXK+lGpLP6dPFQ0Z+mhkAvDR7v7FZGEy000D71+gKOe4aIAMib X/NJa42O1x9cbDE39AEpVMub/QYwtRD/HkcZwNylyY4YKIjsMtxPYYz2Ndt+wm0r e21s/4iy2a1HXen4OBc2KAnpVoZyGnXya6KZ0dzirt8Q1dK606Az/HxifVb/CJQZ L10NSfYTWNPfPlf0gYZmPlZeZDuNeZaHi8bZGqTquL4AAOTdqtQOMEoT+68qd74s 00B9XrA2Jmjvpya2pHQSqvAsuSqZ1E2iRMGMC9r/+UYb54OnXnrl9fOw/TtbrLMm isbB1Ll0hDcJvacikYJ5ck+lGrFLHbYpWtBIrokBHbw9fwl1I9aY5MloBYZwmcTz e65L7M4MvA/y6gSlCi6DS1fmdZ5ecNnBkuW6YseDI884P4c4+YDMOhs/B/rO16v6 In2fSWvb+DSuAPmn+6QDV0msJg34J2oykbKcdDK7F1g7By/yDNVVgHPoWhi4EGBA u09A/2nM0VwhXYNrElJI =LVMf -----END PGP SIGNATURE----- Merge remote-tracking branch 'remotes/awilliam/tags/vfio-update-20180313.0' into staging VFIO updates 2018-03-13 - Display support for vGPUs (Gerd Hoffmann) - Enable new kernel support for mmaps overlapping MSI-X vector table, disable MSI-X emulation on POWER (Alexey Kardashevskiy) # gpg: Signature made Tue 13 Mar 2018 19:48:49 GMT # gpg: using RSA key 239B9B6E3BB08B22 # gpg: Good signature from "Alex Williamson <alex.williamson@redhat.com>" # gpg: aka "Alex Williamson <alex@shazbot.org>" # gpg: aka "Alex Williamson <alwillia@redhat.com>" # gpg: aka "Alex Williamson <alex.l.williamson@gmail.com>" # Primary key fingerprint: 42F6 C04E 540B D1A9 9E7B 8A90 239B 9B6E 3BB0 8B22 * remotes/awilliam/tags/vfio-update-20180313.0: ppc/spapr, vfio: Turn off MSIX emulation for VFIO devices vfio-pci: Allow mmap of MSIX BAR vfio/pci: Relax DMA map errors for MMIO regions vfio/display: adding dmabuf support vfio/display: adding region support vfio/display: core & wireup vfio/common: cleanup in vfio_region_finalize secondary-vga: properly close QemuConsole on unplug console: minimal hotplug suport ui/pixman: add qemu_drm_format_to_pixman() standard-headers: add drm/drm_fourcc.h Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
commit
a57946ff2a
@ -292,6 +292,14 @@ static void pci_secondary_vga_realize(PCIDevice *dev, Error **errp)
|
|||||||
pci_register_bar(&d->dev, 2, PCI_BASE_ADDRESS_SPACE_MEMORY, &d->mmio);
|
pci_register_bar(&d->dev, 2, PCI_BASE_ADDRESS_SPACE_MEMORY, &d->mmio);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void pci_secondary_vga_exit(PCIDevice *dev)
|
||||||
|
{
|
||||||
|
PCIVGAState *d = PCI_VGA(dev);
|
||||||
|
VGACommonState *s = &d->vga;
|
||||||
|
|
||||||
|
graphic_console_close(s->con);
|
||||||
|
}
|
||||||
|
|
||||||
static void pci_secondary_vga_init(Object *obj)
|
static void pci_secondary_vga_init(Object *obj)
|
||||||
{
|
{
|
||||||
/* Expose framebuffer byteorder via QOM */
|
/* Expose framebuffer byteorder via QOM */
|
||||||
@ -361,6 +369,7 @@ static void secondary_class_init(ObjectClass *klass, void *data)
|
|||||||
PCIDeviceClass *k = PCI_DEVICE_CLASS(klass);
|
PCIDeviceClass *k = PCI_DEVICE_CLASS(klass);
|
||||||
|
|
||||||
k->realize = pci_secondary_vga_realize;
|
k->realize = pci_secondary_vga_realize;
|
||||||
|
k->exit = pci_secondary_vga_exit;
|
||||||
k->class_id = PCI_CLASS_DISPLAY_OTHER;
|
k->class_id = PCI_CLASS_DISPLAY_OTHER;
|
||||||
dc->props = secondary_pci_properties;
|
dc->props = secondary_pci_properties;
|
||||||
dc->reset = pci_secondary_vga_reset;
|
dc->reset = pci_secondary_vga_reset;
|
||||||
|
@ -2855,6 +2855,11 @@ static void spapr_set_modern_hotplug_events(Object *obj, bool value,
|
|||||||
spapr->use_hotplug_event_source = value;
|
spapr->use_hotplug_event_source = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool spapr_get_msix_emulation(Object *obj, Error **errp)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
static char *spapr_get_resize_hpt(Object *obj, Error **errp)
|
static char *spapr_get_resize_hpt(Object *obj, Error **errp)
|
||||||
{
|
{
|
||||||
sPAPRMachineState *spapr = SPAPR_MACHINE(obj);
|
sPAPRMachineState *spapr = SPAPR_MACHINE(obj);
|
||||||
@ -2936,6 +2941,8 @@ static void spapr_instance_init(Object *obj)
|
|||||||
object_property_set_description(obj, "vsmt",
|
object_property_set_description(obj, "vsmt",
|
||||||
"Virtual SMT: KVM behaves as if this were"
|
"Virtual SMT: KVM behaves as if this were"
|
||||||
" the host's SMT mode", &error_abort);
|
" the host's SMT mode", &error_abort);
|
||||||
|
object_property_add_bool(obj, "vfio-no-msix-emulation",
|
||||||
|
spapr_get_msix_emulation, NULL, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void spapr_machine_finalizefn(Object *obj)
|
static void spapr_machine_finalizefn(Object *obj)
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
ifeq ($(CONFIG_LINUX), y)
|
ifeq ($(CONFIG_LINUX), y)
|
||||||
obj-$(CONFIG_SOFTMMU) += common.o
|
obj-$(CONFIG_SOFTMMU) += common.o
|
||||||
obj-$(CONFIG_PCI) += pci.o pci-quirks.o
|
obj-$(CONFIG_PCI) += pci.o pci-quirks.o display.o
|
||||||
obj-$(CONFIG_VFIO_CCW) += ccw.o
|
obj-$(CONFIG_VFIO_CCW) += ccw.o
|
||||||
obj-$(CONFIG_SOFTMMU) += platform.o
|
obj-$(CONFIG_SOFTMMU) += platform.o
|
||||||
obj-$(CONFIG_VFIO_XGMAC) += calxeda-xgmac.o
|
obj-$(CONFIG_VFIO_XGMAC) += calxeda-xgmac.o
|
||||||
|
@ -544,18 +544,40 @@ static void vfio_listener_region_add(MemoryListener *listener,
|
|||||||
|
|
||||||
llsize = int128_sub(llend, int128_make64(iova));
|
llsize = int128_sub(llend, int128_make64(iova));
|
||||||
|
|
||||||
|
if (memory_region_is_ram_device(section->mr)) {
|
||||||
|
hwaddr pgmask = (1ULL << ctz64(hostwin->iova_pgsizes)) - 1;
|
||||||
|
|
||||||
|
if ((iova & pgmask) || (int128_get64(llsize) & pgmask)) {
|
||||||
|
error_report("Region 0x%"HWADDR_PRIx"..0x%"HWADDR_PRIx
|
||||||
|
" is not aligned to 0x%"HWADDR_PRIx
|
||||||
|
" and cannot be mapped for DMA",
|
||||||
|
section->offset_within_region,
|
||||||
|
int128_getlo(section->size),
|
||||||
|
pgmask + 1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ret = vfio_dma_map(container, iova, int128_get64(llsize),
|
ret = vfio_dma_map(container, iova, int128_get64(llsize),
|
||||||
vaddr, section->readonly);
|
vaddr, section->readonly);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
error_report("vfio_dma_map(%p, 0x%"HWADDR_PRIx", "
|
error_report("vfio_dma_map(%p, 0x%"HWADDR_PRIx", "
|
||||||
"0x%"HWADDR_PRIx", %p) = %d (%m)",
|
"0x%"HWADDR_PRIx", %p) = %d (%m)",
|
||||||
container, iova, int128_get64(llsize), vaddr, ret);
|
container, iova, int128_get64(llsize), vaddr, ret);
|
||||||
|
if (memory_region_is_ram_device(section->mr)) {
|
||||||
|
/* Allow unexpected mappings not to be fatal for RAM devices */
|
||||||
|
return;
|
||||||
|
}
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
fail:
|
fail:
|
||||||
|
if (memory_region_is_ram_device(section->mr)) {
|
||||||
|
error_report("failed to vfio_dma_map. pci p2p may not work");
|
||||||
|
return;
|
||||||
|
}
|
||||||
/*
|
/*
|
||||||
* On the initfn path, store the first error in the container so we
|
* On the initfn path, store the first error in the container so we
|
||||||
* can gracefully fail. Runtime, there's not much we can do other
|
* can gracefully fail. Runtime, there's not much we can do other
|
||||||
@ -577,6 +599,7 @@ static void vfio_listener_region_del(MemoryListener *listener,
|
|||||||
hwaddr iova, end;
|
hwaddr iova, end;
|
||||||
Int128 llend, llsize;
|
Int128 llend, llsize;
|
||||||
int ret;
|
int ret;
|
||||||
|
bool try_unmap = true;
|
||||||
|
|
||||||
if (vfio_listener_skipped_section(section)) {
|
if (vfio_listener_skipped_section(section)) {
|
||||||
trace_vfio_listener_region_del_skip(
|
trace_vfio_listener_region_del_skip(
|
||||||
@ -629,14 +652,34 @@ static void vfio_listener_region_del(MemoryListener *listener,
|
|||||||
|
|
||||||
trace_vfio_listener_region_del(iova, end);
|
trace_vfio_listener_region_del(iova, end);
|
||||||
|
|
||||||
ret = vfio_dma_unmap(container, iova, int128_get64(llsize));
|
if (memory_region_is_ram_device(section->mr)) {
|
||||||
memory_region_unref(section->mr);
|
hwaddr pgmask;
|
||||||
if (ret) {
|
VFIOHostDMAWindow *hostwin;
|
||||||
error_report("vfio_dma_unmap(%p, 0x%"HWADDR_PRIx", "
|
bool hostwin_found = false;
|
||||||
"0x%"HWADDR_PRIx") = %d (%m)",
|
|
||||||
container, iova, int128_get64(llsize), ret);
|
QLIST_FOREACH(hostwin, &container->hostwin_list, hostwin_next) {
|
||||||
|
if (hostwin->min_iova <= iova && end <= hostwin->max_iova) {
|
||||||
|
hostwin_found = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
assert(hostwin_found); /* or region_add() would have failed */
|
||||||
|
|
||||||
|
pgmask = (1ULL << ctz64(hostwin->iova_pgsizes)) - 1;
|
||||||
|
try_unmap = !((iova & pgmask) || (int128_get64(llsize) & pgmask));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (try_unmap) {
|
||||||
|
ret = vfio_dma_unmap(container, iova, int128_get64(llsize));
|
||||||
|
if (ret) {
|
||||||
|
error_report("vfio_dma_unmap(%p, 0x%"HWADDR_PRIx", "
|
||||||
|
"0x%"HWADDR_PRIx") = %d (%m)",
|
||||||
|
container, iova, int128_get64(llsize), ret);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
memory_region_unref(section->mr);
|
||||||
|
|
||||||
if (container->iommu_type == VFIO_SPAPR_TCE_v2_IOMMU) {
|
if (container->iommu_type == VFIO_SPAPR_TCE_v2_IOMMU) {
|
||||||
vfio_spapr_remove_window(container,
|
vfio_spapr_remove_window(container,
|
||||||
section->offset_within_address_space);
|
section->offset_within_address_space);
|
||||||
@ -858,6 +901,13 @@ void vfio_region_finalize(VFIORegion *region)
|
|||||||
g_free(region->mmaps);
|
g_free(region->mmaps);
|
||||||
|
|
||||||
trace_vfio_region_finalize(region->vbasedev->name, region->nr);
|
trace_vfio_region_finalize(region->vbasedev->name, region->nr);
|
||||||
|
|
||||||
|
region->mem = NULL;
|
||||||
|
region->mmaps = NULL;
|
||||||
|
region->nr_mmaps = 0;
|
||||||
|
region->size = 0;
|
||||||
|
region->flags = 0;
|
||||||
|
region->nr = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void vfio_region_mmaps_set_enabled(VFIORegion *region, bool enabled)
|
void vfio_region_mmaps_set_enabled(VFIORegion *region, bool enabled)
|
||||||
@ -1421,6 +1471,21 @@ int vfio_get_dev_region_info(VFIODevice *vbasedev, uint32_t type,
|
|||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool vfio_has_region_cap(VFIODevice *vbasedev, int region, uint16_t cap_type)
|
||||||
|
{
|
||||||
|
struct vfio_region_info *info = NULL;
|
||||||
|
bool ret = false;
|
||||||
|
|
||||||
|
if (!vfio_get_region_info(vbasedev, region, &info)) {
|
||||||
|
if (vfio_get_region_info_cap(info, cap_type)) {
|
||||||
|
ret = true;
|
||||||
|
}
|
||||||
|
g_free(info);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Interfaces for IBM EEH (Enhanced Error Handling)
|
* Interfaces for IBM EEH (Enhanced Error Handling)
|
||||||
*/
|
*/
|
||||||
|
347
hw/vfio/display.c
Normal file
347
hw/vfio/display.c
Normal file
@ -0,0 +1,347 @@
|
|||||||
|
/*
|
||||||
|
* display support for mdev based vgpu devices
|
||||||
|
*
|
||||||
|
* Copyright Red Hat, Inc. 2017
|
||||||
|
*
|
||||||
|
* Authors:
|
||||||
|
* Gerd Hoffmann
|
||||||
|
*
|
||||||
|
* This work is licensed under the terms of the GNU GPL, version 2. See
|
||||||
|
* the COPYING file in the top-level directory.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "qemu/osdep.h"
|
||||||
|
#include <linux/vfio.h>
|
||||||
|
#include <sys/ioctl.h>
|
||||||
|
|
||||||
|
#include "sysemu/sysemu.h"
|
||||||
|
#include "ui/console.h"
|
||||||
|
#include "qapi/error.h"
|
||||||
|
#include "pci.h"
|
||||||
|
|
||||||
|
#ifndef DRM_PLANE_TYPE_PRIMARY
|
||||||
|
# define DRM_PLANE_TYPE_PRIMARY 1
|
||||||
|
# define DRM_PLANE_TYPE_CURSOR 2
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static void vfio_display_update_cursor(VFIODMABuf *dmabuf,
|
||||||
|
struct vfio_device_gfx_plane_info *plane)
|
||||||
|
{
|
||||||
|
if (dmabuf->pos_x != plane->x_pos || dmabuf->pos_y != plane->y_pos) {
|
||||||
|
dmabuf->pos_x = plane->x_pos;
|
||||||
|
dmabuf->pos_y = plane->y_pos;
|
||||||
|
dmabuf->pos_updates++;
|
||||||
|
}
|
||||||
|
if (dmabuf->hot_x != plane->x_hot || dmabuf->hot_y != plane->y_hot) {
|
||||||
|
dmabuf->hot_x = plane->x_hot;
|
||||||
|
dmabuf->hot_y = plane->y_hot;
|
||||||
|
dmabuf->hot_updates++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static VFIODMABuf *vfio_display_get_dmabuf(VFIOPCIDevice *vdev,
|
||||||
|
uint32_t plane_type)
|
||||||
|
{
|
||||||
|
VFIODisplay *dpy = vdev->dpy;
|
||||||
|
struct vfio_device_gfx_plane_info plane;
|
||||||
|
VFIODMABuf *dmabuf;
|
||||||
|
int fd, ret;
|
||||||
|
|
||||||
|
memset(&plane, 0, sizeof(plane));
|
||||||
|
plane.argsz = sizeof(plane);
|
||||||
|
plane.flags = VFIO_GFX_PLANE_TYPE_DMABUF;
|
||||||
|
plane.drm_plane_type = plane_type;
|
||||||
|
ret = ioctl(vdev->vbasedev.fd, VFIO_DEVICE_QUERY_GFX_PLANE, &plane);
|
||||||
|
if (ret < 0) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
if (!plane.drm_format || !plane.size) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
QTAILQ_FOREACH(dmabuf, &dpy->dmabuf.bufs, next) {
|
||||||
|
if (dmabuf->dmabuf_id == plane.dmabuf_id) {
|
||||||
|
/* found in list, move to head, return it */
|
||||||
|
QTAILQ_REMOVE(&dpy->dmabuf.bufs, dmabuf, next);
|
||||||
|
QTAILQ_INSERT_HEAD(&dpy->dmabuf.bufs, dmabuf, next);
|
||||||
|
if (plane_type == DRM_PLANE_TYPE_CURSOR) {
|
||||||
|
vfio_display_update_cursor(dmabuf, &plane);
|
||||||
|
}
|
||||||
|
return dmabuf;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fd = ioctl(vdev->vbasedev.fd, VFIO_DEVICE_GET_GFX_DMABUF, &plane.dmabuf_id);
|
||||||
|
if (fd < 0) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
dmabuf = g_new0(VFIODMABuf, 1);
|
||||||
|
dmabuf->dmabuf_id = plane.dmabuf_id;
|
||||||
|
dmabuf->buf.width = plane.width;
|
||||||
|
dmabuf->buf.height = plane.height;
|
||||||
|
dmabuf->buf.stride = plane.stride;
|
||||||
|
dmabuf->buf.fourcc = plane.drm_format;
|
||||||
|
dmabuf->buf.fd = fd;
|
||||||
|
if (plane_type == DRM_PLANE_TYPE_CURSOR) {
|
||||||
|
vfio_display_update_cursor(dmabuf, &plane);
|
||||||
|
}
|
||||||
|
|
||||||
|
QTAILQ_INSERT_HEAD(&dpy->dmabuf.bufs, dmabuf, next);
|
||||||
|
return dmabuf;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void vfio_display_free_one_dmabuf(VFIODisplay *dpy, VFIODMABuf *dmabuf)
|
||||||
|
{
|
||||||
|
QTAILQ_REMOVE(&dpy->dmabuf.bufs, dmabuf, next);
|
||||||
|
dpy_gl_release_dmabuf(dpy->con, &dmabuf->buf);
|
||||||
|
close(dmabuf->buf.fd);
|
||||||
|
g_free(dmabuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void vfio_display_free_dmabufs(VFIOPCIDevice *vdev)
|
||||||
|
{
|
||||||
|
VFIODisplay *dpy = vdev->dpy;
|
||||||
|
VFIODMABuf *dmabuf, *tmp;
|
||||||
|
uint32_t keep = 5;
|
||||||
|
|
||||||
|
QTAILQ_FOREACH_SAFE(dmabuf, &dpy->dmabuf.bufs, next, tmp) {
|
||||||
|
if (keep > 0) {
|
||||||
|
keep--;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
assert(dmabuf != dpy->dmabuf.primary);
|
||||||
|
vfio_display_free_one_dmabuf(dpy, dmabuf);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void vfio_display_dmabuf_update(void *opaque)
|
||||||
|
{
|
||||||
|
VFIOPCIDevice *vdev = opaque;
|
||||||
|
VFIODisplay *dpy = vdev->dpy;
|
||||||
|
VFIODMABuf *primary, *cursor;
|
||||||
|
bool free_bufs = false, new_cursor = false;;
|
||||||
|
|
||||||
|
primary = vfio_display_get_dmabuf(vdev, DRM_PLANE_TYPE_PRIMARY);
|
||||||
|
if (primary == NULL) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dpy->dmabuf.primary != primary) {
|
||||||
|
dpy->dmabuf.primary = primary;
|
||||||
|
qemu_console_resize(dpy->con,
|
||||||
|
primary->buf.width, primary->buf.height);
|
||||||
|
dpy_gl_scanout_dmabuf(dpy->con, &primary->buf);
|
||||||
|
free_bufs = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
cursor = vfio_display_get_dmabuf(vdev, DRM_PLANE_TYPE_CURSOR);
|
||||||
|
if (dpy->dmabuf.cursor != cursor) {
|
||||||
|
dpy->dmabuf.cursor = cursor;
|
||||||
|
new_cursor = true;
|
||||||
|
free_bufs = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cursor && (new_cursor || cursor->hot_updates)) {
|
||||||
|
bool have_hot = (cursor->hot_x != 0xffffffff &&
|
||||||
|
cursor->hot_y != 0xffffffff);
|
||||||
|
dpy_gl_cursor_dmabuf(dpy->con, &cursor->buf, have_hot,
|
||||||
|
cursor->hot_x, cursor->hot_y);
|
||||||
|
cursor->hot_updates = 0;
|
||||||
|
} else if (!cursor && new_cursor) {
|
||||||
|
dpy_gl_cursor_dmabuf(dpy->con, NULL, false, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cursor && cursor->pos_updates) {
|
||||||
|
dpy_gl_cursor_position(dpy->con,
|
||||||
|
cursor->pos_x,
|
||||||
|
cursor->pos_y);
|
||||||
|
cursor->pos_updates = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
dpy_gl_update(dpy->con, 0, 0, primary->buf.width, primary->buf.height);
|
||||||
|
|
||||||
|
if (free_bufs) {
|
||||||
|
vfio_display_free_dmabufs(vdev);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static const GraphicHwOps vfio_display_dmabuf_ops = {
|
||||||
|
.gfx_update = vfio_display_dmabuf_update,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int vfio_display_dmabuf_init(VFIOPCIDevice *vdev, Error **errp)
|
||||||
|
{
|
||||||
|
if (!display_opengl) {
|
||||||
|
error_setg(errp, "vfio-display-dmabuf: opengl not available");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
vdev->dpy = g_new0(VFIODisplay, 1);
|
||||||
|
vdev->dpy->con = graphic_console_init(DEVICE(vdev), 0,
|
||||||
|
&vfio_display_dmabuf_ops,
|
||||||
|
vdev);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void vfio_display_dmabuf_exit(VFIODisplay *dpy)
|
||||||
|
{
|
||||||
|
VFIODMABuf *dmabuf;
|
||||||
|
|
||||||
|
if (QTAILQ_EMPTY(&dpy->dmabuf.bufs)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
while ((dmabuf = QTAILQ_FIRST(&dpy->dmabuf.bufs)) != NULL) {
|
||||||
|
vfio_display_free_one_dmabuf(dpy, dmabuf);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ---------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
static void vfio_display_region_update(void *opaque)
|
||||||
|
{
|
||||||
|
VFIOPCIDevice *vdev = opaque;
|
||||||
|
VFIODisplay *dpy = vdev->dpy;
|
||||||
|
struct vfio_device_gfx_plane_info plane = {
|
||||||
|
.argsz = sizeof(plane),
|
||||||
|
.flags = VFIO_GFX_PLANE_TYPE_REGION
|
||||||
|
};
|
||||||
|
pixman_format_code_t format;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = ioctl(vdev->vbasedev.fd, VFIO_DEVICE_QUERY_GFX_PLANE, &plane);
|
||||||
|
if (ret < 0) {
|
||||||
|
error_report("ioctl VFIO_DEVICE_QUERY_GFX_PLANE: %s",
|
||||||
|
strerror(errno));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!plane.drm_format || !plane.size) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
format = qemu_drm_format_to_pixman(plane.drm_format);
|
||||||
|
if (!format) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dpy->region.buffer.size &&
|
||||||
|
dpy->region.buffer.nr != plane.region_index) {
|
||||||
|
/* region changed */
|
||||||
|
vfio_region_exit(&dpy->region.buffer);
|
||||||
|
vfio_region_finalize(&dpy->region.buffer);
|
||||||
|
dpy->region.surface = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dpy->region.surface &&
|
||||||
|
(surface_width(dpy->region.surface) != plane.width ||
|
||||||
|
surface_height(dpy->region.surface) != plane.height ||
|
||||||
|
surface_format(dpy->region.surface) != format)) {
|
||||||
|
/* size changed */
|
||||||
|
dpy->region.surface = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!dpy->region.buffer.size) {
|
||||||
|
/* mmap region */
|
||||||
|
ret = vfio_region_setup(OBJECT(vdev), &vdev->vbasedev,
|
||||||
|
&dpy->region.buffer,
|
||||||
|
plane.region_index,
|
||||||
|
"display");
|
||||||
|
if (ret != 0) {
|
||||||
|
error_report("%s: vfio_region_setup(%d): %s",
|
||||||
|
__func__, plane.region_index, strerror(-ret));
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
ret = vfio_region_mmap(&dpy->region.buffer);
|
||||||
|
if (ret != 0) {
|
||||||
|
error_report("%s: vfio_region_mmap(%d): %s", __func__,
|
||||||
|
plane.region_index, strerror(-ret));
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
assert(dpy->region.buffer.mmaps[0].mmap != NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dpy->region.surface == NULL) {
|
||||||
|
/* create surface */
|
||||||
|
dpy->region.surface = qemu_create_displaysurface_from
|
||||||
|
(plane.width, plane.height, format,
|
||||||
|
plane.stride, dpy->region.buffer.mmaps[0].mmap);
|
||||||
|
dpy_gfx_replace_surface(dpy->con, dpy->region.surface);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* full screen update */
|
||||||
|
dpy_gfx_update(dpy->con, 0, 0,
|
||||||
|
surface_width(dpy->region.surface),
|
||||||
|
surface_height(dpy->region.surface));
|
||||||
|
return;
|
||||||
|
|
||||||
|
err:
|
||||||
|
vfio_region_exit(&dpy->region.buffer);
|
||||||
|
vfio_region_finalize(&dpy->region.buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const GraphicHwOps vfio_display_region_ops = {
|
||||||
|
.gfx_update = vfio_display_region_update,
|
||||||
|
};
|
||||||
|
|
||||||
|
static int vfio_display_region_init(VFIOPCIDevice *vdev, Error **errp)
|
||||||
|
{
|
||||||
|
vdev->dpy = g_new0(VFIODisplay, 1);
|
||||||
|
vdev->dpy->con = graphic_console_init(DEVICE(vdev), 0,
|
||||||
|
&vfio_display_region_ops,
|
||||||
|
vdev);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void vfio_display_region_exit(VFIODisplay *dpy)
|
||||||
|
{
|
||||||
|
if (!dpy->region.buffer.size) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
vfio_region_exit(&dpy->region.buffer);
|
||||||
|
vfio_region_finalize(&dpy->region.buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ---------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
int vfio_display_probe(VFIOPCIDevice *vdev, Error **errp)
|
||||||
|
{
|
||||||
|
struct vfio_device_gfx_plane_info probe;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
memset(&probe, 0, sizeof(probe));
|
||||||
|
probe.argsz = sizeof(probe);
|
||||||
|
probe.flags = VFIO_GFX_PLANE_TYPE_PROBE | VFIO_GFX_PLANE_TYPE_DMABUF;
|
||||||
|
ret = ioctl(vdev->vbasedev.fd, VFIO_DEVICE_QUERY_GFX_PLANE, &probe);
|
||||||
|
if (ret == 0) {
|
||||||
|
return vfio_display_dmabuf_init(vdev, errp);
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(&probe, 0, sizeof(probe));
|
||||||
|
probe.argsz = sizeof(probe);
|
||||||
|
probe.flags = VFIO_GFX_PLANE_TYPE_PROBE | VFIO_GFX_PLANE_TYPE_REGION;
|
||||||
|
ret = ioctl(vdev->vbasedev.fd, VFIO_DEVICE_QUERY_GFX_PLANE, &probe);
|
||||||
|
if (ret == 0) {
|
||||||
|
return vfio_display_region_init(vdev, errp);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (vdev->display == ON_OFF_AUTO_AUTO) {
|
||||||
|
/* not an error in automatic mode */
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
error_setg(errp, "vfio: device doesn't support any (known) display method");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void vfio_display_finalize(VFIOPCIDevice *vdev)
|
||||||
|
{
|
||||||
|
if (!vdev->dpy) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
graphic_console_close(vdev->dpy->con);
|
||||||
|
vfio_display_dmabuf_exit(vdev->dpy);
|
||||||
|
vfio_display_region_exit(vdev->dpy);
|
||||||
|
g_free(vdev->dpy);
|
||||||
|
}
|
@ -1294,6 +1294,15 @@ static void vfio_pci_fixup_msix_region(VFIOPCIDevice *vdev)
|
|||||||
off_t start, end;
|
off_t start, end;
|
||||||
VFIORegion *region = &vdev->bars[vdev->msix->table_bar].region;
|
VFIORegion *region = &vdev->bars[vdev->msix->table_bar].region;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the host driver allows mapping of a MSIX data, we are going to
|
||||||
|
* do map the entire BAR and emulate MSIX table on top of that.
|
||||||
|
*/
|
||||||
|
if (vfio_has_region_cap(&vdev->vbasedev, region->nr,
|
||||||
|
VFIO_REGION_INFO_CAP_MSIX_MAPPABLE)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We expect to find a single mmap covering the whole BAR, anything else
|
* We expect to find a single mmap covering the whole BAR, anything else
|
||||||
* means it's either unsupported or already setup.
|
* means it's either unsupported or already setup.
|
||||||
@ -1572,6 +1581,19 @@ static int vfio_msix_setup(VFIOPCIDevice *vdev, int pos, Error **errp)
|
|||||||
*/
|
*/
|
||||||
memory_region_set_enabled(&vdev->pdev.msix_pba_mmio, false);
|
memory_region_set_enabled(&vdev->pdev.msix_pba_mmio, false);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The emulated machine may provide a paravirt interface for MSIX setup
|
||||||
|
* so it is not strictly necessary to emulate MSIX here. This becomes
|
||||||
|
* helpful when frequently accessed MMIO registers are located in
|
||||||
|
* subpages adjacent to the MSIX table but the MSIX data containing page
|
||||||
|
* cannot be mapped because of a host page size bigger than the MSIX table
|
||||||
|
* alignment.
|
||||||
|
*/
|
||||||
|
if (object_property_get_bool(OBJECT(qdev_get_machine()),
|
||||||
|
"vfio-no-msix-emulation", NULL)) {
|
||||||
|
memory_region_set_enabled(&vdev->pdev.msix_table_mmio, false);
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3015,6 +3037,13 @@ static void vfio_realize(PCIDevice *pdev, Error **errp)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (vdev->display != ON_OFF_AUTO_OFF) {
|
||||||
|
ret = vfio_display_probe(vdev, errp);
|
||||||
|
if (ret) {
|
||||||
|
goto out_teardown;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
vfio_register_err_notifier(vdev);
|
vfio_register_err_notifier(vdev);
|
||||||
vfio_register_req_notifier(vdev);
|
vfio_register_req_notifier(vdev);
|
||||||
vfio_setup_resetfn_quirk(vdev);
|
vfio_setup_resetfn_quirk(vdev);
|
||||||
@ -3035,6 +3064,7 @@ static void vfio_instance_finalize(Object *obj)
|
|||||||
VFIOPCIDevice *vdev = DO_UPCAST(VFIOPCIDevice, pdev, pci_dev);
|
VFIOPCIDevice *vdev = DO_UPCAST(VFIOPCIDevice, pdev, pci_dev);
|
||||||
VFIOGroup *group = vdev->vbasedev.group;
|
VFIOGroup *group = vdev->vbasedev.group;
|
||||||
|
|
||||||
|
vfio_display_finalize(vdev);
|
||||||
vfio_bars_finalize(vdev);
|
vfio_bars_finalize(vdev);
|
||||||
g_free(vdev->emulated_config_bits);
|
g_free(vdev->emulated_config_bits);
|
||||||
g_free(vdev->rom);
|
g_free(vdev->rom);
|
||||||
@ -3123,6 +3153,8 @@ static void vfio_instance_init(Object *obj)
|
|||||||
static Property vfio_pci_dev_properties[] = {
|
static Property vfio_pci_dev_properties[] = {
|
||||||
DEFINE_PROP_PCI_HOST_DEVADDR("host", VFIOPCIDevice, host),
|
DEFINE_PROP_PCI_HOST_DEVADDR("host", VFIOPCIDevice, host),
|
||||||
DEFINE_PROP_STRING("sysfsdev", VFIOPCIDevice, vbasedev.sysfsdev),
|
DEFINE_PROP_STRING("sysfsdev", VFIOPCIDevice, vbasedev.sysfsdev),
|
||||||
|
DEFINE_PROP_ON_OFF_AUTO("display", VFIOPCIDevice,
|
||||||
|
display, ON_OFF_AUTO_AUTO),
|
||||||
DEFINE_PROP_UINT32("x-intx-mmap-timeout-ms", VFIOPCIDevice,
|
DEFINE_PROP_UINT32("x-intx-mmap-timeout-ms", VFIOPCIDevice,
|
||||||
intx.mmap_timeout, 1100),
|
intx.mmap_timeout, 1100),
|
||||||
DEFINE_PROP_BIT("x-vga", VFIOPCIDevice, features,
|
DEFINE_PROP_BIT("x-vga", VFIOPCIDevice, features,
|
||||||
|
@ -133,6 +133,7 @@ typedef struct VFIOPCIDevice {
|
|||||||
#define VFIO_FEATURE_ENABLE_IGD_OPREGION_BIT 2
|
#define VFIO_FEATURE_ENABLE_IGD_OPREGION_BIT 2
|
||||||
#define VFIO_FEATURE_ENABLE_IGD_OPREGION \
|
#define VFIO_FEATURE_ENABLE_IGD_OPREGION \
|
||||||
(1 << VFIO_FEATURE_ENABLE_IGD_OPREGION_BIT)
|
(1 << VFIO_FEATURE_ENABLE_IGD_OPREGION_BIT)
|
||||||
|
OnOffAuto display;
|
||||||
int32_t bootindex;
|
int32_t bootindex;
|
||||||
uint32_t igd_gms;
|
uint32_t igd_gms;
|
||||||
OffAutoPCIBAR msix_relo;
|
OffAutoPCIBAR msix_relo;
|
||||||
@ -147,6 +148,7 @@ typedef struct VFIOPCIDevice {
|
|||||||
bool no_kvm_msi;
|
bool no_kvm_msi;
|
||||||
bool no_kvm_msix;
|
bool no_kvm_msix;
|
||||||
bool no_geforce_quirks;
|
bool no_geforce_quirks;
|
||||||
|
VFIODisplay *dpy;
|
||||||
} VFIOPCIDevice;
|
} VFIOPCIDevice;
|
||||||
|
|
||||||
uint32_t vfio_pci_read_config(PCIDevice *pdev, uint32_t addr, int len);
|
uint32_t vfio_pci_read_config(PCIDevice *pdev, uint32_t addr, int len);
|
||||||
@ -174,4 +176,7 @@ int vfio_pci_igd_opregion_init(VFIOPCIDevice *vdev,
|
|||||||
struct vfio_region_info *info,
|
struct vfio_region_info *info,
|
||||||
Error **errp);
|
Error **errp);
|
||||||
|
|
||||||
|
int vfio_display_probe(VFIOPCIDevice *vdev, Error **errp);
|
||||||
|
void vfio_display_finalize(VFIOPCIDevice *vdev);
|
||||||
|
|
||||||
#endif /* HW_VFIO_VFIO_PCI_H */
|
#endif /* HW_VFIO_VFIO_PCI_H */
|
||||||
|
@ -26,6 +26,7 @@
|
|||||||
#include "exec/memory.h"
|
#include "exec/memory.h"
|
||||||
#include "qemu/queue.h"
|
#include "qemu/queue.h"
|
||||||
#include "qemu/notify.h"
|
#include "qemu/notify.h"
|
||||||
|
#include "ui/console.h"
|
||||||
#ifdef CONFIG_LINUX
|
#ifdef CONFIG_LINUX
|
||||||
#include <linux/vfio.h>
|
#include <linux/vfio.h>
|
||||||
#endif
|
#endif
|
||||||
@ -142,6 +143,27 @@ typedef struct VFIOGroup {
|
|||||||
QLIST_ENTRY(VFIOGroup) container_next;
|
QLIST_ENTRY(VFIOGroup) container_next;
|
||||||
} VFIOGroup;
|
} VFIOGroup;
|
||||||
|
|
||||||
|
typedef struct VFIODMABuf {
|
||||||
|
QemuDmaBuf buf;
|
||||||
|
uint32_t pos_x, pos_y, pos_updates;
|
||||||
|
uint32_t hot_x, hot_y, hot_updates;
|
||||||
|
int dmabuf_id;
|
||||||
|
QTAILQ_ENTRY(VFIODMABuf) next;
|
||||||
|
} VFIODMABuf;
|
||||||
|
|
||||||
|
typedef struct VFIODisplay {
|
||||||
|
QemuConsole *con;
|
||||||
|
struct {
|
||||||
|
VFIORegion buffer;
|
||||||
|
DisplaySurface *surface;
|
||||||
|
} region;
|
||||||
|
struct {
|
||||||
|
QTAILQ_HEAD(, VFIODMABuf) bufs;
|
||||||
|
VFIODMABuf *primary;
|
||||||
|
VFIODMABuf *cursor;
|
||||||
|
} dmabuf;
|
||||||
|
} VFIODisplay;
|
||||||
|
|
||||||
void vfio_put_base_device(VFIODevice *vbasedev);
|
void vfio_put_base_device(VFIODevice *vbasedev);
|
||||||
void vfio_disable_irqindex(VFIODevice *vbasedev, int index);
|
void vfio_disable_irqindex(VFIODevice *vbasedev, int index);
|
||||||
void vfio_unmask_single_irqindex(VFIODevice *vbasedev, int index);
|
void vfio_unmask_single_irqindex(VFIODevice *vbasedev, int index);
|
||||||
@ -171,6 +193,7 @@ int vfio_get_region_info(VFIODevice *vbasedev, int index,
|
|||||||
struct vfio_region_info **info);
|
struct vfio_region_info **info);
|
||||||
int vfio_get_dev_region_info(VFIODevice *vbasedev, uint32_t type,
|
int vfio_get_dev_region_info(VFIODevice *vbasedev, uint32_t type,
|
||||||
uint32_t subtype, struct vfio_region_info **info);
|
uint32_t subtype, struct vfio_region_info **info);
|
||||||
|
bool vfio_has_region_cap(VFIODevice *vbasedev, int region, uint16_t cap_type);
|
||||||
#endif
|
#endif
|
||||||
extern const MemoryListener vfio_prereg_listener;
|
extern const MemoryListener vfio_prereg_listener;
|
||||||
|
|
||||||
|
411
include/standard-headers/drm/drm_fourcc.h
Normal file
411
include/standard-headers/drm/drm_fourcc.h
Normal file
@ -0,0 +1,411 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2011 Intel Corporation
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
||||||
|
* copy of this software and associated documentation files (the "Software"),
|
||||||
|
* to deal in the Software without restriction, including without limitation
|
||||||
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||||
|
* and/or sell copies of the Software, and to permit persons to whom the
|
||||||
|
* Software is furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice (including the next
|
||||||
|
* paragraph) shall be included in all copies or substantial portions of the
|
||||||
|
* Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||||
|
* VA LINUX SYSTEMS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
||||||
|
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
||||||
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
||||||
|
* OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef DRM_FOURCC_H
|
||||||
|
#define DRM_FOURCC_H
|
||||||
|
|
||||||
|
|
||||||
|
#if defined(__cplusplus)
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define fourcc_code(a, b, c, d) ((uint32_t)(a) | ((uint32_t)(b) << 8) | \
|
||||||
|
((uint32_t)(c) << 16) | ((uint32_t)(d) << 24))
|
||||||
|
|
||||||
|
#define DRM_FORMAT_BIG_ENDIAN (1<<31) /* format is big endian instead of little endian */
|
||||||
|
|
||||||
|
/* color index */
|
||||||
|
#define DRM_FORMAT_C8 fourcc_code('C', '8', ' ', ' ') /* [7:0] C */
|
||||||
|
|
||||||
|
/* 8 bpp Red */
|
||||||
|
#define DRM_FORMAT_R8 fourcc_code('R', '8', ' ', ' ') /* [7:0] R */
|
||||||
|
|
||||||
|
/* 16 bpp Red */
|
||||||
|
#define DRM_FORMAT_R16 fourcc_code('R', '1', '6', ' ') /* [15:0] R little endian */
|
||||||
|
|
||||||
|
/* 16 bpp RG */
|
||||||
|
#define DRM_FORMAT_RG88 fourcc_code('R', 'G', '8', '8') /* [15:0] R:G 8:8 little endian */
|
||||||
|
#define DRM_FORMAT_GR88 fourcc_code('G', 'R', '8', '8') /* [15:0] G:R 8:8 little endian */
|
||||||
|
|
||||||
|
/* 32 bpp RG */
|
||||||
|
#define DRM_FORMAT_RG1616 fourcc_code('R', 'G', '3', '2') /* [31:0] R:G 16:16 little endian */
|
||||||
|
#define DRM_FORMAT_GR1616 fourcc_code('G', 'R', '3', '2') /* [31:0] G:R 16:16 little endian */
|
||||||
|
|
||||||
|
/* 8 bpp RGB */
|
||||||
|
#define DRM_FORMAT_RGB332 fourcc_code('R', 'G', 'B', '8') /* [7:0] R:G:B 3:3:2 */
|
||||||
|
#define DRM_FORMAT_BGR233 fourcc_code('B', 'G', 'R', '8') /* [7:0] B:G:R 2:3:3 */
|
||||||
|
|
||||||
|
/* 16 bpp RGB */
|
||||||
|
#define DRM_FORMAT_XRGB4444 fourcc_code('X', 'R', '1', '2') /* [15:0] x:R:G:B 4:4:4:4 little endian */
|
||||||
|
#define DRM_FORMAT_XBGR4444 fourcc_code('X', 'B', '1', '2') /* [15:0] x:B:G:R 4:4:4:4 little endian */
|
||||||
|
#define DRM_FORMAT_RGBX4444 fourcc_code('R', 'X', '1', '2') /* [15:0] R:G:B:x 4:4:4:4 little endian */
|
||||||
|
#define DRM_FORMAT_BGRX4444 fourcc_code('B', 'X', '1', '2') /* [15:0] B:G:R:x 4:4:4:4 little endian */
|
||||||
|
|
||||||
|
#define DRM_FORMAT_ARGB4444 fourcc_code('A', 'R', '1', '2') /* [15:0] A:R:G:B 4:4:4:4 little endian */
|
||||||
|
#define DRM_FORMAT_ABGR4444 fourcc_code('A', 'B', '1', '2') /* [15:0] A:B:G:R 4:4:4:4 little endian */
|
||||||
|
#define DRM_FORMAT_RGBA4444 fourcc_code('R', 'A', '1', '2') /* [15:0] R:G:B:A 4:4:4:4 little endian */
|
||||||
|
#define DRM_FORMAT_BGRA4444 fourcc_code('B', 'A', '1', '2') /* [15:0] B:G:R:A 4:4:4:4 little endian */
|
||||||
|
|
||||||
|
#define DRM_FORMAT_XRGB1555 fourcc_code('X', 'R', '1', '5') /* [15:0] x:R:G:B 1:5:5:5 little endian */
|
||||||
|
#define DRM_FORMAT_XBGR1555 fourcc_code('X', 'B', '1', '5') /* [15:0] x:B:G:R 1:5:5:5 little endian */
|
||||||
|
#define DRM_FORMAT_RGBX5551 fourcc_code('R', 'X', '1', '5') /* [15:0] R:G:B:x 5:5:5:1 little endian */
|
||||||
|
#define DRM_FORMAT_BGRX5551 fourcc_code('B', 'X', '1', '5') /* [15:0] B:G:R:x 5:5:5:1 little endian */
|
||||||
|
|
||||||
|
#define DRM_FORMAT_ARGB1555 fourcc_code('A', 'R', '1', '5') /* [15:0] A:R:G:B 1:5:5:5 little endian */
|
||||||
|
#define DRM_FORMAT_ABGR1555 fourcc_code('A', 'B', '1', '5') /* [15:0] A:B:G:R 1:5:5:5 little endian */
|
||||||
|
#define DRM_FORMAT_RGBA5551 fourcc_code('R', 'A', '1', '5') /* [15:0] R:G:B:A 5:5:5:1 little endian */
|
||||||
|
#define DRM_FORMAT_BGRA5551 fourcc_code('B', 'A', '1', '5') /* [15:0] B:G:R:A 5:5:5:1 little endian */
|
||||||
|
|
||||||
|
#define DRM_FORMAT_RGB565 fourcc_code('R', 'G', '1', '6') /* [15:0] R:G:B 5:6:5 little endian */
|
||||||
|
#define DRM_FORMAT_BGR565 fourcc_code('B', 'G', '1', '6') /* [15:0] B:G:R 5:6:5 little endian */
|
||||||
|
|
||||||
|
/* 24 bpp RGB */
|
||||||
|
#define DRM_FORMAT_RGB888 fourcc_code('R', 'G', '2', '4') /* [23:0] R:G:B little endian */
|
||||||
|
#define DRM_FORMAT_BGR888 fourcc_code('B', 'G', '2', '4') /* [23:0] B:G:R little endian */
|
||||||
|
|
||||||
|
/* 32 bpp RGB */
|
||||||
|
#define DRM_FORMAT_XRGB8888 fourcc_code('X', 'R', '2', '4') /* [31:0] x:R:G:B 8:8:8:8 little endian */
|
||||||
|
#define DRM_FORMAT_XBGR8888 fourcc_code('X', 'B', '2', '4') /* [31:0] x:B:G:R 8:8:8:8 little endian */
|
||||||
|
#define DRM_FORMAT_RGBX8888 fourcc_code('R', 'X', '2', '4') /* [31:0] R:G:B:x 8:8:8:8 little endian */
|
||||||
|
#define DRM_FORMAT_BGRX8888 fourcc_code('B', 'X', '2', '4') /* [31:0] B:G:R:x 8:8:8:8 little endian */
|
||||||
|
|
||||||
|
#define DRM_FORMAT_ARGB8888 fourcc_code('A', 'R', '2', '4') /* [31:0] A:R:G:B 8:8:8:8 little endian */
|
||||||
|
#define DRM_FORMAT_ABGR8888 fourcc_code('A', 'B', '2', '4') /* [31:0] A:B:G:R 8:8:8:8 little endian */
|
||||||
|
#define DRM_FORMAT_RGBA8888 fourcc_code('R', 'A', '2', '4') /* [31:0] R:G:B:A 8:8:8:8 little endian */
|
||||||
|
#define DRM_FORMAT_BGRA8888 fourcc_code('B', 'A', '2', '4') /* [31:0] B:G:R:A 8:8:8:8 little endian */
|
||||||
|
|
||||||
|
#define DRM_FORMAT_XRGB2101010 fourcc_code('X', 'R', '3', '0') /* [31:0] x:R:G:B 2:10:10:10 little endian */
|
||||||
|
#define DRM_FORMAT_XBGR2101010 fourcc_code('X', 'B', '3', '0') /* [31:0] x:B:G:R 2:10:10:10 little endian */
|
||||||
|
#define DRM_FORMAT_RGBX1010102 fourcc_code('R', 'X', '3', '0') /* [31:0] R:G:B:x 10:10:10:2 little endian */
|
||||||
|
#define DRM_FORMAT_BGRX1010102 fourcc_code('B', 'X', '3', '0') /* [31:0] B:G:R:x 10:10:10:2 little endian */
|
||||||
|
|
||||||
|
#define DRM_FORMAT_ARGB2101010 fourcc_code('A', 'R', '3', '0') /* [31:0] A:R:G:B 2:10:10:10 little endian */
|
||||||
|
#define DRM_FORMAT_ABGR2101010 fourcc_code('A', 'B', '3', '0') /* [31:0] A:B:G:R 2:10:10:10 little endian */
|
||||||
|
#define DRM_FORMAT_RGBA1010102 fourcc_code('R', 'A', '3', '0') /* [31:0] R:G:B:A 10:10:10:2 little endian */
|
||||||
|
#define DRM_FORMAT_BGRA1010102 fourcc_code('B', 'A', '3', '0') /* [31:0] B:G:R:A 10:10:10:2 little endian */
|
||||||
|
|
||||||
|
/* packed YCbCr */
|
||||||
|
#define DRM_FORMAT_YUYV fourcc_code('Y', 'U', 'Y', 'V') /* [31:0] Cr0:Y1:Cb0:Y0 8:8:8:8 little endian */
|
||||||
|
#define DRM_FORMAT_YVYU fourcc_code('Y', 'V', 'Y', 'U') /* [31:0] Cb0:Y1:Cr0:Y0 8:8:8:8 little endian */
|
||||||
|
#define DRM_FORMAT_UYVY fourcc_code('U', 'Y', 'V', 'Y') /* [31:0] Y1:Cr0:Y0:Cb0 8:8:8:8 little endian */
|
||||||
|
#define DRM_FORMAT_VYUY fourcc_code('V', 'Y', 'U', 'Y') /* [31:0] Y1:Cb0:Y0:Cr0 8:8:8:8 little endian */
|
||||||
|
|
||||||
|
#define DRM_FORMAT_AYUV fourcc_code('A', 'Y', 'U', 'V') /* [31:0] A:Y:Cb:Cr 8:8:8:8 little endian */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 2 plane RGB + A
|
||||||
|
* index 0 = RGB plane, same format as the corresponding non _A8 format has
|
||||||
|
* index 1 = A plane, [7:0] A
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_XRGB8888_A8 fourcc_code('X', 'R', 'A', '8')
|
||||||
|
#define DRM_FORMAT_XBGR8888_A8 fourcc_code('X', 'B', 'A', '8')
|
||||||
|
#define DRM_FORMAT_RGBX8888_A8 fourcc_code('R', 'X', 'A', '8')
|
||||||
|
#define DRM_FORMAT_BGRX8888_A8 fourcc_code('B', 'X', 'A', '8')
|
||||||
|
#define DRM_FORMAT_RGB888_A8 fourcc_code('R', '8', 'A', '8')
|
||||||
|
#define DRM_FORMAT_BGR888_A8 fourcc_code('B', '8', 'A', '8')
|
||||||
|
#define DRM_FORMAT_RGB565_A8 fourcc_code('R', '5', 'A', '8')
|
||||||
|
#define DRM_FORMAT_BGR565_A8 fourcc_code('B', '5', 'A', '8')
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 2 plane YCbCr
|
||||||
|
* index 0 = Y plane, [7:0] Y
|
||||||
|
* index 1 = Cr:Cb plane, [15:0] Cr:Cb little endian
|
||||||
|
* or
|
||||||
|
* index 1 = Cb:Cr plane, [15:0] Cb:Cr little endian
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_NV12 fourcc_code('N', 'V', '1', '2') /* 2x2 subsampled Cr:Cb plane */
|
||||||
|
#define DRM_FORMAT_NV21 fourcc_code('N', 'V', '2', '1') /* 2x2 subsampled Cb:Cr plane */
|
||||||
|
#define DRM_FORMAT_NV16 fourcc_code('N', 'V', '1', '6') /* 2x1 subsampled Cr:Cb plane */
|
||||||
|
#define DRM_FORMAT_NV61 fourcc_code('N', 'V', '6', '1') /* 2x1 subsampled Cb:Cr plane */
|
||||||
|
#define DRM_FORMAT_NV24 fourcc_code('N', 'V', '2', '4') /* non-subsampled Cr:Cb plane */
|
||||||
|
#define DRM_FORMAT_NV42 fourcc_code('N', 'V', '4', '2') /* non-subsampled Cb:Cr plane */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 3 plane YCbCr
|
||||||
|
* index 0: Y plane, [7:0] Y
|
||||||
|
* index 1: Cb plane, [7:0] Cb
|
||||||
|
* index 2: Cr plane, [7:0] Cr
|
||||||
|
* or
|
||||||
|
* index 1: Cr plane, [7:0] Cr
|
||||||
|
* index 2: Cb plane, [7:0] Cb
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_YUV410 fourcc_code('Y', 'U', 'V', '9') /* 4x4 subsampled Cb (1) and Cr (2) planes */
|
||||||
|
#define DRM_FORMAT_YVU410 fourcc_code('Y', 'V', 'U', '9') /* 4x4 subsampled Cr (1) and Cb (2) planes */
|
||||||
|
#define DRM_FORMAT_YUV411 fourcc_code('Y', 'U', '1', '1') /* 4x1 subsampled Cb (1) and Cr (2) planes */
|
||||||
|
#define DRM_FORMAT_YVU411 fourcc_code('Y', 'V', '1', '1') /* 4x1 subsampled Cr (1) and Cb (2) planes */
|
||||||
|
#define DRM_FORMAT_YUV420 fourcc_code('Y', 'U', '1', '2') /* 2x2 subsampled Cb (1) and Cr (2) planes */
|
||||||
|
#define DRM_FORMAT_YVU420 fourcc_code('Y', 'V', '1', '2') /* 2x2 subsampled Cr (1) and Cb (2) planes */
|
||||||
|
#define DRM_FORMAT_YUV422 fourcc_code('Y', 'U', '1', '6') /* 2x1 subsampled Cb (1) and Cr (2) planes */
|
||||||
|
#define DRM_FORMAT_YVU422 fourcc_code('Y', 'V', '1', '6') /* 2x1 subsampled Cr (1) and Cb (2) planes */
|
||||||
|
#define DRM_FORMAT_YUV444 fourcc_code('Y', 'U', '2', '4') /* non-subsampled Cb (1) and Cr (2) planes */
|
||||||
|
#define DRM_FORMAT_YVU444 fourcc_code('Y', 'V', '2', '4') /* non-subsampled Cr (1) and Cb (2) planes */
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Format Modifiers:
|
||||||
|
*
|
||||||
|
* Format modifiers describe, typically, a re-ordering or modification
|
||||||
|
* of the data in a plane of an FB. This can be used to express tiled/
|
||||||
|
* swizzled formats, or compression, or a combination of the two.
|
||||||
|
*
|
||||||
|
* The upper 8 bits of the format modifier are a vendor-id as assigned
|
||||||
|
* below. The lower 56 bits are assigned as vendor sees fit.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* Vendor Ids: */
|
||||||
|
#define DRM_FORMAT_MOD_NONE 0
|
||||||
|
#define DRM_FORMAT_MOD_VENDOR_NONE 0
|
||||||
|
#define DRM_FORMAT_MOD_VENDOR_INTEL 0x01
|
||||||
|
#define DRM_FORMAT_MOD_VENDOR_AMD 0x02
|
||||||
|
#define DRM_FORMAT_MOD_VENDOR_NVIDIA 0x03
|
||||||
|
#define DRM_FORMAT_MOD_VENDOR_SAMSUNG 0x04
|
||||||
|
#define DRM_FORMAT_MOD_VENDOR_QCOM 0x05
|
||||||
|
#define DRM_FORMAT_MOD_VENDOR_VIVANTE 0x06
|
||||||
|
#define DRM_FORMAT_MOD_VENDOR_BROADCOM 0x07
|
||||||
|
/* add more to the end as needed */
|
||||||
|
|
||||||
|
#define DRM_FORMAT_RESERVED ((1ULL << 56) - 1)
|
||||||
|
|
||||||
|
#define fourcc_mod_code(vendor, val) \
|
||||||
|
((((uint64_t)DRM_FORMAT_MOD_VENDOR_## vendor) << 56) | ((val) & 0x00ffffffffffffffULL))
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Format Modifier tokens:
|
||||||
|
*
|
||||||
|
* When adding a new token please document the layout with a code comment,
|
||||||
|
* similar to the fourcc codes above. drm_fourcc.h is considered the
|
||||||
|
* authoritative source for all of these.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Invalid Modifier
|
||||||
|
*
|
||||||
|
* This modifier can be used as a sentinel to terminate the format modifiers
|
||||||
|
* list, or to initialize a variable with an invalid modifier. It might also be
|
||||||
|
* used to report an error back to userspace for certain APIs.
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_MOD_INVALID fourcc_mod_code(NONE, DRM_FORMAT_RESERVED)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Linear Layout
|
||||||
|
*
|
||||||
|
* Just plain linear layout. Note that this is different from no specifying any
|
||||||
|
* modifier (e.g. not setting DRM_MODE_FB_MODIFIERS in the DRM_ADDFB2 ioctl),
|
||||||
|
* which tells the driver to also take driver-internal information into account
|
||||||
|
* and so might actually result in a tiled framebuffer.
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_MOD_LINEAR fourcc_mod_code(NONE, 0)
|
||||||
|
|
||||||
|
/* Intel framebuffer modifiers */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Intel X-tiling layout
|
||||||
|
*
|
||||||
|
* This is a tiled layout using 4Kb tiles (except on gen2 where the tiles 2Kb)
|
||||||
|
* in row-major layout. Within the tile bytes are laid out row-major, with
|
||||||
|
* a platform-dependent stride. On top of that the memory can apply
|
||||||
|
* platform-depending swizzling of some higher address bits into bit6.
|
||||||
|
*
|
||||||
|
* This format is highly platforms specific and not useful for cross-driver
|
||||||
|
* sharing. It exists since on a given platform it does uniquely identify the
|
||||||
|
* layout in a simple way for i915-specific userspace.
|
||||||
|
*/
|
||||||
|
#define I915_FORMAT_MOD_X_TILED fourcc_mod_code(INTEL, 1)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Intel Y-tiling layout
|
||||||
|
*
|
||||||
|
* This is a tiled layout using 4Kb tiles (except on gen2 where the tiles 2Kb)
|
||||||
|
* in row-major layout. Within the tile bytes are laid out in OWORD (16 bytes)
|
||||||
|
* chunks column-major, with a platform-dependent height. On top of that the
|
||||||
|
* memory can apply platform-depending swizzling of some higher address bits
|
||||||
|
* into bit6.
|
||||||
|
*
|
||||||
|
* This format is highly platforms specific and not useful for cross-driver
|
||||||
|
* sharing. It exists since on a given platform it does uniquely identify the
|
||||||
|
* layout in a simple way for i915-specific userspace.
|
||||||
|
*/
|
||||||
|
#define I915_FORMAT_MOD_Y_TILED fourcc_mod_code(INTEL, 2)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Intel Yf-tiling layout
|
||||||
|
*
|
||||||
|
* This is a tiled layout using 4Kb tiles in row-major layout.
|
||||||
|
* Within the tile pixels are laid out in 16 256 byte units / sub-tiles which
|
||||||
|
* are arranged in four groups (two wide, two high) with column-major layout.
|
||||||
|
* Each group therefore consits out of four 256 byte units, which are also laid
|
||||||
|
* out as 2x2 column-major.
|
||||||
|
* 256 byte units are made out of four 64 byte blocks of pixels, producing
|
||||||
|
* either a square block or a 2:1 unit.
|
||||||
|
* 64 byte blocks of pixels contain four pixel rows of 16 bytes, where the width
|
||||||
|
* in pixel depends on the pixel depth.
|
||||||
|
*/
|
||||||
|
#define I915_FORMAT_MOD_Yf_TILED fourcc_mod_code(INTEL, 3)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Intel color control surface (CCS) for render compression
|
||||||
|
*
|
||||||
|
* The framebuffer format must be one of the 8:8:8:8 RGB formats.
|
||||||
|
* The main surface will be plane index 0 and must be Y/Yf-tiled,
|
||||||
|
* the CCS will be plane index 1.
|
||||||
|
*
|
||||||
|
* Each CCS tile matches a 1024x512 pixel area of the main surface.
|
||||||
|
* To match certain aspects of the 3D hardware the CCS is
|
||||||
|
* considered to be made up of normal 128Bx32 Y tiles, Thus
|
||||||
|
* the CCS pitch must be specified in multiples of 128 bytes.
|
||||||
|
*
|
||||||
|
* In reality the CCS tile appears to be a 64Bx64 Y tile, composed
|
||||||
|
* of QWORD (8 bytes) chunks instead of OWORD (16 bytes) chunks.
|
||||||
|
* But that fact is not relevant unless the memory is accessed
|
||||||
|
* directly.
|
||||||
|
*/
|
||||||
|
#define I915_FORMAT_MOD_Y_TILED_CCS fourcc_mod_code(INTEL, 4)
|
||||||
|
#define I915_FORMAT_MOD_Yf_TILED_CCS fourcc_mod_code(INTEL, 5)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Tiled, NV12MT, grouped in 64 (pixels) x 32 (lines) -sized macroblocks
|
||||||
|
*
|
||||||
|
* Macroblocks are laid in a Z-shape, and each pixel data is following the
|
||||||
|
* standard NV12 style.
|
||||||
|
* As for NV12, an image is the result of two frame buffers: one for Y,
|
||||||
|
* one for the interleaved Cb/Cr components (1/2 the height of the Y buffer).
|
||||||
|
* Alignment requirements are (for each buffer):
|
||||||
|
* - multiple of 128 pixels for the width
|
||||||
|
* - multiple of 32 pixels for the height
|
||||||
|
*
|
||||||
|
* For more information: see https://linuxtv.org/downloads/v4l-dvb-apis/re32.html
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_MOD_SAMSUNG_64_32_TILE fourcc_mod_code(SAMSUNG, 1)
|
||||||
|
|
||||||
|
/* Vivante framebuffer modifiers */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Vivante 4x4 tiling layout
|
||||||
|
*
|
||||||
|
* This is a simple tiled layout using tiles of 4x4 pixels in a row-major
|
||||||
|
* layout.
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_MOD_VIVANTE_TILED fourcc_mod_code(VIVANTE, 1)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Vivante 64x64 super-tiling layout
|
||||||
|
*
|
||||||
|
* This is a tiled layout using 64x64 pixel super-tiles, where each super-tile
|
||||||
|
* contains 8x4 groups of 2x4 tiles of 4x4 pixels (like above) each, all in row-
|
||||||
|
* major layout.
|
||||||
|
*
|
||||||
|
* For more information: see
|
||||||
|
* https://github.com/etnaviv/etna_viv/blob/master/doc/hardware.md#texture-tiling
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_MOD_VIVANTE_SUPER_TILED fourcc_mod_code(VIVANTE, 2)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Vivante 4x4 tiling layout for dual-pipe
|
||||||
|
*
|
||||||
|
* Same as the 4x4 tiling layout, except every second 4x4 pixel tile starts at a
|
||||||
|
* different base address. Offsets from the base addresses are therefore halved
|
||||||
|
* compared to the non-split tiled layout.
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_MOD_VIVANTE_SPLIT_TILED fourcc_mod_code(VIVANTE, 3)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Vivante 64x64 super-tiling layout for dual-pipe
|
||||||
|
*
|
||||||
|
* Same as the 64x64 super-tiling layout, except every second 4x4 pixel tile
|
||||||
|
* starts at a different base address. Offsets from the base addresses are
|
||||||
|
* therefore halved compared to the non-split super-tiled layout.
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_MOD_VIVANTE_SPLIT_SUPER_TILED fourcc_mod_code(VIVANTE, 4)
|
||||||
|
|
||||||
|
/* NVIDIA frame buffer modifiers */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Tegra Tiled Layout, used by Tegra 2, 3 and 4.
|
||||||
|
*
|
||||||
|
* Pixels are arranged in simple tiles of 16 x 16 bytes.
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_MOD_NVIDIA_TEGRA_TILED fourcc_mod_code(NVIDIA, 1)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 16Bx2 Block Linear layout, used by desktop GPUs, and Tegra K1 and later
|
||||||
|
*
|
||||||
|
* Pixels are arranged in 64x8 Groups Of Bytes (GOBs). GOBs are then stacked
|
||||||
|
* vertically by a power of 2 (1 to 32 GOBs) to form a block.
|
||||||
|
*
|
||||||
|
* Within a GOB, data is ordered as 16B x 2 lines sectors laid in Z-shape.
|
||||||
|
*
|
||||||
|
* Parameter 'v' is the log2 encoding of the number of GOBs stacked vertically.
|
||||||
|
* Valid values are:
|
||||||
|
*
|
||||||
|
* 0 == ONE_GOB
|
||||||
|
* 1 == TWO_GOBS
|
||||||
|
* 2 == FOUR_GOBS
|
||||||
|
* 3 == EIGHT_GOBS
|
||||||
|
* 4 == SIXTEEN_GOBS
|
||||||
|
* 5 == THIRTYTWO_GOBS
|
||||||
|
*
|
||||||
|
* Chapter 20 "Pixel Memory Formats" of the Tegra X1 TRM describes this format
|
||||||
|
* in full detail.
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_MOD_NVIDIA_16BX2_BLOCK(v) \
|
||||||
|
fourcc_mod_code(NVIDIA, 0x10 | ((v) & 0xf))
|
||||||
|
|
||||||
|
#define DRM_FORMAT_MOD_NVIDIA_16BX2_BLOCK_ONE_GOB \
|
||||||
|
fourcc_mod_code(NVIDIA, 0x10)
|
||||||
|
#define DRM_FORMAT_MOD_NVIDIA_16BX2_BLOCK_TWO_GOB \
|
||||||
|
fourcc_mod_code(NVIDIA, 0x11)
|
||||||
|
#define DRM_FORMAT_MOD_NVIDIA_16BX2_BLOCK_FOUR_GOB \
|
||||||
|
fourcc_mod_code(NVIDIA, 0x12)
|
||||||
|
#define DRM_FORMAT_MOD_NVIDIA_16BX2_BLOCK_EIGHT_GOB \
|
||||||
|
fourcc_mod_code(NVIDIA, 0x13)
|
||||||
|
#define DRM_FORMAT_MOD_NVIDIA_16BX2_BLOCK_SIXTEEN_GOB \
|
||||||
|
fourcc_mod_code(NVIDIA, 0x14)
|
||||||
|
#define DRM_FORMAT_MOD_NVIDIA_16BX2_BLOCK_THIRTYTWO_GOB \
|
||||||
|
fourcc_mod_code(NVIDIA, 0x15)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Broadcom VC4 "T" format
|
||||||
|
*
|
||||||
|
* This is the primary layout that the V3D GPU can texture from (it
|
||||||
|
* can't do linear). The T format has:
|
||||||
|
*
|
||||||
|
* - 64b utiles of pixels in a raster-order grid according to cpp. It's 4x4
|
||||||
|
* pixels at 32 bit depth.
|
||||||
|
*
|
||||||
|
* - 1k subtiles made of a 4x4 raster-order grid of 64b utiles (so usually
|
||||||
|
* 16x16 pixels).
|
||||||
|
*
|
||||||
|
* - 4k tiles made of a 2x2 grid of 1k subtiles (so usually 32x32 pixels). On
|
||||||
|
* even 4k tile rows, they're arranged as (BL, TL, TR, BR), and on odd rows
|
||||||
|
* they're (TR, BR, BL, TL), where bottom left is start of memory.
|
||||||
|
*
|
||||||
|
* - an image made of 4k tiles in rows either left-to-right (even rows of 4k
|
||||||
|
* tiles) or right-to-left (odd rows of 4k tiles).
|
||||||
|
*/
|
||||||
|
#define DRM_FORMAT_MOD_BROADCOM_VC4_T_TILED fourcc_mod_code(BROADCOM, 1)
|
||||||
|
|
||||||
|
#if defined(__cplusplus)
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* DRM_FOURCC_H */
|
@ -386,6 +386,7 @@ QemuConsole *graphic_console_init(DeviceState *dev, uint32_t head,
|
|||||||
void graphic_console_set_hwops(QemuConsole *con,
|
void graphic_console_set_hwops(QemuConsole *con,
|
||||||
const GraphicHwOps *hw_ops,
|
const GraphicHwOps *hw_ops,
|
||||||
void *opaque);
|
void *opaque);
|
||||||
|
void graphic_console_close(QemuConsole *con);
|
||||||
|
|
||||||
void graphic_hw_update(QemuConsole *con);
|
void graphic_hw_update(QemuConsole *con);
|
||||||
void graphic_hw_invalidate(QemuConsole *con);
|
void graphic_hw_invalidate(QemuConsole *con);
|
||||||
@ -398,6 +399,7 @@ QemuConsole *qemu_console_lookup_by_index(unsigned int index);
|
|||||||
QemuConsole *qemu_console_lookup_by_device(DeviceState *dev, uint32_t head);
|
QemuConsole *qemu_console_lookup_by_device(DeviceState *dev, uint32_t head);
|
||||||
QemuConsole *qemu_console_lookup_by_device_name(const char *device_id,
|
QemuConsole *qemu_console_lookup_by_device_name(const char *device_id,
|
||||||
uint32_t head, Error **errp);
|
uint32_t head, Error **errp);
|
||||||
|
QemuConsole *qemu_console_lookup_unused(void);
|
||||||
bool qemu_console_is_visible(QemuConsole *con);
|
bool qemu_console_is_visible(QemuConsole *con);
|
||||||
bool qemu_console_is_graphic(QemuConsole *con);
|
bool qemu_console_is_graphic(QemuConsole *con);
|
||||||
bool qemu_console_is_fixedsize(QemuConsole *con);
|
bool qemu_console_is_fixedsize(QemuConsole *con);
|
||||||
|
@ -33,6 +33,8 @@
|
|||||||
# define PIXMAN_BE_r8g8b8a8 PIXMAN_r8g8b8a8
|
# define PIXMAN_BE_r8g8b8a8 PIXMAN_r8g8b8a8
|
||||||
# define PIXMAN_BE_x8b8g8r8 PIXMAN_x8b8g8r8
|
# define PIXMAN_BE_x8b8g8r8 PIXMAN_x8b8g8r8
|
||||||
# define PIXMAN_BE_a8b8g8r8 PIXMAN_a8b8g8r8
|
# define PIXMAN_BE_a8b8g8r8 PIXMAN_a8b8g8r8
|
||||||
|
# define PIXMAN_LE_r8g8b8 PIXMAN_b8g8r8
|
||||||
|
# define PIXMAN_LE_a8r8g8b8 PIXMAN_b8g8r8a8
|
||||||
# define PIXMAN_LE_x8r8g8b8 PIXMAN_b8g8r8x8
|
# define PIXMAN_LE_x8r8g8b8 PIXMAN_b8g8r8x8
|
||||||
#else
|
#else
|
||||||
# define PIXMAN_BE_r8g8b8 PIXMAN_b8g8r8
|
# define PIXMAN_BE_r8g8b8 PIXMAN_b8g8r8
|
||||||
@ -44,6 +46,8 @@
|
|||||||
# define PIXMAN_BE_r8g8b8a8 PIXMAN_a8b8g8r8
|
# define PIXMAN_BE_r8g8b8a8 PIXMAN_a8b8g8r8
|
||||||
# define PIXMAN_BE_x8b8g8r8 PIXMAN_r8g8b8x8
|
# define PIXMAN_BE_x8b8g8r8 PIXMAN_r8g8b8x8
|
||||||
# define PIXMAN_BE_a8b8g8r8 PIXMAN_r8g8b8a8
|
# define PIXMAN_BE_a8b8g8r8 PIXMAN_r8g8b8a8
|
||||||
|
# define PIXMAN_LE_r8g8b8 PIXMAN_r8g8b8
|
||||||
|
# define PIXMAN_LE_a8r8g8b8 PIXMAN_a8r8g8b8
|
||||||
# define PIXMAN_LE_x8r8g8b8 PIXMAN_x8r8g8b8
|
# define PIXMAN_LE_x8r8g8b8 PIXMAN_x8r8g8b8
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -51,6 +55,7 @@
|
|||||||
|
|
||||||
PixelFormat qemu_pixelformat_from_pixman(pixman_format_code_t format);
|
PixelFormat qemu_pixelformat_from_pixman(pixman_format_code_t format);
|
||||||
pixman_format_code_t qemu_default_pixman_format(int bpp, bool native_endian);
|
pixman_format_code_t qemu_default_pixman_format(int bpp, bool native_endian);
|
||||||
|
pixman_format_code_t qemu_drm_format_to_pixman(uint32_t drm_format);
|
||||||
int qemu_pixman_get_type(int rshift, int gshift, int bshift);
|
int qemu_pixman_get_type(int rshift, int gshift, int bshift);
|
||||||
pixman_format_code_t qemu_pixman_get_format(PixelFormat *pf);
|
pixman_format_code_t qemu_pixman_get_format(PixelFormat *pf);
|
||||||
bool qemu_pixman_check_format(DisplayChangeListener *dcl,
|
bool qemu_pixman_check_format(DisplayChangeListener *dcl,
|
||||||
|
@ -39,6 +39,7 @@ cp_portable() {
|
|||||||
-e 'input-event-codes' \
|
-e 'input-event-codes' \
|
||||||
-e 'sys/' \
|
-e 'sys/' \
|
||||||
-e 'pvrdma_verbs' \
|
-e 'pvrdma_verbs' \
|
||||||
|
-e 'drm.h' \
|
||||||
> /dev/null
|
> /dev/null
|
||||||
then
|
then
|
||||||
echo "Unexpected #include in input file $f".
|
echo "Unexpected #include in input file $f".
|
||||||
@ -57,6 +58,7 @@ cp_portable() {
|
|||||||
-e 's/__attribute__((packed))/QEMU_PACKED/' \
|
-e 's/__attribute__((packed))/QEMU_PACKED/' \
|
||||||
-e 's/__inline__/inline/' \
|
-e 's/__inline__/inline/' \
|
||||||
-e 's/__BITS_PER_LONG/HOST_LONG_BITS/' \
|
-e 's/__BITS_PER_LONG/HOST_LONG_BITS/' \
|
||||||
|
-e '/\"drm.h\"/d' \
|
||||||
-e '/sys\/ioctl.h/d' \
|
-e '/sys\/ioctl.h/d' \
|
||||||
-e 's/SW_MAX/SW_MAX_/' \
|
-e 's/SW_MAX/SW_MAX_/' \
|
||||||
-e 's/atomic_t/int/' \
|
-e 's/atomic_t/int/' \
|
||||||
@ -152,6 +154,9 @@ for i in "$tmpdir"/include/linux/*virtio*.h "$tmpdir/include/linux/input.h" \
|
|||||||
"$tmpdir/include/linux/pci_regs.h"; do
|
"$tmpdir/include/linux/pci_regs.h"; do
|
||||||
cp_portable "$i" "$output/include/standard-headers/linux"
|
cp_portable "$i" "$output/include/standard-headers/linux"
|
||||||
done
|
done
|
||||||
|
mkdir -p "$output/include/standard-headers/drm"
|
||||||
|
cp_portable "$tmpdir/include/drm/drm_fourcc.h" \
|
||||||
|
"$output/include/standard-headers/drm"
|
||||||
|
|
||||||
rm -rf "$output/include/standard-headers/drivers/infiniband/hw/vmw_pvrdma"
|
rm -rf "$output/include/standard-headers/drivers/infiniband/hw/vmw_pvrdma"
|
||||||
mkdir -p "$output/include/standard-headers/drivers/infiniband/hw/vmw_pvrdma"
|
mkdir -p "$output/include/standard-headers/drivers/infiniband/hw/vmw_pvrdma"
|
||||||
|
79
ui/console.c
79
ui/console.c
@ -1282,11 +1282,16 @@ static QemuConsole *new_console(DisplayState *ds, console_type_t console_type,
|
|||||||
s->console_type = console_type;
|
s->console_type = console_type;
|
||||||
|
|
||||||
consoles = g_realloc(consoles, sizeof(*consoles) * (nb_consoles+1));
|
consoles = g_realloc(consoles, sizeof(*consoles) * (nb_consoles+1));
|
||||||
if (console_type != GRAPHIC_CONSOLE) {
|
if (console_type != GRAPHIC_CONSOLE || qdev_hotplug) {
|
||||||
s->index = nb_consoles;
|
s->index = nb_consoles;
|
||||||
consoles[nb_consoles++] = s;
|
consoles[nb_consoles++] = s;
|
||||||
} else {
|
} else {
|
||||||
/* HACK: Put graphical consoles before text consoles. */
|
/*
|
||||||
|
* HACK: Put graphical consoles before text consoles.
|
||||||
|
*
|
||||||
|
* Only do that for coldplugged devices. After initial device
|
||||||
|
* initialization we will not renumber the consoles any more.
|
||||||
|
*/
|
||||||
for (i = nb_consoles; i > 0; i--) {
|
for (i = nb_consoles; i > 0; i--) {
|
||||||
if (consoles[i - 1]->console_type == GRAPHIC_CONSOLE)
|
if (consoles[i - 1]->console_type == GRAPHIC_CONSOLE)
|
||||||
break;
|
break;
|
||||||
@ -1874,21 +1879,61 @@ QemuConsole *graphic_console_init(DeviceState *dev, uint32_t head,
|
|||||||
int height = 480;
|
int height = 480;
|
||||||
QemuConsole *s;
|
QemuConsole *s;
|
||||||
DisplayState *ds;
|
DisplayState *ds;
|
||||||
|
DisplaySurface *surface;
|
||||||
|
|
||||||
ds = get_alloc_displaystate();
|
ds = get_alloc_displaystate();
|
||||||
trace_console_gfx_new();
|
s = qemu_console_lookup_unused();
|
||||||
s = new_console(ds, GRAPHIC_CONSOLE, head);
|
if (s) {
|
||||||
s->ui_timer = timer_new_ms(QEMU_CLOCK_REALTIME, dpy_set_ui_info_timer, s);
|
trace_console_gfx_reuse(s->index);
|
||||||
|
if (s->surface) {
|
||||||
|
width = surface_width(s->surface);
|
||||||
|
height = surface_height(s->surface);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
trace_console_gfx_new();
|
||||||
|
s = new_console(ds, GRAPHIC_CONSOLE, head);
|
||||||
|
s->ui_timer = timer_new_ms(QEMU_CLOCK_REALTIME,
|
||||||
|
dpy_set_ui_info_timer, s);
|
||||||
|
}
|
||||||
graphic_console_set_hwops(s, hw_ops, opaque);
|
graphic_console_set_hwops(s, hw_ops, opaque);
|
||||||
if (dev) {
|
if (dev) {
|
||||||
object_property_set_link(OBJECT(s), OBJECT(dev), "device",
|
object_property_set_link(OBJECT(s), OBJECT(dev), "device",
|
||||||
&error_abort);
|
&error_abort);
|
||||||
}
|
}
|
||||||
|
|
||||||
s->surface = qemu_create_message_surface(width, height, noinit);
|
surface = qemu_create_message_surface(width, height, noinit);
|
||||||
|
dpy_gfx_replace_surface(s, surface);
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const GraphicHwOps unused_ops = {
|
||||||
|
/* no callbacks */
|
||||||
|
};
|
||||||
|
|
||||||
|
void graphic_console_close(QemuConsole *con)
|
||||||
|
{
|
||||||
|
static const char unplugged[] =
|
||||||
|
"Guest display has been unplugged";
|
||||||
|
DisplaySurface *surface;
|
||||||
|
int width = 640;
|
||||||
|
int height = 480;
|
||||||
|
|
||||||
|
if (con->surface) {
|
||||||
|
width = surface_width(con->surface);
|
||||||
|
height = surface_height(con->surface);
|
||||||
|
}
|
||||||
|
|
||||||
|
trace_console_gfx_close(con->index);
|
||||||
|
object_property_set_link(OBJECT(con), NULL, "device", &error_abort);
|
||||||
|
graphic_console_set_hwops(con, &unused_ops, NULL);
|
||||||
|
|
||||||
|
if (con->gl) {
|
||||||
|
dpy_gl_scanout_disable(con);
|
||||||
|
}
|
||||||
|
surface = qemu_create_message_surface(width, height, unplugged);
|
||||||
|
dpy_gfx_replace_surface(con, surface);
|
||||||
|
}
|
||||||
|
|
||||||
QemuConsole *qemu_console_lookup_by_index(unsigned int index)
|
QemuConsole *qemu_console_lookup_by_index(unsigned int index)
|
||||||
{
|
{
|
||||||
if (index >= nb_consoles) {
|
if (index >= nb_consoles) {
|
||||||
@ -1945,6 +1990,28 @@ QemuConsole *qemu_console_lookup_by_device_name(const char *device_id,
|
|||||||
return con;
|
return con;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QemuConsole *qemu_console_lookup_unused(void)
|
||||||
|
{
|
||||||
|
Object *obj;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < nb_consoles; i++) {
|
||||||
|
if (!consoles[i]) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (consoles[i]->hw_ops != &unused_ops) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
obj = object_property_get_link(OBJECT(consoles[i]),
|
||||||
|
"device", &error_abort);
|
||||||
|
if (obj != NULL) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
return consoles[i];
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
bool qemu_console_is_visible(QemuConsole *con)
|
bool qemu_console_is_visible(QemuConsole *con)
|
||||||
{
|
{
|
||||||
return (con == active_console) || (con->dcls > 0);
|
return (con == active_console) || (con->dcls > 0);
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
#include "qemu/osdep.h"
|
#include "qemu/osdep.h"
|
||||||
#include "qemu-common.h"
|
#include "qemu-common.h"
|
||||||
#include "ui/console.h"
|
#include "ui/console.h"
|
||||||
|
#include "standard-headers/drm/drm_fourcc.h"
|
||||||
|
|
||||||
PixelFormat qemu_pixelformat_from_pixman(pixman_format_code_t format)
|
PixelFormat qemu_pixelformat_from_pixman(pixman_format_code_t format)
|
||||||
{
|
{
|
||||||
@ -88,6 +89,27 @@ pixman_format_code_t qemu_default_pixman_format(int bpp, bool native_endian)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Note: drm is little endian, pixman is native endian */
|
||||||
|
pixman_format_code_t qemu_drm_format_to_pixman(uint32_t drm_format)
|
||||||
|
{
|
||||||
|
static const struct {
|
||||||
|
uint32_t drm_format;
|
||||||
|
pixman_format_code_t pixman;
|
||||||
|
} map[] = {
|
||||||
|
{ DRM_FORMAT_RGB888, PIXMAN_LE_r8g8b8 },
|
||||||
|
{ DRM_FORMAT_ARGB8888, PIXMAN_LE_a8r8g8b8 },
|
||||||
|
{ DRM_FORMAT_XRGB8888, PIXMAN_LE_x8r8g8b8 }
|
||||||
|
};
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < ARRAY_SIZE(map); i++) {
|
||||||
|
if (drm_format == map[i].drm_format) {
|
||||||
|
return map[i].pixman;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
int qemu_pixman_get_type(int rshift, int gshift, int bshift)
|
int qemu_pixman_get_type(int rshift, int gshift, int bshift)
|
||||||
{
|
{
|
||||||
int type = PIXMAN_TYPE_OTHER;
|
int type = PIXMAN_TYPE_OTHER;
|
||||||
|
@ -2,6 +2,8 @@
|
|||||||
|
|
||||||
# ui/console.c
|
# ui/console.c
|
||||||
console_gfx_new(void) ""
|
console_gfx_new(void) ""
|
||||||
|
console_gfx_reuse(int index) "%d"
|
||||||
|
console_gfx_close(int index) "%d"
|
||||||
console_putchar_csi(int esc_param0, int esc_param1, int ch, int nb_esc_params) "escape sequence CSI%d;%d%c, %d parameters"
|
console_putchar_csi(int esc_param0, int esc_param1, int ch, int nb_esc_params) "escape sequence CSI%d;%d%c, %d parameters"
|
||||||
console_putchar_unhandled(int ch) "unhandled escape character '%c'"
|
console_putchar_unhandled(int ch) "unhandled escape character '%c'"
|
||||||
console_txt_new(int w, int h) "%dx%d"
|
console_txt_new(int w, int h) "%dx%d"
|
||||||
|
Loading…
Reference in New Issue
Block a user