home *** CD-ROM | disk | FTP | other *** search
/ OS/2 Shareware BBS: 5 Edit / 05-Edit.zip / e20313sr.zip / emacs / 20.3.1 / src / emxdep.c < prev    next >
C/C++ Source or Header  |  1999-07-31  |  57KB  |  1,988 lines

  1. /* emxdep.c, emx-specific bits of GNU Emacs.
  2.    Copyright (C) 1993-1996 Eberhard Mattes.
  3.  
  4. This file is part of GNU Emacs.
  5.  
  6. GNU Emacs is free software; you can redistribute it and/or modify
  7. it under the terms of the GNU General Public License as published by
  8. the Free Software Foundation; either version 2, or (at your option)
  9. any later version.
  10.  
  11. GNU Emacs is distributed in the hope that it will be useful,
  12. but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  14. GNU General Public License for more details.
  15.  
  16. You should have received a copy of the GNU General Public License
  17. along with GNU Emacs; see the file COPYING.  If not, write to
  18. the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
  19. Boston, MA 02111-1307, USA.  */
  20.  
  21.  
  22. #include <stdio.h>
  23. #include <stdlib.h>
  24. #include <string.h>
  25. #include <fcntl.h>
  26. #include <pwd.h>
  27. #include <termios.h>
  28. #include <errno.h>
  29. #include <sys/types.h>
  30. #include <sys/process.h>
  31. #include <sys/ea.h>
  32. #include <sys/ead.h>
  33. #include <sys/nls.h>
  34.  
  35. #define INCL_DEVICES
  36. #define INCL_DOSPROCESS
  37. #define INCL_DOSSESMGR
  38. #define INCL_DOSMODULEMGR
  39. #define INCL_DOSERRORS
  40. #define INCL_WINSWITCHLIST
  41. #define INCL_KBD
  42. #define INCL_VIO
  43. #define UCHAR OS2UCHAR
  44. #include <os2.h>
  45. #undef UCHAR
  46.  
  47. #include "config.h"
  48. #include "lisp.h"
  49. #include "commands.h"
  50.  
  51. /* Priority class and level.  */
  52. struct priority
  53. {
  54.   ULONG pclass;
  55.   LONG plevel;
  56. };
  57.  
  58. extern Lisp_Object Qicon;
  59.  
  60. Lisp_Object Qascii;
  61. Lisp_Object Qbinary;
  62. Lisp_Object Qbinary_process_input;
  63. Lisp_Object Qbitmap;
  64. Lisp_Object Qmetafile;
  65. Lisp_Object Qea;
  66. Lisp_Object Qmvmt;
  67. Lisp_Object Qmvst;
  68. Lisp_Object Qasn1;
  69. Lisp_Object Qemx_case_map;
  70. Lisp_Object Qpreserve;
  71.  
  72. /* Alist of elements (REGEXP . HANDLER) for program file names for
  73.    which HANDLER is called by call-process and start-process.  */
  74. static Lisp_Object Vprogram_name_handler_alist;
  75.  
  76. /* This variable describes handlers that have "already" had a chance
  77.    to handle the current operation.  */
  78. static Lisp_Object Vinhibit_program_name_handlers;
  79.  
  80. /* List of regular expressions identifying file names for which binary
  81.    mode is to be used.  */
  82. Lisp_Object Vemx_binary_mode_list;
  83.  
  84. /* The system emx is running on: `ms-dos' or `os2'.  */
  85. Lisp_Object Vemx_system_type;
  86.  
  87. /* The priority to be used for child processes.  */
  88. Lisp_Object Vprocess_priority;
  89.  
  90. /* Use binary mode for input pipe to processes and output pipe from
  91.    processes, respectively.  */
  92. int binary_process_input;
  93. int binary_process_output;
  94.  
  95. /* Use binary mode for input pipe to asynchronous processes.  */
  96. int binary_async_process_input;
  97.  
  98. /* Prevent `expand-file-name' from prepending a drive letter.  */
  99. static int remote_expand_file_name;
  100.  
  101. /* The current code page.  */
  102. int cur_code_page;
  103.  
  104. /* Translate file names to lower case if this variable is non-nil.
  105.    Let `expand-file-name' preserve letter case in file names by using
  106.    the `emx-case-map' text property if this variable is `preserve'. */
  107. static Lisp_Object Vcase_fold_file_names;
  108.  
  109. /* Start child processes as sessions (vs. processes). */
  110. static int emx_start_session;
  111.  
  112. /* Send signals to process tree. */
  113. static int emx_kill_tree;
  114.  
  115. /* Process priority class and level to be used for child process, set
  116.    by check_process_priority.  */
  117. static struct priority child_priority;
  118.  
  119. /* Process priority class and level of the main thread of Emacs.  */
  120. static struct priority emacs_priority;
  121.  
  122. /* The following variables are used for accessing text-mode files as
  123.    if they were binary files.  fb_fd is the handle of the current
  124.    `faked binary file' (there can be at most one such file at a time).
  125.    If fb_bd is -1, there is no `faked binary file'.  fb_buf holds the
  126.    complete contents of the file, with cr/lf translated to newline.
  127.    fb_size is the number of translated bytes.  fb_pos is the current
  128.    value of the file position indicator.  */
  129. static int fb_fd;
  130. static char *fb_buf;
  131. static int fb_size;
  132. static int fb_pos;
  133.  
  134. extern Lisp_Object Vprocess_environment;
  135. #ifdef HAVE_PM
  136. extern int pm_pid;
  137. extern int pm_session_started;
  138. extern Lisp_Object pm_list_code_pages ();
  139. #endif /* HAVE_PM */
  140.  
  141.  
  142. int sigblock (int mask)
  143. {
  144.   return 0;
  145. }
  146.  
  147. int sigsetmask (int mask)
  148. {
  149.   return 0;
  150. }
  151.  
  152. int nice (int incr)
  153. {
  154.   return 0;
  155. }
  156.  
  157. int setuid (int id)
  158. {
  159.   return 0;
  160. }
  161.  
  162. #define EMX_UID 1
  163.  
  164. int getuid (void)
  165. {
  166.   return EMX_UID;
  167. }
  168.  
  169. int geteuid (void)
  170. {
  171.   return EMX_UID;
  172. }
  173.  
  174. /* The pw_uid member is not used anywhere in Emacs 19.28.  Avoid,
  175.    however, surprises in future versions. */
  176.  
  177. struct passwd *emxdep_getpwuid (uid_t uid)
  178. {
  179.   struct passwd *p;
  180.  
  181.   p = (getpwuid)(uid);
  182.   if (p)
  183.     p->pw_uid = EMX_UID;
  184.   return p;
  185. }
  186.  
  187. struct passwd *emxdep_getpwnam (const char *name)
  188. {
  189.   struct passwd *p;
  190.  
  191.   p = (getpwnam)(name);
  192.   if (p)
  193.     p->pw_uid = EMX_UID;
  194.   return p;
  195. }
  196.  
  197. int link (char *name1, char *name2)
  198. {
  199.   return (rename (name1, name2));
  200. }
  201.  
  202. int setpgrp (int pid, int pgrp)
  203. {
  204.   return 0;
  205. }
  206.  
  207. int gethostname (char *name, int namelen)
  208. {
  209.   char *sp = getenv ("SYSTEMNAME");
  210.   if (!sp)
  211.     sp = "standalone";
  212.   _strncpy (name, sp, namelen);
  213.   return 0;
  214. }
  215.  
  216. int vfork (void)
  217. {
  218.   return 0;                     /* We're the child process! */
  219. }
  220.  
  221.  
  222. int emx_killpg (gid, signo)
  223. {
  224.   return kill (emx_kill_tree ? gid : -gid, signo);
  225. }
  226.  
  227.  
  228. /* The following functions are used to be able to read a text-mode
  229.    file as if it were a binary file, that is, lseek() works and the
  230.    st_size field of struct stat is exact.  This is done by reading the
  231.    entire file into memory, translating cr/lf to the newline
  232.    character.  Only one file at a time can be a `faked binary file'.  */
  233.  
  234. /* Prepare for reading from FD as if it were the handle of a binary
  235.    file.  SIZE is a pointer to the st_size field of struct stat for
  236.    the file.  This function assumes that FD is opened in text mode.
  237.    Return 0 on success, -1 on error.  */
  238.  
  239. int fb_start (int fd, off_t *size)
  240. {
  241.   if (fb_fd != -1)
  242.     abort ();
  243.   fb_fd = fd;
  244.   fb_pos = 0;
  245.   fb_buf = (char *) xmalloc (*size);
  246.   fb_size = read (fd, fb_buf, *size);
  247.   if (fb_size < 0)
  248.     {
  249.       xfree (fb_buf);
  250.       fb_fd = -1;
  251.       return -1;
  252.     }
  253.   *size = fb_size;
  254.   return 0;
  255. }
  256.  
  257.  
  258. /* Replace read() for `faked binary files'.  If FD is not the handle
  259.    passed to fb_start(), the original read() function is called.  */
  260.  
  261. int fb_read (int fd, void *buf, int nbyte)
  262. {
  263.   if (fd != fb_fd)
  264.     return read (fd, buf, nbyte);
  265.   if (nbyte > fb_size - fb_pos)
  266.     nbyte = fb_size - fb_pos;
  267.   bcopy (fb_buf + fb_pos, buf, nbyte);
  268.   fb_pos += nbyte;
  269.   return nbyte;
  270. }
  271.  
  272.  
  273. /* Replace lseek() for `faked binary files'.  If FD is not the handle
  274.    passed to fb_start(), the original lseek() function is called.  */
  275.  
  276. long fb_lseek (int fd, long offset, int origin)
  277. {
  278.   if (fd != fb_fd)
  279.     return lseek (fd, offset, origin);
  280.   if (origin != SEEK_SET)
  281.     abort ();
  282.   if (offset < 0 || offset > fb_size)
  283.     return (-1);
  284.   fb_pos = offset;
  285.   return fb_pos;
  286. }
  287.  
  288.  
  289. /* Replace close() for `faked binary files'.  If FD is not the handle
  290.    passed to fb_start(), the original close() function is called.
  291.    This function releases the memory allocated by fb_start() and
  292.    removes the special meaning of FD for the functions above.  */
  293.  
  294. int fb_close (int fd)
  295. {
  296.   if (fd == fb_fd)
  297.     {
  298.       xfree (fb_buf);
  299.       fb_fd = -1;
  300.     }
  301.   return close (fd);
  302. }
  303.  
  304.  
  305. static Lisp_Object emx_build_case_fold_string (const unsigned char *p)
  306. {
  307.   unsigned char *down;
  308.   size_t i, j, len;
  309.   Lisp_Object obj, start, end;
  310.  
  311.   len = strlen (p);
  312.  
  313.   if (NILP (Vcase_fold_file_names))
  314.       /* make_string does not take _const_ unsigned char *   JMB */
  315.       return make_string ((unsigned char *)p, len);
  316.  
  317.   down = alloca (len + 1);
  318.   memcpy (down, p, len + 1);
  319.   _nls_strlwr (down);
  320.   obj = make_string (down, len);
  321.   if (!EQ (Vcase_fold_file_names, Qpreserve))
  322.     return obj;
  323.   i = 0;
  324.   if (_fngetdrive (p) != 0)
  325.     i += 2;
  326.   while (i < len)
  327.     if (p[i] == down[i])
  328.       ++i;
  329.     else
  330.       {
  331.         j = i++;
  332.         while (i < len && p[i] != down[i])
  333.           ++i;
  334.         XSETFASTINT (start, j);
  335.         XSETFASTINT (end, i);
  336.         Fput_text_property (start, end, Qemx_case_map, Qt, obj);
  337.       }
  338.   return obj;
  339. }
  340.  
  341.  
  342. Lisp_Object emx_get_original_case_string (Lisp_Object obj)
  343. {
  344.   Lisp_Object start, end, next;
  345.   size_t i, e, len;
  346.   unsigned char *str;
  347.  
  348.   if (!EQ (Vcase_fold_file_names, Qpreserve))
  349.     return obj;
  350.   len = XSTRING (obj)->size;
  351.   XSETFASTINT (start, 0);
  352.   XSETFASTINT (end, len);
  353.   start = Ftext_property_not_all (start, end, Qemx_case_map, Qnil, obj);
  354.   if (NILP (start))
  355.     return obj;
  356.   str = alloca (len + 1);
  357.   memcpy (str, XSTRING (obj)->data, len + 1);
  358.   do
  359.     {
  360.       next = Fnext_single_property_change (start, Qemx_case_map, obj, Qnil);
  361.       if (NILP (next))
  362.         XSETFASTINT (next, len);
  363.       e = XINT (next);
  364.       for (i = XINT (start); i < e; ++i)
  365.         str[i] = _nls_toupper (str[i]);
  366.       start = Fnext_single_property_change (next, Qemx_case_map, obj, Qnil);
  367.     } while (!NILP (start));
  368.   return make_string (str, len);
  369. }
  370.  
  371.  
  372. DEFUN ("emx-original-file-name", Femx_original_file_name, Semx_original_file_name,
  373.   1, 1, 0,
  374.   "Return the original file name for FILENAME, with case mapping undone.\n\
  375. Just return FILENAME if `preserve-file-name-case' is nil.\n\
  376. This function uses the text properties set by `expand-file-name' if\n\
  377. `preserve-file-name-case' is non-nil.")
  378.   (filename)
  379.      Lisp_Object filename;
  380. {
  381.   CHECK_STRING (filename, 0);
  382.   return emx_get_original_case_string (filename);
  383. }
  384.  
  385.  
  386. /* Match "^/[^/:]*[^/:.]:" to check for ange-ftp path. */
  387.  
  388. static int emx_ange_ftp_p (const unsigned char *p)
  389. {
  390.   if (p[0] != '/' || p[1] == ':')
  391.     return 0;
  392.   do
  393.     {
  394.       ++p;
  395.     } while (*p != 0 && *p != '/' && *p != ':');
  396.   return *p == ':' && p[-1] != '.';
  397. }
  398.  
  399.  
  400. #define UNC_P(s) (IS_DIRECTORY_SEP ((s)[0]) && IS_DIRECTORY_SEP ((s)[1]) && !IS_DIRECTORY_SEP ((s)[2]))
  401.  
  402. Lisp_Object emx_expand_file_name (Lisp_Object name, Lisp_Object defalt)
  403. {
  404.   const unsigned char *nm, *p;
  405.   unsigned char *q, *result;
  406.   int trailing_slash, drive, size;
  407.  
  408.   name = emx_get_original_case_string (name);
  409.   nm = XSTRING (name)->data;
  410.  
  411.   /* Use the Unix version of expand-file-name for ange-ftp. */
  412.  
  413.   if (remote_expand_file_name || emx_ange_ftp_p (nm))
  414.     return Qnil;
  415.  
  416.   /* Check for a trailing slash (which will be preserved).  */
  417.  
  418.   if (*nm == 0)
  419.     trailing_slash = 0;
  420.   else
  421.     {
  422.       p = strchr (nm, 0);
  423.       trailing_slash = IS_DIRECTORY_SEP (p[-1]);
  424.     }
  425.   p = nm;
  426.  
  427.   /* Expand "~" and "~user".  */
  428.  
  429.   if (p[0] == '~')
  430.     {
  431.       unsigned char *tem;
  432.  
  433.       ++p;
  434.       while (*p != 0 && !IS_DIRECTORY_SEP (*p))
  435.     ++p;            /* skip user name ("all users are equal") */
  436.       if (*p != 0)
  437.     ++p;            /* skip slash or backslash */
  438.       q = (unsigned char *) egetenv ("HOME");
  439.       if (q == 0 || *q == 0)
  440.         {
  441.           /* HOME not set, use the root directory of the current
  442.              drive.  */
  443.  
  444.           tem = alloca (strlen (p) + 2);
  445.           tem[0] = '/';
  446.           strcpy (tem + 1, p);
  447.         }
  448.       else
  449.         {
  450.           tem = alloca (strlen (q) + strlen (p) + 2);
  451.           strcpy (tem, q);
  452.           q = strchr (tem, 0);
  453.           while (q != tem && IS_DIRECTORY_SEP (q[-1]))
  454.             --q;
  455.           *q++ = '/';
  456.           strcpy (q, p);
  457.         }
  458.       p = tem;
  459.     }
  460.  
  461.   if (!_fngetdrive (p) && !UNC_P (p))
  462.     {
  463.       /* There is no drive letter.  Get the drive letter (or the
  464.          complete directory) from the default directory.  */
  465.  
  466.       unsigned char *tem;
  467.  
  468.       CHECK_STRING (defalt, 1);
  469.       q = XSTRING (defalt)->data;
  470.       if (IS_DIRECTORY_SEP (*p))
  471.         {
  472.           /* It's an absolute pathname lacking a drive letter.
  473.              Prepend the drive letter of the default directory.  */
  474.  
  475.           drive = _fngetdrive (q);
  476.           if (drive)
  477.             {
  478.               while (IS_DIRECTORY_SEP (*p))
  479.                 ++p;
  480.               tem = alloca (strlen (p) + 4);
  481.               tem[0] = (unsigned char)drive;
  482.               tem[1] = ':';
  483.               tem[2] = '/';
  484.               strcpy (tem + 3, p);
  485.               p = tem;
  486.             }
  487.         }
  488.       else
  489.         {
  490.           /* It's a relative pathname.  Prepend the default directory.  */
  491.  
  492.           tem = alloca (strlen (q) + strlen (p) + 2);
  493.           strcpy (tem, q);
  494.           q = strchr (tem, 0);
  495.           if (q != tem && !IS_DIRECTORY_SEP (q[-1]))
  496.             *q++ = '/';
  497.           strcpy (q, p);
  498.           p = tem;
  499.         }
  500.     }
  501.  
  502.   /* Prepend the drive letter (and a slash if there's no slash after
  503.      the drive letter -- a drive letter always denotes the root
  504.      directory).  */
  505.  
  506.   drive = _fngetdrive (p);
  507.   if (drive)
  508.     p += 2;
  509.   else if (UNC_P (p))
  510.     {
  511.       /* Universal Naming Convention (\\server\path) -- don't prepend
  512.          a drive letter.  */
  513.       drive = -1;
  514.     }
  515.   else
  516.     drive = _getdrive ();
  517.  
  518.   if (drive != -1)
  519.     {
  520.       unsigned char *tem;
  521.  
  522.       while (IS_DIRECTORY_SEP (*p))
  523.         ++p;
  524.       tem = alloca (strlen (p) + 4);
  525.       tem[0] = (unsigned char)drive;
  526.       tem[1] = ':';
  527.       tem[2] = '/';
  528.       strcpy (tem + 3, p);
  529.       p = tem;
  530.     }
  531.  
  532.   /* Now remove "/./" and "/../".  As we've created an absolute
  533.      pathname, the output string of _abspath() is not longer than the
  534.      input string.  _abspath() translates backslashes into forward
  535.      slashes.  */
  536.  
  537.   size = strlen (p) + 1;
  538.   result = alloca (size + 1);   /* One extra byte for the trailing slash */
  539.   if (_abspath (result, p, size) != 0)
  540.     error ("Internal error in expand-file-name");
  541.  
  542.   /* Add or remove the trailing slash.  */
  543.  
  544.   q = strchr (result, 0);
  545.   if (q != result && IS_DIRECTORY_SEP (q[-1]))
  546.     {
  547.       if (!trailing_slash && !(_fngetdrive (result) && q == result + 3))
  548.         q[-1] = 0;              /* Remove the trailing slash */
  549.     }
  550.   else if (trailing_slash)
  551.     {
  552.       q[0] = '/'; q[1] = 0;     /* Add a trailing slash */
  553.     }
  554.  
  555.   /* Translate to lower case and build a Lisp string.  */
  556.  
  557.   return emx_build_case_fold_string (result);
  558. }
  559.  
  560.  
  561. Lisp_Object find_program_name_handler (Lisp_Object filename)
  562. {
  563.   /* This function must not munge the match data.  */
  564.   Lisp_Object chain;
  565.   char *tem, *p;
  566.  
  567.   CHECK_STRING (filename, 0);
  568.  
  569.   /* Use a shortcut.  */
  570.  
  571.   if (NILP (Vprogram_name_handler_alist))
  572.     return Qnil;
  573.  
  574.   /* Convert the string to lower case and replace backslashes with
  575.      forward slashes.  This simplifies the regexps considerably.  */
  576.  
  577.   tem = (char *)alloca (XSTRING (filename)->size + 1);
  578.   strcpy (tem, XSTRING (filename)->data);
  579.   _nls_strlwr (tem);
  580.   for (p = tem; *p != 0; ++p)
  581.     if (*p == '\\')
  582.       *p = '/';
  583.   filename = build_string (tem);
  584.  
  585.   for (chain = Vprogram_name_handler_alist; CONSP (chain);
  586.        chain = XCONS (chain)->cdr)
  587.     {
  588.       Lisp_Object elt;
  589.       elt = XCONS (chain)->car;
  590.       if (CONSP (elt))
  591.     {
  592.       Lisp_Object string;
  593.       string = XCONS (elt)->car;
  594.       if (STRINGP (string)
  595.           && fast_string_match (string, filename) >= 0
  596.               && NILP (Fmemq (XCONS (elt)->cdr,
  597.                               Vinhibit_program_name_handlers)))
  598.             return XCONS (elt)->cdr;
  599.     }
  600.       QUIT;
  601.     }
  602.   return Qnil;
  603. }
  604.  
  605.  
  606. /* Set the priority of process or thread ID (depending on SCOPE) to
  607.    the values found in PRTY.  */
  608.  
  609. static void set_priority (const struct priority *prty, ULONG scope, ULONG id)
  610. {
  611.   if (_osmode == OS2_MODE)
  612.     {
  613.       DosSetPriority (scope, prty->pclass, -31, id);
  614.       if (prty->plevel != 0)
  615.         DosSetPriority (scope, prty->pclass, prty->plevel, id);
  616.     }
  617. }
  618.  
  619.  
  620. /* Decode the priority class and level given by PCLASS and PLEVEL,
  621.    respectively.  Store the result to *PRTY.  Signal an error if input
  622.    is invalid.  */
  623.  
  624. static void decode_priority (struct priority *prty, Lisp_Object pclass,
  625.                              Lisp_Object plevel)
  626. {
  627.   if (NILP (pclass))
  628.     prty->pclass = PRTYC_NOCHANGE;
  629.   else if (EQ (pclass, intern ("idle-time")))
  630.     prty->pclass = PRTYC_IDLETIME;
  631.   else if (EQ (pclass, intern ("regular")))
  632.     prty->pclass = PRTYC_REGULAR;
  633.   else if (EQ (pclass, intern ("foreground-server")))
  634.     prty->pclass = PRTYC_FOREGROUNDSERVER;
  635.   else
  636.     error ("Invalid priority class");
  637.   if (NILP (plevel))
  638.     prty->plevel = 0;
  639.   else
  640.     {
  641.       CHECK_NUMBER (plevel, 1);
  642.       prty->plevel = XINT (plevel);
  643.       if (prty->plevel < 0 || prty->plevel > 31)
  644.         error ("Invalid priority level");
  645.     }
  646. }
  647.  
  648.  
  649. /* Check the value of process-priority and store the priority in
  650.    child_priority.  This is done before starting the process as
  651.    signalling an error after successfully starting the child process
  652.    isn't a good idea.  */
  653.  
  654. void check_process_priority (void)
  655. {
  656.   if (CONSP (Vprocess_priority))
  657.     decode_priority (&child_priority, Fcar (Vprocess_priority),
  658.                      Fcdr (Vprocess_priority));
  659.   else
  660.     decode_priority (&child_priority, Vprocess_priority, Qnil);
  661. }
  662.  
  663.  
  664. /* Stolen from child_setup of callproc.c and hacked severly.  */
  665.  
  666. int emx_child_setup (in, out, err, new_argv, set_pgrp, current_dir)
  667.      int in, out, err;
  668.      register char **new_argv;
  669.      int set_pgrp;
  670.      Lisp_Object current_dir;
  671. {
  672.   int saved_in, saved_out, saved_err;
  673.   int saved_errno;
  674.   char *org_cwd = 0;
  675.   char org_cwd_buf[512];
  676.   char **env, *p;
  677.   int pid, mode;
  678.  
  679.   {
  680.     register unsigned char *temp;
  681.     register int i;
  682.  
  683.     i = XSTRING (current_dir)->size;
  684.     temp = (unsigned char *) alloca (i + 2);
  685.     bcopy (XSTRING (current_dir)->data, temp, i);
  686.     if (i > 1 && IS_DIRECTORY_SEP (temp[i-1]) && temp[i-2] != ':')
  687.       --i;
  688.     temp[i] = 0;
  689.     org_cwd = _getcwd2 (org_cwd_buf, sizeof (org_cwd_buf));
  690.     _chdir2 (temp);
  691.   }
  692.  
  693.   /* Set `env' to a vector of the strings in Vprocess_environment.  */
  694.   {
  695.     register Lisp_Object tem;
  696.     register char **new_env;
  697.     register int new_length;
  698.  
  699.     new_length = 0;
  700.     for (tem = Vprocess_environment;
  701.          CONSP (tem) && STRINGP (XCONS (tem)->car);
  702.          tem = XCONS (tem)->cdr)
  703.       new_length++;
  704.  
  705.     /* new_length + 1 to include terminating 0 */
  706.     env = new_env = (char **) alloca ((new_length + 1) * sizeof (char *));
  707.  
  708.     /* Copy the Vprocess_alist strings into new_env.  */
  709.     for (tem = Vprocess_environment;
  710.          CONSP  (tem) && STRINGP (XCONS (tem)->car);
  711.      tem = XCONS (tem)->cdr)
  712.       *new_env++ = (char *) XSTRING (XCONS (tem)->car)->data;
  713.     *new_env = 0;
  714.   }
  715.  
  716.   saved_in = dup (0);
  717.   saved_out = dup (1);
  718.   saved_err = dup (2);
  719.  
  720.   if (saved_in == -1 || saved_out == -1 || saved_err == -1)
  721.     {
  722.       if (saved_in != -1) close (saved_in);
  723.       if (saved_out != -1) close (saved_out);
  724.       if (saved_err != -1) close (saved_err);
  725.       return -1;
  726.     }
  727.  
  728.   fcntl (saved_in, F_SETFD, 1);
  729.   fcntl (saved_out, F_SETFD, 1);
  730.   fcntl (saved_err, F_SETFD, 1);
  731.  
  732.   close (0);
  733.   close (1);
  734.   close (2);
  735.  
  736.   dup2 (in, 0);
  737.   dup2 (out, 1);
  738.   dup2 (err, 2);
  739.  
  740.   /* Close Emacs's descriptors that this process should not have.  */
  741.   close_process_descs ();
  742.  
  743.   /* Set the priority.  Note that check_process_priority must have
  744.      been called.  Changing the priority after starting the process
  745.      yields ERROR_NOT_DESCENDANT, therefore we let the child process
  746.      inherit the priority.  */
  747.   if (!NILP (Vprocess_priority))
  748.     set_priority (&child_priority, PRTYS_THREAD, 0);
  749.  
  750.   if (_osmode == OS2_MODE && emx_start_session)
  751.     mode = P_SESSION | P_MINIMIZE | P_BACKGROUND;
  752.   else
  753.     mode = P_NOWAIT;
  754.   pid = spawnvpe (mode, new_argv[0], new_argv, env);
  755.   saved_errno = errno;
  756.  
  757.   /* Restore our priority.  */
  758.   if (!NILP (Vprocess_priority))
  759.     set_priority (&emacs_priority, PRTYS_THREAD, 0);
  760.  
  761.   dup2 (saved_in, 0); close (saved_in);
  762.   dup2 (saved_out, 1); close (saved_out);
  763.   dup2 (saved_err, 2); close (saved_err);
  764.   if (err != out)
  765.     close (err);
  766.   if (in == out)                /* pty */
  767.     close (out);
  768.   if (org_cwd != 0)
  769.     _chdir2 (org_cwd);
  770.   errno = saved_errno;
  771.   return pid;
  772. }
  773.  
  774.  
  775. void emx_proc_input_pipe (int fd)
  776. {
  777.   if (fd >= 0 && binary_process_input)
  778.     setmode (fd, O_BINARY);
  779. }
  780.  
  781.  
  782. void emx_proc_output_pipe (int fd)
  783. {
  784.   if (fd >= 0 && binary_process_output)
  785.     setmode (fd, O_BINARY);
  786. }
  787.  
  788.  
  789. void emx_setup_start_process (void)
  790. {
  791.   if (!binary_process_input && binary_async_process_input)
  792.     specbind (Qbinary_process_input, Qt);
  793. }
  794.  
  795.  
  796. /* emx 0.9b (revision index 40) does not support tcsetattr() for PTYs.
  797.    Turn off ISIG and OPOST by using DosDevIOCtl. */
  798.  
  799. #define IOCTL_XF86SUP           0x76
  800. #define XF86SUP_TIOCSETA        0x48
  801. #define XF86SUP_TIOCGETA        0x65
  802.  
  803. struct pt_termios
  804. {
  805.   unsigned short c_iflag;
  806.   unsigned short c_oflag;
  807.   unsigned short c_cflag;
  808.   unsigned short c_lflag;
  809.   unsigned char    c_cc[NCCS];
  810.   long reserved[4];
  811. };
  812.  
  813. void emx_setup_slave_pty (int fd)
  814. {
  815.   ULONG rc, parm_length, data_length;
  816.   struct pt_termios tio;
  817.  
  818.   data_length = sizeof (tio);
  819.   rc = DosDevIOCtl (fd, IOCTL_XF86SUP, XF86SUP_TIOCGETA,
  820.                     NULL, 0, NULL,
  821.                     &tio, data_length, &data_length);
  822.   if (rc == 0)
  823.     {
  824.       parm_length = sizeof (tio);
  825.       tio.c_lflag &= ~(ISIG|ECHO);
  826.       tio.c_oflag &= ~OPOST;
  827.       rc = DosDevIOCtl (fd, IOCTL_XF86SUP, XF86SUP_TIOCSETA,
  828.                         &tio, parm_length, &parm_length,
  829.                         NULL, 0, NULL);
  830.     }
  831. }
  832.  
  833.  
  834. /* Delete temporary file when a process ends.  Note that
  835.    emx_async_delete_now is called in the SIGCLD handler, and may
  836.    interrupt emx_async_delete_add.  Therefore we don't deallocate or
  837.    modify the list in emx_async_delete_now.  We cannot use the PID as
  838.    delete_temp_file of callproc.c does not know the PID. */
  839.  
  840. struct async_delete
  841. {
  842.   struct async_delete *next;
  843.   int active, length;
  844.   char *name;
  845. };
  846.  
  847. static struct async_delete *async_delete_head;
  848.  
  849. void emx_async_delete_add (char *name)
  850. {
  851.   struct async_delete *p;
  852.   int length;
  853.  
  854.   length = strlen (name);
  855.   for (p = async_delete_head; p; p = p->next)
  856.     if (!p->active && p->length <= length)
  857.       break;
  858.   if (!p)
  859.     for (p = async_delete_head; p; p = p->next)
  860.       if (!p->active)
  861.         break;
  862.   if (p)
  863.     {
  864.       if (p->length < length)
  865.         {
  866.           free (p->name);
  867.           p->name = (char *)xmalloc (length + 1);
  868.           p->length = length;
  869.         }
  870.       strcpy (p->name, name);
  871.       p->active = 1;
  872.     }
  873.   else
  874.     {
  875.       p = (struct async_delete *)xmalloc (sizeof (struct async_delete));
  876.       p->name = (char *)xmalloc (length + 1);
  877.       p->length = length;
  878.       strcpy (p->name, name);
  879.       p->active = 1;
  880.       p->next = async_delete_head;
  881.       async_delete_head = p;
  882.     }
  883.  
  884.   /* SIGCLD may occur between the first attempt to delete the file
  885.      (before calling this function) and this point. */
  886.  
  887.   if (unlink (name) == 0)
  888.     p->active = 0;
  889. }
  890.  
  891.  
  892. /* Try to delete the temporary files. */
  893.  
  894. void emx_async_delete_now (void)
  895. {
  896.   struct async_delete *p;
  897.   int again = 1;
  898.  
  899.   while (again)
  900.     {
  901.       again = 0;
  902.       for (p = async_delete_head; p; p = p->next)
  903.         if (p->active && !(unlink (p->name) != 0 && errno == EACCES))
  904.           {
  905.             p->active = 0;
  906.             again = 1;
  907.           }
  908.     }
  909. }
  910.  
  911.  
  912. void emx_exec_name (char **dst)
  913. {
  914.   PTIB ptib;
  915.   PPIB ppib;
  916.   char name[280], *p;
  917.  
  918.   if (_osmode == OS2_MODE)
  919.     {
  920.       DosGetInfoBlocks (&ptib, &ppib);
  921.       DosQueryModuleName (ppib->pib_hmte, sizeof (name), name);
  922.       _nls_strlwr (name);
  923.       *dst = strdup (name);
  924.     }
  925.   for (p = *dst; *p != 0; ++p)
  926.     if (*p == '\\')
  927.       *p = '/';
  928. }
  929.  
  930.  
  931. DEFUN ("emx-binary-mode-p", Femx_binary_mode_p, Semx_binary_mode_p, 1, 1, 0,
  932.   "Return t if binary mode should be used for FILENAME.\n\
  933. Otherwise, return nil.\n\
  934. Binary mode is used if one of the regular expressions in\n\
  935. `emx-binary-mode-list' match FILENAME.")
  936.   (filename)
  937.     Lisp_Object filename;
  938. {
  939.   /* This function must not munge the match data.  */
  940.   Lisp_Object chain;
  941.  
  942.   CHECK_STRING (filename, 0);
  943.  
  944.   for (chain = Vemx_binary_mode_list; CONSP (chain);
  945.        chain = XCONS (chain)->cdr)
  946.     {
  947.       Lisp_Object elt;
  948.       elt = XCONS (chain)->car;
  949.       if (STRINGP (elt) && fast_string_match (elt, filename) >= 0)
  950.         return Qt;
  951.       QUIT;
  952.     }
  953.   return Qnil;
  954. }
  955.  
  956.  
  957. #ifdef HAVE_PM
  958.  
  959. DEFUN ("pm-session-bond", Fpm_session_bond, Spm_session_bond, 1, 1, 0,
  960.   "Establish or break bond between emacs.exe and pmemacs.exe sessions.\n\
  961. Non-nil BOND establishes the bond, nil BOND breaks the bond.")
  962.   (bond)
  963.      Lisp_Object bond;
  964. {
  965.   HSWITCH hSwitch;
  966.   SWCNTRL data;
  967.   STATUSDATA status;
  968.  
  969.   if (!pm_session_started)
  970.     error ("PM Emacs connection not established");
  971.  
  972.   hSwitch = WinQuerySwitchHandle (NULLHANDLE, pm_pid);
  973.   status.Length = sizeof (status);
  974.   status.SelectInd = SET_SESSION_UNCHANGED;
  975.   status.BondInd = (NILP (bond) ? SET_SESSION_NO_BOND : SET_SESSION_BOND);
  976.   if (hSwitch != NULLHANDLE && WinQuerySwitchEntry (hSwitch, &data) == 0)
  977.     DosSetSession (data.idSession, &status);
  978.   return Qnil;
  979. }
  980.  
  981. #endif /* HAVE_PM */
  982.  
  983. DEFUN ("remove-from-window-list", Fremove_from_window_list,
  984.   Sremove_from_window_list, 0, 0, 0,
  985.   "Remove Emacs from the Window List.")
  986.   ()
  987. {
  988.   HSWITCH hSwitch;
  989.  
  990.   hSwitch = WinQuerySwitchHandle (NULLHANDLE, getpid ());
  991.   WinRemoveSwitchEntry (hSwitch);
  992.   return Qnil;
  993. }
  994.  
  995.  
  996. DEFUN ("filesystem-type", Ffilesystem_type, Sfilesystem_type,
  997.   1, 1, 0,
  998.   "Return a string identifying the filesystem type of PATH.\n\
  999. Filesystem types include FAT, HPFS, LAN, CDFS, NFS and NETWARE.")
  1000.   (string)
  1001.      Lisp_Object string;
  1002. {
  1003.   char drive[3], type[16];
  1004.   int d;
  1005.  
  1006.   CHECK_STRING (string, 0);
  1007.   d = _fngetdrive (XSTRING (string)->data);
  1008.   if (d == 0)
  1009.     d = _getdrive ();
  1010.   drive[0] = (char)d;
  1011.   drive[1] = ':';
  1012.   drive[2] = 0;
  1013.   if (_filesys (drive, type, sizeof (type)) != 0)
  1014.     error ("_filesys() failed");
  1015.   return build_string (type);
  1016. }
  1017.  
  1018.  
  1019. DEFUN ("file-name-valid-p", Ffile_name_valid_p, Sfile_name_valid_p,
  1020.   1, 1, 0,
  1021.   "Return t if STRING is a valid file name.\n\
  1022. Whether a file name is valid or not depends on the file system.\n\
  1023. This is a special feature of GNU Emacs for emx.")
  1024.   (string)
  1025.      Lisp_Object string;
  1026. {
  1027.   int i;
  1028.   unsigned char *name;
  1029.  
  1030.   CHECK_STRING (string, 0);
  1031.   name = XSTRING (string)->data;
  1032.   if (_osmode == OS2_MODE)
  1033.     {
  1034.       i = open (name, O_RDONLY);
  1035.       if (i >= 0)
  1036.         {
  1037.           close (i);
  1038.           return Qt;
  1039.         }
  1040.       i = _syserrno ();
  1041.       return (i != 15 && i != 123 && i != 206) ? Qt : Qnil;
  1042.     }
  1043.   else
  1044.     {
  1045.       if (_fngetdrive (name) != 0)
  1046.         name += 2;
  1047.       if (*name == 0)
  1048.         return Qnil;
  1049.       if (strpbrk (name, " \"'*+,:;<=>?[]|") != NULL)
  1050.         return Qnil;
  1051.       for (i = 0; name[i] != 0; ++i)
  1052.         if (name[i] < 0x20)
  1053.           return Qnil;
  1054.       for (;;)
  1055.         {
  1056.           i = 0;
  1057.           while (*name != 0 && !IS_DIRECTORY_SEP (*name) && *name != '.')
  1058.             ++i, ++name;
  1059.           if (i > 8)
  1060.             return Qnil;
  1061.           if (*name == '.')
  1062.             {
  1063.               ++name;
  1064.               if (i == 0)
  1065.                 {
  1066.                   if (*name == '.')
  1067.                     ++name;
  1068.                   if (*name != 0 && !IS_DIRECTORY_SEP (*name))
  1069.                     return Qnil;
  1070.                 }
  1071.               i = 0;
  1072.               while (*name != 0 && !IS_DIRECTORY_SEP (*name) && *name != '.')
  1073.                 ++i, ++name;
  1074.               if (i > 3)
  1075.                 return Qnil;
  1076.             }
  1077.           if (*name == 0)
  1078.             return Qt;
  1079.           if (!IS_DIRECTORY_SEP (*name))
  1080.             return Qnil;
  1081.           ++name;
  1082.         }
  1083.     }
  1084. }
  1085.  
  1086.  
  1087. DEFUN ("keyboard-type", Fkeyboard_type, Skeyboard_type,
  1088.   0, 0, 0,
  1089.   "Return information about the keyboard.\n\
  1090. The value is a list of the form (COUNTRY SUBCOUNTRY CODEPAGE), where\n\
  1091.   COUNTRY is the country code of the keyboard layout (a string),\n\
  1092.     for instance \"US\".\n\
  1093.   SUBCOUNTRY is the subcountry code (a string), for instance \"103 \".\n\
  1094.   CODEPAGE is the codepage (a number), on which the current keyboard\n\
  1095.     translation table is based, for instance 437.\n\
  1096. This function is currently implemented under OS/2 only.\n\
  1097. If the keyboard information cannot be retrieved (because Emacs is\n\
  1098. running under MS-DOS, for instance), nil is returned.")
  1099.   ()
  1100. {
  1101.   ULONG plen, dlen, action;
  1102.   HFILE handle;
  1103.   struct
  1104.     {
  1105.       USHORT length;
  1106.       USHORT codepage;
  1107.       UCHAR strings[8];
  1108.     } kd;
  1109.   Lisp_Object value;
  1110.  
  1111.   value = Qnil;
  1112.   if (_osmode == OS2_MODE
  1113.       && DosOpen ("KBD$", &handle, &action, 0, 0,
  1114.                   OPEN_ACTION_FAIL_IF_NEW | OPEN_ACTION_OPEN_IF_EXISTS,
  1115.                   OPEN_ACCESS_READONLY | OPEN_SHARE_DENYNONE,
  1116.                   NULL) == 0)
  1117.     {
  1118.       kd.length = sizeof (kd);
  1119.       dlen = sizeof (kd); plen = 0;
  1120.       if (DosDevIOCtl (handle, 4, 0x7b, NULL, plen, &plen,
  1121.                        &kd, dlen, &dlen) == 0)
  1122.         value = Fcons (build_string (kd.strings),
  1123.                        Fcons (build_string (strchr (kd.strings, 0) + 1),
  1124.                               Fcons (make_number (kd.codepage), Qnil)));
  1125.       DosClose (handle);
  1126.     }
  1127.   return value;
  1128. }
  1129.  
  1130.  
  1131. DEFUN ("emacs-priority", Femacs_priority, Semacs_priority,
  1132.   1, 2, 0,
  1133.   "Set the priority of the Emacs process.\n\
  1134. PCLASS selects the priority class.  Possible values are\n\
  1135.   nil (no change),\n\
  1136.   idle-time (idle-time priority class -- you don't want to use this),\n\
  1137.   regular (regular priority class -- this is the OS/2 default,\n\
  1138.     a priority boost is applied if the process is in the foreground), and\n\
  1139.   foreground-server (fixed-high priority class -- use with care).\n\
  1140. PLEVEL is nil (same as 0) or a number between 0 and 31 which indicates\n\
  1141.   the priority level within the priority class.  Level 31 has the highest\n\
  1142.   priority in each class, the default value assigned by OS/2 is 0.\n\
  1143. Child processes inherit the priority unless process-priority is non-nil.\n\
  1144. This function is implemented under OS/2 only.")
  1145.   (pclass, plevel)
  1146.      Lisp_Object pclass, plevel;
  1147. {
  1148.   struct priority prty;
  1149.  
  1150.   decode_priority (&prty, pclass, plevel);
  1151.   set_priority (&prty, PRTYS_THREAD, 0);
  1152.   emacs_priority = prty;
  1153.   return Qnil;
  1154. }
  1155.  
  1156.  
  1157. /* This function is called on program startup.  */
  1158.  
  1159. void emx_setup (void)
  1160. {
  1161.   int min_rev = 40;
  1162.  
  1163.   /* Check the emx.dll revision index. */
  1164.  
  1165.   if (_osmode == OS2_MODE && _emx_rev < min_rev)
  1166.     {
  1167.       ULONG rc;
  1168.       HMODULE hmod;
  1169.       char name[CCHMAXPATH];
  1170.       char fail[9];
  1171.  
  1172.       fprintf (stderr, "Your emx.dll has revision index %d.\n"
  1173.                "Emacs requires revision index %d (version 0.9b, fix00) "
  1174.                "or later.\n",
  1175.                _emx_rev, min_rev);
  1176.       rc = DosLoadModule (fail, sizeof (fail), "emx", &hmod);
  1177.       if (rc == 0)
  1178.         {
  1179.           rc = DosQueryModuleName (hmod, sizeof (name), name);
  1180.           if (rc == 0)
  1181.             fprintf (stderr, "Please delete or update `%s'.\n", name);
  1182.           DosFreeModule (hmod);
  1183.         }
  1184.       exit (1);
  1185.     }
  1186.  
  1187.   /* No `faked binary file' is currently in used.  */
  1188.  
  1189.   fb_fd = -1;
  1190.  
  1191.   /* Initialize the priority to be used after starting a child
  1192.      process.  At the moment, I'm too lazy for looking in TIB2...  */
  1193.  
  1194.   emacs_priority.pclass = PRTYC_REGULAR;
  1195.   emacs_priority.plevel = 0;
  1196.  
  1197.   /* Initialize the current code page.  */
  1198.  
  1199.   cur_code_page = 0;
  1200.   if (_osmode == OS2_MODE)
  1201.     {
  1202.       ULONG rc, cp, len;
  1203.  
  1204.       len = 0;
  1205.       rc = DosQueryCp (sizeof (cp), &cp, &len);
  1206.       if ((rc == 0 || rc == ERROR_CPLIST_TOO_SMALL) && len == sizeof (cp))
  1207.         cur_code_page = cp;
  1208.     }
  1209. }
  1210.  
  1211.  
  1212. DEFUN ("current-code-page", Fcurrent_code_page, Scurrent_code_page, 0, 0, 0,
  1213.   "Return the current code page identifier.\n\
  1214. The value is nil if the code page identifier cannot be retrieved.\n\
  1215. Code pages are not yet supported under MS-DOS.")
  1216.   ()
  1217. {
  1218.   if (_osmode == DOS_MODE || cur_code_page == 0)
  1219.     return Qnil;
  1220.   return make_number (cur_code_page);
  1221. }
  1222.  
  1223.  
  1224. DEFUN ("list-code-pages", Flist_code_pages, Slist_code_pages, 0, 0, 0,
  1225.   "Return a list of available code pages.\n\
  1226. Code pages are not yet supported under MS-DOS.")
  1227.   ()
  1228. {
  1229.   ULONG rc, acp[8], len;
  1230.   Lisp_Object list;
  1231.   int i;
  1232.  
  1233.   if (_osmode == DOS_MODE)
  1234.     return Qnil;
  1235. #ifdef HAVE_PM
  1236.   if (pm_session_started)
  1237.     return pm_list_code_pages ();
  1238. #else /* not HAVE_PM
  1239.   return Qnil;                  /* TODO */
  1240. #endif /* not HAVE_PM */
  1241.   len = 0;
  1242.   rc = DosQueryCp (sizeof (acp), acp, &len);
  1243.   if ((rc != 0 && rc != ERROR_CPLIST_TOO_SMALL) || len < sizeof (acp[0]))
  1244.     return Qnil;
  1245.   list = Qnil;
  1246.   for (i = len / sizeof (acp[0]) - 1; i >= 1; --i)
  1247.     list = Fcons (make_number (acp[i]), list);
  1248.   return list;
  1249. }
  1250.  
  1251.  
  1252. DEFUN ("set-code-page", Fset_code_page, Sset_code_page, 1, 1, "NCode page: ",
  1253.   "Set the code page to CODE-PAGE.\n\
  1254. Code pages are not yet supported under MS-DOS.")
  1255.   (code_page)
  1256.      Lisp_Object code_page;
  1257. {
  1258.   CHECK_NUMBER (code_page, 0);
  1259.   if (_osmode == DOS_MODE)
  1260.     error ("Cannot set code page under MS-DOS");
  1261. #ifndef HAVE_PM
  1262.   error ("Cannot set code page under X");
  1263. #else /* HAVE_PM */
  1264.   if (XINT (code_page) >= 1 && XINT (code_page) <= 32767)
  1265.     {
  1266.       USHORT rc, cp;
  1267.  
  1268.       cp = XINT (code_page);
  1269.       if (cp == cur_code_page)
  1270.         return Qnil;
  1271.       if (pm_session_started)
  1272.         {
  1273.           if (pm_set_code_page (cp))
  1274.             {
  1275.               cur_code_page = cp;
  1276.               return Qnil;
  1277.             }
  1278.         }
  1279.       else
  1280.         {
  1281.           rc = VioSetCp (0, cp, 0);
  1282.           if (rc == 0)
  1283.             {
  1284.               rc = KbdSetCp (0, cp, 0);
  1285.               if (rc != 0)
  1286.                 VioSetCp (0, (USHORT)cur_code_page, 0);
  1287.             }
  1288.           if (rc == 0)
  1289.             {
  1290.               cur_code_page = cp;
  1291.               return Qnil;
  1292.             }
  1293.         }
  1294.     }
  1295.   error ("Invalid code page");
  1296. #endif /* HAVE_PM */
  1297. }
  1298.  
  1299.  
  1300. DEFUN ("set-cursor-size", Fset_cursor_size, Sset_cursor_size, 2, 2,
  1301.   "NFirst row: \nLast row: ",
  1302.   "Set the text-mode cursor size.\n\
  1303. The cursor will occupy rows START through END.  Negative arguments\n\
  1304. are percentages.")
  1305.   (start, end)
  1306.      Lisp_Object start, end;
  1307. {
  1308.   VIOCURSORINFO vci1, vci2, *pvci;
  1309.  
  1310.   CHECK_NUMBER (start, 0);
  1311.   CHECK_NUMBER (end, 0);
  1312.   if (_osmode == DOS_MODE)
  1313.     error ("Cannot set cursor size under MS-DOS");
  1314.   if (XINT (start) < -100 || XINT (start) > 31)
  1315.     error ("Invalid cursor start row");
  1316.   if (XINT (end) < -100 || XINT (end) > 31)
  1317.     error ("Invalid cursor end row");
  1318.   pvci = _THUNK_PTR_STRUCT_OK (&vci1) ? &vci1 : &vci2;
  1319.  
  1320.   if (VioGetCurType (pvci, 0) != 0)
  1321.     error ("VioGetCurType failed");
  1322.   pvci->yStart = (USHORT)XINT (start);
  1323.   pvci->cEnd = (USHORT)XINT (end);
  1324.   if (VioSetCurType (pvci, 0) != 0)
  1325.     error ("VioSetCurType failed");
  1326.   return Qnil;
  1327. }
  1328.  
  1329.  
  1330. DEFUN ("get-cursor-size", Fget_cursor_size, Sget_cursor_size, 0, 0, 0,
  1331.   "Return the text-mode cursor size: (START . END).\n\
  1332. START is the start row, END is the end row of the cursor.\n\
  1333. Return nil if the cursor size cannot be retrieved.")
  1334.   ()
  1335. {
  1336.   VIOCURSORINFO vci1, vci2, *pvci;
  1337.  
  1338.   if (_osmode == DOS_MODE)
  1339.     return Qnil;
  1340.   pvci = _THUNK_PTR_STRUCT_OK (&vci1) ? &vci1 : &vci2;
  1341.   if (VioGetCurType (pvci, 0) != 0)
  1342.     return Qnil;
  1343.   return Fcons (make_number (pvci->yStart), make_number (pvci->cEnd));
  1344. }
  1345.  
  1346.  
  1347. static int emx_longname_mismatch (const char *fname, const char *longname)
  1348. {
  1349.   return _fncmp (_getname (fname), longname) != 0;
  1350. }
  1351.  
  1352.  
  1353. void emx_fix_longname (const char *fname)
  1354. {
  1355.   struct _ea ea;
  1356.   char *longname;
  1357.  
  1358.   if (_ea_get (&ea, fname, -1, ".LONGNAME") == 0 && ea.value != NULL)
  1359.     {
  1360.       if (ea.size >= 4 && *(unsigned short *)ea.value == EAT_ASCII)
  1361.         {
  1362.           longname = alloca (ea.size - 4 + 1);
  1363.           memcpy (longname, (const char *)ea.value + 4, ea.size - 4);
  1364.           longname[ea.size-4] = 0;
  1365.           if (emx_longname_mismatch (fname, longname))
  1366.             _ea_remove (fname, -1, ".LONGNAME");
  1367.         }
  1368.       else
  1369.         _ea_remove (fname, -1, ".LONGNAME");
  1370.     }
  1371. }
  1372.  
  1373.  
  1374. void emx_fix_ead_longname (_ead ead, const char *fname)
  1375. {
  1376.   int idx, size;
  1377.   const void *val;
  1378.   char *longname;
  1379.  
  1380.   idx = _ead_find (ead, ".LONGNAME");
  1381.   if (idx != -1)
  1382.     {
  1383.       val = _ead_get_value (ead, idx);
  1384.       size = _ead_value_size (ead, idx);
  1385.       if (val != NULL && size >= 4 && *(unsigned short *)val == EAT_ASCII)
  1386.         {
  1387.           longname = alloca (size - 4 + 1);
  1388.           memcpy (longname, (const char *)val + 4, size - 4);
  1389.           longname[size-4] = 0;
  1390.           if (emx_longname_mismatch (fname, longname))
  1391.             _ead_delete (ead, idx);
  1392.         }
  1393.       else
  1394.         _ead_delete (ead, idx);
  1395.     }
  1396. }
  1397.  
  1398.  
  1399. static Lisp_Object ea_to_lisp (struct _ea *pea)
  1400. {
  1401.   Lisp_Object elts[4];
  1402.   USHORT type, size;
  1403.  
  1404.   if (!pea->value)
  1405.     return Qnil;
  1406.  
  1407.   elts[0] = make_number (pea->flags);
  1408.   if (pea->size < 4)
  1409.     elts[1] = elts[2] = Qnil;
  1410.   else
  1411.     {
  1412.       type = ((USHORT *)pea->value)[0];
  1413.       size = ((USHORT *)pea->value)[1];
  1414.       switch (type)
  1415.         {
  1416.         case EAT_ASCII:
  1417.           elts[1] = Qascii;
  1418.           break;
  1419.         case EAT_BINARY:
  1420.           elts[1] = Qbinary;
  1421.           break;
  1422.         case EAT_BITMAP:
  1423.           elts[1] = Qbitmap;
  1424.           break;
  1425.         case EAT_METAFILE:
  1426.           elts[1] = Qmetafile;
  1427.           break;
  1428.         case EAT_ICON:
  1429.           elts[1] = Qicon;
  1430.           break;
  1431.         case EAT_EA:
  1432.           elts[1] = Qea;
  1433.           break;
  1434.         case EAT_MVMT:
  1435.           elts[1] = Qmvmt;
  1436.           break;
  1437.         case EAT_MVST:
  1438.           elts[1] = Qmvst;
  1439.           break;
  1440.         case EAT_ASN1:
  1441.           elts[1] = Qasn1;
  1442.           break;
  1443.         default:
  1444.           elts[1] = make_number (type);
  1445.           break;
  1446.         }
  1447.       elts[2] = make_number (size);
  1448.     }
  1449.   elts[3] = make_string (pea->value, pea->size);
  1450.   return Fvector (4, elts);
  1451. }
  1452.  
  1453.  
  1454. static void ea_from_lisp (struct _ea *pea, Lisp_Object vec)
  1455. {
  1456.   Lisp_Object l_flags, l_type, l_size, l_value;
  1457.   int value_size;
  1458.   void *value;
  1459.  
  1460.   if (XVECTOR (vec)->size != 4)
  1461.     error ("EA vector does not have 4 elements");
  1462.  
  1463.   l_flags = XVECTOR (vec)->contents[0];
  1464.   l_type  = XVECTOR (vec)->contents[1];
  1465.   l_size  = XVECTOR (vec)->contents[2];
  1466.   l_value = XVECTOR (vec)->contents[3];
  1467.  
  1468.   if (!STRINGP (l_value))
  1469.     error ("Invalid EA vector");
  1470.   value_size = XSTRING (l_value)->size;
  1471.   value = alloca (value_size);
  1472.   bcopy (XSTRING (l_value)->data, value, value_size);
  1473.  
  1474.   if (!INTEGERP (l_flags) || XINT (l_flags) < 0 || XINT (l_flags > 255))
  1475.     error ("Invalid EA vector");
  1476.   pea->flags = XINT (l_flags);
  1477.  
  1478.   if (!NILP (l_type))
  1479.     {
  1480.       USHORT type;
  1481.  
  1482.       if (INTEGERP (l_type))
  1483.         {
  1484.           if (XINT (l_type) < 0 || XINT (l_type) > 65535)
  1485.             error ("Invalid EA vector");
  1486.           type = XINT (l_type);
  1487.         }
  1488.       else if (EQ (l_type, Qascii))
  1489.         type = EAT_ASCII;
  1490.       else if (EQ (l_type, Qbinary))
  1491.         type = EAT_BINARY;
  1492.       else if (EQ (l_type, Qbitmap))
  1493.         type = EAT_BITMAP;
  1494.       else if (EQ (l_type, Qmetafile))
  1495.         type = EAT_METAFILE;
  1496.       else if (EQ (l_type, Qicon))
  1497.         type = EAT_ICON;
  1498.       else if (EQ (l_type, Qea))
  1499.         type = EAT_EA;
  1500.       else if (EQ (l_type, Qmvmt))
  1501.         type = EAT_MVMT;
  1502.       else if (EQ (l_type, Qmvst))
  1503.         type = EAT_MVST;
  1504.       else if (EQ (l_type, Qasn1))
  1505.         type = EAT_ASN1;
  1506.       else
  1507.         error ("Invalid EA vector");
  1508.       if (value_size >= 4)
  1509.         ((USHORT *)value)[0] = type;
  1510.     }
  1511.  
  1512.   if (!NILP (l_size))
  1513.     {
  1514.       if (!INTEGERP (l_size) || XINT (l_size) < 0 || XINT (l_size > 65535 - 4))
  1515.         error ("Invalid EA vector");
  1516.       if (value_size >= 4)
  1517.         ((USHORT *)value)[1] = (USHORT)XINT (l_size);
  1518.     }
  1519.   pea->size = value_size;
  1520.   pea->value = xmalloc (value_size);
  1521.   bcopy (value, pea->value, value_size);
  1522. }
  1523.  
  1524.  
  1525. static Lisp_Object
  1526. ead_unwind (obj)
  1527.      Lisp_Object obj;
  1528. {
  1529.   _ead ead;
  1530.   char *s;
  1531.  
  1532.   s = XSTRING (obj)->data;
  1533.   sscanf (s, "%p", &ead);
  1534.   _ead_destroy (ead);
  1535.   return Qnil;
  1536. }
  1537.  
  1538.  
  1539. static void ead_record_unwind_protect (_ead ead)
  1540. {
  1541.   char buf[64];
  1542.  
  1543.   /* _ead is a pointer type.  Make a string from its value to turn it
  1544.      into a Lisp object.  I don't dare to use XPNTR et al.  */
  1545.   sprintf (buf, "%p", ead);
  1546.   record_unwind_protect (ead_unwind, build_string (buf));
  1547. }
  1548.  
  1549.  
  1550. DEFUN ("get-ea", Fget_ea, Sget_ea, 2, 2, 0,
  1551.   "Get from file or directory PATH the extended attribute NAME.\n\
  1552. Return nil if there is no such extended attribute.  Otherwise,\n\
  1553. return a vector [FLAGS TYPE SIZE VALUE], where FLAGS is the flags byte\n\
  1554. of the extended attribute, TYPE identifies the type of the extended\n\
  1555. attribute (`ascii', `binary', `bitmap', `metafile', `icon', `ea',\n\
  1556. `mvmt', `mvst', `asn1', or a number), SIZE is the contents of the\n\
  1557. size field of the value and VALUE is the (binary) value\n\
  1558. of the extended attribute, including the type and size fields.\n\
  1559. TYPE and SIZE are nil if the value is too short.")
  1560.   (path, name)
  1561.      Lisp_Object path, name;
  1562. {
  1563.   struct _ea ea;
  1564.   Lisp_Object result;
  1565.  
  1566.   CHECK_STRING (path, 0);
  1567.   CHECK_STRING (name, 1);
  1568.   path = Fexpand_file_name (path, Qnil);
  1569.  
  1570.   if (_ea_get (&ea, XSTRING (path)->data, 0, XSTRING (name)->data) != 0)
  1571.     report_file_error ("Getting extended attribute", Fcons (path, Qnil));
  1572.   result = ea_to_lisp (&ea);
  1573.   _ea_free (&ea);
  1574.   return result;
  1575. }
  1576.  
  1577.  
  1578. DEFUN ("get-ea-string", Fget_ea_string, Sget_ea_string, 2, 2, 0,
  1579.   "Get from file or directory PATH the extended attribute NAME.\n\
  1580. Return nil if there is no such extended attribute, or if the value\n\
  1581. of the extended attribute is not a string.")
  1582.   (path, name)
  1583.      Lisp_Object path, name;
  1584. {
  1585.   struct _ea ea;
  1586.   Lisp_Object result;
  1587.  
  1588.   CHECK_STRING (path, 0);
  1589.   CHECK_STRING (name, 1);
  1590.   path = Fexpand_file_name (path, Qnil);
  1591.  
  1592.   if (_ea_get (&ea, XSTRING (path)->data, 0, XSTRING (name)->data) != 0)
  1593.     report_file_error ("Getting extended attribute", Fcons (path, Qnil));
  1594.   if (ea.value && ea.size >= 4 && *(USHORT *)ea.value == EAT_ASCII)
  1595.     result = make_string (ea.value + 4, ea.size - 4);
  1596.   else
  1597.     result = Qnil;
  1598.   _ea_free (&ea);
  1599.   return result;
  1600. }
  1601.  
  1602.  
  1603. DEFUN ("get-ea-list", Fget_ea_list, Sget_ea_list, 1, 1, 0,
  1604.   "Get from file or directory PATH all extended attributes.\n\
  1605. Return a list of elements (NAME . VECTOR), one element for each\n\
  1606. extended attribute.  NAME is the name of the extended attribute,\n\
  1607. VECTOR is a vector [FLAGS TYPE SIZE VALUE], where FLAGS is the flags\n\
  1608. byte of the extended attribute, TYPE identifies the type of the extended\n\
  1609. attribute (`ascii', `binary', `bitmap', `metafile', `icon', `ea',\n\
  1610. `mvmt', `mvst', `asn1', or a number), SIZE is the contents of the\n\
  1611. size field of the value and VALUE is the (binary) value\n\
  1612. of the extended attribute, including the type and size fields.\n\
  1613. TYPE and SIZE are nil if the value is too short.")
  1614.   (path)
  1615.      Lisp_Object path;
  1616. {
  1617.   _ead ead;
  1618.   struct _ea ea;
  1619.   Lisp_Object result, vec, name;
  1620.   int i, n;
  1621.   int count = specpdl_ptr - specpdl;
  1622.  
  1623.   CHECK_STRING (path, 0);
  1624.   path = Fexpand_file_name (path, Qnil);
  1625.  
  1626.   ead = _ead_create ();
  1627.   if (!ead)
  1628.     report_file_error ("Getting extended attribute", Fcons (path, Qnil));
  1629.  
  1630.   ead_record_unwind_protect (ead);
  1631.  
  1632.   if (_ead_read (ead, XSTRING (path)->data, 0, 0) != 0)
  1633.     report_file_error ("Getting extended attribute", Fcons (path, Qnil));
  1634.  
  1635.   result = Qnil;
  1636.   n = _ead_count (ead);
  1637.   for (i = n; i >= 1; --i)
  1638.     {
  1639.       ea.flags = _ead_get_flags (ead, i);
  1640.       ea.size = _ead_value_size (ead, i);
  1641.       ea.value = (void *)_ead_get_value (ead, i);
  1642.       vec = ea_to_lisp (&ea);
  1643.       /* build_string does not take _const_ char * JMB */
  1644.       name = build_string ((char *)_ead_get_name (ead, i));
  1645.       result = Fcons (Fcons (name, vec), result);
  1646.     }
  1647.  
  1648.   return unbind_to (count, result);
  1649. }
  1650.  
  1651.  
  1652. DEFUN ("put-ea", Fput_ea, Sput_ea, 3, 3, 0,
  1653.   "Attach an extended attribute to the file or directory PATH.\n\
  1654. NAME is the name of the extended attribute.  DATA is a vector\n\
  1655. [FLAGS TYPE SIZE VALUE] containing the value, where FLAGS is the\n\
  1656. flags byte of the extended attribute, TYPE identifies the type of\n\
  1657. the extended attribute (`ascii', `binary', `bitmap', `metafile',\n\
  1658. `icon', `ea', `mvmt', `mvst', `asn1', or a number), SIZE is the\n\
  1659. size the value and VALUE is the (binary) value of the extended\n\
  1660. attribute, including the type and size fields.  If TYPE is non-nil,\n\
  1661. it will override the type field of VALUE.  If SIZE is non-nil, it\n\
  1662. will override the size field of VALUE.")
  1663.   (path, name, value)
  1664.      Lisp_Object path, name, value;
  1665. {
  1666.   struct _ea ea;
  1667.  
  1668.   CHECK_STRING (path, 0);
  1669.   CHECK_STRING (name, 1);
  1670.   CHECK_VECTOR (value, 2);
  1671.   path = Fexpand_file_name (path, Qnil);
  1672.  
  1673.   ea_from_lisp (&ea, value);
  1674.   if (_ea_put (&ea, XSTRING (path)->data, 0, XSTRING (name)->data) != 0)
  1675.     {
  1676.       xfree (ea.value);
  1677.       report_file_error ("Attaching extended attribute", Fcons (path, Qnil));
  1678.     }
  1679.   xfree (ea.value);
  1680.   return Qnil;
  1681. }
  1682.  
  1683.  
  1684. DEFUN ("put-ea-list", Fput_ea_list, Sput_ea_list, 2, 4, 0,
  1685.   "Attach a list of extended attributes to the file or directory PATH.\n\
  1686. EAS is a list of elements (NAME . VECTOR), one element for each\n\
  1687. extended attribute.  NAME is the name of the extended attribute,\n\
  1688. VECTOR is a vector [FLAGS TYPE SIZE VALUE], where FLAGS is the flags\n\
  1689. byte of the extended attribute, TYPE identifies the type of the extended\n\
  1690. attribute (`ascii', `binary', `bitmap', `metafile', `icon', `ea', `mvmt',\n\
  1691. `mvst', `asn1', or a number), SIZE is the size the value and VALUE is\n\
  1692. the (binary) value of the extended attribute, including the type and size\n\
  1693. fields.  If TYPE is non-nil, it will override the type field of VALUE.\n\
  1694. If SIZE is non-nil, it will override the size field of VALUE.\n\
  1695. If the optional third argument REPLACE is non-nil, the existing extended\n\
  1696. attributes of the file or directory are removed before attaching the\n\
  1697. new extended attributes.\n\
  1698. If the optional fourth argument FIXLONGNAME is non-nil, the .LONGNAME\n\
  1699. extended attribute will be removed if it doesn't match the file name.")
  1700.   (path, eas, replace, fixlongname)
  1701.      Lisp_Object path, eas, replace, fixlongname;
  1702. {
  1703.   _ead ead;
  1704.   struct _ea ea;
  1705.   Lisp_Object elt, name, vec;
  1706.   int rc;
  1707.   int count = specpdl_ptr - specpdl;
  1708.  
  1709.   CHECK_STRING (path, 0);
  1710.   path = Fexpand_file_name (path, Qnil);
  1711.  
  1712.   ead = _ead_create ();
  1713.   if (!ead)
  1714.     report_file_error ("Attaching extended attributes", Fcons (path, Qnil));
  1715.  
  1716.   ead_record_unwind_protect (ead);
  1717.  
  1718.   while (CONSP (eas))
  1719.     {
  1720.       elt = XCONS (eas)->car;
  1721.       CHECK_CONS (elt, 1);
  1722.       name = XCONS (elt)->car;
  1723.       vec = XCONS (elt)->cdr;
  1724.       CHECK_STRING (name, 1);
  1725.       CHECK_VECTOR (vec, 1);
  1726.       ea_from_lisp (&ea, vec);
  1727.       /* When using emx 0.8h, this requires the C library of emxfix07.  */
  1728.       rc = _ead_add (ead, XSTRING (name)->data, ea.flags, ea.value, ea.size);
  1729.       xfree (ea.value);
  1730.       if (rc < 0)
  1731.         report_file_error ("Attaching extended attributes",
  1732.                            Fcons (path, Qnil));
  1733.       eas = XCONS (eas)->cdr;
  1734.     }
  1735.   if (!NILP (eas))
  1736.     CHECK_CONS (eas, 1);
  1737.  
  1738.   if (!NILP (fixlongname))
  1739.     emx_fix_ead_longname (ead, XSTRING (path)->data);
  1740.  
  1741.   if (_ead_write (ead, XSTRING (path)->data, 0,
  1742.                   NILP (replace) ? _EAD_MERGE : 0) != 0)
  1743.     report_file_error ("Attaching extended attributes", Fcons (path, Qnil));
  1744.  
  1745.   if (NILP (replace) && !NILP (fixlongname))
  1746.     emx_fix_longname (XSTRING (path)->data);
  1747.  
  1748.   return unbind_to (count, Qnil);
  1749. }
  1750.  
  1751.  
  1752. DEFUN ("put-ea-string", Fput_ea_string, Sput_ea_string, 3, 3, 0,
  1753.   "Attach an extended attribute (string) to the file or directory PATH.\n\
  1754. NAME is the name of the extended attribute.  VALUE is a string\n\
  1755. containing the value.  The flags byte of the extended attribute is\n\
  1756. set to 0, the type is set to `ascii', and the size is computed from\n\
  1757. the length of the string.")
  1758.   (path, name, value)
  1759.      Lisp_Object path, name, value;
  1760. {
  1761.   struct _ea ea;
  1762.   char *str;
  1763.   int size;
  1764.  
  1765.   CHECK_STRING (path, 0);
  1766.   CHECK_STRING (name, 1);
  1767.   CHECK_STRING (value, 2);
  1768.   path = Fexpand_file_name (path, Qnil);
  1769.  
  1770.   size = XSTRING (value)->size;
  1771.   if (size > 65535 - 4)
  1772.     error ("Value of extended attribute too long");
  1773.   str = alloca (4 + size);
  1774.   bcopy (XSTRING (value)->data, str + 4, size);
  1775.   ((USHORT *)str)[0] = EAT_ASCII;
  1776.   ((USHORT *)str)[1] = (USHORT)size;
  1777.   ea.flags = 0;
  1778.   ea.size = 4 + size;
  1779.   ea.value = str;
  1780.   if (_ea_put (&ea, XSTRING (path)->data, 0, XSTRING (name)->data) != 0)
  1781.     report_file_error ("Attaching extended attribute", Fcons (path, Qnil));
  1782.   return Qnil;
  1783. }
  1784.  
  1785.  
  1786. DEFUN ("remove-ea", Fremove_ea, Sremove_ea, 2, 2, 0,
  1787.   "Remove extended attribute NAME from the file or directory PATH.")
  1788.   (path, name)
  1789.      Lisp_Object path, name;
  1790. {
  1791.   CHECK_STRING (path, 0);
  1792.   CHECK_STRING (name, 1);
  1793.   path = Fexpand_file_name (path, Qnil);
  1794.  
  1795.   if (_ea_remove (XSTRING (path)->data, 0, XSTRING (name)->data) != 0)
  1796.     report_file_error ("Removing extended attribute", Fcons (path, Qnil));
  1797.   return Qnil;
  1798. }
  1799.  
  1800.  
  1801. DEFUN ("remove-all-eas", Fremove_all_eas, Sremove_all_eas, 1, 1, 0,
  1802.   "Remove all extended attribute from the file or directory PATH.")
  1803.   (path)
  1804.      Lisp_Object path;
  1805. {
  1806.   _ead ead;
  1807.   int rc;
  1808.  
  1809.   CHECK_STRING (path, 0);
  1810.   path = Fexpand_file_name (path, Qnil);
  1811.  
  1812.   ead = _ead_create ();
  1813.   if (!ead)
  1814.     report_file_error ("Removing all extended attributes", Fcons (path, Qnil));
  1815.   rc = _ead_write (ead, XSTRING (path)->data, 0, 0);
  1816.   _ead_destroy (ead);
  1817.   if (rc != 0)
  1818.     report_file_error ("Removing all extended attributes", Fcons (path, Qnil));
  1819.   return Qnil;
  1820. }
  1821.  
  1822.  
  1823. void emx_resize_vio (int *pheight, int *pwidth)
  1824. {
  1825.   int w, h;
  1826.   VIOMODEINFO vmi;
  1827.   ULONG rc;
  1828.  
  1829.   if (_osmode == DOS_MODE) return;
  1830.   h = *pheight; w = *pwidth;
  1831.   if (w <= 40)
  1832.     w = 40;
  1833.   else if (w <= 80)
  1834.     w = 80;
  1835.   else
  1836.     w = 132;
  1837.   if (h <= 2)
  1838.     h = 2;
  1839.  
  1840.   vmi.cb = sizeof (vmi);
  1841.   rc = VioGetMode (&vmi, 0);
  1842.   if (rc != 0) return;
  1843.   vmi.col = w;
  1844.   vmi.row = h;
  1845.   rc = VioSetMode (&vmi, 0);
  1846.   if (rc != 0) return;
  1847.   rc = VioGetMode (&vmi, 0);
  1848.   if (rc != 0) return;
  1849.   *pwidth = vmi.col;
  1850.   *pheight = vmi.row;
  1851. }
  1852.  
  1853.  
  1854. void
  1855. syms_of_emxdep ()
  1856. {
  1857.   Qascii = intern ("ascii");
  1858.   staticpro (&Qascii);
  1859.   Qbinary = intern ("binary");
  1860.   staticpro (&Qbinary);
  1861.   Qbinary_process_input = intern ("binary-process-input");
  1862.   staticpro (&Qbinary_process_input);
  1863.   Qbitmap = intern ("bitmap");
  1864.   staticpro (&Qbitmap);
  1865.   Qmetafile = intern ("metafile");
  1866.   staticpro (&Qmetafile);
  1867.   Qea = intern ("ea");
  1868.   staticpro (&Qea);
  1869.   Qmvmt = intern ("mvmt");
  1870.   staticpro (&Qmvmt);
  1871.   Qmvst = intern ("mvst");
  1872.   staticpro (&Qmvst);
  1873.   Qasn1 = intern ("asn1");
  1874.   staticpro (&Qasn1);
  1875.   Qemx_case_map = intern ("emx-case-map");
  1876.   staticpro (&Qemx_case_map);
  1877.   Qpreserve = intern ("preserve");
  1878.   staticpro (&Qpreserve);
  1879.  
  1880.   DEFVAR_LISP ("emx-binary-mode-list", &Vemx_binary_mode_list,
  1881.     "*List of regular expressions.  Binary mode is used for files\n\
  1882. matching one of the regular expressions.");
  1883.   Vemx_binary_mode_list = Qnil;
  1884.  
  1885.   DEFVAR_BOOL ("binary-process-input", &binary_process_input,
  1886.     "*Non-nil means write process input in binary mode.\n\
  1887. nil means use file type of the buffer for writing process input.\n\
  1888. binary-process-input is examined by call-process and start-process.");
  1889.   binary_process_input = 0;
  1890.  
  1891.   DEFVAR_BOOL ("binary-process-output", &binary_process_output,
  1892.     "*Non-nil means read process output in binary mode.\n\
  1893. nil means use file type of the buffer for reading process output.\n\
  1894. binary-process-output is examined by call-process and start-process.");
  1895.   binary_process_output = 0;
  1896.  
  1897.   DEFVAR_BOOL ("binary-async-process-input", &binary_async_process_input,
  1898.     "*Default value for `binary-process-input' if `binary-process-input'\n\
  1899. is nil; used for `start-process' only.");
  1900.   binary_async_process_input = 1;
  1901.  
  1902.   DEFVAR_LISP ("emx-system-type", &Vemx_system_type,
  1903.     "Underlying operating system (or program loader).\n\
  1904. Possible values are `os2' and `ms-dos'.");
  1905.   Vemx_system_type = intern (_osmode == OS2_MODE ? "os2" : "ms-dos");
  1906.  
  1907.   DEFVAR_LISP ("process-priority", &Vprocess_priority,
  1908.     "Priority to be assigned to child processes.\n\
  1909. Possible values are nil (child processes inherit the priority of Emacs),\n\
  1910. a priority class, or (PCLASS . PLEVEL), where PCLASS is a priority\n\
  1911. class and PLEVEL is a priority level.  Available priority classes are:\n\
  1912.   idle-time (idle-time priority class),\n\
  1913.   regular (regular priority class -- this is the OS/2 default,\n\
  1914.     a priority boost is applied if the process is in the foreground), and\n\
  1915.   foreground-server (fixed-high priority class).\n\
  1916. If PLEVEL is omitted or is nil, a priority level of 0 is used.\n\
  1917. Otherwise, PLEVEL shall be a number between 0 and 31 which indicates\n\
  1918.   the priority level within the priority class.  Level 31 has the highest\n\
  1919.   priority in each class, the default value assigned by OS/2 is 0.");
  1920.   Vprocess_priority = Qnil;
  1921.  
  1922.   DEFVAR_LISP ("case-fold-file-names", &Vcase_fold_file_names,
  1923.     "*`t' if all file names should be translated to lower case.\n\
  1924. `nil' if file names should not be translated to lower case.\n\
  1925. `preserve' if `expand-file-name' should use text attributes to preserve\n\
  1926. case in file names translated to lower case.  This feature is experimental.");
  1927.   Vcase_fold_file_names = Qt;
  1928.  
  1929.   DEFVAR_LISP ("program-name-handler-alist", &Vprogram_name_handler_alist,
  1930.     "*Alist of elements (REGEXP . HANDLER) for programs handled specially.\n\
  1931. If a file name matches REGEXP, then HANDLER is called to take over the work\n\
  1932. of call-process and start-process.\n\
  1933. \n\
  1934. The first argument given to HANDLER is a list containing the name of the\n\
  1935. primitive to be handled and the arguments of the primitive.\n\
  1936. For call-process, the list is (call-process INFILE BUFFER DISPLAY);\n\
  1937. for start-process, the list is (start-process NAME BUFFER).  The second\n\
  1938. argument passed to HANDLER is the file name of the program; the third\n\
  1939. argument is a list of the program arguments.  For example, if you do\n\
  1940.     (call-process PROGRAM \"infile\" t nil ARG1 ARG2)\n\
  1941. and PROGRAM is handled by HANDLER, then HANDLER is called like this:\n\
  1942.     (funcall HANDLER '(call-process \"infile\" t nil) PROGRAM '(ARG1 ARG2))\n\
  1943. HANDLER may modify the structure of its first and third arguments.");
  1944.   Vprogram_name_handler_alist = Qnil;
  1945.  
  1946.   DEFVAR_LISP ("inhibit-program-name-handlers", &Vinhibit_program_name_handlers,
  1947.     "A list of program name handlers that temporarily should not be used.");
  1948.   Vinhibit_program_name_handlers = Qnil;
  1949.  
  1950.   DEFVAR_BOOL ("emx-start-session", &emx_start_session,
  1951.     "*Non-nil to let `start-process' create separate sessions.\n\
  1952. Nil to let `start-process' run processes in the same session as Emacs.");
  1953.   emx_start_session = 1;
  1954.  
  1955.   DEFVAR_BOOL ("remote-expand-file-name", &remote_expand_file_name,
  1956.     "Non-nil to prevent `expand-file-name' from prepending a drive letter.");
  1957.   remote_expand_file_name = 0;
  1958.  
  1959.   DEFVAR_BOOL ("emx-kill-tree", &emx_kill_tree,
  1960.     "Non-nil if signals for process groups should be sent to process tree.\n\
  1961. Nil to send signals to the target process only.");
  1962.   emx_kill_tree = 1;
  1963.  
  1964.   defsubr (&Scurrent_code_page);
  1965.   defsubr (&Slist_code_pages);
  1966.   defsubr (&Sset_code_page);
  1967.   defsubr (&Sset_cursor_size);
  1968.   defsubr (&Sget_cursor_size);
  1969.   defsubr (&Semacs_priority);
  1970.   defsubr (&Sfilesystem_type);
  1971.   defsubr (&Skeyboard_type);
  1972.   defsubr (&Sfile_name_valid_p);
  1973.   defsubr (&Sremove_from_window_list);
  1974. #ifdef HAVE_PM
  1975.   defsubr (&Spm_session_bond);
  1976. #endif /* HAVE_PM */
  1977.   defsubr (&Semx_binary_mode_p);
  1978.   defsubr (&Sget_ea);
  1979.   defsubr (&Sget_ea_string);
  1980.   defsubr (&Sget_ea_list);
  1981.   defsubr (&Sput_ea);
  1982.   defsubr (&Sput_ea_string);
  1983.   defsubr (&Sput_ea_list);
  1984.   defsubr (&Sremove_ea);
  1985.   defsubr (&Sremove_all_eas);
  1986.   defsubr (&Semx_original_file_name);
  1987. }
  1988.