1*5c4a5fe1SAndy Fiddaman /*-
2*5c4a5fe1SAndy Fiddaman * SPDX-License-Identifier: BSD-2-Clause
3*5c4a5fe1SAndy Fiddaman *
4*5c4a5fe1SAndy Fiddaman * Copyright (c) 2011 NetApp, Inc.
5*5c4a5fe1SAndy Fiddaman * All rights reserved.
6*5c4a5fe1SAndy Fiddaman *
7*5c4a5fe1SAndy Fiddaman * Redistribution and use in source and binary forms, with or without
8*5c4a5fe1SAndy Fiddaman * modification, are permitted provided that the following conditions
9*5c4a5fe1SAndy Fiddaman * are met:
10*5c4a5fe1SAndy Fiddaman * 1. Redistributions of source code must retain the above copyright
11*5c4a5fe1SAndy Fiddaman * notice, this list of conditions and the following disclaimer.
12*5c4a5fe1SAndy Fiddaman * 2. Redistributions in binary form must reproduce the above copyright
13*5c4a5fe1SAndy Fiddaman * notice, this list of conditions and the following disclaimer in the
14*5c4a5fe1SAndy Fiddaman * documentation and/or other materials provided with the distribution.
15*5c4a5fe1SAndy Fiddaman *
16*5c4a5fe1SAndy Fiddaman * THIS SOFTWARE IS PROVIDED BY NETAPP, INC ``AS IS'' AND
17*5c4a5fe1SAndy Fiddaman * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18*5c4a5fe1SAndy Fiddaman * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19*5c4a5fe1SAndy Fiddaman * ARE DISCLAIMED. IN NO EVENT SHALL NETAPP, INC OR CONTRIBUTORS BE LIABLE
20*5c4a5fe1SAndy Fiddaman * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21*5c4a5fe1SAndy Fiddaman * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22*5c4a5fe1SAndy Fiddaman * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23*5c4a5fe1SAndy Fiddaman * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24*5c4a5fe1SAndy Fiddaman * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25*5c4a5fe1SAndy Fiddaman * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26*5c4a5fe1SAndy Fiddaman * SUCH DAMAGE.
27*5c4a5fe1SAndy Fiddaman */
28*5c4a5fe1SAndy Fiddaman /*
29*5c4a5fe1SAndy Fiddaman * This file and its contents are supplied under the terms of the
30*5c4a5fe1SAndy Fiddaman * Common Development and Distribution License ("CDDL"), version 1.0.
31*5c4a5fe1SAndy Fiddaman * You may only use this file in accordance with the terms of version
32*5c4a5fe1SAndy Fiddaman * 1.0 of the CDDL.
33*5c4a5fe1SAndy Fiddaman *
34*5c4a5fe1SAndy Fiddaman * A full copy of the text of the CDDL should have accompanied this
35*5c4a5fe1SAndy Fiddaman * source. A copy of the CDDL is also available via the Internet at
36*5c4a5fe1SAndy Fiddaman * http://www.illumos.org/license/CDDL.
37*5c4a5fe1SAndy Fiddaman *
38*5c4a5fe1SAndy Fiddaman * Copyright 2020 Oxide Computer Company
39*5c4a5fe1SAndy Fiddaman */
40*5c4a5fe1SAndy Fiddaman
41*5c4a5fe1SAndy Fiddaman
42*5c4a5fe1SAndy Fiddaman #include <sys/param.h>
43*5c4a5fe1SAndy Fiddaman #include <sys/linker_set.h>
44*5c4a5fe1SAndy Fiddaman #include <sys/_iovec.h>
45*5c4a5fe1SAndy Fiddaman #include <sys/mman.h>
46*5c4a5fe1SAndy Fiddaman
47*5c4a5fe1SAndy Fiddaman #include <x86/psl.h>
48*5c4a5fe1SAndy Fiddaman #include <x86/segments.h>
49*5c4a5fe1SAndy Fiddaman
50*5c4a5fe1SAndy Fiddaman #include <machine/vmm.h>
51*5c4a5fe1SAndy Fiddaman #include <vmmapi.h>
52*5c4a5fe1SAndy Fiddaman
53*5c4a5fe1SAndy Fiddaman #include <stdio.h>
54*5c4a5fe1SAndy Fiddaman #include <string.h>
55*5c4a5fe1SAndy Fiddaman #include <assert.h>
56*5c4a5fe1SAndy Fiddaman
57*5c4a5fe1SAndy Fiddaman #include "bhyverun.h"
58*5c4a5fe1SAndy Fiddaman #include "config.h"
59*5c4a5fe1SAndy Fiddaman #include "inout.h"
60*5c4a5fe1SAndy Fiddaman
61*5c4a5fe1SAndy Fiddaman SET_DECLARE(inout_port_set, struct inout_port);
62*5c4a5fe1SAndy Fiddaman
63*5c4a5fe1SAndy Fiddaman #define MAX_IOPORTS (1 << 16)
64*5c4a5fe1SAndy Fiddaman
65*5c4a5fe1SAndy Fiddaman #define VERIFY_IOPORT(port, size) \
66*5c4a5fe1SAndy Fiddaman assert((port) >= 0 && (size) > 0 && ((port) + (size)) <= MAX_IOPORTS)
67*5c4a5fe1SAndy Fiddaman
68*5c4a5fe1SAndy Fiddaman struct inout_handler {
69*5c4a5fe1SAndy Fiddaman const char *name;
70*5c4a5fe1SAndy Fiddaman int flags;
71*5c4a5fe1SAndy Fiddaman inout_func_t handler;
72*5c4a5fe1SAndy Fiddaman void *arg;
73*5c4a5fe1SAndy Fiddaman };
74*5c4a5fe1SAndy Fiddaman
75*5c4a5fe1SAndy Fiddaman static struct inout_handler inout_handlers[MAX_IOPORTS];
76*5c4a5fe1SAndy Fiddaman
77*5c4a5fe1SAndy Fiddaman static int
default_inout(struct vmctx * ctx __unused,int in,int port __unused,int bytes,uint32_t * eax,void * arg __unused)78*5c4a5fe1SAndy Fiddaman default_inout(struct vmctx *ctx __unused, int in,
79*5c4a5fe1SAndy Fiddaman int port __unused, int bytes, uint32_t *eax, void *arg __unused)
80*5c4a5fe1SAndy Fiddaman {
81*5c4a5fe1SAndy Fiddaman if (in) {
82*5c4a5fe1SAndy Fiddaman switch (bytes) {
83*5c4a5fe1SAndy Fiddaman case 4:
84*5c4a5fe1SAndy Fiddaman *eax = 0xffffffff;
85*5c4a5fe1SAndy Fiddaman break;
86*5c4a5fe1SAndy Fiddaman case 2:
87*5c4a5fe1SAndy Fiddaman *eax = 0xffff;
88*5c4a5fe1SAndy Fiddaman break;
89*5c4a5fe1SAndy Fiddaman case 1:
90*5c4a5fe1SAndy Fiddaman *eax = 0xff;
91*5c4a5fe1SAndy Fiddaman break;
92*5c4a5fe1SAndy Fiddaman }
93*5c4a5fe1SAndy Fiddaman }
94*5c4a5fe1SAndy Fiddaman
95*5c4a5fe1SAndy Fiddaman return (0);
96*5c4a5fe1SAndy Fiddaman }
97*5c4a5fe1SAndy Fiddaman
98*5c4a5fe1SAndy Fiddaman static void
register_default_iohandler(int start,int size)99*5c4a5fe1SAndy Fiddaman register_default_iohandler(int start, int size)
100*5c4a5fe1SAndy Fiddaman {
101*5c4a5fe1SAndy Fiddaman struct inout_port iop;
102*5c4a5fe1SAndy Fiddaman
103*5c4a5fe1SAndy Fiddaman VERIFY_IOPORT(start, size);
104*5c4a5fe1SAndy Fiddaman
105*5c4a5fe1SAndy Fiddaman bzero(&iop, sizeof(iop));
106*5c4a5fe1SAndy Fiddaman iop.name = "default";
107*5c4a5fe1SAndy Fiddaman iop.port = start;
108*5c4a5fe1SAndy Fiddaman iop.size = size;
109*5c4a5fe1SAndy Fiddaman iop.flags = IOPORT_F_INOUT | IOPORT_F_DEFAULT;
110*5c4a5fe1SAndy Fiddaman iop.handler = default_inout;
111*5c4a5fe1SAndy Fiddaman
112*5c4a5fe1SAndy Fiddaman register_inout(&iop);
113*5c4a5fe1SAndy Fiddaman }
114*5c4a5fe1SAndy Fiddaman
115*5c4a5fe1SAndy Fiddaman int
emulate_inout(struct vmctx * ctx,struct vcpu * vcpu,struct vm_inout * inout)116*5c4a5fe1SAndy Fiddaman emulate_inout(struct vmctx *ctx, struct vcpu *vcpu, struct vm_inout *inout)
117*5c4a5fe1SAndy Fiddaman {
118*5c4a5fe1SAndy Fiddaman struct inout_handler handler;
119*5c4a5fe1SAndy Fiddaman inout_func_t hfunc;
120*5c4a5fe1SAndy Fiddaman void *harg;
121*5c4a5fe1SAndy Fiddaman int error;
122*5c4a5fe1SAndy Fiddaman uint8_t bytes;
123*5c4a5fe1SAndy Fiddaman bool in;
124*5c4a5fe1SAndy Fiddaman
125*5c4a5fe1SAndy Fiddaman bytes = inout->bytes;
126*5c4a5fe1SAndy Fiddaman in = (inout->flags & INOUT_IN) != 0;
127*5c4a5fe1SAndy Fiddaman
128*5c4a5fe1SAndy Fiddaman assert(bytes == 1 || bytes == 2 || bytes == 4);
129*5c4a5fe1SAndy Fiddaman
130*5c4a5fe1SAndy Fiddaman handler = inout_handlers[inout->port];
131*5c4a5fe1SAndy Fiddaman hfunc = handler.handler;
132*5c4a5fe1SAndy Fiddaman harg = handler.arg;
133*5c4a5fe1SAndy Fiddaman
134*5c4a5fe1SAndy Fiddaman if (hfunc == default_inout &&
135*5c4a5fe1SAndy Fiddaman get_config_bool_default("x86.strictio", false))
136*5c4a5fe1SAndy Fiddaman return (-1);
137*5c4a5fe1SAndy Fiddaman
138*5c4a5fe1SAndy Fiddaman if (in) {
139*5c4a5fe1SAndy Fiddaman if (!(handler.flags & IOPORT_F_IN))
140*5c4a5fe1SAndy Fiddaman return (-1);
141*5c4a5fe1SAndy Fiddaman } else {
142*5c4a5fe1SAndy Fiddaman if (!(handler.flags & IOPORT_F_OUT))
143*5c4a5fe1SAndy Fiddaman return (-1);
144*5c4a5fe1SAndy Fiddaman }
145*5c4a5fe1SAndy Fiddaman
146*5c4a5fe1SAndy Fiddaman error = hfunc(ctx, in, inout->port, bytes, &inout->eax, harg);
147*5c4a5fe1SAndy Fiddaman return (error);
148*5c4a5fe1SAndy Fiddaman }
149*5c4a5fe1SAndy Fiddaman
150*5c4a5fe1SAndy Fiddaman void
init_inout(void)151*5c4a5fe1SAndy Fiddaman init_inout(void)
152*5c4a5fe1SAndy Fiddaman {
153*5c4a5fe1SAndy Fiddaman struct inout_port **iopp, *iop;
154*5c4a5fe1SAndy Fiddaman
155*5c4a5fe1SAndy Fiddaman /*
156*5c4a5fe1SAndy Fiddaman * Set up the default handler for all ports
157*5c4a5fe1SAndy Fiddaman */
158*5c4a5fe1SAndy Fiddaman register_default_iohandler(0, MAX_IOPORTS);
159*5c4a5fe1SAndy Fiddaman
160*5c4a5fe1SAndy Fiddaman /*
161*5c4a5fe1SAndy Fiddaman * Overwrite with specified handlers
162*5c4a5fe1SAndy Fiddaman */
163*5c4a5fe1SAndy Fiddaman SET_FOREACH(iopp, inout_port_set) {
164*5c4a5fe1SAndy Fiddaman iop = *iopp;
165*5c4a5fe1SAndy Fiddaman assert(iop->port < MAX_IOPORTS);
166*5c4a5fe1SAndy Fiddaman inout_handlers[iop->port].name = iop->name;
167*5c4a5fe1SAndy Fiddaman inout_handlers[iop->port].flags = iop->flags;
168*5c4a5fe1SAndy Fiddaman inout_handlers[iop->port].handler = iop->handler;
169*5c4a5fe1SAndy Fiddaman inout_handlers[iop->port].arg = NULL;
170*5c4a5fe1SAndy Fiddaman }
171*5c4a5fe1SAndy Fiddaman }
172*5c4a5fe1SAndy Fiddaman
173*5c4a5fe1SAndy Fiddaman int
register_inout(struct inout_port * iop)174*5c4a5fe1SAndy Fiddaman register_inout(struct inout_port *iop)
175*5c4a5fe1SAndy Fiddaman {
176*5c4a5fe1SAndy Fiddaman int i;
177*5c4a5fe1SAndy Fiddaman
178*5c4a5fe1SAndy Fiddaman VERIFY_IOPORT(iop->port, iop->size);
179*5c4a5fe1SAndy Fiddaman
180*5c4a5fe1SAndy Fiddaman /*
181*5c4a5fe1SAndy Fiddaman * Verify that the new registration is not overwriting an already
182*5c4a5fe1SAndy Fiddaman * allocated i/o range.
183*5c4a5fe1SAndy Fiddaman */
184*5c4a5fe1SAndy Fiddaman if ((iop->flags & IOPORT_F_DEFAULT) == 0) {
185*5c4a5fe1SAndy Fiddaman for (i = iop->port; i < iop->port + iop->size; i++) {
186*5c4a5fe1SAndy Fiddaman if ((inout_handlers[i].flags & IOPORT_F_DEFAULT) == 0)
187*5c4a5fe1SAndy Fiddaman return (-1);
188*5c4a5fe1SAndy Fiddaman }
189*5c4a5fe1SAndy Fiddaman }
190*5c4a5fe1SAndy Fiddaman
191*5c4a5fe1SAndy Fiddaman for (i = iop->port; i < iop->port + iop->size; i++) {
192*5c4a5fe1SAndy Fiddaman inout_handlers[i].name = iop->name;
193*5c4a5fe1SAndy Fiddaman inout_handlers[i].flags = iop->flags;
194*5c4a5fe1SAndy Fiddaman inout_handlers[i].handler = iop->handler;
195*5c4a5fe1SAndy Fiddaman inout_handlers[i].arg = iop->arg;
196*5c4a5fe1SAndy Fiddaman }
197*5c4a5fe1SAndy Fiddaman
198*5c4a5fe1SAndy Fiddaman return (0);
199*5c4a5fe1SAndy Fiddaman }
200*5c4a5fe1SAndy Fiddaman
201*5c4a5fe1SAndy Fiddaman int
unregister_inout(struct inout_port * iop)202*5c4a5fe1SAndy Fiddaman unregister_inout(struct inout_port *iop)
203*5c4a5fe1SAndy Fiddaman {
204*5c4a5fe1SAndy Fiddaman
205*5c4a5fe1SAndy Fiddaman VERIFY_IOPORT(iop->port, iop->size);
206*5c4a5fe1SAndy Fiddaman assert(inout_handlers[iop->port].name == iop->name);
207*5c4a5fe1SAndy Fiddaman
208*5c4a5fe1SAndy Fiddaman register_default_iohandler(iop->port, iop->size);
209*5c4a5fe1SAndy Fiddaman
210*5c4a5fe1SAndy Fiddaman return (0);
211*5c4a5fe1SAndy Fiddaman }
212