1 /* MS-DOS specific Lisp utilities. Coded by Manabu Higashida, 1991.
2 Major changes May-July 1993 Morten Welinder (only 10% original code left)
3 Copyright (C) 1991, 1993 Free Software Foundation, Inc.
5 This file is part of GNU Emacs.
7 GNU Emacs is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 2, or (at your option)
12 GNU Emacs is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with GNU Emacs; see the file COPYING. If not, write to
19 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
20 Boston, MA 02111-1307, USA. */
26 /* The entire file is within this conditional */
33 #include "termhooks.h"
39 DEFUN ("mode25", Fmode25
, Smode25
, 0, 0, "", "\
40 Changes the number of rows to 25.")
46 if (!inhibit_window_system
)
51 int86 (0x10, ®s
, ®s
);
54 int86 (0x10, ®s
, ®s
);
57 int86 (0x10, ®s
, ®s
);
59 int86 (0x10, ®s
, ®s
);
60 Fset_frame_size (Fselected_frame (), ScreenCols (), ScreenRows ());
63 if (have_mouse
) mouse_init ();
67 DEFUN ("mode4350", Fmode4350
, Smode4350
, 0, 0, "", "\
68 Changes the number of rows to 43 (EGA) or 50 (VGA).")
74 if (!inhibit_window_system
)
79 int86 (0x10, ®s
, ®s
);
82 int86 (0x10, ®s
, ®s
);
85 int86 (0x10, ®s
, ®s
);
88 int86 (0x10, ®s
, ®s
);
89 Fset_frame_size (Fselected_frame (), ScreenCols (), ScreenRows ());
92 if (have_mouse
) mouse_init ();
96 DEFUN ("int86", Fint86
, Sint86
, 2, 2, 0,
97 "Call specific MSDOS interrupt number INTERRUPT with REGISTERS.\n\
98 Return the updated REGISTER vector.\n\
100 INTERRUPT should be an integer in the range 0 to 255.\n\
101 REGISTERS should be a vector produced by `make-register' and\n\
102 `set-register-value'.")
103 (interrupt
, registers
)
104 Lisp_Object interrupt
, registers
;
108 union REGS inregs
, outregs
;
111 CHECK_NUMBER (interrupt
, 0);
112 no
= (unsigned long) XINT (interrupt
);
113 CHECK_VECTOR (registers
, 1);
114 if (no
< 0 || no
> 0xff || XVECTOR (registers
)-> size
!= 8)
116 for (i
= 0; i
< 8; i
++)
117 CHECK_NUMBER (XVECTOR (registers
)->contents
[i
], 1);
119 inregs
.x
.ax
= (unsigned long) XFASTINT (XVECTOR (registers
)->contents
[0]);
120 inregs
.x
.bx
= (unsigned long) XFASTINT (XVECTOR (registers
)->contents
[1]);
121 inregs
.x
.cx
= (unsigned long) XFASTINT (XVECTOR (registers
)->contents
[2]);
122 inregs
.x
.dx
= (unsigned long) XFASTINT (XVECTOR (registers
)->contents
[3]);
123 inregs
.x
.si
= (unsigned long) XFASTINT (XVECTOR (registers
)->contents
[4]);
124 inregs
.x
.di
= (unsigned long) XFASTINT (XVECTOR (registers
)->contents
[5]);
125 inregs
.x
.cflag
= (unsigned long) XFASTINT (XVECTOR (registers
)->contents
[6]);
126 inregs
.x
.flags
= (unsigned long) XFASTINT (XVECTOR (registers
)->contents
[7]);
128 int86 (no
, &inregs
, &outregs
);
130 XVECTOR (registers
)->contents
[0] = make_number (outregs
.x
.ax
);
131 XVECTOR (registers
)->contents
[1] = make_number (outregs
.x
.bx
);
132 XVECTOR (registers
)->contents
[2] = make_number (outregs
.x
.cx
);
133 XVECTOR (registers
)->contents
[3] = make_number (outregs
.x
.dx
);
134 XVECTOR (registers
)->contents
[4] = make_number (outregs
.x
.si
);
135 XVECTOR (registers
)->contents
[5] = make_number (outregs
.x
.di
);
136 XVECTOR (registers
)->contents
[6] = make_number (outregs
.x
.cflag
);
137 XVECTOR (registers
)->contents
[7] = make_number (outregs
.x
.flags
);
142 DEFUN ("msdos-memget", Fdos_memget
, Sdos_memget
, 2, 2, 0,
143 "Read DOS memory at offset ADDRESS into VECTOR.\n\
144 Return the updated VECTOR.")
146 Lisp_Object address
, vector
;
153 CHECK_NUMBER (address
, 0);
154 offs
= (unsigned long) XINT (address
);
155 CHECK_VECTOR (vector
, 1);
156 len
= XVECTOR (vector
)-> size
;
157 if (len
< 1 || len
> 2048 || address
< 0 || address
> 0xfffff - len
)
160 dosmemget (offs
, len
, buf
);
162 for (i
= 0; i
< len
; i
++)
163 XVECTOR (vector
)->contents
[i
] = make_number (buf
[i
]);
168 DEFUN ("msdos-memput", Fdos_memput
, Sdos_memput
, 2, 2, 0,
169 "Write DOS memory at offset ADDRESS from VECTOR.")
171 Lisp_Object address
, vector
;
178 CHECK_NUMBER (address
, 0);
179 offs
= (unsigned long) XINT (address
);
180 CHECK_VECTOR (vector
, 1);
181 len
= XVECTOR (vector
)-> size
;
182 if (len
< 1 || len
> 2048 || address
< 0 || address
> 0xfffff - len
)
186 for (i
= 0; i
< len
; i
++)
188 CHECK_NUMBER (XVECTOR (vector
)->contents
[i
], 1);
189 buf
[i
] = (unsigned char) XFASTINT (XVECTOR (vector
)->contents
[i
]) & 0xFF;
192 dosmemput (buf
, len
, offs
);
196 DEFUN ("msdos-set-keyboard", Fmsdos_set_keyboard
, Smsdos_set_keyboard
, 1, 2, 0,
197 "Set keyboard layout according to COUNTRY-CODE.\n\
198 If the optional argument ALLKEYS is non-nil, the keyboard is mapped for\n\
199 all keys; otherwise it is only used when the ALT key is pressed.\n\
200 The current keyboard layout is available in dos-keyboard-code.")
201 (country_code
, allkeys
)
202 Lisp_Object country_code
;
204 CHECK_NUMBER (country_code
, 0);
205 if (!dos_set_keyboard (XINT (country_code
), !NILP (allkeys
)))
210 #ifndef HAVE_X_WINDOWS
211 /* Later we might want to control the mouse interface with this function,
212 e.g., with respect to non-80 column screen modes. */
214 DEFUN ("msdos-mouse-p", Fmsdos_mouse_p
, Smsdos_mouse_p
, 0, 0, 0, "\
215 Report whether a mouse is present.")
224 DEFUN ("set-mouse-position", Fset_mouse_position
, Sset_mouse_position
, 3, 3, 0,
225 "Move the mouse pointer to the center of character cell (X,Y) in FRAME.\n\
226 WARNING: If you use this under X windows,\n\
227 you should call `unfocus-frame' afterwards.")
229 Lisp_Object frame
, x
, y
;
231 mouse_moveto (XINT (x
), XINT (y
));
235 /* Function to translate colour names to integers. See lisp/term/pc-win.el
236 for its definition. */
238 Lisp_Object Qmsdos_color_translate
;
242 DEFUN ("msdos-mouse-init", Fmsdos_mouse_init
, Smsdos_mouse_init
, 0, 0, "",
243 "Initialize and enable mouse if available.")
255 DEFUN ("msdos-mouse-enable", Fmsdos_mouse_enable
, Smsdos_mouse_enable
, 0, 0, "",
256 "Enable mouse if available.")
264 return have_mouse
? Qt
: Qnil
;
267 DEFUN ("msdos-mouse-disable", Fmsdos_mouse_disable
, Smsdos_mouse_disable
, 0, 0, "",
268 "Disable mouse if available.")
272 if (have_mouse
) have_mouse
= -1;
276 DEFUN ("insert-startup-screen", Finsert_startup_screen
, Sinsert_startup_screen
, 0, 0, "", "\
277 Insert copy of screen contents prior to starting emacs.\n\
278 Return nil if startup screen is not available.")
285 if (!dos_get_saved_screen (&s
, &rows
, &cols
))
288 for (i
= 0; i
< rows
; i
++)
290 for (j
= 0; j
< cols
; j
++)
295 insert_char ('\n', 1);
303 int dos_country_code
;
305 int dos_timezone_offset
;
306 int dos_decimal_point
;
307 int dos_keyboard_layout
;
308 unsigned char dos_country_info
[DOS_COUNTRY_INFO
];
314 Lisp_Object Vdos_version
;
315 Lisp_Object Vdos_display_scancodes
;
321 _go32_dpmi_seginfo info
;
322 _go32_dpmi_registers dpmiregs
;
324 #ifndef SYSTEM_MALLOC
325 get_lim_data (); /* why the hell isn't this called elsewhere? */
329 intdos (®s
, ®s
);
330 Vdos_version
= Fcons (make_number (regs
.h
.al
), make_number (regs
.h
.ah
));
332 /* Obtain the country code by calling Dos via Dpmi. Don't rely on GO32. */
333 info
.size
= (sizeof(dos_country_info
) + 15) / 16;
334 if (_go32_dpmi_allocate_dos_memory (&info
))
335 dos_country_code
= 1;
338 dpmiregs
.x
.ax
= 0x3800;
339 dpmiregs
.x
.ds
= info
.rm_segment
;
341 dpmiregs
.x
.ss
= dpmiregs
.x
.sp
= 0;
342 _go32_dpmi_simulate_int (0x21, &dpmiregs
);
343 dos_country_code
= dpmiregs
.x
.bx
;
344 dosmemget (info
.rm_segment
* 16, DOS_COUNTRY_INFO
, dos_country_info
);
345 _go32_dpmi_free_dos_memory (&info
);
347 dos_set_keyboard (dos_country_code
, 0);
350 intdos (®s
, ®s
);
352 /* Estimate code page from country code */
353 switch (dos_country_code
)
355 case 45: /* Denmark */
356 case 47: /* Norway */
364 dos_codepage
= regs
.x
.bx
& 0xffff;
373 defsubr (&Smode4350
);
375 defsubr (&Sdos_memget
);
376 defsubr (&Sdos_memput
);
377 defsubr (&Smsdos_mouse_init
);
378 defsubr (&Smsdos_mouse_enable
);
379 defsubr (&Smsdos_set_keyboard
);
380 defsubr (&Sinsert_startup_screen
);
381 defsubr (&Smsdos_mouse_disable
);
382 #ifndef HAVE_X_WINDOWS
383 defsubr (&Smsdos_mouse_p
);
384 defsubr (&Sset_mouse_position
);
386 Qmsdos_color_translate
= intern ("msdos-color-translate");
387 staticpro (&Qmsdos_color_translate
);
390 DEFVAR_INT ("dos-country-code", &dos_country_code
,
391 "The country code returned by Dos when Emacs was started.\n\
392 Usually this is the international telephone prefix.");
394 DEFVAR_INT ("dos-codepage", &dos_codepage
,
395 "The codepage active when Emacs was started.\n\
396 The following are known:\n\
398 850 Multilingual (Latin I)\n\
399 852 Slavic (Latin II)\n\
403 863 Canada (French)\n\
404 865 Norway/Denmark");
406 DEFVAR_INT ("dos-timezone-offset", &dos_timezone_offset
,
407 "The current timezone offset to UTC in minutes.
408 Implicitly modified when the TZ variable is changed.");
410 DEFVAR_LISP ("dos-version", &Vdos_version
,
411 "The (MAJOR . MINOR) Dos version (subject to modification with setver).");
413 DEFVAR_LISP ("dos-display-scancodes", &Vdos_display_scancodes
,
414 "*When non-nil, the keyboard scan-codes are displayed at the bottom right\n\
415 corner of the display (typically at the end of the mode line).\n\
416 The output format is: scan code:char code*modifiers.");
417 Vdos_display_scancodes
= Qnil
;
419 DEFVAR_INT ("dos-hyper-key", &dos_hyper_key
,
420 "*If set to 1, use right ALT key as hyper key.\n\
421 If set to 2, use right CTRL key as hyper key.");
424 DEFVAR_INT ("dos-super-key", &dos_super_key
,
425 "*If set to 1, use right ALT key as super key.\n\
426 If set to 2, use right CTRL key as super key.");
429 DEFVAR_INT ("dos-keypad-mode", &dos_keypad_mode
,
430 "*Controls what key code is returned by a key in the numeric keypad.\n\
431 The `numlock ON' action is only taken if no modifier keys are pressed.\n\
432 The value is an integer constructed by adding the following bits together:\n\
434 0x00 Digit key returns digit (if numlock ON)\n\
435 0x01 Digit key returns kp-digit (if numlock ON)\n\
436 0x02 Digit key returns M-digit (if numlock ON)\n\
437 0x03 Digit key returns edit key (if numlock ON)\n\
439 0x00 Grey key returns char (if numlock ON)\n\
440 0x04 Grey key returns kp-key (if numlock ON)\n\
442 0x00 Digit key returns digit (if numlock OFF)\n\
443 0x10 Digit key returns kp-digit (if numlock OFF)\n\
444 0x20 Digit key returns M-digit (if numlock OFF)\n\
445 0x30 Digit key returns edit key (if numlock OFF)\n\
447 0x00 Grey key returns char (if numlock OFF)\n\
448 0x40 Grey key returns kp-key (if numlock OFF)\n\
450 0x200 ALT-0..ALT-9 in top-row produces shifted codes.");
451 dos_keypad_mode
= 0x75;
453 DEFVAR_INT ("dos-keyboard-layout", &dos_keyboard_layout
,
454 "Contains the country code for the current keyboard layout.\n\
455 Use msdos-set-keyboard to select another keyboard layout.");
456 dos_keyboard_layout
= 1; /* US */
458 DEFVAR_INT ("dos-decimal-point", &dos_decimal_point
,
459 "If non-zero, it contains the character to be returned when the\n\
460 decimal point key in the numeric keypad is pressed when Num Lock is on.\n\
461 If zero, the decimal point key returns the country code specific value.");
462 dos_decimal_point
= 0;