MOUSE.C 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613
  1. /*
  2. THE COMPUTER CODE CONTAINED HEREIN IS THE SOLE PROPERTY OF PARALLAX
  3. SOFTWARE CORPORATION ("PARALLAX"). PARALLAX, IN DISTRIBUTING THE CODE TO
  4. END-USERS, AND SUBJECT TO ALL OF THE TERMS AND CONDITIONS HEREIN, GRANTS A
  5. ROYALTY-FREE, PERPETUAL LICENSE TO SUCH END-USERS FOR USE BY SUCH END-USERS
  6. IN USING, DISPLAYING, AND CREATING DERIVATIVE WORKS THEREOF, SO LONG AS
  7. SUCH USE, DISPLAY OR CREATION IS FOR NON-COMMERCIAL, ROYALTY OR REVENUE
  8. FREE PURPOSES. IN NO EVENT SHALL THE END-USER USE THE COMPUTER CODE
  9. CONTAINED HEREIN FOR REVENUE-BEARING PURPOSES. THE END-USER UNDERSTANDS
  10. AND AGREES TO THE TERMS HEREIN AND ACCEPTS THE SAME BY USE OF THIS FILE.
  11. COPYRIGHT 1993-1998 PARALLAX SOFTWARE CORPORATION. ALL RIGHTS RESERVED.
  12. */
  13. /*
  14. * $Source: f:/miner/source/bios/rcs/mouse.c $
  15. * $Revision: 1.11 $
  16. * $Author: john $
  17. * $Date: 1995/02/10 18:52:17 $
  18. *
  19. * Functions to access Mouse and Cyberman...
  20. *
  21. * $Log: mouse.c $
  22. * Revision 1.11 1995/02/10 18:52:17 john
  23. * Fixed bug with mouse not getting closed.
  24. *
  25. * Revision 1.10 1995/02/02 11:10:33 john
  26. * Changed a bunch of mouse stuff around to maybe get
  27. * around PS/2 mouse hang.
  28. *
  29. * Revision 1.9 1995/01/14 19:19:52 john
  30. * Fixed signed short error cmp with -1 that caused mouse
  31. * to break under Watcom 10.0
  32. *
  33. * Revision 1.8 1994/12/27 12:38:23 john
  34. * Made mouse use temporary dos buffer instead of
  35. *
  36. * allocating its own.
  37. *
  38. *
  39. * Revision 1.7 1994/12/05 23:54:53 john
  40. * Fixed bug with mouse_get_delta only returning positive numbers..
  41. *
  42. * Revision 1.6 1994/11/18 23:18:18 john
  43. * Changed some shorts to ints.
  44. *
  45. * Revision 1.5 1994/09/13 12:34:02 john
  46. * Added functions to get down count and state.
  47. *
  48. * Revision 1.4 1994/08/29 20:52:19 john
  49. * Added better cyberman support; also, joystick calibration
  50. * value return funcctiionn,
  51. *
  52. * Revision 1.3 1994/08/24 18:54:32 john
  53. * *** empty log message ***
  54. *
  55. * Revision 1.2 1994/08/24 18:53:46 john
  56. * Made Cyberman read like normal mouse; added dpmi module; moved
  57. * mouse from assembly to c. Made mouse buttons return time_down.
  58. *
  59. * Revision 1.1 1994/08/24 13:56:37 john
  60. * Initial revision
  61. *
  62. *
  63. */
  64. #pragma off (unreferenced)
  65. static char rcsid[] = "$Id: mouse.c 1.11 1995/02/10 18:52:17 john Exp $";
  66. #pragma on (unreferenced)
  67. #include <stdlib.h>
  68. #include <stdio.h>
  69. #include <conio.h>
  70. #include <dos.h>
  71. #include <i86.h>
  72. #include <string.h>
  73. #include "error.h"
  74. #include "fix.h"
  75. #include "dpmi.h"
  76. #include "mouse.h"
  77. #include "timer.h"
  78. #define ME_CURSOR_MOVED (1<<0)
  79. #define ME_LB_P (1<<1)
  80. #define ME_LB_R (1<<2)
  81. #define ME_RB_P (1<<3)
  82. #define ME_RB_R (1<<4)
  83. #define ME_MB_P (1<<5)
  84. #define ME_MB_R (1<<6)
  85. #define ME_OB_P (1<<7)
  86. #define ME_OB_R (1<<8)
  87. #define ME_X_C (1<<9)
  88. #define ME_Y_C (1<<10)
  89. #define ME_Z_C (1<<11)
  90. #define ME_P_C (1<<12)
  91. #define ME_B_C (1<<13)
  92. #define ME_H_C (1<<14)
  93. #define ME_O_C (1<<15)
  94. #define MOUSE_MAX_BUTTONS 11
  95. typedef struct event_info {
  96. short x;
  97. short y;
  98. short z;
  99. short pitch;
  100. short bank;
  101. short heading;
  102. ushort button_status;
  103. ushort device_dependant;
  104. } event_info;
  105. typedef struct mouse_info {
  106. fix ctime;
  107. ubyte cyberman;
  108. int num_buttons;
  109. ubyte pressed[MOUSE_MAX_BUTTONS];
  110. fix time_went_down[MOUSE_MAX_BUTTONS];
  111. fix time_held_down[MOUSE_MAX_BUTTONS];
  112. uint num_downs[MOUSE_MAX_BUTTONS];
  113. uint num_ups[MOUSE_MAX_BUTTONS];
  114. event_info *x_info;
  115. ushort button_status;
  116. } mouse_info;
  117. typedef struct cyberman_info {
  118. ubyte device_type;
  119. ubyte major_version;
  120. ubyte minor_version;
  121. ubyte x_descriptor;
  122. ubyte y_descriptor;
  123. ubyte z_descriptor;
  124. ubyte pitch_descriptor;
  125. ubyte roll_descriptor;
  126. ubyte yaw_descriptor;
  127. ubyte reserved;
  128. } cyberman_info;
  129. static mouse_info Mouse;
  130. static int Mouse_installed = 0;
  131. #pragma off (check_stack)
  132. void _loadds far mouse_handler (int m_ax, int mbx, int mcx, int mdx, int msi, int mdi)
  133. {
  134. #pragma aux mouse_handler parm [EAX] [EBX] [ECX] [EDX] [ESI] [EDI]
  135. Mouse.ctime = timer_get_fixed_secondsX();
  136. if (m_ax & ME_LB_P) { // left button pressed
  137. if (!Mouse.pressed[MB_LEFT]) {
  138. Mouse.pressed[MB_LEFT] = 1;
  139. Mouse.time_went_down[MB_LEFT] = Mouse.ctime;
  140. }
  141. Mouse.num_downs[MB_LEFT]++;
  142. } else if (m_ax & ME_LB_R ) { // left button released
  143. if (Mouse.pressed[MB_LEFT]) {
  144. Mouse.pressed[MB_LEFT] = 0;
  145. Mouse.time_held_down[MB_LEFT] += Mouse.ctime-Mouse.time_went_down[MB_LEFT];
  146. }
  147. Mouse.num_ups[MB_LEFT]++;
  148. }
  149. if (m_ax & ME_RB_P ) { // right button pressed
  150. if (!Mouse.pressed[MB_RIGHT]) {
  151. Mouse.pressed[MB_RIGHT] = 1;
  152. Mouse.time_went_down[MB_RIGHT] = Mouse.ctime;
  153. }
  154. Mouse.num_downs[MB_RIGHT]++;
  155. } else if (m_ax & ME_RB_R ) {// right button released
  156. if (Mouse.pressed[MB_RIGHT]) {
  157. Mouse.pressed[MB_RIGHT] = 0;
  158. Mouse.time_held_down[MB_RIGHT] += Mouse.ctime-Mouse.time_went_down[MB_RIGHT];
  159. }
  160. Mouse.num_ups[MB_RIGHT]++;
  161. }
  162. if (m_ax & ME_MB_P ) { // middle button pressed
  163. if (!Mouse.pressed[MB_MIDDLE]) {
  164. Mouse.pressed[MB_MIDDLE] = 1;
  165. Mouse.time_went_down[MB_MIDDLE] = Mouse.ctime;
  166. }
  167. Mouse.num_downs[MB_MIDDLE]++;
  168. } else if (m_ax & ME_MB_R ) { // middle button released
  169. if (Mouse.pressed[MB_MIDDLE]) {
  170. Mouse.pressed[MB_MIDDLE] = 0;
  171. Mouse.time_held_down[MB_MIDDLE] += Mouse.ctime-Mouse.time_went_down[MB_MIDDLE];
  172. }
  173. Mouse.num_ups[MB_MIDDLE]++;
  174. }
  175. if (Mouse.cyberman && (m_ax & (ME_Z_C|ME_P_C|ME_B_C|ME_H_C))) {
  176. Mouse.x_info = (event_info *)((msi & 0xFFFF) << 4);
  177. if (m_ax & ME_Z_C ) { // z axis changed
  178. if (Mouse.pressed[MB_Z_UP]) {
  179. // z up released
  180. Mouse.pressed[MB_Z_UP] = 0;
  181. Mouse.time_held_down[MB_Z_UP] += Mouse.ctime-Mouse.time_went_down[MB_Z_UP];
  182. Mouse.num_ups[MB_Z_UP]++;
  183. } else if ( Mouse.x_info->z>0 ) {
  184. // z up pressed
  185. Mouse.pressed[MB_Z_UP] = 1;
  186. Mouse.time_went_down[MB_Z_UP]=Mouse.ctime;
  187. Mouse.num_downs[MB_Z_UP]++;
  188. }
  189. if (Mouse.pressed[MB_Z_DOWN]) {
  190. // z down released
  191. Mouse.pressed[MB_Z_DOWN] = 0;
  192. Mouse.time_held_down[MB_Z_DOWN] += Mouse.ctime-Mouse.time_went_down[MB_Z_DOWN];
  193. Mouse.num_ups[MB_Z_DOWN]++;
  194. } else if ( Mouse.x_info->z<0 ) {
  195. // z down pressed
  196. Mouse.pressed[MB_Z_DOWN] = 1;
  197. Mouse.time_went_down[MB_Z_DOWN]=Mouse.ctime;
  198. Mouse.num_downs[MB_Z_DOWN]++;
  199. }
  200. }
  201. if (m_ax & ME_P_C ) { // pitch changed
  202. if (Mouse.pressed[MB_PITCH_BACKWARD]) {
  203. // pitch backward released
  204. Mouse.pressed[MB_PITCH_BACKWARD] = 0;
  205. Mouse.time_held_down[MB_PITCH_BACKWARD] += Mouse.ctime-Mouse.time_went_down[MB_PITCH_BACKWARD];
  206. Mouse.num_ups[MB_PITCH_BACKWARD]++;
  207. } else if ( Mouse.x_info->pitch>0 ) {
  208. // pitch backward pressed
  209. Mouse.pressed[MB_PITCH_BACKWARD] = 1;
  210. Mouse.time_went_down[MB_PITCH_BACKWARD]=Mouse.ctime;
  211. Mouse.num_downs[MB_PITCH_BACKWARD]++;
  212. }
  213. if (Mouse.pressed[MB_PITCH_FORWARD]) {
  214. // pitch forward released
  215. Mouse.pressed[MB_PITCH_FORWARD] = 0;
  216. Mouse.time_held_down[MB_PITCH_FORWARD] += Mouse.ctime-Mouse.time_went_down[MB_PITCH_FORWARD];
  217. Mouse.num_ups[MB_PITCH_FORWARD]++;
  218. } else if ( Mouse.x_info->pitch<0 ) {
  219. // pitch forward pressed
  220. Mouse.pressed[MB_PITCH_FORWARD] = 1;
  221. Mouse.time_went_down[MB_PITCH_FORWARD]=Mouse.ctime;
  222. Mouse.num_downs[MB_PITCH_FORWARD]++;
  223. }
  224. }
  225. if (m_ax & ME_B_C ) { // bank changed
  226. if (Mouse.pressed[MB_BANK_LEFT]) {
  227. // bank left released
  228. Mouse.pressed[MB_BANK_LEFT] = 0;
  229. Mouse.time_held_down[MB_BANK_LEFT] += Mouse.ctime-Mouse.time_went_down[MB_BANK_LEFT];
  230. Mouse.num_ups[MB_BANK_LEFT]++;
  231. } else if ( Mouse.x_info->bank>0 ) {
  232. // bank left pressed
  233. Mouse.pressed[MB_BANK_LEFT] = 1;
  234. Mouse.time_went_down[MB_BANK_LEFT]=Mouse.ctime;
  235. Mouse.num_downs[MB_BANK_LEFT]++;
  236. }
  237. if (Mouse.pressed[MB_BANK_RIGHT]) {
  238. // bank right released
  239. Mouse.pressed[MB_BANK_RIGHT] = 0;
  240. Mouse.time_held_down[MB_BANK_RIGHT] += Mouse.ctime-Mouse.time_went_down[MB_BANK_RIGHT];
  241. Mouse.num_ups[MB_BANK_RIGHT]++;
  242. } else if ( Mouse.x_info->bank<0 ) {
  243. // bank right pressed
  244. Mouse.pressed[MB_BANK_RIGHT] = 1;
  245. Mouse.time_went_down[MB_BANK_RIGHT]=Mouse.ctime;
  246. Mouse.num_downs[MB_BANK_RIGHT]++;
  247. }
  248. }
  249. if (m_ax & ME_H_C ) { // heading changed
  250. if (Mouse.pressed[MB_HEAD_LEFT]) {
  251. // head left released
  252. Mouse.pressed[MB_HEAD_LEFT] = 0;
  253. Mouse.time_held_down[MB_HEAD_LEFT] += Mouse.ctime-Mouse.time_went_down[MB_HEAD_LEFT];
  254. Mouse.num_ups[MB_HEAD_LEFT]++;
  255. } else if ( Mouse.x_info->heading>0 ) {
  256. // head left pressed
  257. Mouse.pressed[MB_HEAD_LEFT] = 1;
  258. Mouse.time_went_down[MB_HEAD_LEFT]=Mouse.ctime;
  259. Mouse.num_downs[MB_HEAD_LEFT]++;
  260. }
  261. if (Mouse.pressed[MB_HEAD_RIGHT]) {
  262. // head right released
  263. Mouse.pressed[MB_HEAD_RIGHT] = 0;
  264. Mouse.time_held_down[MB_HEAD_RIGHT] += Mouse.ctime-Mouse.time_went_down[MB_HEAD_RIGHT];
  265. Mouse.num_ups[MB_HEAD_RIGHT]++;
  266. } else if ( Mouse.x_info->heading<0 ) {
  267. // head right pressed
  268. Mouse.pressed[MB_HEAD_RIGHT] = 1;
  269. Mouse.time_went_down[MB_HEAD_RIGHT]=Mouse.ctime;
  270. Mouse.num_downs[MB_HEAD_RIGHT]++;
  271. }
  272. }
  273. }
  274. }
  275. void mouse_handler_end (void) // dummy functions
  276. {
  277. }
  278. #pragma on (check_stack)
  279. //--------------------------------------------------------
  280. // returns 0 if no mouse
  281. // else number of buttons
  282. int mouse_init(int enable_cyberman)
  283. {
  284. dpmi_real_regs rr;
  285. cyberman_info *ci;
  286. struct SREGS sregs;
  287. union REGS inregs, outregs;
  288. ubyte *Mouse_dos_mem;
  289. if (Mouse_installed)
  290. return Mouse.num_buttons;
  291. if (_dos_getvect(0x33) == NULL) {
  292. // No mouse driver loaded
  293. return 0;
  294. }
  295. // Reset the mouse driver
  296. memset( &inregs, 0, sizeof(inregs) );
  297. inregs.w.ax = 0;
  298. int386(0x33, &inregs, &outregs);
  299. if (outregs.w.ax != 0xffff)
  300. return 0;
  301. Mouse.num_buttons = outregs.w.bx;
  302. Mouse.cyberman = 0;
  303. // Enable mouse driver
  304. memset( &inregs, 0, sizeof(inregs) );
  305. inregs.w.ax = 0x0020;
  306. int386(0x33, &inregs, &outregs);
  307. if (outregs.w.ax != 0xffff )
  308. return 0;
  309. if ( enable_cyberman ) {
  310. Mouse_dos_mem = dpmi_get_temp_low_buffer( 64 );
  311. if (Mouse_dos_mem==NULL) {
  312. printf( "Unable to allocate DOS buffer in mouse.c\n" );
  313. } else {
  314. // Check for Cyberman...
  315. memset( &rr, 0, sizeof(dpmi_real_regs) );
  316. rr.es = DPMI_real_segment(Mouse_dos_mem);
  317. rr.edx = DPMI_real_offset(Mouse_dos_mem);
  318. rr.eax = 0x53c1;
  319. dpmi_real_int386x( 0x33, &rr );
  320. if (rr.eax==1) {
  321. // SWIFT functions supported
  322. ci = (cyberman_info *)Mouse_dos_mem;
  323. if (ci->device_type==1) { // Cyberman
  324. Mouse.cyberman = 1;
  325. //printf( "Cyberman mouse detected\n" );
  326. Mouse.num_buttons = 11;
  327. }
  328. }
  329. }
  330. }
  331. if (!dpmi_lock_region(&Mouse,sizeof(mouse_info))) {
  332. printf( "Unable to lock mouse data region" );
  333. exit(1);
  334. }
  335. if (!dpmi_lock_region((void near *)mouse_handler,(char *)mouse_handler_end - (char near *)mouse_handler)) {
  336. printf( "Unable to lock mouse handler" );
  337. exit(1);
  338. }
  339. // Install mouse handler
  340. memset( &inregs, 0, sizeof(inregs));
  341. memset( &sregs, 0, sizeof(sregs));
  342. inregs.w.ax = 0xC;
  343. inregs.w.cx = ME_LB_P|ME_LB_R|ME_RB_P|ME_RB_R|ME_MB_P|ME_MB_R; // watch all 3 button ups/downs
  344. if (Mouse.cyberman)
  345. inregs.w.cx |= ME_Z_C| ME_P_C| ME_B_C| ME_H_C; // if using a cyberman, also watch z, pitch, bank, heading.
  346. inregs.x.edx = FP_OFF(mouse_handler);
  347. sregs.es = FP_SEG(mouse_handler);
  348. int386x(0x33, &inregs, &outregs, &sregs);
  349. Mouse_installed = 1;
  350. atexit( mouse_close );
  351. mouse_flush();
  352. return Mouse.num_buttons;
  353. }
  354. void mouse_close()
  355. {
  356. struct SREGS sregs;
  357. union REGS inregs, outregs;
  358. if (Mouse_installed) {
  359. Mouse_installed = 0;
  360. // clear mouse handler by setting flags to 0.
  361. memset( &inregs, 0, sizeof(inregs));
  362. memset( &sregs, 0, sizeof(sregs));
  363. inregs.w.ax = 0xC;
  364. inregs.w.cx = 0; // disable event handler by setting to zero.
  365. inregs.x.edx = 0;
  366. sregs.es = 0;
  367. int386x(0x33, &inregs, &outregs, &sregs);
  368. }
  369. }
  370. void mouse_set_limits( int x1, int y1, int x2, int y2 )
  371. {
  372. union REGS inregs, outregs;
  373. if (!Mouse_installed) return;
  374. memset( &inregs, 0, sizeof(inregs));
  375. inregs.w.ax = 0x7; // Set Horizontal Limits for Pointer
  376. inregs.w.cx = x1;
  377. inregs.w.dx = x2;
  378. int386(0x33, &inregs, &outregs);
  379. memset( &inregs, 0, sizeof(inregs));
  380. inregs.w.ax = 0x8; // Set Vertical Limits for Pointer
  381. inregs.w.cx = y1;
  382. inregs.w.dx = y2;
  383. int386(0x33, &inregs, &outregs);
  384. }
  385. void mouse_get_pos( int *x, int *y)
  386. {
  387. union REGS inregs, outregs;
  388. if (!Mouse_installed) {
  389. *x = *y = 0;
  390. return;
  391. }
  392. memset( &inregs, 0, sizeof(inregs));
  393. inregs.w.ax = 0x3; // Get Mouse Position and Button Status
  394. int386(0x33, &inregs, &outregs);
  395. *x = (short)outregs.w.cx;
  396. *y = (short)outregs.w.dx;
  397. }
  398. void mouse_get_delta( int *dx, int *dy )
  399. {
  400. union REGS inregs, outregs;
  401. if (!Mouse_installed) {
  402. *dx = *dy = 0;
  403. return;
  404. }
  405. memset( &inregs, 0, sizeof(inregs));
  406. inregs.w.ax = 0xb; // Read Mouse motion counters
  407. int386(0x33, &inregs, &outregs);
  408. *dx = (short)outregs.w.cx;
  409. *dy = (short)outregs.w.dx;
  410. }
  411. int mouse_get_btns()
  412. {
  413. int i;
  414. uint flag=1;
  415. int status = 0;
  416. if (!Mouse_installed)
  417. return 0;
  418. for (i=0; i<MOUSE_MAX_BUTTONS; i++ ) {
  419. if (Mouse.pressed[i])
  420. status |= flag;
  421. flag <<= 1;
  422. }
  423. return status;
  424. }
  425. void mouse_set_pos( int x, int y)
  426. {
  427. union REGS inregs, outregs;
  428. if (!Mouse_installed)
  429. return;
  430. memset( &inregs, 0, sizeof(inregs));
  431. inregs.w.ax = 0x4; // Set Mouse Pointer Position
  432. inregs.w.cx = x;
  433. inregs.w.dx = y;
  434. int386(0x33, &inregs, &outregs);
  435. }
  436. void mouse_flush()
  437. {
  438. int i;
  439. fix CurTime;
  440. if (!Mouse_installed)
  441. return;
  442. _disable();
  443. //Clear the mouse data
  444. CurTime =timer_get_fixed_secondsX();
  445. for (i=0; i<MOUSE_MAX_BUTTONS; i++ ) {
  446. Mouse.pressed[i] = 0;
  447. Mouse.time_went_down[i] = CurTime;
  448. Mouse.time_held_down[i] = 0;
  449. Mouse.num_downs[i]=0;
  450. Mouse.num_ups[i]=0;
  451. }
  452. _enable();
  453. }
  454. // Returns how many times this button has went down since last call.
  455. int mouse_button_down_count(int button)
  456. {
  457. int count;
  458. if (!Mouse_installed)
  459. return 0;
  460. _disable();
  461. count = Mouse.num_downs[button];
  462. Mouse.num_downs[button]=0;
  463. _enable();
  464. return count;
  465. }
  466. // Returns 1 if this button is currently down
  467. int mouse_button_state(int button)
  468. {
  469. int state;
  470. if (!Mouse_installed)
  471. return 0;
  472. _disable();
  473. state = Mouse.pressed[button];
  474. _enable();
  475. return state;
  476. }
  477. // Returns how long this button has been down since last call.
  478. fix mouse_button_down_time(int button)
  479. {
  480. fix time_down, time;
  481. if (!Mouse_installed)
  482. return 0;
  483. _disable();
  484. if ( !Mouse.pressed[button] ) {
  485. time_down = Mouse.time_held_down[button];
  486. Mouse.time_held_down[button] = 0;
  487. } else {
  488. time = timer_get_fixed_secondsX();
  489. time_down = time - Mouse.time_went_down[button];
  490. Mouse.time_went_down[button] = time;
  491. }
  492. _enable();
  493. return time_down;
  494. }
  495. void mouse_get_cyberman_pos( int *x, int *y )
  496. {
  497. dpmi_real_regs rr;
  498. event_info * ei;
  499. ubyte *Mouse_dos_mem;
  500. if ( (!Mouse_installed) || (!Mouse.cyberman) ) {
  501. *x = *y = 0;
  502. return;
  503. }
  504. Mouse_dos_mem = dpmi_get_temp_low_buffer( 64 );
  505. if ( !Mouse_dos_mem ) {
  506. *x = *y = 0;
  507. return;
  508. }
  509. memset( &rr, 0, sizeof(dpmi_real_regs) );
  510. rr.es = DPMI_real_segment(Mouse_dos_mem);
  511. rr.edx = DPMI_real_offset(Mouse_dos_mem);
  512. rr.eax = 0x5301;
  513. dpmi_real_int386x( 0x33, &rr );
  514. ei = (event_info *)Mouse_dos_mem;
  515. *x = (((ei->x+8128)*256)/(8064+8128+1)) - 127;
  516. *y = (((ei->y+8128)*256)/(8064+8128+1)) - 127;
  517. }
  518.