diff options
Diffstat (limited to '')
-rw-r--r-- | src/os_vms.c | 853 |
1 files changed, 853 insertions, 0 deletions
diff --git a/src/os_vms.c b/src/os_vms.c new file mode 100644 index 0000000..23854dd --- /dev/null +++ b/src/os_vms.c @@ -0,0 +1,853 @@ +/* vi:set ts=8 sts=4 sw=4 noet: + * + * VIM - Vi IMproved by Bram Moolenaar + * VMS port by Henk Elbers + * VMS deport by Zoltan Arpadffy + * + * 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. + */ + +#include "vim.h" + +// define _generic_64 for use in time functions +#if !defined(VAX) && !defined(PROTO) +# include <gen64def.h> +#else +// based on Alpha's gen64def.h; the file is absent on VAX +typedef struct _generic_64 { +# pragma __nomember_alignment + __union { // You can treat me as... + // long long is not available on VAXen + // unsigned __int64 gen64$q_quadword; ...a single 64-bit value, or + + unsigned int gen64$l_longword [2]; // ...two 32-bit values, or + unsigned short int gen64$w_word [4]; // ...four 16-bit values + } gen64$r_quad_overlay; +} GENERIC_64; +#endif + +typedef struct +{ + char class; + char type; + short width; + union + { + struct + { + char _basic[3]; + char length; + } y; + int basic; + } x; + int extended; +} TT_MODE; + +typedef struct +{ + short buflen; + short itemcode; + char *bufadrs; + int *retlen; +} ITEM; + +typedef struct +{ + ITEM equ; + int nul; +} ITMLST1; + +typedef struct +{ + ITEM index; + ITEM string; + int nul; +} ITMLST2; + +static TT_MODE orgmode; +static short iochan; // TTY I/O channel +static short iosb[4]; // IO status block + +static int vms_match_num = 0; +static int vms_match_free = 0; +static char_u **vms_fmatch = NULL; +static char *Fspec_Rms; // rms file spec, passed implicitly between routines + + + +static TT_MODE get_tty(void); +static void set_tty(int row, int col); + +#define EXPL_ALLOC_INC 64 + +#define EQN(S1,S2,LN) (strncmp(S1,S2,LN) == 0) +#define SKIP_FOLLOWING_SLASHES(Str) do { while (Str[1] == '/') ++Str; } while (0) + + +/* + * vul_desc vult een descriptor met een string en de lengte + * hier van. + */ + static void +vul_desc(DESC *des, char *str) +{ + des->dsc$b_dtype = DSC$K_DTYPE_T; + des->dsc$b_class = DSC$K_CLASS_S; + des->dsc$a_pointer = str; + des->dsc$w_length = str ? strlen(str) : 0; +} + +/* + * vul_item vult een item met een aantal waarden + */ + static void +vul_item(ITEM *itm, short len, short cod, char *adr, int *ret) +{ + itm->buflen = len; + itm->itemcode = cod; + itm->bufadrs = adr; + itm->retlen = ret; +} + + void +mch_settmode(tmode_T tmode) +{ + int status; + + if ( tmode == TMODE_RAW ) + set_tty(0, 0); + else + { + switch (orgmode.width) + { + case 132: OUT_STR_NF((char_u *)"\033[?3h\033>"); break; + case 80: OUT_STR_NF((char_u *)"\033[?3l\033>"); break; + default: break; + } + out_flush(); + status = sys$qiow(0, iochan, IO$_SETMODE, iosb, 0, 0, + &orgmode, sizeof(TT_MODE), 0,0,0,0); + if (status!=SS$_NORMAL || (iosb[0]&0xFFFF)!=SS$_NORMAL) + return; + (void)sys$dassgn(iochan); + iochan = 0; + } +} + + static void +set_tty(int row, int col) +{ + int status; + TT_MODE newmode; // New TTY mode bits + static short first_time = TRUE; + + if (first_time) + { + orgmode = get_tty(); + first_time = FALSE; + } + newmode = get_tty(); + if (col) + newmode.width = col; + if (row) + newmode.x.y.length = row; + newmode.x.basic |= (TT$M_NOECHO | TT$M_HOSTSYNC); + newmode.x.basic &= ~TT$M_TTSYNC; + newmode.extended |= TT2$M_PASTHRU; + status = sys$qiow(0, iochan, IO$_SETMODE, iosb, 0, 0, + &newmode, sizeof(newmode), 0, 0, 0, 0); + if (status!=SS$_NORMAL || (iosb[0]&0xFFFF)!=SS$_NORMAL) + return; +} + + static TT_MODE +get_tty(void) +{ + + static $DESCRIPTOR(odsc,"SYS$OUTPUT"); // output descriptor + + int status; + TT_MODE tt_mode; + + if (!iochan) + status = sys$assign(&odsc,&iochan,0,0); + + status = sys$qiow(0, iochan, IO$_SENSEMODE, iosb, 0, 0, + &tt_mode, sizeof(tt_mode), 0, 0, 0, 0); + if (status != SS$_NORMAL || (iosb[0] & 0xFFFF) != SS$_NORMAL) + { + tt_mode.width = 0; + tt_mode.type = 0; + tt_mode.class = 0; + tt_mode.x.basic = 0; + tt_mode.x.y.length = 0; + tt_mode.extended = 0; + } + return(tt_mode); +} + +/* + * Get the current window size in Rows and Columns. + */ + int +mch_get_shellsize(void) +{ + TT_MODE tmode; + + tmode = get_tty(); // get size from VMS + Columns = tmode.width; + Rows = tmode.x.y.length; + return OK; +} + +/* + * Try to set the window size to Rows and new_Columns. + */ + void +mch_set_shellsize(void) +{ + set_tty(Rows, Columns); + switch (Columns) + { + case 132: OUT_STR_NF((char_u *)"\033[?3h\033>"); break; + case 80: OUT_STR_NF((char_u *)"\033[?3l\033>"); break; + default: break; + } + out_flush(); + screen_start(); +} + + char_u * +mch_getenv(char_u *lognam) +{ + DESC d_file_dev, d_lognam ; + static char buffer[LNM$C_NAMLENGTH+1]; + char_u *cp = NULL; + unsigned long attrib; + int lengte = 0, dum = 0, idx = 0; + ITMLST2 itmlst; + char *sbuf = NULL; + + vul_desc(&d_lognam, (char *)lognam); + vul_desc(&d_file_dev, "LNM$FILE_DEV"); + attrib = LNM$M_CASE_BLIND; + vul_item(&itmlst.index, sizeof(int), LNM$_INDEX, (char *)&idx, &dum); + vul_item(&itmlst.string, LNM$C_NAMLENGTH, LNM$_STRING, buffer, &lengte); + itmlst.nul = 0; + if (sys$trnlnm(&attrib, &d_file_dev, &d_lognam, NULL,&itmlst) == SS$_NORMAL) + { + buffer[lengte] = '\0'; + if (cp = alloc(lengte + 1)) + strcpy((char *)cp, buffer); + return(cp); + } + else if ((sbuf = getenv((char *)lognam))) + { + lengte = strlen(sbuf) + 1; + cp = alloc(lengte); + if (cp) + strcpy((char *)cp, sbuf); + return cp; + } + else + return(NULL); +} + +/* + * mch_setenv VMS version of setenv() + */ + int +mch_setenv(char *var, char *value, int x) +{ + int res, dum; + long attrib = 0L; + char acmode = PSL$C_SUPER; // needs SYSNAM privilege + DESC tabnam, lognam; + ITMLST1 itmlst; + + vul_desc(&tabnam, "LNM$JOB"); + vul_desc(&lognam, var); + vul_item(&itmlst.equ, value ? strlen(value) : 0, value ? LNM$_STRING : 0, + value, &dum); + itmlst.nul = 0; + res = sys$crelnm(&attrib, &tabnam, &lognam, &acmode, &itmlst); + return((res == 1) ? 0 : -1); +} + + int +vms_sys(char *cmd, char *out, char *inp) +{ + DESC cdsc, odsc, idsc; + long status; + + if (cmd) + vul_desc(&cdsc, cmd); + if (out) + vul_desc(&odsc, out); + if (inp) + vul_desc(&idsc, inp); + + lib$spawn(cmd ? &cdsc : NULL, // command string + inp ? &idsc : NULL, // input file + out ? &odsc : NULL, // output file + 0, 0, 0, &status, 0, 0, 0, 0, 0, 0); + return status; +} + +/* + * Convert string to lowercase - most often filename + */ + char * +vms_tolower( char *name ) +{ + int i,nlen = strlen(name); + for (i = 0; i < nlen; i++) + name[i] = TOLOWER_ASC(name[i]); + return name; +} + +/* + * Convert VMS system() or lib$spawn() return code to Unix-like exit value. + */ + int +vms_sys_status(int status) +{ + if (status != SS$_NORMAL && (status & STS$M_SUCCESS) == 0) + return status; // Command failed. + return 0; +} + +/* + * vms_read() + * function for low level char input + * + * Returns: input length + */ + int +vms_read(char *inbuf, size_t nbytes) +{ + int status, function, len; + TT_MODE tt_mode; + ITEM itmlst[2]; // terminates on everything + static long trm_mask[8] = {-1, -1, -1, -1, -1, -1, -1, -1}; + + // whatever happened earlier we need an iochan here + if (!iochan) + tt_mode = get_tty(); + + // important: clean the inbuf + memset(inbuf, 0, nbytes); + + // set up the itemlist for the first read + vul_item(&itmlst[0], 0, TRM$_MODIFIERS, + (char *)( TRM$M_TM_NOECHO | TRM$M_TM_NOEDIT | + TRM$M_TM_NOFILTR | TRM$M_TM_TRMNOECHO | + TRM$M_TM_NORECALL) , 0); + vul_item(&itmlst[1], sizeof(trm_mask), TRM$_TERM, (char *)&trm_mask, 0); + + // wait forever for a char + function = (IO$_READLBLK | IO$M_EXTEND); + status = sys$qiow(0, iochan, function, &iosb, 0, 0, + inbuf, nbytes-1, 0, 0, &itmlst, sizeof(itmlst)); + len = strlen(inbuf); // how many chars we got? + + // read immediately the rest in the IO queue + function = (IO$_READLBLK | IO$M_TIMED | IO$M_ESCAPE | IO$M_NOECHO | IO$M_NOFILTR); + status = sys$qiow(0, iochan, function, &iosb, 0, 0, + inbuf+len, nbytes-1-len, 0, 0, 0, 0); + + len = strlen(inbuf); // return the total length + + return len; +} + +/* + * vms_wproc() is called for each matching filename by decc$to_vms(). + * We want to save each match for later retrieval. + * + * Returns: 1 - continue finding matches + * 0 - stop trying to find any further matches + */ + static int +vms_wproc(char *name, int val) +{ + int i; + static int vms_match_alloced = 0; + + if (val == DECC$K_FOREIGN ) // foreign non VMS files are not counting + return 1; + + // accept all DECC$K_FILE and DECC$K_DIRECTORY + if (vms_match_num == 0) + { + // first time through, setup some things + if (NULL == vms_fmatch) + { + vms_fmatch = ALLOC_MULT(char_u *, EXPL_ALLOC_INC); + if (!vms_fmatch) + return 0; + vms_match_alloced = EXPL_ALLOC_INC; + vms_match_free = EXPL_ALLOC_INC; + } + else + { + // re-use existing space + vms_match_free = vms_match_alloced; + } + } + + // make matches look uniform + vms_remove_version(name); + name=vms_tolower(name); + + // if name already exists, don't add it + for (i = 0; i<vms_match_num; i++) + { + if (0 == STRCMP((char_u *)name,vms_fmatch[i])) + return 1; + } + if (--vms_match_free == 0) + { + char_u **old_vms_fmatch = vms_fmatch; + + // add more space to store matches + vms_match_alloced += EXPL_ALLOC_INC; + vms_fmatch = vim_realloc(old_vms_fmatch, + sizeof(char **) * vms_match_alloced); + if (!vms_fmatch) + { + vim_free(old_vms_fmatch); + return 0; + } + vms_match_free = EXPL_ALLOC_INC; + } + vms_fmatch[vms_match_num] = vim_strsave((char_u *)name); + + ++vms_match_num; + return 1; +} + +/* + * mch_expand_wildcards this code does wild-card pattern + * matching NOT using the shell + * + * return OK for success, FAIL for error (you may lose some + * memory) and put an error message in *file. + * + * num_pat number of input patterns + * pat array of pointers to input patterns + * num_file pointer to number of matched file names + * file pointer to array of pointers to matched file names + * + */ + int +mch_expand_wildcards(int num_pat, char_u **pat, int *num_file, char_u ***file, int flags) +{ + int i, cnt = 0; + char_u buf[MAXPATHL]; + char *result; + int dir; + int files_alloced, files_free; + + *num_file = 0; // default: no files found + files_alloced = EXPL_ALLOC_INC; + files_free = EXPL_ALLOC_INC; + *file = ALLOC_MULT(char_u *, files_alloced); + if (*file == NULL) + { + *num_file = 0; + return FAIL; + } + for (i = 0; i < num_pat; i++) + { + // expand environment var or home dir + if (vim_strchr(pat[i],'$') || vim_strchr(pat[i],'~')) + expand_env(pat[i],buf,MAXPATHL); + else + STRCPY(buf,pat[i]); + + vms_match_num = 0; // reset collection counter + result = decc$translate_vms(vms_fixfilename(buf)); + if ( (int) result == 0 || (int) result == -1 ) + { + cnt = 0; + } + else + { + cnt = decc$to_vms(result, vms_wproc, 1 /*allow wild*/ , (flags & EW_DIR ? 0:1 ) /*allow directory*/) ; + } + if (cnt > 0) + cnt = vms_match_num; + + if (cnt < 1) + continue; + + for (i = 0; i < cnt; i++) + { + // files should exist if expanding interactively + if (!(flags & EW_NOTFOUND) && mch_getperm(vms_fmatch[i]) < 0) + continue; + + // do not include directories + dir = (mch_isdir(vms_fmatch[i])); + if (( dir && !(flags & EW_DIR)) || (!dir && !(flags & EW_FILE))) + continue; + + // Skip files that are not executable if we check for that. + if (!dir && (flags & EW_EXEC) + && !mch_can_exe(vms_fmatch[i], NULL, !(flags & EW_SHELLCMD))) + continue; + + // allocate memory for pointers + if (--files_free < 1) + { + char_u **old_file = *file; + + files_alloced += EXPL_ALLOC_INC; + *file = vim_realloc(old_file, sizeof(char_u **) * files_alloced); + if (*file == NULL) + { + vim_free(old_file); + *file = (char_u **)""; + *num_file = 0; + return(FAIL); + } + files_free = EXPL_ALLOC_INC; + } + + (*file)[*num_file++] = vms_fmatch[i]; + } + } + return OK; +} + + int +mch_expandpath(garray_T *gap, char_u *path, int flags) +{ + int i,cnt = 0; + char *result; + + vms_match_num = 0; + // the result from the decc$translate_vms needs to be handled + // otherwise it might create ACCVIO error in decc$to_vms + result = decc$translate_vms(vms_fixfilename(path)); + if ( (int) result == 0 || (int) result == -1 ) + { + cnt = 0; + } + else + { + cnt = decc$to_vms(result, vms_wproc, 1 /*allow_wild*/, (flags & EW_DIR ? 0:1 ) /*allow directory*/); + } + if (cnt > 0) + cnt = vms_match_num; + for (i = 0; i < cnt; i++) + { + if (mch_getperm(vms_fmatch[i]) >= 0) // add existing file + addfile(gap, vms_fmatch[i], flags); + } + return cnt; +} + +/* + * attempt to translate a mixed unix-vms file specification to pure vms + */ + static void +vms_unix_mixed_filespec(char *in, char *out) +{ + char *lastcolon; + char *end_of_dir; + char ch; + int len; + char *out_str=out; + + // copy vms filename portion up to last colon + // (node and/or disk) + lastcolon = strrchr(in, ':'); // find last colon + if (lastcolon != NULL) + { + len = lastcolon - in + 1; + strncpy(out, in, len); + out += len; + in += len; + } + + end_of_dir = NULL; // default: no directory + + // start of directory portion + ch = *in; + if ((ch == '[') || (ch == '/') || (ch == '<')) // start of directory(s) ? + { + ch = '['; + SKIP_FOLLOWING_SLASHES(in); + } + else if (EQN(in, "../", 3)) // Unix parent directory? + { + *out++ = '['; + *out++ = '-'; + end_of_dir = out; + ch = '.'; + in += 2; + SKIP_FOLLOWING_SLASHES(in); + } + else + { // not a special character + while (EQN(in, "./", 2)) // Ignore Unix "current dir" + { + in += 2; + SKIP_FOLLOWING_SLASHES(in); + } + if (strchr(in, '/') == NULL) // any more Unix directories ? + { + strcpy(out, in); // No - get rest of the spec + return; + } + else + { + *out++ = '['; // Yes, denote a Vms subdirectory + ch = '.'; + --in; + } + } + + // if we get here, there is a directory part of the filename + + // initialize output file spec + *out++ = ch; + ++in; + + while (*in != '\0') + { + ch = *in; + if ((ch == ']') || (ch == '/') || (ch == '>') ) // end of (sub)directory ? + { + end_of_dir = out; + ch = '.'; + SKIP_FOLLOWING_SLASHES(in); + } + else if (EQN(in, "../", 3)) // Unix parent directory? + { + *out++ = '-'; + end_of_dir = out; + ch = '.'; + in += 2; + SKIP_FOLLOWING_SLASHES(in); + } + else + { + while (EQN(in, "./", 2)) // Ignore Unix "current dir" + { + end_of_dir = out; + in += 2; + SKIP_FOLLOWING_SLASHES(in); + ch = *in; + } + } + + // Place next character into output file spec + *out++ = ch; + ++in; + } + + *out = '\0'; // Terminate output file spec + + if (end_of_dir != NULL) // Terminate directory portion + *end_of_dir = ']'; +} + +/* + * for decc$to_vms in vms_fixfilename + */ + static int +vms_fspec_proc(char *fil, int val) +{ + strcpy(Fspec_Rms,fil); + return(1); +} + +/* + * change unix and mixed filenames to VMS + */ + void * +vms_fixfilename(void *instring) +{ + static char *buf = NULL; + static size_t buflen = 0; + size_t len; + + // get a big-enough buffer + len = strlen(instring) + 1; + if (len > buflen) + { + buflen = len + 128; + buf = vim_realloc(buf, buflen * sizeof(char)); + } + +#ifdef DEBUG + char *tmpbuf = NULL; + tmpbuf = ALLOC_MULT(char, buflen); + strcpy(tmpbuf, instring); +#endif + + Fspec_Rms = buf; // for decc$to_vms + + if (strchr(instring,'/') == NULL) + // It is already a VMS file spec + strcpy(buf, instring); + else if (strchr(instring,'"') == NULL) // password in the path? + { + // Seems it is a regular file, let guess that it is pure Unix fspec + if ( (strchr(instring,'[') == NULL) && (strchr(instring,'<') == NULL) && + (strchr(instring,']') == NULL) && (strchr(instring,'>') == NULL) && + (strchr(instring,':') == NULL) ) + { + // It must be a truly unix fspec + decc$to_vms(instring, vms_fspec_proc, 0, 0); + } + else + { + // It is a mixed fspec + vms_unix_mixed_filespec(instring, buf); + } + } + else + // we have a password in the path + // decc$ functions can not handle + // this is our only hope to resolv + vms_unix_mixed_filespec(instring, buf); + + return buf; +} + +/* + * Remove version number from file name + * we need it in some special cases as: + * creating swap file name and writing new file + */ + void +vms_remove_version(void * fname) +{ + char_u *cp; + char_u *fp; + + if ((cp = vim_strchr( fname, ';')) != NULL) // remove version + *cp = '\0'; + else if ((cp = vim_strrchr( fname, '.')) != NULL ) + { + if ((fp = vim_strrchr( fname, ']')) != NULL ) + {;} + else if ((fp = vim_strrchr( fname, '>')) != NULL ) + {;} + else + fp = fname; + + while ( *fp != '\0' && fp < cp ) + if ( *fp++ == '.' ) + *cp = '\0'; + } + return ; +} + +struct typeahead_st { + unsigned short numchars; + unsigned char firstchar; + unsigned char reserved0; + unsigned long reserved1; +} typeahead; + +/* + * Wait "msec" msec until a character is available from file descriptor "fd". + * "msec" == 0 will check for characters once. + * "msec" == -1 will block until a character is available. + */ + int +RealWaitForChar( + int fd UNUSED, // always read from iochan + long msec, + int *check_for_gpm UNUSED, + int *interrupted) +{ + int status; + struct _generic_64 time_curr; + struct _generic_64 time_diff; + struct _generic_64 time_out; + unsigned int convert_operation = LIB$K_DELTA_SECONDS_F; + float sec =(float) msec/1000; + + // make sure the iochan is set + if (!iochan) + get_tty(); + + if (sec > 0) + { + // time-out specified; convert it to absolute time + // sec>0 requirement of lib$cvtf_to_internal_time() + + // get current time (number of 100ns ticks since the VMS Epoch) + status = sys$gettim(&time_curr); + if (status != SS$_NORMAL) + return 0; // error + // construct the delta time +#if __G_FLOAT==0 +# ifndef VAX + // IEEE is default on IA64, but can be used on Alpha too - but not on VAX + status = lib$cvts_to_internal_time( + &convert_operation, &sec, &time_diff); +# endif +#else // default on Alpha and VAX + status = lib$cvtf_to_internal_time( + &convert_operation, &sec, &time_diff); +#endif + if (status != LIB$_NORMAL) + return 0; // error + // add them up + status = lib$add_times( + &time_curr, + &time_diff, + &time_out); + if (status != LIB$_NORMAL) + return 0; // error + } + + while (TRUE) + { + // select() + status = sys$qiow(0, iochan, IO$_SENSEMODE | IO$M_TYPEAHDCNT, iosb, + 0, 0, &typeahead, 8, 0, 0, 0, 0); + if (status != SS$_NORMAL || (iosb[0] & 0xFFFF) != SS$_NORMAL) + return 0; // error + + if (typeahead.numchars) + return 1; // ready to read + + // there's nothing to read; what now? + if (msec == 0) + { + // immediate time-out; return impatiently + return 0; + } + else if (msec < 0) + { + // no time-out; wait on indefinitely + return 1; // fakeout to force a wait in vms_read() + } + else + { + // time-out needs to be checked + status = sys$gettim(&time_curr); + if (status != SS$_NORMAL) + return 0; // error + + status = lib$sub_times( + &time_out, + &time_curr, + &time_diff); + if (status != LIB$_NORMAL) + return 0; // error, incl. time_diff < 0 (i.e. time-out) + + // otherwise wait some more + } + } +} |