VirtualBox

source: vbox/trunk/src/VBox/Devices/PC/BIOS/ps2mouse.c@ 69500

Last change on this file since 69500 was 69500, checked in by vboxsync, 7 years ago

*: scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.0 KB
Line 
1/*
2 * Copyright (C) 2006-2017 Oracle Corporation
3 *
4 * This file is part of VirtualBox Open Source Edition (OSE), as
5 * available from http://www.virtualbox.org. This file is free software;
6 * you can redistribute it and/or modify it under the terms of the GNU
7 * General Public License (GPL) as published by the Free Software
8 * Foundation, in version 2 as it comes in the "COPYING" file of the
9 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
10 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
11 * --------------------------------------------------------------------
12 *
13 * This code is based on:
14 *
15 * ROM BIOS for use with Bochs/Plex86/QEMU emulation environment
16 *
17 * Copyright (C) 2002 MandrakeSoft S.A.
18 *
19 * MandrakeSoft S.A.
20 * 43, rue d'Aboukir
21 * 75002 Paris - France
22 * http://www.linux-mandrake.com/
23 * http://www.mandrakesoft.com/
24 *
25 * This library is free software; you can redistribute it and/or
26 * modify it under the terms of the GNU Lesser General Public
27 * License as published by the Free Software Foundation; either
28 * version 2 of the License, or (at your option) any later version.
29 *
30 * This library is distributed in the hope that it will be useful,
31 * but WITHOUT ANY WARRANTY; without even the implied warranty of
32 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
33 * Lesser General Public License for more details.
34 *
35 * You should have received a copy of the GNU Lesser General Public
36 * License along with this library; if not, write to the Free Software
37 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
38 *
39 */
40
41
42#include <stdint.h>
43#include "biosint.h"
44#include "inlines.h"
45
46
47#if DEBUG_INT15_MS
48# define BX_DEBUG_INT15_MS(...) BX_DEBUG(__VA_ARGS__)
49#else
50# define BX_DEBUG_INT15_MS(...)
51#endif
52
53#if DEBUG_INT74
54# define BX_DEBUG_INT74(...) BX_DEBUG(__VA_ARGS__)
55#else
56# define BX_DEBUG_INT74(...)
57#endif
58
59#if BX_USE_PS2_MOUSE
60
61static const char panic_msg_keyb_buffer_full[] = "%s: keyboard input buffer full\n";
62
63uint8_t send_to_mouse_ctrl(uint8_t sendbyte)
64{
65 BX_DEBUG_INT15_MS("send %02x to mouse:\n", sendbyte);
66 // wait for chance to write to ctrl
67 if (inb(0x64) & 0x02)
68 BX_PANIC(panic_msg_keyb_buffer_full,"sendmouse");
69 outb(0x64, 0xD4);
70 outb(0x60, sendbyte);
71 return(0);
72}
73
74
75uint8_t get_mouse_data(uint8_t __far *data)
76{
77 int retries = 10000; /* ~150ms timeout */
78 uint8_t response;
79
80 while ((inb(0x64) & 0x21) != 0x21 && retries)
81 {
82 /* Wait until the 15us refresh counter toggles. */
83 response = inb(0x61) & 0x10;
84 while((inb(0x61) & 0x10) == response)
85 ;
86 --retries;
87 }
88
89 if (!retries)
90 return(1);
91
92 response = inb(0x60);
93 *data = response;
94 return(0);
95}
96
97void set_kbd_command_byte(uint8_t command_byte)
98{
99 if (inb(0x64) & 0x02)
100 BX_PANIC(panic_msg_keyb_buffer_full,"setkbdcomm");
101
102 outb(0x64, 0x60); // write command byte
103 outb(0x60, command_byte);
104}
105
106
107void BIOSCALL int74_function(volatile uint16_t make_farcall, volatile uint16_t Z,
108 volatile uint16_t Y, volatile uint16_t X, volatile uint16_t status)
109{
110 uint16_t ebda_seg=read_word(0x0040,0x000E);
111 uint8_t in_byte, index, package_count;
112 uint8_t mouse_flags_1, mouse_flags_2;
113
114 BX_DEBUG_INT74("entering int74_function\n");
115 make_farcall = 0;
116
117 in_byte = inb(0x64);
118 if ( (in_byte & 0x21) != 0x21 ) {
119 return;
120 }
121 in_byte = inb(0x60);
122 BX_DEBUG_INT74("int74: read byte %02x\n", in_byte);
123
124 mouse_flags_1 = read_byte(ebda_seg, 0x0026);
125 mouse_flags_2 = read_byte(ebda_seg, 0x0027);
126
127 if ( (mouse_flags_2 & 0x80) != 0x80 ) {
128 return;
129 }
130
131 package_count = mouse_flags_2 & 0x07;
132 index = mouse_flags_1 & 0x07;
133 write_byte(ebda_seg, 0x28 + index, in_byte);
134
135 if ( index >= package_count ) {
136 BX_DEBUG_INT74("int74_function: make_farcall=1\n");
137 status = read_byte(ebda_seg, 0x0028 + 0);
138 X = read_byte(ebda_seg, 0x0028 + 1);
139 Y = read_byte(ebda_seg, 0x0028 + 2);
140 Z = 0;
141 mouse_flags_1 = 0;
142 // check if far call handler installed
143 if (mouse_flags_2 & 0x80)
144 make_farcall = 1;
145 }
146 else {
147 mouse_flags_1++;
148 }
149 write_byte(ebda_seg, 0x0026, mouse_flags_1);
150}
151
152void BIOSCALL int15_function_mouse(pusha_regs_t regs, uint16_t ES, uint16_t DS, volatile uint16_t FLAGS)
153{
154 uint16_t ebda_seg=read_word(0x0040,0x000E);
155 uint8_t mouse_flags_1, mouse_flags_2;
156 uint16_t mouse_driver_seg;
157 uint16_t mouse_driver_offset;
158 uint8_t mouse_cmd;
159 uint8_t ret, mouse_data1, mouse_data2, mouse_data3;
160
161 BX_DEBUG_INT15_MS("int15 AX=%04x\n",regs.u.r16.ax);
162
163 // Return Codes status in AH
164 // =========================
165 // 00: success
166 // 01: invalid subfunction (AL > 7)
167 // 02: invalid input value (out of allowable range)
168 // 03: interface error
169 // 04: resend command received from mouse controller,
170 // device driver should attempt command again
171 // 05: cannot enable mouse, since no far call has been installed
172 // 80/86: mouse service not implemented
173
174 if (regs.u.r8.al > 7) {
175 BX_DEBUG_INT15_MS("unsupported subfn\n");
176 // invalid function
177 SET_CF();
178 regs.u.r8.ah = 1;
179 return;
180 }
181
182 // Valid subfn; disable AUX input and IRQ12, assume no error
183 set_kbd_command_byte(0x65);
184 CLEAR_CF();
185 regs.u.r8.ah = 0;
186
187 switch (regs.u.r8.al) {
188 case 0: // Disable/Enable Mouse
189 BX_DEBUG_INT15_MS("case 0: ");
190 if (regs.u.r8.bh > 1) {
191 BX_DEBUG_INT15_MS("INT 15h C2 AL=0, BH=%02x\n", (unsigned) regs.u.r8.bh);
192 // invalid subfunction
193 SET_CF();
194 regs.u.r8.ah = 1;
195 break;
196 }
197 mouse_flags_2 = read_byte(ebda_seg, 0x0027);
198 if ( (mouse_flags_2 & 0x80) == 0 ) {
199 BX_DEBUG_INT15_MS("INT 15h C2 Enable/Disable Mouse, no far call handler\n");
200 SET_CF();
201 regs.u.r8.ah = 5; // no far call installed
202 break;
203 }
204 if (regs.u.r8.bh == 0) {
205 BX_DEBUG_INT15_MS("Disable Mouse\n");
206 mouse_cmd = 0xF5; // disable mouse command
207 } else {
208 BX_DEBUG_INT15_MS("Enable Mouse\n");
209 mouse_cmd = 0xF4; // enable mouse command
210 }
211
212 ret = send_to_mouse_ctrl(mouse_cmd); // disable mouse command
213 if (ret == 0) {
214 ret = get_mouse_data(&mouse_data1);
215 if ( (ret == 0) || (mouse_data1 == 0xFA) ) {
216 // success
217 break;
218 }
219 }
220
221 // interface error
222 SET_CF();
223 regs.u.r8.ah = 3;
224 break;
225
226 case 5: // Initialize Mouse
227 // Valid package sizes are 1 to 8
228 if ( (regs.u.r8.bh < 1) || (regs.u.r8.bh > 8) ) {
229 SET_CF();
230 regs.u.r8.ah = 2; // invalid input
231 break;
232 }
233 mouse_flags_2 = read_byte(ebda_seg, 0x0027);
234 mouse_flags_2 = (mouse_flags_2 & 0xf8) | (regs.u.r8.bh - 1);
235 write_byte(ebda_seg, 0x0027, mouse_flags_2);
236 // fall through!
237
238 case 1: // Reset Mouse
239 BX_DEBUG_INT15_MS("case 1 or 5:\n");
240 // clear current package byte index
241 mouse_flags_1 = read_byte(ebda_seg, 0x0026);
242 mouse_flags_1 = mouse_flags_1 & 0xf8;
243 write_byte(ebda_seg, 0x0026, mouse_flags_1);
244 ret = send_to_mouse_ctrl(0xFF); // reset mouse command
245 if (ret == 0) {
246 ret = get_mouse_data(&mouse_data3);
247 // if no mouse attached, it will return RESEND
248 if (mouse_data3 == 0xfe) {
249 SET_CF();
250 regs.u.r8.ah = 4; // resend
251 break;
252 }
253 if (mouse_data3 != 0xfa)
254 BX_PANIC("Mouse reset returned %02x (should be ack)\n", (unsigned)mouse_data3);
255 if ( ret == 0 ) {
256 ret = get_mouse_data(&mouse_data1);
257 if ( ret == 0 ) {
258 ret = get_mouse_data(&mouse_data2);
259 if ( ret == 0 ) {
260 // success
261 regs.u.r8.bl = mouse_data1;
262 regs.u.r8.bh = mouse_data2;
263 break;
264 }
265 }
266 }
267 }
268
269 // interface error
270 SET_CF();
271 regs.u.r8.ah = 3;
272 break;
273
274 case 2: // Set Sample Rate
275 BX_DEBUG_INT15_MS("case 2:\n");
276 switch (regs.u.r8.bh) {
277 case 0: mouse_data1 = 10; break; // 10 reports/sec
278 case 1: mouse_data1 = 20; break; // 20 reports/sec
279 case 2: mouse_data1 = 40; break; // 40 reports/sec
280 case 3: mouse_data1 = 60; break; // 60 reports/sec
281 case 4: mouse_data1 = 80; break; // 80 reports/sec
282 case 5: mouse_data1 = 100; break; // 100 reports/sec (default)
283 case 6: mouse_data1 = 200; break; // 200 reports/sec
284 default: mouse_data1 = 0;
285 }
286 if (mouse_data1 > 0) {
287 ret = send_to_mouse_ctrl(0xF3); // set sample rate command
288 if (ret == 0) {
289 ret = get_mouse_data(&mouse_data2);
290 ret = send_to_mouse_ctrl(mouse_data1);
291 ret = get_mouse_data(&mouse_data2);
292 // success
293 } else {
294 // interface error
295 SET_CF();
296 regs.u.r8.ah = 3;
297 }
298 } else {
299 // invalid input
300 SET_CF();
301 regs.u.r8.ah = 2;
302 }
303 break;
304
305 case 3: // Set Resolution
306 BX_DEBUG_INT15_MS("case 3:\n");
307 // BX:
308 // 0 = 25 dpi, 1 count per millimeter
309 // 1 = 50 dpi, 2 counts per millimeter
310 // 2 = 100 dpi, 4 counts per millimeter
311 // 3 = 200 dpi, 8 counts per millimeter
312 if (regs.u.r8.bh < 4) {
313 ret = send_to_mouse_ctrl(0xE8); // set resolution command
314 if (ret == 0) {
315 ret = get_mouse_data(&mouse_data1);
316 if (mouse_data1 != 0xfa)
317 BX_PANIC("Mouse status returned %02x (should be ack)\n", (unsigned)mouse_data1);
318 ret = send_to_mouse_ctrl(regs.u.r8.bh);
319 ret = get_mouse_data(&mouse_data1);
320 if (mouse_data1 != 0xfa)
321 BX_PANIC("Mouse status returned %02x (should be ack)\n", (unsigned)mouse_data1);
322 // success
323 } else {
324 // interface error
325 SET_CF();
326 regs.u.r8.ah = 3;
327 }
328 } else {
329 // invalid input
330 SET_CF();
331 regs.u.r8.ah = 2;
332 }
333 break;
334
335 case 4: // Get Device ID
336 BX_DEBUG_INT15_MS("case 4:\n");
337 ret = send_to_mouse_ctrl(0xF2); // get mouse ID command
338 if (ret == 0) {
339 ret = get_mouse_data(&mouse_data1);
340 ret = get_mouse_data(&mouse_data2);
341 regs.u.r8.bh = mouse_data2;
342 // success
343 } else {
344 // interface error
345 SET_CF();
346 regs.u.r8.ah = 3;
347 }
348 break;
349
350 case 6: // Return Status & Set Scaling Factor...
351 BX_DEBUG_INT15_MS("case 6:\n");
352 switch (regs.u.r8.bh) {
353 case 0: // Return Status
354 ret = send_to_mouse_ctrl(0xE9); // get mouse info command
355 if (ret == 0) {
356 ret = get_mouse_data(&mouse_data1);
357 if (mouse_data1 != 0xfa)
358 BX_PANIC("Mouse status returned %02x (should be ack)\n", (unsigned)mouse_data1);
359 if (ret == 0) {
360 ret = get_mouse_data(&mouse_data1);
361 if ( ret == 0 ) {
362 ret = get_mouse_data(&mouse_data2);
363 if ( ret == 0 ) {
364 ret = get_mouse_data(&mouse_data3);
365 if ( ret == 0 ) {
366 regs.u.r8.bl = mouse_data1;
367 regs.u.r8.cl = mouse_data2;
368 regs.u.r8.dl = mouse_data3;
369 // success
370 break;
371 }
372 }
373 }
374 }
375 }
376
377 // interface error
378 SET_CF();
379 regs.u.r8.ah = 3;
380 break;
381
382 case 1: // Set Scaling Factor to 1:1
383 case 2: // Set Scaling Factor to 2:1
384 if (regs.u.r8.bh == 1) {
385 ret = send_to_mouse_ctrl(0xE6);
386 } else {
387 ret = send_to_mouse_ctrl(0xE7);
388 }
389 if (ret == 0) {
390 get_mouse_data(&mouse_data1);
391 ret = (mouse_data1 != 0xFA);
392 }
393 if (ret != 0) {
394 // interface error
395 SET_CF();
396 regs.u.r8.ah = 3;
397 }
398 break;
399
400 default:
401 BX_PANIC("INT 15h C2 AL=6, BH=%02x\n", (unsigned) regs.u.r8.bh);
402 // invalid subfunction
403 SET_CF();
404 regs.u.r8.ah = 1;
405 }
406 break;
407
408 case 7: // Set Mouse Handler Address
409 BX_DEBUG_INT15_MS("case 7:\n");
410 mouse_driver_seg = ES;
411 mouse_driver_offset = regs.u.r16.bx;
412 write_word(ebda_seg, 0x0022, mouse_driver_offset);
413 write_word(ebda_seg, 0x0024, mouse_driver_seg);
414 mouse_flags_2 = read_byte(ebda_seg, 0x0027);
415 if (mouse_driver_offset == 0 && mouse_driver_seg == 0) {
416 /* remove handler */
417 if ( (mouse_flags_2 & 0x80) != 0 ) {
418 mouse_flags_2 &= ~0x80;
419 }
420 }
421 else {
422 /* install handler */
423 mouse_flags_2 |= 0x80;
424 }
425 write_byte(ebda_seg, 0x0027, mouse_flags_2);
426 break;
427
428 default:
429 BX_PANIC("INT 15h C2 default case entered\n");
430 // invalid subfunction
431 SET_CF();
432 regs.u.r8.ah = 1;
433 }
434 BX_DEBUG_INT15_MS("returning cf = %u, ah = %02x\n", (unsigned)GET_CF(), (unsigned)regs.u.r8.ah);
435 // Re-enable AUX input and IRQ12
436 set_kbd_command_byte(0x47);
437}
438#endif // BX_USE_PS2_MOUSE
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette