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