1*1b2c77f6Sbouyer /* $NetBSD: genfb_xen.c,v 1.2 2023/10/17 16:09:12 bouyer Exp $ */
24a96dd4fSbouyer
34a96dd4fSbouyer /*
44a96dd4fSbouyer * Copyright (c) 2023 Manuel Bouyer.
54a96dd4fSbouyer *
64a96dd4fSbouyer * Redistribution and use in source and binary forms, with or without
74a96dd4fSbouyer * modification, are permitted provided that the following conditions
84a96dd4fSbouyer * are met:
94a96dd4fSbouyer * 1. Redistributions of source code must retain the above copyright
104a96dd4fSbouyer * notice, this list of conditions and the following disclaimer.
114a96dd4fSbouyer * 2. Redistributions in binary form must reproduce the above copyright
124a96dd4fSbouyer * notice, this list of conditions and the following disclaimer in the
134a96dd4fSbouyer * documentation and/or other materials provided with the distribution.
144a96dd4fSbouyer *
154a96dd4fSbouyer * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
164a96dd4fSbouyer * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
174a96dd4fSbouyer * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
184a96dd4fSbouyer * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
194a96dd4fSbouyer * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
204a96dd4fSbouyer * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
214a96dd4fSbouyer * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
224a96dd4fSbouyer * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
234a96dd4fSbouyer * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
244a96dd4fSbouyer * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
254a96dd4fSbouyer *
264a96dd4fSbouyer */
274a96dd4fSbouyer
284a96dd4fSbouyer #include <sys/cdefs.h>
29*1b2c77f6Sbouyer __KERNEL_RCSID(0, "$NetBSD: genfb_xen.c,v 1.2 2023/10/17 16:09:12 bouyer Exp $");
304a96dd4fSbouyer
314a96dd4fSbouyer
324a96dd4fSbouyer #include <sys/device.h>
334a96dd4fSbouyer #include <xen/include/xen.h>
344a96dd4fSbouyer #include <xen/include/hypervisor.h>
354a96dd4fSbouyer #include <xen/include/public/xen.h>
364a96dd4fSbouyer #include <arch/x86/include/genfb_machdep.h>
374a96dd4fSbouyer #include <arch/x86/include/bootinfo.h>
384a96dd4fSbouyer
394a96dd4fSbouyer static struct btinfo_framebuffer _xen_genfb_btinfo = {0};
404a96dd4fSbouyer
414a96dd4fSbouyer const struct btinfo_framebuffer *
xen_genfb_getbtinfo(void)424a96dd4fSbouyer xen_genfb_getbtinfo(void)
434a96dd4fSbouyer {
444a96dd4fSbouyer dom0_vga_console_info_t *d0_consi;
45*1b2c77f6Sbouyer int info_size;
464a96dd4fSbouyer
474a96dd4fSbouyer if (!xendomain_is_dom0())
484a96dd4fSbouyer return NULL;
494a96dd4fSbouyer
504a96dd4fSbouyer if (_xen_genfb_btinfo.common.type == BTINFO_FRAMEBUFFER)
514a96dd4fSbouyer return &_xen_genfb_btinfo;
524a96dd4fSbouyer
53*1b2c77f6Sbouyer #ifdef XENPVHVM
54*1b2c77f6Sbouyer struct xen_platform_op op = {
55*1b2c77f6Sbouyer .cmd = XENPF_get_dom0_console,
56*1b2c77f6Sbouyer };
57*1b2c77f6Sbouyer info_size = HYPERVISOR_platform_op(&op);
58*1b2c77f6Sbouyer if (info_size < sizeof(dom0_vga_console_info_t)) {
59*1b2c77f6Sbouyer printf("XENPF_get_dom0_console fail %d\n", info_size);
60*1b2c77f6Sbouyer return NULL;
61*1b2c77f6Sbouyer }
62*1b2c77f6Sbouyer d0_consi = &op.u.dom0_console;
63*1b2c77f6Sbouyer #else
644a96dd4fSbouyer d0_consi = (void *)((char *)&xen_start_info +
654a96dd4fSbouyer xen_start_info.console.dom0.info_off);
66*1b2c77f6Sbouyer info_size = xen_start_info.console.dom0.info_size;
67*1b2c77f6Sbouyer #endif
684a96dd4fSbouyer
694a96dd4fSbouyer if (d0_consi->video_type != XEN_VGATYPE_VESA_LFB &&
704a96dd4fSbouyer d0_consi->video_type != XEN_VGATYPE_EFI_LFB)
714a96dd4fSbouyer return NULL;
724a96dd4fSbouyer
734a96dd4fSbouyer _xen_genfb_btinfo.common.type = BTINFO_FRAMEBUFFER;
744a96dd4fSbouyer _xen_genfb_btinfo.common.len = sizeof(struct btinfo_framebuffer);
754a96dd4fSbouyer _xen_genfb_btinfo.physaddr = d0_consi->u.vesa_lfb.lfb_base;
76*1b2c77f6Sbouyer if (info_size >=
774a96dd4fSbouyer offsetof(dom0_vga_console_info_t, u.vesa_lfb.ext_lfb_base)) {
784a96dd4fSbouyer _xen_genfb_btinfo.physaddr |=
794a96dd4fSbouyer (uint64_t)d0_consi->u.vesa_lfb.ext_lfb_base << 32;
804a96dd4fSbouyer }
814a96dd4fSbouyer _xen_genfb_btinfo.flags = 0;
824a96dd4fSbouyer _xen_genfb_btinfo.width = d0_consi->u.vesa_lfb.width;
834a96dd4fSbouyer _xen_genfb_btinfo.height = d0_consi->u.vesa_lfb.height;
844a96dd4fSbouyer _xen_genfb_btinfo.stride = d0_consi->u.vesa_lfb.bytes_per_line;
854a96dd4fSbouyer _xen_genfb_btinfo.depth = d0_consi->u.vesa_lfb.bits_per_pixel;
864a96dd4fSbouyer _xen_genfb_btinfo.rnum = d0_consi->u.vesa_lfb.red_pos;
874a96dd4fSbouyer _xen_genfb_btinfo.gnum = d0_consi->u.vesa_lfb.green_pos;
884a96dd4fSbouyer _xen_genfb_btinfo.bnum = d0_consi->u.vesa_lfb.blue_pos;
894a96dd4fSbouyer _xen_genfb_btinfo.rpos = d0_consi->u.vesa_lfb.red_size;
904a96dd4fSbouyer _xen_genfb_btinfo.gpos = d0_consi->u.vesa_lfb.green_size;
914a96dd4fSbouyer _xen_genfb_btinfo.bpos = d0_consi->u.vesa_lfb.blue_size;
924a96dd4fSbouyer _xen_genfb_btinfo.vbemode = 0; /* XXX */
934a96dd4fSbouyer
944a96dd4fSbouyer return &_xen_genfb_btinfo;
954a96dd4fSbouyer }
96