/* vi:set ts=8 sts=4 sw=4: * * VIM - Vi IMproved by Bram Moolenaar * * Do ":help uganda" in Vim to read copying and usage conditions. * Do ":help credits" in Vim to see a list of people who contributed. * See README.txt for an overview of the Vim source code. */ /* * os_msdos.c * * MSDOS system-dependent routines. * A cheap plastic imitation of the amiga dependent code. * A lot in this file was made by Juergen Weigert (jw). * * DJGPP changes by Gert van Antwerpen * Faster text screens by John Lange (jlange@zilker.net) * Windows clipboard functionality added by David Kotchan (dk) * * Some functions are also used for Win16 (MS-Windows 3.1). */ #include #include "vim.h" #include #ifdef HAVE_FCNTL_H # include #endif /* * MS-DOS only code, not used for Win16. */ #ifndef WIN16 #include #ifdef DJGPP # include # include # include # include # ifdef FEAT_CLIPBOARD # include # endif #else # include #endif #if defined(DJGPP) || defined(PROTO) # define _cdecl /* DJGPP doesn't have this */ #endif static int cbrk_pressed = FALSE; /* set by ctrl-break interrupt */ static int ctrlc_pressed = FALSE; /* set when ctrl-C or ctrl-break detected */ static int delayed_redraw = FALSE; /* set when ctrl-C detected */ static int bioskey_read = _NKEYBRD_READ; /* bioskey() argument: read key */ static int bioskey_ready = _NKEYBRD_READY; /* bioskey() argument: key ready? */ #ifdef FEAT_MOUSE static int mouse_avail = FALSE; /* mouse present */ static int mouse_active; /* mouse enabled */ static int mouse_hidden; /* mouse not shown */ static int mouse_click = -1; /* mouse status */ static int mouse_last_click = -1; /* previous status at click */ static int mouse_x = -1; /* mouse x coodinate */ static int mouse_y = -1; /* mouse y coodinate */ static long mouse_click_time = 0; /* biostime() of last click */ static int mouse_click_count = 0; /* count for multi-clicks */ static int mouse_click_x = 0; /* x of previous mouse click */ static int mouse_click_y = 0; /* y of previous mouse click */ static linenr_T mouse_topline = 0; /* w_topline at previous mouse click */ #ifdef FEAT_DIFF static int mouse_topfill = 0; /* w_topfill at previous mouse click */ #endif static int mouse_x_div = 8; /* column = x coord / mouse_x_div */ static int mouse_y_div = 8; /* line = y coord / mouse_y_div */ #endif #define BIOSTICK 55 /* biostime() increases one tick about every 55 msec */ static int orig_attr = 0x0700; /* video attributes when starting */ static int S_iLeft = 0; /* Scroll window; these are 1 offset */ static int S_iTop = 0; static int S_iRight = 0; static int S_iBottom = 0; /* * Need to remember the values, because we set horizontal and vertical * edges separately. */ static void mywindow(int iLeft, int iTop, int iRight, int iBottom) { S_iLeft = iLeft; S_iTop = iTop; S_iRight = iRight; S_iBottom = iBottom; window(iLeft, iTop, iRight, iBottom); } #ifdef DJGPP /* * For DJGPP, use our own functions for fast text screens. JML 1/18/98 */ unsigned long S_ulScreenBase = 0xb8000; unsigned short S_uiAttribute = 0; int S_iCurrentRow = 0; /* These are 0 offset */ int S_iCurrentColumn = 0; short S_selVideo; /* Selector for DJGPP direct video transfers */ /* * Use burst writes to improve mch_write speed - VJN 01/10/99 */ unsigned short S_linebuffer[8000]; /* enough for 160x50 */ unsigned short S_blankbuffer[256]; /* max length of console line */ unsigned short *S_linebufferpos = S_linebuffer; int S_iBufferRow; int S_iBufferColumn; static void myflush(void) { if (S_linebufferpos != S_linebuffer) { _dosmemputw(S_linebuffer, (S_linebufferpos - S_linebuffer), S_ulScreenBase + S_iBufferRow * (Columns << 1) + (S_iBufferColumn << 1)); S_linebufferpos = S_linebuffer; } } static void mygotoxy(int x, int y) { S_iCurrentRow = y - 1; S_iCurrentColumn = x - 1; } /* * Set the system cursor to our cursor position. */ static void set_sys_cursor(void) { if (term_console && full_screen) { myflush(); gotoxy(S_iCurrentColumn + 1, S_iCurrentRow + 1); } } static void setblankbuffer(unsigned short uiValue) { int i; static unsigned short olduiValue = 0; if (olduiValue != uiValue) { /* Load blank line buffer with spaces */ for (i = 0; i < Columns; ++i) S_blankbuffer[i] = uiValue; olduiValue = uiValue; } } static void myclreol(void) { /* Clear to end of line */ setblankbuffer(S_uiAttribute | ' '); _dosmemputw(S_blankbuffer, S_iRight - S_iCurrentColumn, S_ulScreenBase + (S_iCurrentRow) * (Columns << 1) + (S_iCurrentColumn << 1)); } static void myclrscr(void) { /* Clear whole screen */ short iColumn; int endpoint = (Rows * Columns) << 1; setblankbuffer(S_uiAttribute | ' '); for (iColumn = 0; iColumn < endpoint; iColumn += (Columns << 1)) _dosmemputw(S_blankbuffer, Columns, S_ulScreenBase + iColumn); } static void mydelline(void) { short iRow, iColumn; iColumn = (S_iLeft - 1) << 1; /* Copy the lines underneath */ for (iRow = S_iCurrentRow; iRow < S_iBottom - 1; iRow++) movedata(S_selVideo, (((iRow + 1) * Columns) << 1) + iColumn, S_selVideo, ((iRow * Columns) << 1) + iColumn, (S_iRight - S_iLeft + 1) << 1); /* Clear the new row */ setblankbuffer(S_uiAttribute | ' '); _dosmemputw(S_blankbuffer, (S_iRight - S_iLeft) + 1, S_ulScreenBase + (S_iBottom - 1) * (Columns << 1) + iColumn); } static void myinsline(void) { short iRow, iColumn; iColumn = (S_iLeft - 1) << 1; /* Copy the lines underneath */ for (iRow = S_iBottom - 1; iRow >= S_iTop; iRow--) movedata(S_selVideo, (((iRow - 1) * Columns) << 1) + iColumn, S_selVideo, ((iRow * Columns) << 1) + iColumn, (S_iRight - S_iLeft + 1) << 1); /* Clear the new row */ setblankbuffer(S_uiAttribute | ' '); _dosmemputw(S_blankbuffer, (S_iRight - S_iLeft) + 1, S_ulScreenBase + (S_iTop - 1) * (Columns << 1) + iColumn); } /* * Scroll the screen one line up, clear the last line. */ static void myscroll(void) { short iRow, iColumn; iColumn = (S_iLeft - 1) << 1; /* Copy the screen */ for (iRow = S_iTop; iRow < S_iBottom; iRow++) movedata(S_selVideo, ((iRow * Columns) << 1) + iColumn, S_selVideo, (((iRow - 1) * Columns) << 1) + iColumn, (S_iRight - S_iLeft + 1) << 1); /* Clear the bottom row */ setblankbuffer(S_uiAttribute | ' '); _dosmemputw(S_blankbuffer, (S_iRight - S_iLeft) + 1, S_ulScreenBase + (S_iBottom - 1) * (Columns << 1) + iColumn); } static int myputch(int iChar) { unsigned short uiValue; if (iChar == '\n') { myflush(); if (S_iCurrentRow >= S_iBottom - S_iTop) myscroll(); else { S_iCurrentColumn = S_iLeft - 1; S_iCurrentRow++; } } else if (iChar == '\r') { myflush(); S_iCurrentColumn = S_iLeft - 1; } else if (iChar == '\b') { myflush(); if (S_iCurrentColumn >= S_iLeft) S_iCurrentColumn--; } else if (iChar == 7) { sound(440); /* short beep */ delay(200); nosound(); } else { uiValue = S_uiAttribute | (unsigned char)iChar; /* * Normal char - are we starting to buffer? */ if (S_linebufferpos == S_linebuffer) { S_iBufferColumn = S_iCurrentColumn; S_iBufferRow = S_iCurrentRow; } *S_linebufferpos++ = uiValue; S_iCurrentColumn++; if (S_iCurrentColumn >= S_iRight && S_iCurrentRow >= S_iBottom - S_iTop) { myflush(); myscroll(); S_iCurrentColumn = S_iLeft - 1; S_iCurrentRow++; } } return 0; } static void mytextinit(struct text_info *pTextinfo) { S_selVideo = __dpmi_segment_to_descriptor(S_ulScreenBase >> 4); S_uiAttribute = pTextinfo->normattr << 8; } static void get_screenbase(void) { static union REGS regs; /* old Hercules grafic card has different base address (Macewicz) */ regs.h.ah = 0x0f; (void)int86(0x10, ®s, ®s); /* int 10 0f */ if (regs.h.al == 0x07) /* video mode 7 -- hercules mono */ S_ulScreenBase = 0xb0000; else S_ulScreenBase = 0xb8000; } static void mytextattr(int iAttribute) { S_uiAttribute = (unsigned short)iAttribute << 8; } static void mynormvideo(void) { mytextattr(orig_attr); } static void mytextcolor(int iTextColor) { S_uiAttribute = (unsigned short)((S_uiAttribute & 0xf000) | (unsigned short)iTextColor << 8); } static void mytextbackground(int iBkgColor) { S_uiAttribute = (unsigned short)((S_uiAttribute & 0x0f00) | (unsigned short)(iBkgColor << 12)); } /* * Getdigits: Get a number from a string and skip over it. * Note: the argument is a pointer to a char_u pointer! */ static long mygetdigits(pp) char_u **pp; { char_u *p; long retval = 0; p = *pp; if (*p == '-') /* skip negative sign */ ++p; while (VIM_ISDIGIT(*p)) { retval = (retval * 10) + (*p - '0'); ++p; } if (**pp == '-') /* process negative sign */ retval = -retval; *pp = p; return retval; } #else # define mygotoxy gotoxy # define myputch putch # define myscroll scroll # define mynormvideo normvideo # define mytextattr textattr # define mytextcolor textcolor # define mytextbackground textbackground # define mygetdigits getdigits # define myclreol clreol # define myclrscr clrscr # define myinsline insline # define mydelline delline #endif static const struct { char_u scancode; char_u metakey; } altkey_table[] = { {0x1e, 0xe1}, /* a */ {0x30, 0xe2}, /* b */ {0x2e, 0xe3}, /* c */ {0x20, 0xe4}, /* d */ {0x12, 0xe5}, /* e */ {0x21, 0xe6}, /* f */ {0x22, 0xe7}, /* g */ {0x23, 0xe8}, /* h */ {0x17, 0xe9}, /* i */ {0x24, 0xea}, /* j */ {0x25, 0xeb}, /* k */ {0x26, 0xec}, /* l */ {0x32, 0xed}, /* m */ {0x31, 0xee}, /* n */ {0x18, 0xef}, /* o */ {0x19, 0xf0}, /* p */ {0x10, 0xf1}, /* q */ {0x13, 0xf2}, /* r */ {0x1f, 0xf3}, /* s */ {0x14, 0xf4}, /* t */ {0x16, 0xf5}, /* u */ {0x2f, 0xf6}, /* v */ {0x11, 0xf7}, /* w */ {0x2d, 0xf8}, /* x */ {0x15, 0xf9}, /* y */ {0x2c, 0xfa}, /* z */ {0x78, 0xb1}, /* 1 */ {0x79, 0xb2}, /* 2 */ {0x7a, 0xb3}, /* 3 */ {0x7b, 0xb4}, /* 4 */ {0x7c, 0xb5}, /* 5 */ {0x7d, 0xb6}, /* 6 */ {0x7e, 0xb7}, /* 7 */ {0x7f, 0xb8}, /* 8 */ {0x80, 0xb9}, /* 9 */ {0x81, 0xb0}, /* 0 */ }; /* * Translate extended keycodes into meta-chars where applicable */ static int translate_altkeys(int rawkey) { int i, c; if ((rawkey & 0xff) == 0) { c = (rawkey >> 8); for (i = sizeof(altkey_table) / sizeof(altkey_table[0]); --i >= 0; ) { if (c == altkey_table[i].scancode) return (int)altkey_table[i].metakey; } } return rawkey; } /* * Set normal fg/bg color, based on T_ME. Called whem t_me has been set. */ void mch_set_normal_colors() { char_u *p; int n; cterm_normal_fg_color = (orig_attr & 0xf) + 1; cterm_normal_bg_color = ((orig_attr >> 4) & 0xf) + 1; if (T_ME[0] == ESC && T_ME[1] == '|') { p = T_ME + 2; n = getdigits(&p); if (*p == 'm' && n > 0) { cterm_normal_fg_color = (n & 0xf) + 1; cterm_normal_bg_color = ((n >> 4) & 0xf) + 1; } } } #if defined(MCH_CURSOR_SHAPE) || defined(PROTO) /* * Save/restore the shape of the cursor. * call with FALSE to save, TRUE to restore */ static void mch_restore_cursor_shape(int restore) { static union REGS regs; static int saved = FALSE; if (restore) { if (saved) regs.h.ah = 0x01; /*Set Cursor*/ else return; } else { regs.h.ah = 0x03; /*Get Cursor*/ regs.h.bh = 0x00; /*Page */ saved = TRUE; } (void)int86(0x10, ®s, ®s); } /* * Set the shape of the cursor. * 'thickness' can be from 0 (thin) to 7 (block) */ static void mch_set_cursor_shape(int thickness) { union REGS regs; regs.h.ch = 7 - thickness; /*Starting Line*/ regs.h.cl = 7; /*Ending Line*/ regs.h.ah = 0x01; /*Set Cursor*/ (void)int86(0x10, ®s, ®s); } void mch_update_cursor(void) { int idx; int thickness; /* * How the cursor is drawn depends on the current mode. */ idx = get_shape_idx(FALSE); if (shape_table[idx].shape == SHAPE_BLOCK) thickness = 7; else thickness = (7 * shape_table[idx].percentage + 90) / 100; mch_set_cursor_shape(thickness); } #endif /* * Return amount of memory currently available. */ long_u mch_avail_mem(int special) { #ifdef DJGPP return _go32_dpmi_remaining_virtual_memory(); #else return coreleft(); #endif } #ifdef FEAT_MOUSE /* * Set area where mouse can be moved to: The whole screen. * Rows and Columns must be valid when calling! */ static void mouse_area(void) { union REGS regs; if (mouse_avail) { regs.x.cx = 0; /* mouse visible between cx and dx */ regs.x.dx = Columns * mouse_x_div - 1; regs.x.ax = 7; (void)int86(0x33, ®s, ®s); regs.x.cx = 0; /* mouse visible between cx and dx */ regs.x.dx = Rows * mouse_y_div - 1; regs.x.ax = 8; (void)int86(0x33, ®s, ®s); } } static void show_mouse(int on) { static int was_on = FALSE; union REGS regs; if (mouse_avail) { if (!mouse_active || mouse_hidden) on = FALSE; /* * Careful: Each switch on must be compensated by exactly one switch * off */ if ((on && !was_on) || (!on && was_on)) { was_on = on; regs.x.ax = on ? 1 : 2; int86(0x33, ®s, ®s); /* show mouse */ if (on) mouse_area(); } } } #endif /* * Version of kbhit() and getch() that use direct console I/O. * This avoids trouble with CTRL-P and the like, and should work over a telnet * connection (it works for Xvi). */ static int cons_key = -1; /* * Try to get one character directly from the console. * If there is a key, it is stored in cons_key. * Only call when cons_key is -1! */ static void cons_getkey(void) { union REGS regs; /* call DOS function 6: Direct console I/O */ regs.h.ah = 0x06; regs.h.dl = 0xff; (void)intdos(®s, ®s); if ((regs.x.flags & 0x40) == 0) /* zero flag not set? */ cons_key = (regs.h.al & 0xff); } /* * Return TRUE if a character is available. */ static int cons_kbhit(void) { if (cons_key < 0) cons_getkey(); return (cons_key >= 0); } /* * Return a character from the console. * Should only be called when vim_kbhit() returns TRUE. */ static int cons_getch(void) { int c = -1; if (cons_key < 0) cons_getkey(); c = cons_key; cons_key = -1; return c; } #ifdef DJGPP /* * DJGPP provides a kbhit() function that goes to the BIOS instead of DOS. * This doesn't work for terminals connected to a serial port. * Redefine kbhit() here to make it work. */ static int vim_kbhit(void) { union REGS regs; regs.h.ah = 0x0b; (void)intdos(®s, ®s); return regs.h.al; } #ifdef kbhit # undef kbhit /* might have been defined in conio.h */ #endif #define kbhit() vim_kbhit() #endif /* * Simulate WaitForChar() by slowly polling with bioskey(1) or kbhit(). * * If Vim should work over the serial line after a 'ctty com1' we must use * kbhit() and getch(). (jw) * Usually kbhit() is not used, because then CTRL-C and CTRL-P * will be catched by DOS (mool). * * return TRUE if a character is available, FALSE otherwise */ #define FOREVER 1999999999L static int WaitForChar(long msec) { union REGS regs; long starttime = 0; int x, y; if (msec != 0) starttime = biostime(0, 0L); for (;;) { #ifdef FEAT_MOUSE long clicktime; static int old_status = 0; if (mouse_avail && mouse_active && mouse_click < 0) { regs.x.ax = 3; int86(0x33, ®s, ®s); /* check mouse status */ /* only recognize button-down and button-up event */ x = regs.x.cx / mouse_x_div; y = regs.x.dx / mouse_y_div; if ((old_status == 0) != (regs.x.bx == 0)) { if (old_status) /* button up */ mouse_click = MOUSE_RELEASE; else /* button down */ { /* * Translate MSDOS mouse events to Vim mouse events. * TODO: should handle middle mouse button, by pressing * left and right at the same time. */ if (regs.x.bx & MSDOS_MOUSE_LEFT) mouse_click = MOUSE_LEFT; else if (regs.x.bx & MSDOS_MOUSE_RIGHT) mouse_click = MOUSE_RIGHT; else if (regs.x.bx & MSDOS_MOUSE_MIDDLE) mouse_click = MOUSE_MIDDLE; /* * Find out if this is a multi-click */ clicktime = biostime(0, 0L); if (mouse_click_x == x && mouse_click_y == y && mouse_topline == curwin->w_topline #ifdef FEAT_DIFF && mouse_topfill == curwin->w_topfill #endif && mouse_click_count != 4 && mouse_click == mouse_last_click && clicktime < mouse_click_time + p_mouset / BIOSTICK) ++mouse_click_count; else mouse_click_count = 1; mouse_click_time = clicktime; mouse_last_click = mouse_click; mouse_click_x = x; mouse_click_y = y; mouse_topline = curwin->w_topline; #ifdef FEAT_DIFF mouse_topfill = curwin->w_topfill; #endif SET_NUM_MOUSE_CLICKS(mouse_click, mouse_click_count); } } else if (old_status && (x != mouse_x || y != mouse_y)) mouse_click = MOUSE_DRAG; old_status = regs.x.bx; if (mouse_hidden && mouse_x >= 0 && (mouse_x != x || mouse_y != y)) { mouse_hidden = FALSE; show_mouse(TRUE); } mouse_x = x; mouse_y = y; } #endif if ((p_consk ? cons_kbhit() : p_biosk ? bioskey(bioskey_ready) : kbhit()) || cbrk_pressed #ifdef FEAT_MOUSE || mouse_click >= 0 #endif ) return TRUE; /* * Use biostime() to wait until our time is done. * We busy-wait here. Unfortunately, delay() and usleep() have been * reported to give problems with the original Windows 95. This is * fixed in service pack 1, but not everybody installed that. * The DJGPP implementation of usleep() uses a busy-wait loop too. */ if (msec == 0 || (msec != FOREVER && biostime(0, 0L) > starttime + msec / BIOSTICK)) break; #ifdef DJGPP /* Yield the CPU to the next process. */ __dpmi_yield(); #endif } return FALSE; } /* * don't do anything for about "msec" msec */ void mch_delay( long msec, int ignoreinput) { long starttime; if (ignoreinput) { /* * We busy-wait here. Unfortunately, delay() and usleep() have been * reported to give problems with the original Windows 95. This is * fixed in service pack 1, but not everybody installed that. */ starttime = biostime(0, 0L); while (biostime(0, 0L) < starttime + msec / BIOSTICK) ; } else WaitForChar(msec); } /* * mch_write(): write the output buffer to the screen */ void mch_write( char_u *s, int len) { char_u *p; int row, col; if (term_console && full_screen) while (len--) { /* translate ESC | sequences into bios calls */ if (p_wd) /* testing: wait a bit for each char */ WaitForChar(p_wd); if (s[0] == '\n') #ifdef DJGPP { myflush(); S_iCurrentColumn = S_iLeft - 1; } #else myputch('\r'); #endif else if (s[0] == ESC && len > 1 && s[1] == '|') { switch (s[2]) { #ifdef DJGPP case 'B': ScreenVisualBell(); goto got3; #endif case 'J': #ifdef DJGPP myflush(); #endif myclrscr(); goto got3; case 'K': #ifdef DJGPP myflush(); #endif myclreol(); goto got3; case 'L': #ifdef DJGPP myflush(); #endif myinsline(); goto got3; case 'M': #ifdef DJGPP myflush(); #endif mydelline(); got3: s += 3; len -= 2; continue; case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': p = s + 2; row = mygetdigits(&p); /* no check for length! */ if (p > s + len) break; if (*p == ';') { ++p; col = mygetdigits(&p); /* no check for length! */ if (p > s + len) break; if (*p == 'H' || *p == 'r' || *p == 'V') { #ifdef DJGPP myflush(); #endif if (*p == 'H') /* set cursor position */ mygotoxy(col, row); else if (*p == 'V') mywindow(row, S_iTop, col, S_iBottom); else /* set scroll region */ mywindow(S_iLeft, row, S_iRight, col); len -= p - s; s = p + 1; continue; } } else if (*p == 'm' || *p == 'f' || *p == 'b') { if (*p == 'm') /* set color */ { if (row == 0) mynormvideo();/* reset color */ else mytextattr(row); } else if (*p == 'f') /* set foreground color */ mytextcolor(row); else /* set background color */ mytextbackground(row); len -= p - s; s = p + 1; continue; } } } myputch(*s++); } else { write(1, s, (unsigned)len); } } /* * mch_inchar(): low level input funcion. * Get a characters from the keyboard. * If time == 0 do not wait for characters. * If time == n wait a short time for characters. * If time == -1 wait forever for characters. * * return the number of characters obtained */ int mch_inchar( char_u *buf, int maxlen, long time, int tb_change_cnt) { int len = 0; int c; int tmp_c; static int nextchar = 0; /* may keep character when maxlen == 1 */ #ifdef FEAT_AUTOCMD static int once_already = 0; #endif /* * if we got a ctrl-C when we were busy, there will be a "^C" somewhere * on the sceen, so we need to redisplay it. */ if (delayed_redraw) { delayed_redraw = FALSE; update_screen(CLEAR); setcursor(); out_flush(); } /* return remaining character from last call */ if (nextchar) { *buf = nextchar; nextchar = 0; return 1; } #ifdef FEAT_MOUSE if (time != 0) show_mouse(TRUE); #endif #ifdef DJGPP set_sys_cursor(); #endif if (time >= 0) { if (WaitForChar(time) == 0) /* no character available */ { #ifdef FEAT_MOUSE show_mouse(FALSE); #endif #ifdef FEAT_AUTOCMD once_already = 0; #endif return 0; } } else /* time == -1 */ { #ifdef FEAT_AUTOCMD if (once_already == 2) updatescript(0); else if (once_already == 1) { setcursor(); once_already = 2; return 0; } else #endif /* * If there is no character available within 2 seconds (default) * write the autoscript file to disk */ if (WaitForChar(p_ut) == 0) { #ifdef FEAT_AUTOCMD if (has_cursorhold() && get_real_state() == NORMAL_BUSY) { apply_autocmds(EVENT_CURSORHOLD, NULL, NULL, FALSE, curbuf); update_screen(VALID); once_already = 1; return 0; } else #endif updatescript(0); } } WaitForChar(FOREVER); /* wait for key or mouse click */ /* * Try to read as many characters as there are, until the buffer is full. */ /* * we will get at least one key. Get more if they are available * After a ctrl-break we have to read a 0 (!) from the buffer. * bioskey(1) will return 0 if no key is available and when a * ctrl-break was typed. When ctrl-break is hit, this does not always * implies a key hit. */ cbrk_pressed = FALSE; #ifdef FEAT_MOUSE if (mouse_click >= 0 && maxlen >= 5) { len = 5; *buf++ = ESC + 128; *buf++ = 'M'; *buf++ = mouse_click; *buf++ = mouse_x + '!'; *buf++ = mouse_y + '!'; mouse_click = -1; } else #endif { #ifdef FEAT_MOUSE mouse_hidden = TRUE; #endif if (p_biosk && !p_consk) { while ((len == 0 || bioskey(bioskey_ready)) && len < maxlen) { c = translate_altkeys(bioskey(bioskey_read)); /* get the key */ /* * translate a few things for inchar(): * 0x0000 == CTRL-break -> 3 (CTRL-C) * 0x0300 == CTRL-@ -> NUL * 0xnn00 == extended key code -> K_NUL, nn * 0xnne0 == enhanced keyboard -> K_NUL, nn * K_NUL -> K_NUL, 3 */ if (c == 0) c = 3; else if (c == 0x0300) c = NUL; else if ((c & 0xff) == 0 || c == K_NUL || c == 0x4e2b || c == 0x4a2d || c == 0x372a || ((c & 0xff) == 0xe0 && c != 0xe0)) { if (c == K_NUL) c = 3; else c >>= 8; *buf++ = K_NUL; ++len; } if (len < maxlen) { *buf++ = c; len++; #ifdef FEAT_MBYTE /* Convert from 'termencoding' to 'encoding'. Only * translate normal characters, not key codes. */ if (input_conv.vc_type != CONV_NONE && (len == 1 || buf[-2] != K_NUL)) len += convert_input(buf - 1, 1, maxlen - len + 1) - 1; #endif } else nextchar = c; } } else { while ((len == 0 || (p_consk ? cons_kbhit() : kbhit())) && len < maxlen) { switch (c = (p_consk ? cons_getch() : getch())) { case 0: /* NUL means that there is another character. * Get it immediately, because kbhit() doesn't always * return TRUE for the second character. */ if (p_consk) c = cons_getch(); else c = getch(); tmp_c = translate_altkeys(c << 8); if (tmp_c == (c << 8)) { *buf++ = K_NUL; ++len; } else c = tmp_c; break; case K_NUL: *buf++ = K_NUL; ++len; c = 3; break; case 3: cbrk_pressed = TRUE; /*FALLTHROUGH*/ default: break; } if (len < maxlen) { *buf++ = c; ++len; } else nextchar = c; } } } #ifdef FEAT_MOUSE show_mouse(FALSE); #endif beep_count = 0; /* may beep again now that we got some chars */ #ifdef FEAT_AUTOCMD once_already = 0; #endif return len; } /* * return non-zero if a character is available */ int mch_char_avail(void) { return WaitForChar(0L); } #ifdef DJGPP # define INT_ARG int #else # define INT_ARG #endif /* * function for ctrl-break interrupt */ static void interrupt #ifdef DJGPP catch_cbrk(int a) #else catch_cbrk(void) #endif { cbrk_pressed = TRUE; ctrlc_pressed = TRUE; } #ifndef DJGPP /* * ctrl-break handler for DOS. Never called when a ctrl-break is typed, because * we catch interrupt 1b. If you type ctrl-C while Vim is waiting for a * character this function is not called. When a ctrl-C is typed while Vim is * busy this function may be called. By that time a ^C has been displayed on * the screen, so we have to redisplay the screen. We can't do that here, * because we may be called by DOS. The redraw is in mch_inchar(). */ static int _cdecl cbrk_handler(void) { delayed_redraw = TRUE; return 1; /* resume operation after ctrl-break */ } /* * function for critical error interrupt * For DOS 1 and 2 return 0 (Ignore). * For DOS 3 and later return 3 (Fail) */ static void interrupt catch_cint(bp, di, si, ds, es, dx, cx, bx, ax) unsigned bp, di, si, ds, es, dx, cx, bx, ax; { ax = (ax & 0xff00); /* set AL to 0 */ if (_osmajor >= 3) ax |= 3; /* set AL to 3 */ } #endif /* * Set the interrupt vectors for use with Vim on or off. * on == TRUE means as used within Vim */ static void set_interrupts(int on) { static int saved_cbrk; #ifndef DJGPP static void interrupt (*old_cint)(); #endif static void interrupt (*old_cbrk)(INT_ARG); if (on) { saved_cbrk = getcbrk(); /* save old ctrl-break setting */ setcbrk(0); /* do not check for ctrl-break */ #ifdef DJGPP old_cbrk = signal(SIGINT, catch_cbrk); /* critical error interrupt */ #else old_cint = getvect(0x24); /* save old critical error interrupt */ setvect(0x24, catch_cint); /* install our critical error interrupt */ old_cbrk = getvect(0x1B); /* save old ctrl-break interrupt */ setvect(0x1B, catch_cbrk); /* install our ctrl-break interrupt */ ctrlbrk(cbrk_handler); /* vim's ctrl-break handler */ #endif if (term_console) out_str(T_ME); /* set colors */ } else { setcbrk(saved_cbrk); /* restore ctrl-break setting */ #ifdef DJGPP signal(SIGINT,old_cbrk); /* critical error interrupt */ #else setvect(0x24, old_cint); /* restore critical error interrupt */ setvect(0x1B, old_cbrk); /* restore ctrl-break interrupt */ #endif /* restore ctrl-break handler, how ??? */ if (term_console) mynormvideo(); /* restore screen colors */ } } /* * We have no job control, fake it by starting a new shell. */ void mch_suspend(void) { suspend_shell(); } extern int _fmode; /* * Prepare window for use by Vim. */ void mch_init(void) { union REGS regs; #if defined(DJGPP) && defined(FEAT_CLIPBOARD) __dpmi_regs dpmi_regs; #endif /* * Get the video attributes at the cursor. These will be used as the * default attributes. */ regs.h.ah = 0x08; regs.h.bh = 0x00; /* video page 0 */ int86(0x10, ®s, ®s); orig_attr = regs.h.ah; mynormvideo(); if (cterm_normal_fg_color == 0) cterm_normal_fg_color = (orig_attr & 0xf) + 1; if (cterm_normal_bg_color == 0) cterm_normal_bg_color = ((orig_attr >> 4) & 0xf) + 1; term_console = TRUE; /* assume using the console for the things here */ _fmode = O_BINARY; /* we do our own CR-LF translation */ out_flush(); set_interrupts(TRUE); /* catch interrupts */ #ifdef DJGPP /* * Use Long File Names by default, if $LFN not set. */ if (getenv("LFN") == NULL) putenv("LFN=y"); get_screenbase(); #endif #ifdef FEAT_MOUSE /* find out if a MS compatible mouse is available */ regs.x.ax = 0; (void)int86(0x33, ®s, ®s); mouse_avail = regs.x.ax; /* best guess for mouse coordinate computations */ mch_get_shellsize(); if (Columns <= 40) mouse_x_div = 16; if (Rows == 30) mouse_y_div = 16; #endif /* * Try switching to 16 colors for background, instead of 8 colors and * blinking. Does this always work? Can the old value be restored? */ regs.x.ax = 0x1003; regs.h.bl = 0x00; regs.h.bh = 0x00; int86(0x10, ®s, ®s); /* * Test if we have an enhanced AT keyboard. Write 0xFFFF to the keyboard * buffer and try to read it back. If we can't in 16 tries, it's an old * type XT keyboard. */ regs.h.ah = 0x05; regs.x.cx = 0xffff; int86(0x16, ®s, ®s); if (regs.h.al != 1) /* skip this when keyboard buffer is full */ { int i; for (i = 0; i < 16; ++i) { regs.h.ah = 0x10; int86(0x16, ®s, ®s); if (regs.x.ax == 0xffff) break; } if (i == 16) /* 0xffff not read, must be old keyboard */ { bioskey_read = 0; bioskey_ready = 1; } } #ifdef MCH_CURSOR_SHAPE /* Save the old cursor shape */ mch_restore_cursor_shape(FALSE); /* Initialise the cursor shape */ mch_update_cursor(); #endif #if defined(DJGPP) && defined(FEAT_CLIPBOARD) /* * Check to see if the Windows clipboard is available, ie. are we * running from a DOS session within Windows. Obviously, the Windows * clipboard will not be available if we're running under pure DOS. * * int 0x2f, AX = 0x1700 identifies the Windows version we're running * under. Upon return from the interrupt, if AX is unchanged, we're * running under pure DOS and no Windows clipboard is available. * * Remark: could use int86() here but __dpmi_int() is recommended in * the DJGPP docs, since int86() doesn't cover all available interrupts. */ dpmi_regs.x.ax = 0x1700; if (__dpmi_int(0x2f, &dpmi_regs) == -1) /* real-mode interrupt failed? */ dpmi_regs.x.ax = 0x1700; /* force failure */ if (dpmi_regs.x.ax == 0x1700) /* no change in AX? */ clip_init(FALSE); /* no clipboard available, too bad */ else /* else, running under Windows, OK */ clip_init(TRUE); /* clipboard is available */ #endif } int mch_check_win( int argc, char **argv) { /* store argv[0], may be used for $VIM */ if (*argv[0] != NUL) exe_name = FullName_save((char_u *)argv[0], FALSE); /* * Try the DOS search path. The executable may in * fact be called differently, so try this last. */ if (exe_name == NULL || *exe_name == NUL) exe_name = searchpath("vim.exe"); if (isatty(1)) return OK; return FAIL; } /* * Return TRUE if the input comes from a terminal, FALSE otherwise. */ int mch_input_isatty(void) { if (isatty(read_cmd_fd)) return TRUE; return FALSE; } #if defined(USE_FNAME_CASE) || defined(PROTO) /* * fname_case(): Set the case of the file name, if it already exists. * TODO: should expand short to long file names. Need to use DOS interrupts, * see DJGPP sources libc/dos/dir/findfirs.c. */ void fname_case(char_u *name, int len) { char_u *tail; struct ffblk fb; slash_adjust(name); if (findfirst(name, &fb, 0) == 0) { tail = gettail(name); if (len == 0 ? STRLEN(tail) == STRLEN(fb.ff_name) : (tail - name) + STRLEN(fb.ff_name) < len) STRCPY(tail, fb.ff_name); } } #endif /* * return process ID */ long mch_get_pid(void) { return (long)0; } /* * Change default drive (just like _chdrive of Borland C 3.1) */ static int change_drive(int drive) { union REGS regs; regs.h.ah = 0x0e; regs.h.dl = drive - 1; intdos(®s, ®s); /* set default drive */ regs.h.ah = 0x19; intdos(®s, ®s); /* get default drive */ if (regs.h.al == drive - 1) return 0; return -1; } /* * Get absolute file name into buffer 'buf' of length 'len' bytes. * All slashes are replaced with backslashes, to avoid trouble when comparing * file names. When 'shellslash' set do it the other way around. * * return FAIL for failure, OK otherwise */ int mch_FullName( char_u *fname, char_u *buf, int len, int force) { if (!force && mch_isFullName(fname)) /* already expanded */ { STRNCPY(buf, fname, len); buf[len - 1] = NUL; slash_adjust(buf); return OK; } #ifdef __BORLANDC__ /* Only Borland C++ has this */ if (_fullpath((char *)buf, (char *)fname, len - 1) == NULL) return FAIL; return OK; #else /* almost the same as mch_FullName() in os_unix.c */ { # if 1 char_u fullpath[MAXPATHL]; if (!_truename(fname, fullpath)) return FAIL; slash_adjust(fullpath); /* Only needed when 'shellslash' set */ STRNCPY(buf, fullpath, len); buf[len - 1] = NUL; return OK; # else /* Old code, to be deleted... */ int l; char_u olddir[MAXPATHL]; char_u *p, *q; int c; int retval = OK; *buf = 0; /* * change to the directory for a moment, * and then do the getwd() (and get back to where we were). * This will get the correct path name with "../" things. */ p = vim_strrchr(fname, '/'); q = vim_strrchr(fname, '\\'); if (q != NULL && (p == NULL || q > p)) p = q; q = vim_strrchr(fname, ':'); if (q != NULL && (p == NULL || q > p)) p = q; if (p != NULL) { if (getcwd(olddir, MAXPATHL) == NULL) { p = NULL; /* can't get current dir: don't chdir */ retval = FAIL; } else { if (p == fname) /* /fname */ q = p + 1; /* -> / */ else if (q + 1 == p) /* ... c:\foo */ q = p + 1; /* -> c:\ */ else /* but c:\foo\bar */ q = p; /* -> c:\foo */ c = *q; /* truncate at start of fname */ *q = NUL; # ifdef DJGPP STRCPY(buf, fname); slash_adjust(buf); /* needed when fname starts with \ */ if (mch_chdir(buf)) /* change to the directory */ # else if (mch_chdir(fname)) /* change to the directory */ # endif retval = FAIL; else { fname = q; if (c == psepc) /* if we cut the name at a */ fname++; /* '\', don't add it again */ } *q = c; } } if (getcwd(buf, len) == NULL) { retval = FAIL; *buf = NUL; } # ifdef USE_FNAME_CASE else { char_u *head; char_u *tail; struct ffblk fb; int c; int added; /* Apparently "longna~1" isn't expanded by getcwd(), at least not * for DJGPP. Expand it here. Have to do each dirname * separately. */ slash_adjust(buf); head = buf; if (isalpha(*head) && head[1] == ':') head += 2; /* skip "c:" */ while (*head != NUL) { /* Advance "head" to the start of a dirname and "tail" to just * after it. */ while (*head == '/' || *head == '\\') ++head; for (tail = head; *tail != NUL; ++tail) if (*tail == '/' || *tail == '\\') break; c = *tail; *tail = NUL; if (findfirst(buf, &fb, FA_DIREC) == 0) { added = STRLEN(fb.ff_name); if ((head - buf) + added + STRLEN(tail + 1) + 2 < len) { added -= (tail - head); if (added != 0) mch_memmove(tail + 1 + added, tail + 1, STRLEN(tail + 1) + 1); STRCPY(head, fb.ff_name); tail += added; } } *tail = c; head = tail; } } # endif if (p != NULL) mch_chdir(olddir); /* * Concatenate the file name to the path. */ if (*fname != NUL) { l = STRLEN(buf); if (l > 0 && buf[l - 1] != '/' && buf[l - 1] != '\\') strcat(buf, pseps); strcat(buf, fname); } return retval; # endif } #endif } /* * Replace all slashes by backslashes. * This used to be the other way around, but MS-DOS sometimes has problems * with slashes (e.g. in a command name). We can't have mixed slashes and * backslashes, because comparing file names will not work correctly. The * commands that use a file name should try to avoid the need to type a * backslash twice. * When 'shellslash' set do it the other way around. */ void slash_adjust(char_u *p) { #ifdef OLD_DJGPP /* this seems to have been fixed in DJGPP 2.01 */ /* DJGPP can't handle a file name that starts with a backslash, and when it * starts with a slash there should be no backslashes */ if (*p == '\\' || *p == '/') while (*p) { if (*p == '\\') *p = '/'; #ifdef FEAT_MBYTE if (has_mbyte) p += (*mb_ptr2len_check)(p); else #endif ++p; } else #endif while (*p) { if (*p == psepcN) *p = psepc; #ifdef FEAT_MBYTE if (has_mbyte) p += (*mb_ptr2len_check)(p); else #endif ++p; } } /* * Return TRUE if "fname" does not depend on the current directory. */ int mch_isFullName(char_u *fname) { /* A name like "d:/foo" and "//server/share" is absolute */ return (fname[0] != NUL && fname[1] == ':' && (fname[2] == '/' || fname[2] == '\\')) || (fname[0] == fname[1] && (fname[0] == '/' || fname[0] == '\\')); } void mch_early_init(void) { } /* * Careful: mch_exit() may be called before mch_init()! */ void mch_exit(int r) { settmode(TMODE_COOK); stoptermcap(); set_interrupts(FALSE); /* restore interrupts */ #ifdef DJGPP set_sys_cursor(); #endif /* Somehow outputting CR-NL causes the original colors to be restored */ out_char('\r'); out_char('\n'); out_flush(); ml_close_all(TRUE); /* remove all memfiles */ #ifdef MCH_CURSOR_SHAPE mch_restore_cursor_shape(TRUE); #endif exit(r); } /* * set the tty in (raw) ? "raw" : "cooked" mode * Does not change the tty, as bioskey() and kbhit() work raw all the time. */ void mch_settmode(int tmode) { } #ifdef FEAT_MOUSE void mch_setmouse(int on) { mouse_active = on; mouse_hidden = TRUE; /* dont show it until moved */ } #endif /* * set screen mode * return FAIL for failure, OK otherwise */ int mch_screenmode(char_u *arg) { int mode; int i; static char *(names[]) = {"BW40", "C40", "BW80", "C80", "MONO", "C4350"}; static int modes[] = { BW40, C40, BW80, C80, MONO, C4350}; mode = -1; if (VIM_ISDIGIT(*arg)) /* mode number given */ mode = atoi((char *)arg); else { for (i = 0; i < sizeof(names) / sizeof(char_u *); ++i) if (stricmp(names[i], (char *)arg) == 0) { mode = modes[i]; break; } } if (mode == -1) { EMSG("E362: Unsupported screen mode"); return FAIL; } textmode(mode); /* use Borland function */ #ifdef DJGPP /* base address may have changed */ get_screenbase(); #endif /* Screen colors may have changed. */ out_str(T_ME); #ifdef FEAT_MOUSE if (mode <= 1 || mode == 4 || mode == 5 || mode == 13 || mode == 0x13) mouse_x_div = 16; else mouse_x_div = 8; if (mode == 0x11 || mode == 0x12) mouse_y_div = 16; else if (mode == 0x10) mouse_y_div = 14; else mouse_y_div = 8; shell_resized(); #endif return OK; } /* * Structure used by Turbo-C/Borland-C to store video parameters. */ #ifndef DJGPP extern struct text_info _video; #endif /* * try to get the real window size * return FAIL for failure, OK otherwise */ int mch_get_shellsize(void) { struct text_info textinfo; /* * The screenwidth is returned by the BIOS OK. * The screenheight is in a location in the bios RAM, if the display is * EGA or VGA. */ if (!term_console) return FAIL; gettextinfo(&textinfo); Columns = textinfo.screenwidth; Rows = textinfo.screenheight; #ifndef DJGPP if (textinfo.currmode > 10) Rows = *(char far *)MK_FP(0x40, 0x84) + 1; #endif if (Columns < MIN_COLUMNS || Rows < MIN_LINES) { /* these values are overwritten by termcap size or default */ Columns = 80; Rows = 25; return FAIL; } #ifdef DJGPP mytextinit(&textinfo); /* Added by JML, 1/15/98 */ #endif return OK; } /* * Set the active window for delline/insline. */ static void set_window(void) { if (term_console) { #ifndef DJGPP _video.screenheight = Rows; #endif mywindow(1, 1, Columns, Rows); } screen_start(); } void mch_set_shellsize(void) { /* Should try to set the window size to Rows and Columns. * May involve switching display mode.... * We assume the user knows the size and just use it. */ } /* * Rows and/or Columns has changed. */ void mch_new_shellsize() { #ifdef FEAT_MOUSE /* best guess for mouse coordinate computations */ if (Columns <= 40) mouse_x_div = 16; if (Rows == 30) mouse_y_div = 16; #endif set_window(); #ifdef FEAT_MOUSE mouse_area(); /* set area where mouse can go */ #endif } #if defined(DJGPP) || defined(PROTO) /* * Check the number of Columns with a BIOS call. This avoids a crash of the * DOS console when 'columns' is set to a too large value. */ void mch_check_columns() { static union REGS regs; regs.h.ah = 0x0f; (void)int86(0x10, ®s, ®s); if ((unsigned)Columns > (unsigned)regs.h.ah) Columns = (unsigned)regs.h.ah; } #endif /* * call shell, return FAIL for failure, OK otherwise * options: SHELL_*, see vim.h. */ int mch_call_shell( char_u *cmd, int options) { int x; int tmode = cur_tmode; #ifndef DJGPP char_u *newcmd; #endif out_flush(); #ifdef DJGPP set_sys_cursor(); #endif if (options & SHELL_COOKED) settmode(TMODE_COOK); /* set to normal mode */ set_interrupts(FALSE); /* restore interrupts */ #ifdef DJGPP /* ignore signals while external command is running */ signal(SIGINT, SIG_IGN); signal(SIGHUP, SIG_IGN); signal(SIGQUIT, SIG_IGN); signal(SIGTERM, SIG_IGN); #endif if (cmd == NULL) x = system((char *)p_sh); else { #ifdef DJGPP /* * Use 'shell' for system(). */ setenv("SHELL", (char *)p_sh, 1); x = system(cmd); #else /* we use "command" to start the shell, slow but easy */ newcmd = alloc(STRLEN(p_sh) + STRLEN(p_shcf) + STRLEN(cmd) + 3); if (newcmd == NULL) x = -1; else { sprintf((char *)newcmd, "%s %s %s", p_sh, p_shcf, cmd); x = system((char *)newcmd); vim_free(newcmd); } #endif } #ifdef DJGPP signal(SIGINT, SIG_DFL); signal(SIGHUP, SIG_DFL); signal(SIGQUIT, SIG_DFL); signal(SIGTERM, SIG_DFL); #endif if (tmode == TMODE_RAW) settmode(TMODE_RAW); /* set to raw mode */ set_interrupts(TRUE); /* catch interrupts */ if (x && !(options & SHELL_SILENT) && !emsg_silent) { MSG_PUTS("\nshell returned "); msg_outnum((long)x); msg_putchar('\n'); } return x; } /* * check for an "interrupt signal": CTRL-break or CTRL-C */ void mch_breakcheck(void) { if (ctrlc_pressed) { ctrlc_pressed = FALSE; got_int = TRUE; } } /* * Return TRUE if "p" contain a wildcard that can be expanded by * dos_expandpath(). */ int mch_has_exp_wildcard(char_u *p) { for ( ; *p; ++p) { if (vim_strchr((char_u *)"?*[", *p) != NULL || (*p == '~' && p[1] != NUL)) return TRUE; #ifdef FEAT_MBYTE if (has_mbyte) p += (*mb_ptr2len_check)(p) - 1; #endif } return FALSE; } /* * Return TRUE if "p" contain a wildcard or a "~1" kind of thing (could be a * shortened file name). */ int mch_has_wildcard(char_u *p) { for ( ; *p; ++p) { if (vim_strchr((char_u *) # ifdef VIM_BACKTICK "?*$[`" # else "?*$[" # endif , *p) != NULL || (*p == '~' && p[1] != NUL)) return TRUE; #ifdef FEAT_MBYTE if (has_mbyte) p += (*mb_ptr2len_check)(p) - 1; #endif } return FALSE; } /* * Change directory to "path". * The normal chdir() does not change the default drive. This one does. * Return 0 for success, -1 for failure. */ int mch_chdir(char *path) { if (path[0] == NUL) /* just checking... */ return 0; if (path[1] == ':') /* has a drive name */ { if (change_drive(TOLOWER_ASC(path[0]) - 'a' + 1)) return -1; /* invalid drive name */ path += 2; } if (*path == NUL) /* drive name only */ return 0; return chdir(path); /* let the normal chdir() do the rest */ } #ifdef DJGPP /* * mch_rename() works around a bug in rename (aka MoveFile) in * Windows 95: rename("foo.bar", "foo.bar~") will generate a * file whose short file name is "FOO.BAR" (its long file name will * be correct: "foo.bar~"). Because a file can be accessed by * either its SFN or its LFN, "foo.bar" has effectively been * renamed to "foo.bar", which is not at all what was wanted. This * seems to happen only when renaming files with three-character * extensions by appending a suffix that does not include ".". * Windows NT gets it right, however, with an SFN of "FOO~1.BAR". * This works like mch_rename in os_win32.c, but is a bit simpler. * * Like rename(), returns 0 upon success, non-zero upon failure. * Should probably set errno appropriately when errors occur. */ int mch_rename(const char *OldFile, const char *NewFile) { char_u *TempFile; int retval; int fd; /* rename() works correctly without long file names, so use that */ if (!_USE_LFN) return rename(OldFile, NewFile); if ((TempFile = alloc((unsigned)(STRLEN(OldFile) + 13))) == NULL) return -1; STRCPY(TempFile, OldFile); STRCPY(gettail(TempFile), "axlqwqhy.ba~"); if (rename(OldFile, TempFile)) retval = -1; else { /* now create an empty file called OldFile; this prevents * the operating system using OldFile as an alias (SFN) * if we're renaming within the same directory. For example, * we're editing a file called filename.asc.txt by its SFN, * filena~1.txt. If we rename filena~1.txt to filena~1.txt~ * (i.e., we're making a backup while writing it), the SFN * for filena~1.txt~ will be filena~1.txt, by default, which * will cause all sorts of problems later in buf_write. So, we * create an empty file called filena~1.txt and the system will have * to find some other SFN for filena~1.txt~, such as filena~2.txt */ if ((fd = open(OldFile, O_RDWR|O_CREAT|O_EXCL, 0444)) < 0) return -1; retval = rename(TempFile, NewFile); close(fd); mch_remove((char_u *)OldFile); /* If renaming to NewFile failed, rename TempFile back to OldFile, so * that it looks like nothing happened. */ if (retval) rename(TempFile, OldFile); } vim_free(TempFile); return retval; /* success */ } #endif #if defined(DJGPP) || defined(PROTO) /* * setlocale() for DJGPP with MS-DOS codepage support * Author: Cyril Slobin * * Scaled down a lot for use by Vim: Only support setlocale(LC_ALL, ""). */ #undef setlocale #include #include #include #define UPCASE (__dj_ISALNUM | __dj_ISALPHA | __dj_ISGRAPH | __dj_ISPRINT | __dj_ISUPPER) #define LOCASE (__dj_ISALNUM | __dj_ISALPHA | __dj_ISGRAPH | __dj_ISPRINT | __dj_ISLOWER) char * djgpp_setlocale(void) { __dpmi_regs regs; struct { char id; unsigned short off, seg; } __attribute__ ((packed)) info; unsigned char buffer[0x82], lower, upper; int i; regs.x.ax = 0x6502; regs.x.bx = 0xffff; regs.x.dx = 0xffff; regs.x.cx = 5; regs.x.es = __tb >> 4; regs.x.di = __tb & 0xf; __dpmi_int(0x21, ®s); if (regs.x.flags & 1) return NULL; dosmemget(__tb, 5, &info); dosmemget((info.seg << 4) + info.off, 0x82, buffer); if (*(short *)buffer != 0x80) return NULL; /* Fix problem of underscores being replaced with y-umlaut. (Levin) */ if (buffer[26] == 0x5f) buffer[26] = 0x98; for (i = 0; i < 0x80; i++) { lower = i + 0x80; upper = (buffer+2)[i]; if (lower != upper) { __dj_ctype_flags[lower+1] = LOCASE; __dj_ctype_toupper[lower+1] = upper; if (__dj_ctype_flags[upper+1] == 0) __dj_ctype_flags[upper+1] = UPCASE; if (__dj_ctype_tolower[upper+1] == upper) __dj_ctype_tolower[upper+1] = lower; } } return "C"; } #if defined(FEAT_CLIPBOARD) || defined(PROTO) /* * Clipboard stuff, for cutting and pasting text to other windows. * * Implementation of DOS/Windows clipboard data transfer * by David Kotchan (dkotchan@sympatico.ca) */ #define CF_TEXT 0x01 /* Windows clipboard format: Windows (ANSI) text */ #define CF_OEMTEXT 0x07 /* Windows clipboard format: OEM (DOS) text */ #define CF_VIMCLIP 0x04 /* trick: SYLK clipboard format for VimClipboard */ static int Win16OpenClipboard(void); static int Win16CloseClipboard(void); static int Win16EmptyClipboard(void); static char_u *Win16GetClipboardData(int clip_data_format); static int Win16SetClipboardData(int clip_data_format, char_u *clip_data, int clip_data_size, int clip_data_type); /* * Make vim the owner of the current selection. Return OK upon success. */ int clip_mch_own_selection(VimClipboard *cbd) { /* * Never actually own the clipboard. If another application sets the * clipboard, we don't want to think that we still own it. */ return FAIL; } /* * Make vim NOT the owner of the current selection. */ void clip_mch_lose_selection(VimClipboard *cbd) { /* Nothing needs to be done here */ } /* * Read the Windows clipboard text and put it in Vim's clipboard register. */ void clip_mch_request_selection(VimClipboard *cbd) { int type = MCHAR; char_u *pAllocated = NULL; char_u *pClipText = NULL; int clip_data_format = 0; if (Win16OpenClipboard()) { /* Check for Vim's own clipboard format first. The CF_VIMCLIP format * is just ordinary text (like CF_TEXT) except prepended by the * selection type (as a single character). Note that under DOS we * actually cannot define a custom CF_VIMCLIP clipboard format; we * use instead one of the existing Windows-defined formats, usually * "DIF" or "SYLK". See Win16GetClipboardData() for details. * * Note that Win16GetClipboardData() returns the address of the memory * block it allocated. This is not necessary the start of the * clipboard text data: there may be other bytes ahead of the * text (particularly for CF_VIMCLIP) which are used for data * management. So pClipText is not necessarily == pAllocated. */ if ((pAllocated = Win16GetClipboardData(CF_VIMCLIP)) != NULL) { clip_data_format = CF_VIMCLIP; pClipText = pAllocated; switch (*pClipText++) /* after ++, pClipText points to text */ { default: case 'L': type = MLINE; break; case 'C': type = MCHAR; break; #ifdef FEAT_VISUAL case 'B': type = MBLOCK; break; #endif } } /* Otherwise, check for the normal Windows text formats. There are * two of these: CF_TEXT (common) and CF_OEMTEXT (used for DOS * compatibility). Experiments show that, under the DOS/Windows * clipboard interface, writing CF_TEXT data to the clipboard * automatically creates a CF_OEMTEXT format as well. */ else if ((pAllocated = Win16GetClipboardData(CF_TEXT)) != NULL) { clip_data_format = CF_TEXT; pClipText = pAllocated; type = (vim_strchr((char*)pClipText, '\r') != NULL) ? MLINE : MCHAR; } else if ((pAllocated = Win16GetClipboardData(CF_OEMTEXT)) != NULL) { clip_data_format = CF_OEMTEXT; pClipText = pAllocated; type = (vim_strchr((char*)pClipText, '\r') != NULL) ? MLINE : MCHAR; } /* Did we get anything? */ if (pClipText != NULL) { char_u *pDest; char_u *pStart; char_u *pEnd; long_u clip_data_size = 0; /* The Windows clipboard normally stores its text lines terminated * by . But Vim uses only , so translate the * into . Also, watch for possible null bytes at the end of * pClipText. These are padding added by "get_clipboard_data" * (int 0x2f, AX= 0x1705) in order to round the data size up to the * next multiple of 32 bytes. See Win16GetClipboardData() for * details. */ pDest = strstr( pClipText, "\r\n" ); /* find first */ if (pDest != NULL) /* found one? */ { pStart = pDest + 1; /* points to after */ pEnd = strstr( pStart, "\r\n" );/* find next */ while (pEnd != NULL) /* found one? */ { memmove(pDest, pStart, (long)(pEnd - pStart)); /* exclude */ pDest += (long)(pEnd - pStart); /* new destination */ pStart = pEnd + 1; /* new starting point */ pEnd = strstr(pStart, "\r\n"); /* find next */ } /* Fell out of while() loop: no more pairs. Just copy * the rest of the data, up to the first null byte. */ pEnd = strchr(pStart, '\0'); /* find first null */ memmove(pDest, pStart, (long)(pEnd - pStart)); /* exclude nul */ pDest += (long)(pEnd - pStart); *pDest = '\0'; /* terminate */ /* Now that all pairs have been "compressed" into just * 's, determine the true text length. */ clip_data_size = (long_u)(pDest - pClipText); } else { /* no pairs at all */ /* Since the data may have been padded with trailing nulls, * determine the true string length. */ clip_data_size = STRLEN(pClipText); /* true data length */ } /* Copy the cleaned-up data over to Vim's clipboard "*" register. */ clip_yank_selection(type, pClipText, clip_data_size, cbd); /* Free the memory that Win16GetClipboardData() allocated. */ vim_free(pAllocated); } Win16CloseClipboard(); } // end if (Win16OpenClipboard()) } /* * Send the currently selected Vim text to the Windows clipboard. */ void clip_mch_set_selection( VimClipboard *cbd ) { char_u *pClipData = NULL; long_u clip_data_size; int clip_data_type; /* If the '*' register isn't already filled in, fill it in now. */ cbd->owned = TRUE; clip_get_selection(cbd); cbd->owned = FALSE; /* * clip_convert_selection() returns a pointer to a buffer containing * the text to send to the Windows clipboard, together with a count * of the number of characters (bytes) in the buffer. The function's * return value is the 'type' of selection: MLINE, MCHAR, or MBLOCK; * or -1 for failure. */ clip_data_type = clip_convert_selection(&pClipData, &clip_data_size, cbd); if (clip_data_type < 0) /* could not convert? */ return; /* early exit */ if (Win16OpenClipboard()) { if (Win16EmptyClipboard()) { int sentOK; sentOK = Win16SetClipboardData(CF_TEXT, pClipData, clip_data_size, clip_data_type); sentOK = Win16SetClipboardData(CF_VIMCLIP, pClipData, clip_data_size, clip_data_type) && sentOK; if (!sentOK) { /* one or both of Win16SetClipboardData() failed. */ /* Technically we don't know why Win16SetClipboardData() * failed, but almost always it will be because there wasn't * enough DOS memory to bufer the data, so report that as the * problem. * * We report the error here (instead of in * Win16SetClipboardData()) because we don't want the error * reported twice. */ EMSG("E450: Selection too large, cannot allocate DOS buffer"); } } Win16CloseClipboard(); } /* release memory allocated by clip_convert_selection() */ vim_free(pClipData); return; } /* * Win16OpenClipboard: open the Windows clipboard. The clipboard must be open * before it can be communicated with at all. Return TRUE on success, * FALSE on failure. */ static int Win16OpenClipboard(void) { __dpmi_regs dpmi_regs; long start_time; int tick_count; /* int 02xf, AX = 0x1701 attempts to open the Windows clipboard. Upon * return from the interrupt, if AX is non-zero, the clipboard was * successfully opened. If AX is zero, the clipboard could not be opened * because it is currently in use by another process. * * Remark: other DOS programs I (dk) have written that use the Windows * clipboard sometimes encounter the problem that the clipboard cannot * be opened even though it is demonstrably not in use by any other * process. In all cases, repeated attempts to open the clipboard * eventually succeed, but the initial attempt occasionally fails. * * The problem is intermittent and appears to be related to DOS being * "busy" at certain unpredictable times. DOS maintains two internal * flags that indicate whether it's busy: InDOS and CritErr. The * location of InDOS can be found by calling int 0x21, AH = 0x34. The * location of CritErr can be found by calling int 0x21, AX = 0x5d06. * If either of these flags is set, DOS is "busy" and cannot be * interrupted. See "Undocumented DOS" by Schulman et al for details. * * However here I take the easier approach that if the first call to open * the clipboard does not succeed, just try again. In fact, try once per * biostime() clock tick, up to 18 times (about one second). */ tick_count = 0; dpmi_regs.x.ax = 0x1701; /* open Windows clipboard */ if (__dpmi_int(0x2f, &dpmi_regs) == -1) { /* real-mode interrupt failed? */ return FALSE; /* FALSE --> clipboard not open */ } /* wait up to one second */ while (dpmi_regs.x.ax == 0 && tick_count++ < 18) { /* Wait one clock tick (18.2 ticks/sec = 55 msec per tick). * * We busy-wait here. Unfortunately, delay() and usleep() have been * reported to give problems with the original Windows 95. This is * fixed in service pack 1, but not everybody installed that. */ start_time = biostime(0, 0L); while (biostime(0, 0L) == start_time) ; dpmi_regs.x.ax = 0x1701; /* open Windows clipboard */ if (__dpmi_int(0x2f, &dpmi_regs) == -1) { /* real-mode interrupt failed? */ return FALSE; /* FALSE --> clipboard not open */ } } /* Couldn't open the clipboard, even after 18 attempts? */ if (tick_count >= 18 && dpmi_regs.x.ax == 0) return FALSE; /* FALSE --> clipboard not open */ return TRUE; /* TRUE --> clipboard opened successfully, OK */ } /* * Win16CloseClipboard: close the Windows clipboard. Return TRUE on * success, FALSE on failure. This function can always be called, * whether the clipboard is open or not. */ static int Win16CloseClipboard(void) { __dpmi_regs dpmi_regs; /* Close the clipboard. This interrupt can always be called, even * if the clipboard is already closed. */ dpmi_regs.x.ax = 0x1708; /* close the clipboard */ if (__dpmi_int(0x2f, &dpmi_regs) == -1) { /* real-mode interrupt failed? */ return FALSE; /* FALSE --> clipboard could not be closed */ } return TRUE; /* TRUE --> clipboard closed successfully, OK */ } /* * Win16EmptyClipboard: empty the (previously opened) Windows clipboard. * Return TRUE on success, FALSE on failure. */ static int Win16EmptyClipboard(void) { __dpmi_regs dpmi_regs; /* int 02xf, AX = 0x1702 attempts to empty the Windows clipboard. Upon * return from the interrupt, if AX == 0, the clipboard could not be * emptied (for some reason). */ dpmi_regs.x.ax = 0x1702; /* empty the Windows clipboard */ if (__dpmi_int(0x2f, &dpmi_regs) == -1) { /* real-mode interrupt failed? */ return FALSE; /* FALSE --> clipboard could not be emptied */ } /* Did we succeed in clearing the clipboard? */ if (dpmi_regs.x.ax == 0) return FALSE; /* FALSE --> clipboard could not be emptied */ return TRUE; /* TRUE --> clipboard was emptied, OK */ } /* * FreeDOSMemory: a helper function to free memory previously * allocated by a call to __dpmi_allocate_dos_memory(). */ static void FreeDOSMemory(int protected_mode_selector) { /* Free the DOS buffer and release the DPMI prot-mode selector. * * It's important that DOS memory be properly released because * there's only a limited amount of it. Therefore, if the call * to __dpmi_free_dos_memory() fails, emit an error message * unconditionally. */ if (__dpmi_free_dos_memory(protected_mode_selector) == -1) EMSG("E451: could not free DOS memory buffer (DJGPP)"); } /* * Win16GetClipboardData: query the Windows clipboard as to whether data * is available in a particular clipboard format. If data is * available, allocate a buffer for it and read the data from the * clipboard into the buffer. Return a pointer to the buffer. If * no data is available in the requested format, return NULL. * * This routine allocates memory to hold the retrieved clipboard * data. It's the caller's responsibility to free this memory * once it's finished using it. The memory should be freed by * calling vim_free(). */ static char_u * Win16GetClipboardData(int clip_data_format) { __dpmi_regs dpmi_regs; int real_mode_segment_address; int protected_mode_selector; char_u *clip_data_buffer; long_u clip_data_size; /* We only handle clipboard formats we recognize, others are ignored. * * It's not possible to create a custom clipboard format for VimClipboard * data under DOS, so one of the predefined Windows formats had to be * used for CF_VIMCLIP. Two obscure formats, popular when Windows 3.0 * came out but no longer in much use today, are the DIF and SYLK formats. * DIF is the Data Interchange Format, SYLK is the Symbolic Link format. * They are both text formats and either one can be hijacked for use as * "the VimClipboard format". Of course, this conflicts with anyone who * still *is* using DIF or SYLK data formats, but that will be very few * people. * * I (dk) chose SYLK as the more obscure format because it was used * mostly for Microsoft Multiplan (the pre-cursor to Excel) and it's not * likely Multiplan is used anywhere much anymore. Mind you, Excel can * still export to both DIF and SYLK formats. */ switch (clip_data_format) { case CF_VIMCLIP: /* Vim's own special clipboard format */ case CF_TEXT: /* Windows text */ case CF_OEMTEXT: /* DOS (OEM) text */ /* int 02xf, AX = 0x1704 returns the number of bytes of data currently * on the Windows clipboard, for the specified format. Upon return * from the interrupt, DX:AX = the number of bytes, rounded up to the * nearest multiple of 32. */ dpmi_regs.x.ax = 0x1704; /* get size of clipbd data */ dpmi_regs.x.dx = clip_data_format; if (__dpmi_int(0x2f, &dpmi_regs) == -1) { /* real-mode interrupt failed? */ return NULL; /* early exit */ } /* Did we get anything? If not, this is not an error. */ if (dpmi_regs.x.dx == 0 && dpmi_regs.x.ax == 0) { /* no CF_VIMCLIP data? */ return NULL; /* early exit */ } /* There is data available in the requested clipboard format. * * Calculate data size. Remember this is rounded up to the nearest * multiple of 32, so clip_data_size is actually an upper limit. * The extra bytes, if any, are set to null (0x00) when the data is * read from the clipboard. (Later:) actually I'm no longer sure * this is strictly true: the end-of-data is marked by a null, but * the extra bytes appear to sometimes be null, sometimes not. * They may just be garbage. */ clip_data_size = dpmi_regs.x.ax + (dpmi_regs.x.dx << 16); /* Allocate memory to retrieve the data. The buffer has to lie in the * DOS memory region (in the first 1 MByte of address space) because * the Windows clipboard interface expects a 16-bit segment:offset * pointer to a buffer address within the DOS region. Must therefore * use __dpmi_allocate_dos_memory() instead of lalloc() or alloc(). */ real_mode_segment_address = __dpmi_allocate_dos_memory( (clip_data_size + 15) >> 4, /* buffer size, in 16-byte paragraphs */ &protected_mode_selector); /* prot-mode selector for the address */ if (real_mode_segment_address == -1) { /* memory allocation failed. */ /* Technically we don't know why the allocation failed, but * almost always it will be because there wasn't enough DOS * memory to satisfy the request, so report that as the problem. * On my system, DJGPP is able to satisfy a DOS allocation request * up to about 600K in size. This depends on your HIMEM.SYS and * EMM386.EXE settings however. */ EMSG("E452: Clipboard data too large, cannot allocate DOS buffer"); return NULL; /* early exit */ } /* Copy data from the clipboard into the buffer. Experiments show that * the Windows clipboard is smart enough to handle data transfers * larger than 64K properly, even though the buffer address is a 16-bit * segment:offset (which would normally limit the block size to 64K * unless ES gets incremented). */ dpmi_regs.x.ax = 0x1705; /* get clipboard data */ dpmi_regs.x.dx = clip_data_format; /* CF_VIMCLIP */ dpmi_regs.x.es = real_mode_segment_address; /* buffer ad: segment */ dpmi_regs.x.bx = 0; /* buffer ad: offset */ if (__dpmi_int( 0x2f, &dpmi_regs) == -1) { /* real-mode interrupt failed? */ EMSG("E453: could not copy clipboard data to DOS buffer"); FreeDOSMemory(protected_mode_selector); /* clean up DOS mem */ return NULL; /* early exit */ } /* Clipboard data is now in DOS memory in the buffer pointed to by * ES:BX. Copy this into ordinary memory that Vim can access (ie. * prot-mode memory). Allocate one extra byte to ensure the text * is terminated properly (in case it was somehow corrupted). */ clip_data_buffer = (char_u *)lalloc(clip_data_size + 1, TRUE); if (clip_data_buffer == NULL) { /* allocation failed? */ EMSG("E454: could not allocate clipboard memory buffer"); FreeDOSMemory(protected_mode_selector); /* clean up DOS mem */ return NULL; /* early exit */ } *(clip_data_buffer + clip_data_size) = '\0'; /* ensure terminated */ /* Copy the data from DOS memory to Vim-accessible memory. */ movedata( /* DJGPP version of memcpy() */ protected_mode_selector, 0, /* source: DOS ad (via selector) */ _my_ds(), (unsigned)clip_data_buffer, /* target: normal mem address */ clip_data_size); /* how many bytes */ /* Free the DOS buffer and release the DPMI prot-mode selector. */ FreeDOSMemory(protected_mode_selector); /* clean up DOS memory */ return clip_data_buffer; /* return pointer to allocated buffer */ default: /* unknown clipboard format */ return NULL; } } /* * Win16SetClipboardData: send 'clip_data_size' bytes of data from the buffer * pointed to by 'clip_data', to the Windows clipboard. The data is * registered with the clipboard as being in the 'clip_data_format' * format. */ static int Win16SetClipboardData( int clip_data_format, char_u *clip_data, int clip_data_size, int clip_data_type) { __dpmi_regs dpmi_regs; int real_mode_segment_address; int protected_mode_selector; long_u protected_mode_offset = 0L; int total_size = clip_data_size; char_u *clip_sel_type; /* If we're using the CF_VIMCLIP custom format, allocate an extra * byte for clip_sel_type, which is a character indicating the type * of text selection: MLINE, MCHAR, or MBLOCK. */ if (clip_data_format == CF_VIMCLIP) total_size++; /* extra byte for marker */ /* Data cannot be sent directly from a Vim string (pClipData) to * the Windows clipboard, because the Windows clipboard interface * expects a 16-bit (DOS) segment:offset address for the source * buffer. Therefore we must create a "transfer buffer" in the DOS * memory region (in the first 1 MByte of address space) and copy * the Vim string into that. From there, the data can then be sent * to the Windows clipboard. * * To allocate DOS memory, we must use __dpmi_allocate_dos_memory() * instead of lalloc() or alloc(). If the allocation fails, it will * almost invariably be because there is not enough DOS memory * available to accommodate the size of clip_data. There is nothing * we can do about this, we simply have to fail. */ real_mode_segment_address = __dpmi_allocate_dos_memory( (total_size + 15) >> 4, /* buffer size, in 16-byte paragraphs */ &protected_mode_selector); /* prot-mode selector for the address */ if (real_mode_segment_address == -1) { /* memory allocation failed. */ /* Technically we don't know why the allocation failed, but * almost always it will be because there wasn't enough DOS * memory to satisfy the request. On my system, DJGPP is able * to satisfy a DOS allocation request up to about 600K in size. * This depends however on HIMEM.SYS and EMM386.EXE settings. */ return FALSE; /* early exit */ } /* Copy data from Vim's buffer (clip_data) into the DOS transfer buffer. * This can be larger than 64K; movedata() takes care of crossing any * 16-bit segment boundaries. * * If we're using Vim's custom clipboard format, we must copy one extra * byte to indicate the type of selection: line, character, or block. */ if (clip_data_format == CF_VIMCLIP) { switch (clip_data_type) { default: case MLINE: clip_sel_type = "L"; break; case MCHAR: clip_sel_type = "C"; break; #ifdef FEAT_VISUAL case MBLOCK: clip_sel_type = "B"; break; #endif } movedata( _my_ds(), (unsigned)clip_sel_type, /* source: normal memory address */ protected_mode_selector, 0, /* target: DOS ad (via selector) */ 1); /* how many bytes to copy */ protected_mode_offset += STRLEN(clip_sel_type); /* allow for marker */ } movedata( _my_ds(), (unsigned)clip_data, /* source: normal memory address */ protected_mode_selector, /* target: DOS address (via selector) */ protected_mode_offset, /* non-zero, if using clip_sel_type */ clip_data_size); /* how many bytes to copy */ /* Send data from the DOS transfer buffer to the Windows clipboard. * int 02xf, AX = 0x1703 sends SI:CX bytes of data from the buffer * at ES:BX, to the clipboard. */ dpmi_regs.x.ax = 0x1703; /* send clipboard data */ dpmi_regs.x.dx = clip_data_format; /* flag: format of the data */ dpmi_regs.x.si = ((total_size >> 16) & 0x0000ffffL); /* hi word of data size */ dpmi_regs.x.cx = (total_size & 0x0000ffffL); /* lo word of data size */ dpmi_regs.x.es = real_mode_segment_address; /* buffer address: segment */ dpmi_regs.x.bx = 0; /* buffer address: offset */ if (__dpmi_int(0x2f, &dpmi_regs) == -1) { /* real-mode interrupt failed. */ FreeDOSMemory(protected_mode_selector); /* clean up DOS memory */ return FALSE; /* early exit */ } /* Free the DOS buffer and release the DPMI prot-mode selector. */ FreeDOSMemory(protected_mode_selector); /* clean up DOS memory */ return TRUE; /* TRUE --> data successfully sent to clipboard */ } #endif /* FEAT_CLIPBOARD */ #endif /* DJGPP */ /* * End of MS-DOS only code */ #endif /* WIN16 */ /* common MS-DOS and Win16 code follows */ static int vim_chmod(char_u *name) { char_u *p; int f; int c = 0; /* chmod() can't handle a file name with a trailing slash, remove it. * But don't remove it for "/" or "c:/". */ p = name + STRLEN(name); if (p > name) --p; if (p > name && (*p == '\\' || *p == '/') && p[-1] != ':') { c = *p; /* remove trailing (back)slash */ *p = NUL; } else p = NULL; #if defined(__BORLANDC__) && (__BORLANDC__ > 0x410) /* this also sets the archive bit, supported by Borland C 4.0 and later, * where __BORLANDC__ is 0x450 (3.1 is 0x410) */ f = _rtl_chmod((char *)name, 0, 0); #else f = _chmod((char *)name, 0, 0); #endif if (p != NULL) *p = c; /* put back (back)slash */ return f; } /* * get file permissions for 'name' * Returns -1 for error. * Returns FA_attributes defined in dos.h */ long mch_getperm(char_u *name) { return (long)vim_chmod(name); /* get file mode */ } /* * set file permission for 'name' to 'perm' * * return FAIL for failure, OK otherwise */ int mch_setperm( char_u *name, long perm) { perm |= FA_ARCH; /* file has changed, set archive bit */ #if defined(__BORLANDC__) && (__BORLANDC__ > 0x410) return (_rtl_chmod((char *)name, 1, (int)perm) == -1 ? FAIL : OK); #else return (_chmod((char *)name, 1, (int)perm) == -1 ? FAIL : OK); #endif } /* * Set hidden flag for "name". */ void mch_hide(char_u *name) { /* DOS 6.2 share.exe causes "seek error on file write" errors when making * the swap file hidden. Thus don't do it. */ } /* * return TRUE if "name" is a directory * return FALSE if "name" is not a directory * return FALSE for error * * beware of a trailing (back)slash */ int mch_isdir(char_u *name) { int f; f = vim_chmod(name); if (f == -1) return FALSE; /* file does not exist at all */ if ((f & FA_DIREC) == 0) return FALSE; /* not a directory */ return TRUE; } #if defined(FEAT_EVAL) || defined(PROTO) /* * Return 1 if "name" can be executed, 0 if not. * Return -1 if unknown. */ int mch_can_exe(name) char_u *name; { return (searchpath(name) != NULL); } #endif /* * Check what "name" is: * NODE_NORMAL: file or directory (or doesn't exist) * NODE_WRITABLE: writable device, socket, fifo, etc. * NODE_OTHER: non-writable things */ int mch_nodetype(char_u *name) { if (STRICMP(name, "AUX") == 0 || STRICMP(name, "CON") == 0 || STRICMP(name, "CLOCK$") == 0 || STRICMP(name, "NUL") == 0 || STRICMP(name, "PRN") == 0 || ((STRNICMP(name, "COM", 3) == 0 || STRNICMP(name, "LPT", 3) == 0) && VIM_ISDIGIT(name[3]) && name[4] == NUL)) return NODE_WRITABLE; /* TODO: NODE_OTHER? */ return NODE_NORMAL; } /* * Get name of current directory into buffer 'buf' of length 'len' bytes. * Return OK for success, FAIL for failure. */ int mch_dirname( char_u *buf, int len) { #ifdef DJGPP if (getcwd((char *)buf, len) == NULL) return FAIL; /* turn the '/'s returned by DJGPP into '\'s */ slash_adjust(buf); return OK; #else return (getcwd((char *)buf, len) != NULL ? OK : FAIL); #endif } /* * this version of remove is not scared by a readonly (backup) file * * returns -1 on error, 0 otherwise (just like remove()) */ int mch_remove(char_u *name) { (void)mch_setperm(name, 0); /* default permissions */ return unlink((char *)name); } /* * Special version of getenv(): Use uppercase name. */ char_u * mch_getenv(char_u *name) { int i; #define MAXENVLEN 50 char_u var_copy[MAXENVLEN + 1]; char_u *p; char_u *res; /* * Take a copy of the argument, and force it to upper case before passing * to getenv(). On DOS systems, getenv() doesn't like lower-case argument * (unlike Win32 et al.) If the name is too long to fit in var_copy[] * allocate memory. */ if ((i = STRLEN(name)) > MAXENVLEN) p = alloc(i + 1); else p = var_copy; if (p == NULL) p = name; /* out of memory, fall back to unmodified name */ else { for (i = 0; name[i] != NUL; ++i) p[i] = toupper(name[i]); p[i] = NUL; } res = (char_u *)getenv((char *)p); if (p != var_copy && p != name) vim_free(p); return res; } /* * Insert user name in s[len]. */ int mch_get_user_name( char_u *s, int len) { *s = NUL; return FAIL; } /* * Insert host name is s[len]. */ void mch_get_host_name( char_u *s, int len) { #ifdef DJGPP STRNCPY(s, "PC (32 bits Vim)", len); #else STRNCPY(s, "PC (16 bits Vim)", len); #endif s[len - 1] = NUL; /* make sure it's terminated */ }