xref: /netbsd-src/sys/arch/xen/xen/genfb_xen.c (revision 1b2c77f6a3a4d386a295c61e52a15f6c51dcc552)
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