JOYSTICK.C
上传用户:lllw2006
上传日期:2007-01-05
资源大小:39k
文件大小:20k
源码类别:

游戏

开发平台:

DOS

  1. /*
  2. ** Joystick routines.
  3. **
  4. ** This file is part of:
  5. **
  6. ** Joystick Library for Dos.
  7. ** Version: 1.1
  8. **
  9. ** Last modified: June 15th, 1997
  10. **
  11. ** (C) 1997 Simone Zanella Productions.
  12. **
  13. ** E-mail: szanella@dsi.unive.it
  14. **         zanella@prometeo.it
  15. **
  16. ** Web:    http://www.dsi.unive.it/~szanella/index.htm
  17. **         http://members.tripod.com/~szanella/index.htm
  18. **
  19. ** Changes from version 1.0:
  20. **
  21. ** - changed calibration to manual, to avoid incompatibilities and
  22. **   to get a faster reading;
  23. ** - added support for 4 and 6 buttons joysticks;
  24. ** - added parameter mode to function InstallJoystick, to set
  25. **   the number of buttons for joysticks;
  26. ** - added parameter skipjoy to function InstallJoystick, to skip
  27. **   installation/calibration of one joystick;
  28. ** - added parameter calibfile to function InstallJoystick, to
  29. **   allow calibration saving/reloading;
  30. ** - changed to integer parameters 'first' and 'second' in function
  31. **   ReadJoystick, to keep flags for fire buttons 5 and 6;
  32. ** - removed tolerances from function InstallJoystick.
  33. **
  34. ** --------------------------------------------------------
  35. **
  36. ** This is freeware: use it as you like, but ALWAYS include
  37. ** the original author's name in the docs of your program.
  38. **
  39. ** --------------------------------------------------------
  40. **
  41. ** Usage:
  42. **
  43. ** - include "joystick.h" at the beginning of your program and
  44. **   compile and link the module "joystick.c"
  45. **
  46. ** - at program startup, use:
  47. **
  48. **   unsigned char j, mode;
  49. **
  50. **   // Set mode to the kind of joystick requested:
  51. **   // JOY_2BUTTONS    standard 2-buttons joysticks
  52. **   // JOY_4BUTTONS    4-buttons joysticks (fire 3 and 4 are other joystick's
  53. **   //                 fire buttons)
  54. **   // JOY_6BUTTONS    6-buttons joystick (fire 3 & 4 as above, fire 5 and 6
  55. **   //                 connected to joystick #2 directions)
  56. **   //
  57. **   // Set calibfile to the name of a file to be used for saving/reloading
  58. **   // calibration (NULL to force re-calibration at each run).
  59. **   //
  60. **   // Set skipjoy to:
  61. **   // JOY_NONE    if all joysticks connected should be detected and calibrated;
  62. **   // JOY_SECOND  if only first joystick is used and thus should be calibrated.
  63. **
  64. **   j = InstallJoystick(mode, skipjoy, calibfile);
  65. **
  66. ** - use the value j to determine number of joysticks connected:
  67. **
  68. **   j & JOY_FIRST    first joystick is present
  69. **   j & JOY_SECOND   second joystick is present
  70. **   j & JOY_BOTH     both joysticks present (or one 6-buttons joystick)
  71. **
  72. ** - when you want to read the joystick(s), use:
  73. **
  74. **   unsigned int fj, sj;
  75. **
  76. **   ReadJoystick(&fj, &sj);
  77. **
  78. **   Then, you can determine if a direction is activated by:
  79. **
  80. **   fj & JOY_xx
  81. **
  82. **   where "xx" can be: UP, DN (down), LT (left), RT (right)
  83. **
  84. **   You can test the fire buttons in a similar way:
  85. **
  86. **   fj & JOY_AF    first button
  87. **   fj & JOY_BF    second button
  88. **   fj & JOY_CF    third button (only when mode = JOY_4BUTTONS)
  89. **   fj & JOY_DF    fourth button (only when mode = JOY_4BUTTONS)
  90. **   fj & JOY_EF    fifth button (only joystick 1, when mode = JOY_6BUTTONS)
  91. **   fj & JOY_FF    sixth button (only joystick 1, when mode = JOY_6BUTTONS)
  92. **
  93. **   Of course, replace "fj" above with "sj" to test second joystick.
  94. **
  95. ** If you want to poll joystick at regular intervals, you can chain the
  96. ** system timer (int 8).
  97. */
  98. #include "joystick.h"
  99. #include <stdlib.h>
  100. #include <dos.h>
  101. #include <limits.h>
  102. #include <conio.h>
  103. #include <io.h>
  104. #include <fcntl.h>
  105. #include <stat.h>
  106. #include <stdio.h>
  107. /*
  108. #ifndef __BORLANDC__
  109. #include <pc.h>
  110. #endif
  111. */
  112. #define JOYPORT              0x201
  113. /* Number of divisions, for determining thresholds */
  114. #define SEGMENTS             2
  115. /* Tolerance, to determine direction on 6-buttons joysticks */
  116. #define JOY_TOL              100
  117.  /* Constant for timed joy reading */
  118. #define JOYCOUNTER           60000
  119. /* Pseudo-functions */
  120. #define GetBit(c, n)         ( c & ( 1 << (n) ) )
  121. /* Static variables used for storing:
  122. ** - x and y coordinates;
  123. ** - threshold values;
  124. ** - status (ready/not ready);
  125. ** - mode (2, 4 or 6 buttons);
  126. ** - flags.
  127. */
  128. static int x[2], y[2], calib[2][4];
  129. static unsigned char jready[2], joymode = JOY_2BUTTONS;
  130. static unsigned int jstatus[2];
  131. /* Maps for fire buttons (internal) */
  132. static unsigned char fire1[2] = {16, 64};
  133. static unsigned char fire2[2] = {32, 128};
  134. /****************************************************************************/
  135. /************** Internal functions: should NOT be called by user ************/
  136. /****************************************************************************/
  137. static unsigned char _readjport(void)
  138. {
  139.   /*
  140.   ** Read joystick port, setting x and y coordinates for each joystick;
  141.   ** the status of the 4 buttons is returned.
  142.   */
  143.   register long count;
  144.   register unsigned char st, mask;
  145.   mask = 0;
  146.   if (jready[0])
  147.     mask |= 3;
  148.   if (jready[1])
  149.     mask |= 12;
  150.   count = x[0] = y[0] = x[1] = y[1] = 0;
  151.   disable();
  152.   /* Read port for JOYCOUNTER cycles */
  153.   /* Don't count previous reading (VERY FAST machines) */
  154.   do st = inportb(JOYPORT);
  155.   while (st & mask);
  156.   outportb(JOYPORT, 0xFF);
  157.   do
  158.   {
  159.     st = inportb(JOYPORT);
  160.     if (st & 1)
  161.       x[0]++;
  162.     if (st & 2)
  163.       y[0]++;
  164.     if (st & 4)
  165.       x[1]++;
  166.     if (st & 8)
  167.       y[1]++;
  168.     count++;
  169.   } while ((st & mask) && count <= JOYCOUNTER);
  170.   enable();
  171.   return st;
  172. }
  173. static unsigned char _readjbutton(const unsigned char j)
  174. {
  175.   /*
  176.   ** Read joystick, returning 0 if no fire button is pressed.
  177.   */
  178.   register unsigned char st, mask, pressed;
  179.   if (j)
  180.     mask = 12;
  181.   else
  182.     mask = 3;
  183.   outportb(JOYPORT, 0xFF);
  184.   st = inportb(JOYPORT);
  185.   pressed = (j) ? (!(st & 64) || !(st & 128)) : (!(st & 16) || !(st & 32));
  186.   while (st & mask)
  187.     st = inportb(JOYPORT);
  188.   return pressed;
  189. }
  190. static unsigned char _readjpos(const unsigned char j)
  191. {
  192.   /*
  193.   ** Wait for the user to press a joystick button, then read joystick
  194.   ** port; return 0 if user aborted calibration.
  195.   */
  196.   delay(1000);
  197.   /* Wait for button pressed */
  198.   while (!_readjbutton(j))
  199.     if (kbhit())
  200.       if (getch() == 0x1B)
  201.         return 0;
  202.   _readjport();
  203.   /* Wait for button released */
  204.   while (_readjbutton(j));
  205.   return 1;
  206. }
  207. static unsigned int _manualdir(const unsigned char j)
  208. {
  209.   /*
  210.   ** Direction check for manual calibration.
  211.   */
  212.   register unsigned int jstat = 0;
  213.   if (x[j] < calib[j][0])
  214.     jstat |= JOY_LT; /* Left */
  215.   else
  216.     jstat |= (x[j] > calib[j][1]) ? JOY_RT : 0; /* Right or center */
  217.   if (y[j] < calib[j][2])
  218.     jstat |= JOY_UP; /* Up */
  219.   else
  220.     jstat |= (y[j] > calib[j][3]) ? JOY_DN : 0; /* Down or center */
  221.   return jstat;
  222. }
  223. unsigned char _usercalib(const unsigned char j)
  224. {
  225.   /*
  226.   ** Print out messages for performing manual calibration for joystick j;
  227.   ** set calibration values and return 1 if successfull, 0 if user aborted.
  228.   */
  229.   unsigned char i, l;
  230.   int centx[2], centy[2], minx[2], maxx[2], miny[2], maxy[2];
  231.   int ocentx, ocenty;
  232.   char *msg[]={"Center stick and press a button (Esc abort)...",
  233.                "Move stick to upper left and press a button (Esc abort)...",
  234.                "Move stick to lower right and press a button (Esc abort)...",
  235.                "While pressing button 5, press button 1 or 2 (Esc abort)...",
  236.                "While pressing button 6, press button 1 or 2 (Esc abort)..."};
  237.   printf("Calibration of joystick %d..n", j + 1);
  238.   for (i = 0; i < 3; i++)
  239.   {
  240.     printf(msg[i]);
  241. #ifndef __BORLANDC__
  242.     fflush(stdout);
  243. #endif
  244.     l = _readjpos(j);
  245.     printf("n");
  246.     if (!l)
  247.       return 0;
  248.     /* Insert values for center, maximum, minimum variables */
  249.     switch (i)
  250.     {
  251.       case 0:
  252.         centx[j] = x[j];
  253.         centy[j] = y[j];
  254.         ocentx = x[1];
  255.         ocenty = y[1];
  256.         break;
  257.       case 1:
  258.         minx[j] = x[j];
  259.         miny[j] = y[j];
  260.         break;
  261.       default:
  262.         maxx[j] = x[j];
  263.         maxy[j] = y[j];
  264.         break;
  265.     }
  266.   }
  267.   if (joymode == JOY_6BUTTONS)
  268.   {
  269.     unsigned int jstat;
  270.     for (i = 0; i < 2; i++)
  271.     {
  272.       /* We must calibrate pseudo-directions corresponding to buttons 5 and 6 */
  273.       printf(msg[3 + i]);
  274. #ifndef __BORLANDC__
  275.       fflush(stdout);
  276. #endif
  277.       delay(1000);
  278.   retry:
  279.       /* Wait for button pressed */
  280.       while (!_readjbutton(j))
  281.         if (kbhit())
  282.           if (getch() == 0x1B)
  283.             return 0;
  284.       _readjport();
  285.       jstat = 0;
  286.       if (x[1] < ocentx - JOY_TOL)
  287.         jstat |= JOY_LT; /* Left */
  288.       else
  289.         jstat |= (x[1] > ocentx + JOY_TOL) ? JOY_RT : 0; /* Right or center */
  290.       if (y[1] < ocenty - JOY_TOL)
  291.         jstat |= JOY_UP; /* Up */
  292.       else
  293.         jstat |= (y[1] > ocenty + JOY_TOL) ? JOY_DN : 0; /* Down or center */
  294.       /* Wait for button released */
  295.       while (_readjbutton(j));
  296.       if (jstat & (i ? (JOY_LT | JOY_RT) : (JOY_UP | JOY_DN)))
  297.         if (i) /* Button 6: can be LEFT or RIGHT */
  298.           if (x[1] > ocentx) /* Button corresponds to RIGHT */
  299.           {
  300.             calib[1][1] = ocentx + ((x[1] - ocentx) / SEGMENTS);
  301.             calib[1][0] = INT_MIN;
  302.           }
  303.           else /* Button corresponds to LEFT */
  304.           {
  305.             calib[1][1] = INT_MAX;
  306.             calib[1][0] = ocentx - ((ocentx - x[1]) / SEGMENTS);
  307.           }
  308.         else /* Button 5: can be UP or DOWN */
  309.           if (y[1] > ocenty) /* Button corresponds to DOWN */
  310.           {
  311.             calib[1][3] = ocenty + ((y[1] - ocenty) / SEGMENTS);
  312.             calib[1][2] = INT_MIN;
  313.           }
  314.           else /* Button corresponds to UP */
  315.           {
  316.             calib[1][3] = INT_MAX;
  317.             calib[1][2] = ocenty - ((ocenty - y[1]) / SEGMENTS);
  318.           }
  319.       else
  320.         goto retry;
  321.       printf("n");
  322.     }
  323.   }
  324.   printf("n");
  325.   /* line BEFORE WHICH left is stuffed */
  326.   calib[j][0] = centx[j] - ((centx[j] - minx[j]) / SEGMENTS);
  327.   /* line AFTER WHICH right is stuffed */
  328.   calib[j][1] = centx[j] + ((maxx[j] - centx[j]) / SEGMENTS);
  329.   /* line BEFORE WHICH up is stuffed */
  330.   calib[j][2] = centy[j] - ((centy[j] - miny[j]) / SEGMENTS);
  331.   /* line AFTER WHICH down is stuffed */
  332.   calib[j][3] = centy[j] + ((maxy[j] - centy[j]) / SEGMENTS);
  333.   return 1;
  334. }
  335. static void _readjoy(void)
  336. {
  337.   /* Set flags for each joystick (directions and fire buttons) */
  338.   register int  v;
  339.   register unsigned char st, j;
  340.   st = _readjport();
  341.   for (j = 0; j < 2; j++)
  342.     if (jready[j])
  343.     {
  344.       jstatus[j] = 0;
  345.       if (!(st & fire1[j]))
  346.         jstatus[j] |= JOY_AF;
  347.       if (!(st & fire2[j]))
  348.         jstatus[j] |= JOY_BF;
  349.       jstatus[j] |= _manualdir(j);
  350.       if (joymode) /* 4 buttons or 6 buttons */
  351.       {
  352.         v = j ? 0 : 1;
  353.         if (!(st & fire1[v]))
  354.           jstatus[j] |= JOY_CF;
  355.         if (!(st & fire2[v]))
  356.           jstatus[j] |= JOY_DF;
  357.         /*
  358.         ** Buttons 5 and 6 are connected to directions on second joystick.
  359.         ** Only one 6-button joystick can be connected, so we use jstatus[0]
  360.         ** instead of jstatus[j].
  361.         */
  362.         if (joymode == JOY_6BUTTONS)
  363.         {
  364.           v = _manualdir(1);
  365.           if (v & (JOY_UP | JOY_DN))
  366.             jstatus[0] |= JOY_EF;
  367.           if (v & (JOY_LT | JOY_RT))
  368.             jstatus[0] |= JOY_FF;
  369.           j = 2;
  370.         }
  371.       }
  372.     }
  373. }
  374. static unsigned char _joyready(const unsigned char j)
  375. {
  376.   /*
  377.   ** Probe joystick, returning 1 if it is present, 0 otherwise.
  378.   ** Presence is determined in this way:
  379.   ** - if position (x and y) and both fire buttons read through bios are 0,
  380.   **   joystick is absent;
  381.   ** - otherwise, it is present.
  382.   **
  383.   ** If 'j' is 0, probes the first joystick; if 'j' is 1, probes the second.
  384.   */
  385.   union REGS regs;
  386.   register int x, y, fa, fb;
  387.   /* Read position */
  388.   regs.h.ah = 0x84;
  389.   regs.x.dx = 1;
  390.   int86(0x15, &regs, &regs);
  391.   if (regs.x.cflag)
  392.     return 0; /* Error: joystick not found */
  393.   if (j)
  394.   {
  395.     x = regs.x.cx;
  396.     y = regs.x.dx;
  397.   }
  398.   else
  399.   {
  400.     x = regs.x.ax;
  401.     y = regs.x.bx;
  402.   }
  403.   /* Read fire buttons */
  404.   regs.h.ah = 0x84;
  405.   regs.x.dx = 0;
  406.   int86(0x15, &regs, &regs);
  407.   if (regs.x.cflag)
  408.     return 0; /* Error: joystick not found */
  409.   if (j)
  410.   {
  411.     fa = !GetBit(regs.x.ax, 6);
  412.     fb = !GetBit(regs.x.ax, 7);
  413.   }
  414.   else
  415.   {
  416.     fa = !GetBit(regs.x.ax, 4);
  417.     fb = !GetBit(regs.x.ax, 5);
  418.   }
  419.   if (x + y + fa + fb == 0)
  420.     return 0;
  421.   else
  422.     return 1;
  423. }
  424. static void _savecalib(const char *filename, const unsigned char njoy)
  425. {
  426.   /*
  427.   ** Store number of joysticks connected, joystick mode and
  428.   ** calibration informations into filename (which is created
  429.   ** or overwritten). No errors are reported; if filename is NULL,
  430.   ** calibration is not saved.
  431.   */
  432.   if (filename != NULL)
  433.   {
  434.     register int i, l;
  435.     int h;
  436.     struct calibrecord {
  437.       unsigned char njoy;
  438.       unsigned char joymode;
  439.       int calib[2][4];
  440.     } cr;
  441.     cr.njoy = njoy;
  442.     cr.joymode = joymode;
  443.     for (i = 0; i < 2; i++)
  444.       for (l = 0; l < 4; l++)
  445.         cr.calib[i][l] = calib[i][l];
  446.     h = open(filename, O_WRONLY | O_BINARY | O_CREAT | O_TRUNC, S_IWRITE);
  447.     if (h != -1)
  448.     {
  449.       write(h, (void *) &cr, sizeof(cr));
  450.       close(h);
  451.     }
  452.   }
  453. }
  454. static unsigned char _checkcalib(const unsigned char j)
  455. {
  456.   /*
  457.   ** Check if joystick j is centered, according to current calibration;
  458.   ** returns JOY_FIRST or JOY_SECOND (according to j) if it needs to
  459.   ** be re-calibrated, JOY_NONE otherwise.
  460.   ** This routine does two checks, at one-second interval each:
  461.   ** if both return joystick off-centered, the program assumes that
  462.   ** calibration is needed.
  463.   */
  464.   int try = 0;
  465.   unsigned int jstat;
  466. retry:
  467.   _readjport();
  468.   jstat = _manualdir(j);
  469.   if (jstat)
  470.     if (try)
  471.       return j + 1;
  472.     else
  473.     {
  474.       delay(1000);
  475.       goto retry;
  476.     }
  477.   return JOY_NONE;
  478. }
  479. static unsigned char _loadcalib(const char *filename, const unsigned char njoy)
  480. {
  481.   /*
  482.   ** Load calibration and returns:
  483.   **
  484.   ** JOY_NONE       if mode and joy # are correct and joys appear centered
  485.   ** JOY_FIRST      if joy #1 must be re-calibrated
  486.   ** JOY_SECOND     if joy #2 must be re-calibrated
  487.   ** JOY_BOTH       if both joys need to be re-calibrated
  488.   */
  489.   if (filename != NULL)
  490.   {
  491.     int h, i, l;
  492.     unsigned char oldnjoy;
  493.     struct calibrecord {
  494.       unsigned char njoy;
  495.       unsigned char joymode;
  496.       int calib[2][4];
  497.     } cr;
  498.     h = open(filename, O_RDONLY | O_BINARY, S_IREAD);
  499.     /* File not found: re-calibration */
  500.     if (h == -1)
  501.       return njoy;
  502.     /* Wrong size: remove file and force re-calibration */
  503.     if (filelength(h) != sizeof(cr))
  504.     {
  505.       close(h);
  506.       remove(filename);
  507.       return njoy;
  508.     }
  509.     l = read(h, (void *) &cr, sizeof(cr)) != sizeof(cr);
  510.     close(h);
  511.     /* Error reading: re-calibration */
  512.     if (l)
  513.       return njoy;
  514.     /* Mode incompatibility: re-calibration */
  515.     if ((joymode == JOY_6BUTTONS && cr.joymode != JOY_6BUTTONS) ||
  516.         (cr.joymode == JOY_6BUTTONS && joymode != JOY_6BUTTONS))
  517.       return njoy;
  518.     /* Different number of joysticks: re-calibration for only the new joystick */
  519.     oldnjoy = njoy;
  520.     for (i = 0; i < 2; i++)
  521.       for (l = 0; l < 4; l++)
  522.         calib[i][l] = cr.calib[i][l];
  523.     if (oldnjoy != cr.njoy)
  524.     {
  525.       unsigned char t = 0;
  526.       for (i = 1; i < 3; i++)
  527.         if ((oldnjoy & i) && (cr.njoy & i))
  528.           t |= _checkcalib(i - 1);
  529.         else
  530.           if (oldnjoy & i)
  531.             t |= i;
  532.       return t;
  533.     }
  534.     return JOY_NONE;
  535.   }
  536.   else
  537.     return njoy;   /* NULL file specified: re-calibration */
  538. }
  539. /****************************************************************************/
  540. /******************* External functions: joystick interface *****************/
  541. /****************************************************************************/
  542. unsigned char InstallJoystick(const unsigned char mode, const unsigned char skipjoy, const char *calibfile)
  543. {
  544.   /*
  545.   ** This function checks and returns the number of installed joysticks.
  546.   **
  547.   ** 'mode' can be:
  548.   **
  549.   ** JOY_2BUTTONS   2-buttons joysticks (one or two)
  550.   ** JOY_4BUTTONS   4-buttons joysticks (one or two)
  551.   ** JOY_6BUTTONS   6-buttons joystick (only one)
  552.   **
  553.   ** 'skipjoy' can be:
  554.   **
  555.   ** JOY_NONE    all joysticks present are installed and calibrated
  556.   ** JOY_FIRST   skip installation/calibration for first joystick
  557.   ** JOY_SECOND  skip installation/calibration for second joystick
  558.   **
  559.   ** Most of the times, skipjoy will be JOY_NONE (all joysticks installed
  560.   ** and calibrated) or JOY_SECOND (only first joystick is installed and
  561.   ** calibrated, e.g. when your program uses just one joystick).
  562.   **
  563.   ** 'calibfile' is the name of the file from which calibration must be
  564.   ** loaded (if exists) or to which it will be saved (if does NOT exist).
  565.   ** If 'calibfile' is NULL, calibration won't be saved, i.e. the user will
  566.   ** have to calibrate joystick(s) at each run.
  567.   ** If 'mode' is JOY_2BUTTONS or JOY_4BUTTONS and mode determined from
  568.   ** configuration file is JOY_6BUTTONS (or viceversa), calibration will be
  569.   ** forced; the same will happen if number of joysticks connected is
  570.   ** different or if the calibration appears incorrect.
  571.   ** If calibration is aborted for one joystick, that joystick will be
  572.   ** reported as not connected (it will be re-calibrated the next time you
  573.   ** run the program).
  574.   **
  575.   ** Values returned
  576.   ** JOY_NONE    if no joystick is installed
  577.   ** JOY_FIRST   if only joystick 1 is installed
  578.   ** JOY_SECOND  if only joystick 2 is installed
  579.   ** JOY_BOTH    if both joysticks (or one 6-buttons joystick) are installed
  580.   **
  581.   ** Central values are set at current position.
  582.   */
  583.   register unsigned char retval = 0, j;
  584.   unsigned char j2calib;
  585.   joymode = mode;
  586.   /* Determine joysticks attached */
  587.   for (j = 0; j < 2; j++)
  588.     if ((jready[j] = _joyready(j)) != 0)
  589.       retval += j + 1;
  590.   /* Revert to 4-buttons mode if 6-buttons joystick not found */
  591.   if (joymode == JOY_6BUTTONS && retval < JOY_BOTH)
  592.     joymode = JOY_4BUTTONS;
  593.   /* Prompt the user to calibrate joystick(s) */
  594.   j2calib = _loadcalib(calibfile, retval);
  595.   for (j = 0; j < 2; j++)
  596.   {
  597.     if ((skipjoy & (j + 1)) || (jready[j] && (j2calib & (j + 1)) && !_usercalib(j)))
  598.     {
  599.       jready[j] = 0;
  600.       retval -= j + 1;
  601.     }
  602.     /* No need to calibrate second (virtual) joystick when in 6-buttons mode */
  603.     if (joymode == JOY_6BUTTONS)
  604.       j = 2;
  605.   }
  606.   if (retval)
  607.     _savecalib(calibfile, retval);
  608.   return retval;
  609. }
  610. void ReadJoystick(unsigned int *first, unsigned int *second)
  611. {
  612.   /*
  613.   ** Read both joysticks, returning into "first" and "second" an
  614.   ** unsigned integer with the following flags set:
  615.   **
  616.   ** JOY_UP    Joystick moved up
  617.   ** JOY_DN    Joystick moved down
  618.   ** JOY_LT    Joystick moved to left
  619.   ** JOY_RT    Joystick moved to right
  620.   ** JOY_AF    Button 1 pressed
  621.   ** JOY_BF    Button 2 pressed
  622.   ** JOY_CF    Button 3 pressed (joymode = JOY_4BUTTONS or JOY_6BUTTONS)
  623.   ** JOY_DF    Button 4 pressed (joymode = JOY_4BUTTONS or JOY_6BUTTONS)
  624.   ** JOY_EF    Button 5 pressed (joymode = JOY_6BUTTONS)
  625.   ** JOY_FF    Button 6 pressed (joymode = JOY_6BUTTONS)
  626.   **
  627.   ** If a NULL pointer is passed, no data is stored (useful if data
  628.   ** for only one joystick are needed).
  629.   */
  630.   _readjoy();
  631.   if (first)
  632.     *first = jstatus[0];
  633.   if (second)
  634.     *second = jstatus[1];
  635. }