xref: /openbsd/sys/arch/amd64/pci/vga_post.c (revision 4c9b2a68)
1*4c9b2a68Sjmatthew /* $OpenBSD: vga_post.c,v 1.11 2021/01/03 02:29:28 jmatthew Exp $ */
2e208dfa2Spirofti /* $NetBSD: vga_post.c,v 1.12 2009/03/15 21:32:36 cegger Exp $ */
3e208dfa2Spirofti 
4e208dfa2Spirofti /*-
5e208dfa2Spirofti  * Copyright (c) 2007 Joerg Sonnenberger <joerg@NetBSD.org>.
6e208dfa2Spirofti  * All rights reserved.
7e208dfa2Spirofti  *
8e208dfa2Spirofti  * Redistribution and use in source and binary forms, with or without
9e208dfa2Spirofti  * modification, are permitted provided that the following conditions
10e208dfa2Spirofti  * are met:
11e208dfa2Spirofti  *
12e208dfa2Spirofti  * 1. Redistributions of source code must retain the above copyright
13e208dfa2Spirofti  *    notice, this list of conditions and the following disclaimer.
14e208dfa2Spirofti  * 2. Redistributions in binary form must reproduce the above copyright
15e208dfa2Spirofti  *    notice, this list of conditions and the following disclaimer in
16e208dfa2Spirofti  *    the documentation and/or other materials provided with the
17e208dfa2Spirofti  *    distribution.
18e208dfa2Spirofti  *
19e208dfa2Spirofti  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20e208dfa2Spirofti  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21e208dfa2Spirofti  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
22e208dfa2Spirofti  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE
23e208dfa2Spirofti  * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
24e208dfa2Spirofti  * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
25e208dfa2Spirofti  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
26e208dfa2Spirofti  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
27e208dfa2Spirofti  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
28e208dfa2Spirofti  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
29e208dfa2Spirofti  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30e208dfa2Spirofti  * SUCH DAMAGE.
31e208dfa2Spirofti  */
32e208dfa2Spirofti 
33e208dfa2Spirofti #include <sys/param.h>
34e208dfa2Spirofti #include <sys/device.h>
35e208dfa2Spirofti #include <sys/malloc.h>
3631a7ed3dSmpi 
37e208dfa2Spirofti #include <uvm/uvm_extern.h>
38e208dfa2Spirofti 
39e208dfa2Spirofti #include <machine/pio.h>
40e208dfa2Spirofti 
414b013528Sderaadt #include <dev/x86emu/x86emu.h>
424b013528Sderaadt #include <dev/x86emu/x86emu_regs.h>
43e208dfa2Spirofti 
44e208dfa2Spirofti #define	BASE_MEMORY	65536	/* How much memory to allocate in Real Mode */
45e208dfa2Spirofti 
46e208dfa2Spirofti struct vga_post {
476bf2980cSderaadt 	struct x86emu emu;
48e208dfa2Spirofti 	vaddr_t sys_image;
49e208dfa2Spirofti 	uint32_t initial_eax;
50e208dfa2Spirofti 	uint8_t bios_data[PAGE_SIZE];
51e208dfa2Spirofti 	struct pglist ram_backing;
52e208dfa2Spirofti };
53e208dfa2Spirofti 
54e208dfa2Spirofti #ifdef DDB
55e208dfa2Spirofti static struct vga_post *ddb_vgapostp;
56e208dfa2Spirofti void ddb_vgapost(void);
57e208dfa2Spirofti #endif
58e208dfa2Spirofti 
59e208dfa2Spirofti static uint8_t
vm86_emu_inb(struct x86emu * emu,uint16_t port)606bf2980cSderaadt vm86_emu_inb(struct x86emu *emu, uint16_t port)
61e208dfa2Spirofti {
62e208dfa2Spirofti 	if (port == 0xb2) /* APM scratch register */
63e208dfa2Spirofti 		return 0;
64e208dfa2Spirofti 
65e208dfa2Spirofti 	if (port >= 0x80 && port < 0x88) /* POST status register */
66e208dfa2Spirofti 		return 0;
67e208dfa2Spirofti 
68e208dfa2Spirofti 	return inb(port);
69e208dfa2Spirofti }
70e208dfa2Spirofti 
71e208dfa2Spirofti static uint16_t
vm86_emu_inw(struct x86emu * emu,uint16_t port)726bf2980cSderaadt vm86_emu_inw(struct x86emu *emu, uint16_t port)
73e208dfa2Spirofti {
74e208dfa2Spirofti 	if (port >= 0x80 && port < 0x88) /* POST status register */
75e208dfa2Spirofti 		return 0;
76e208dfa2Spirofti 
77e208dfa2Spirofti 	return inw(port);
78e208dfa2Spirofti }
79e208dfa2Spirofti 
80e208dfa2Spirofti static uint32_t
vm86_emu_inl(struct x86emu * emu,uint16_t port)816bf2980cSderaadt vm86_emu_inl(struct x86emu *emu, uint16_t port)
82e208dfa2Spirofti {
83e208dfa2Spirofti 	if (port >= 0x80 && port < 0x88) /* POST status register */
84e208dfa2Spirofti 		return 0;
85e208dfa2Spirofti 
86e208dfa2Spirofti 	return inl(port);
87e208dfa2Spirofti }
88e208dfa2Spirofti 
89e208dfa2Spirofti static void
vm86_emu_outb(struct x86emu * emu,uint16_t port,uint8_t val)906bf2980cSderaadt vm86_emu_outb(struct x86emu *emu, uint16_t port, uint8_t val)
91e208dfa2Spirofti {
92e208dfa2Spirofti 	if (port == 0xb2) /* APM scratch register */
93e208dfa2Spirofti 		return;
94e208dfa2Spirofti 
95e208dfa2Spirofti 	if (port >= 0x80 && port < 0x88) /* POST status register */
96e208dfa2Spirofti 		return;
97e208dfa2Spirofti 
98e208dfa2Spirofti 	outb(port, val);
99e208dfa2Spirofti }
100e208dfa2Spirofti 
101e208dfa2Spirofti static void
vm86_emu_outw(struct x86emu * emu,uint16_t port,uint16_t val)1026bf2980cSderaadt vm86_emu_outw(struct x86emu *emu, uint16_t port, uint16_t val)
103e208dfa2Spirofti {
104e208dfa2Spirofti 	if (port >= 0x80 && port < 0x88) /* POST status register */
105e208dfa2Spirofti 		return;
106e208dfa2Spirofti 
107e208dfa2Spirofti 	outw(port, val);
108e208dfa2Spirofti }
109e208dfa2Spirofti 
110e208dfa2Spirofti static void
vm86_emu_outl(struct x86emu * emu,uint16_t port,uint32_t val)1116bf2980cSderaadt vm86_emu_outl(struct x86emu *emu, uint16_t port, uint32_t val)
112e208dfa2Spirofti {
113e208dfa2Spirofti 	if (port >= 0x80 && port < 0x88) /* POST status register */
114e208dfa2Spirofti 		return;
115e208dfa2Spirofti 
116e208dfa2Spirofti 	outl(port, val);
117e208dfa2Spirofti }
118e208dfa2Spirofti 
119e208dfa2Spirofti struct vga_post *
vga_post_init(int bus,int device,int function)120e208dfa2Spirofti vga_post_init(int bus, int device, int function)
121e208dfa2Spirofti {
122e208dfa2Spirofti 	struct vga_post *sc;
123e208dfa2Spirofti 	vaddr_t iter;
124e208dfa2Spirofti 	struct vm_page *pg;
125e208dfa2Spirofti 	vaddr_t sys_image, sys_bios_data;
126e208dfa2Spirofti 	int err;
127e208dfa2Spirofti 
128*4c9b2a68Sjmatthew 	sys_bios_data = (vaddr_t)km_alloc(PAGE_SIZE, &kv_any, &kp_none,
129*4c9b2a68Sjmatthew 	    &kd_nowait);
130e208dfa2Spirofti 	if (sys_bios_data == 0)
131e208dfa2Spirofti 		return NULL;
132e208dfa2Spirofti 
133*4c9b2a68Sjmatthew 	sys_image = (vaddr_t)km_alloc(1024 * 1024, &kv_any, &kp_none,
134*4c9b2a68Sjmatthew 	    &kd_nowait);
135e208dfa2Spirofti 	if (sys_image == 0) {
136*4c9b2a68Sjmatthew 		km_free((void *)sys_bios_data, PAGE_SIZE, &kv_any, &kp_none);
137e208dfa2Spirofti 		return NULL;
138e208dfa2Spirofti 	}
139e208dfa2Spirofti 	sc = malloc(sizeof(*sc), M_DEVBUF, M_WAITOK|M_ZERO);
140e208dfa2Spirofti 
1417113f419Smiod 	TAILQ_INIT(&sc->ram_backing);
142e208dfa2Spirofti 	err = uvm_pglistalloc(BASE_MEMORY, 0, (paddr_t)-1, 0, 0,
143e208dfa2Spirofti 	    &sc->ram_backing, BASE_MEMORY/PAGE_SIZE, UVM_PLA_WAITOK);
144e208dfa2Spirofti 	if (err) {
145*4c9b2a68Sjmatthew 		km_free((void *)sc->sys_image, 1024 * 1024, &kv_any, &kp_none);
146bae2bd50Sderaadt 		free(sc, M_DEVBUF, sizeof(*sc));
147e208dfa2Spirofti 		return NULL;
148e208dfa2Spirofti 	}
149e208dfa2Spirofti 
150e208dfa2Spirofti 	sc->sys_image = sys_image;
151e208dfa2Spirofti 	sc->emu.sys_private = sc;
152e208dfa2Spirofti 
1531e8cdc2eSderaadt 	pmap_kenter_pa(sys_bios_data, 0, PROT_READ);
154e208dfa2Spirofti 	pmap_update(pmap_kernel());
155e208dfa2Spirofti 	memcpy((void *)sc->bios_data, (void *)sys_bios_data, PAGE_SIZE);
156e208dfa2Spirofti 	pmap_kremove(sys_bios_data, PAGE_SIZE);
157*4c9b2a68Sjmatthew 	km_free((void *)sys_bios_data, PAGE_SIZE, &kv_any, &kp_none);
158e208dfa2Spirofti 
159e208dfa2Spirofti 	iter = 0;
160e208dfa2Spirofti 	TAILQ_FOREACH(pg, &sc->ram_backing, pageq) {
161e208dfa2Spirofti 		pmap_kenter_pa(sc->sys_image + iter, VM_PAGE_TO_PHYS(pg),
1621e8cdc2eSderaadt 		    PROT_READ | PROT_WRITE);
163e208dfa2Spirofti 		iter += PAGE_SIZE;
164e208dfa2Spirofti 	}
1657113f419Smiod 	KASSERT(iter == BASE_MEMORY);
166e208dfa2Spirofti 
167e208dfa2Spirofti 	for (iter = 640 * 1024; iter < 1024 * 1024; iter += PAGE_SIZE)
168e208dfa2Spirofti 		pmap_kenter_pa(sc->sys_image + iter, iter,
1691e8cdc2eSderaadt 		    PROT_READ | PROT_WRITE);
170e208dfa2Spirofti 	pmap_update(pmap_kernel());
171e208dfa2Spirofti 
172e208dfa2Spirofti 	memset(&sc->emu, 0, sizeof(sc->emu));
1736bf2980cSderaadt 	x86emu_init_default(&sc->emu);
174e208dfa2Spirofti 	sc->emu.emu_inb = vm86_emu_inb;
175e208dfa2Spirofti 	sc->emu.emu_inw = vm86_emu_inw;
176e208dfa2Spirofti 	sc->emu.emu_inl = vm86_emu_inl;
177e208dfa2Spirofti 	sc->emu.emu_outb = vm86_emu_outb;
178e208dfa2Spirofti 	sc->emu.emu_outw = vm86_emu_outw;
179e208dfa2Spirofti 	sc->emu.emu_outl = vm86_emu_outl;
180e208dfa2Spirofti 
181e208dfa2Spirofti 	sc->emu.mem_base = (char *)sc->sys_image;
182e208dfa2Spirofti 	sc->emu.mem_size = 1024 * 1024;
183e208dfa2Spirofti 
184e208dfa2Spirofti 	sc->initial_eax = bus * 256 + device * 8 + function;
185e208dfa2Spirofti #ifdef DDB
186e208dfa2Spirofti 	ddb_vgapostp = sc;
187e208dfa2Spirofti #endif
188e208dfa2Spirofti 	return sc;
189e208dfa2Spirofti }
190e208dfa2Spirofti 
191e208dfa2Spirofti void
vga_post_call(struct vga_post * sc)192e208dfa2Spirofti vga_post_call(struct vga_post *sc)
193e208dfa2Spirofti {
194e208dfa2Spirofti 	sc->emu.x86.R_EAX = sc->initial_eax;
195e208dfa2Spirofti 	sc->emu.x86.R_EDX = 0x00000080;
196e208dfa2Spirofti 	sc->emu.x86.R_DS = 0x0040;
197e208dfa2Spirofti 	sc->emu.x86.register_flags = 0x3200;
198e208dfa2Spirofti 
199e208dfa2Spirofti 	memcpy((void *)sc->sys_image, sc->bios_data, PAGE_SIZE);
200e208dfa2Spirofti 
201e208dfa2Spirofti 	/* stack is at the end of the first 64KB */
202e208dfa2Spirofti 	sc->emu.x86.R_SS = 0;
203e208dfa2Spirofti 	sc->emu.x86.R_ESP = 0;
204e208dfa2Spirofti 
205e208dfa2Spirofti 	/* Jump straight into the VGA BIOS POST code */
2066bf2980cSderaadt 	x86emu_exec_call(&sc->emu, 0xc000, 0x0003);
207e208dfa2Spirofti }
208e208dfa2Spirofti 
209e208dfa2Spirofti void
vga_post_free(struct vga_post * sc)210e208dfa2Spirofti vga_post_free(struct vga_post *sc)
211e208dfa2Spirofti {
212e208dfa2Spirofti 	uvm_pglistfree(&sc->ram_backing);
213e208dfa2Spirofti 	pmap_kremove(sc->sys_image, 1024 * 1024);
214*4c9b2a68Sjmatthew 	km_free((void *)sc->sys_image, 1024 * 1024, &kv_any, &kp_none)
215e208dfa2Spirofti 	pmap_update(pmap_kernel());
216bae2bd50Sderaadt 	free(sc, M_DEVBUF, sizeof(*sc));
217e208dfa2Spirofti }
218e208dfa2Spirofti 
219e208dfa2Spirofti #ifdef DDB
220e208dfa2Spirofti void
ddb_vgapost(void)221e208dfa2Spirofti ddb_vgapost(void)
222e208dfa2Spirofti {
223e208dfa2Spirofti 
224e208dfa2Spirofti 	if (ddb_vgapostp)
225e208dfa2Spirofti 		vga_post_call(ddb_vgapostp);
226e208dfa2Spirofti 	else
227e208dfa2Spirofti 		printf("ddb_vgapost: vga_post not initialized\n");
228e208dfa2Spirofti }
229e208dfa2Spirofti #endif
230