home *** CD-ROM | disk | FTP | other *** search
/ back2roots/padua / padua.7z / padua / shell / csh519s.lha / comm2.c < prev    next >
C/C++ Source or Header  |  1992-02-25  |  33KB  |  1,438 lines

  1. /*
  2.  * COMM2.C
  3.  *
  4.  * (c)1986 Matthew Dillon     9 October 1986
  5.  *
  6.  * Version 2.07M by Steve Drew 10-Sep-87
  7.  * Version 4.01A by Carlo Borreo & Cesare Dieni 17-Feb-90
  8.  * Version 5.00L by Urban Mueller 17-Feb-91
  9.  *
  10.  */
  11.  
  12. #include "shell.h"
  13.  
  14. /* comm2.c */
  15. static long dptrtosecs(DPTR *d);
  16. static long timeof(char *s);
  17. static int evalif(void);
  18. static int clinum(char *name);
  19. static int copydir(long srcdir, long destdir, int recur);
  20. static int copyfile(char *srcname, long srcdir, char *destname, long destdir);
  21. static int file_date(struct DateStamp *date, char *name);
  22. static void changedisk(struct MsgPort *task);
  23. static int func_array( char *fav[], int fac);
  24. static int func_int(int i);
  25. static int func_bool(int i);
  26. static int func_string(char *str);
  27. static int commas(char *av[], int ac, int n);
  28. static int wordset(char **av, int ac, char **(*func)(char **,int,char**,int,int*,int));
  29. static int split_arg(char **av, int ac);
  30. static char *info_part(char **av, int ac, int n, char *buf);
  31.  
  32. /* Casting conveniences */
  33. #define BPTR_TO_C(strtag, var)  ((struct strtag *)(BADDR( (ULONG) var)))
  34. #define PROC(task)              ((struct Process *)task)
  35. #define CLI(proc)               (BPTR_TO_C(CommandLineInterface, proc->pr_CLI))
  36.  
  37. /* Externs */
  38. extern int has_wild;                    /* flag set if any arg has a ? or * */
  39.  
  40. int
  41. do_abortline( void )
  42. {
  43.     Exec_abortline = 1;
  44.     return 0;
  45. }
  46.  
  47. int
  48. do_error( void )
  49. {
  50.     return atoi(av[1]);
  51. }
  52.  
  53. int
  54. do_return( void )
  55. {
  56.     int retcode=(ac<2 ? 0 : atoi(av[1]));
  57.  
  58.     Exec_abortline = 1;
  59.     if (Src_stack) {
  60.         Src_abort[Src_stack-1]=1;
  61.         return retcode;
  62.     } else
  63.         main_exit(retcode);
  64.     return 0;
  65. }
  66.  
  67. /*
  68.  * STRHEAD
  69.  *
  70.  * place a string into a variable removing everything after and including
  71.  * the 'break' character
  72.  *
  73.  * strhead varname breakchar string
  74.  *
  75.  */
  76.  
  77. int
  78. do_strhead( void )
  79. {
  80.     char *s;
  81.     if (*av[2] && (s=index( av[3], *av[2]))) 
  82.         *s='\0';
  83.     set_var (LEVEL_SET, av[1], av[3]);
  84.     return 0;
  85. }
  86.  
  87. int
  88. do_strtail( void )
  89. {
  90.     char *s;
  91.     if (*av[2] && (s=index(av[3],*av[2]))) s++; else s=av[3];
  92.     set_var (LEVEL_SET, av[1], s);
  93.     return 0;
  94. }
  95.  
  96. static long
  97. dptrtosecs(DPTR *d)
  98. {
  99.     struct DateStamp *ds=(&d->fib->fib_Date);
  100.     return ds->ds_Days*86400 + ds->ds_Minute*60 + ds->ds_Tick/TICKS_PER_SECOND;
  101. }
  102.  
  103. static long
  104. timeof(char *s)
  105. {
  106.     DPTR *d;
  107.     int  dummy;
  108.     long n;
  109.  
  110.     if ( (d=dopen(s,&dummy))==NULL ) return 0L;
  111.     n=dptrtosecs(d);
  112.     dclose(d);
  113.     return n;
  114. }
  115.  
  116. /*
  117.  * if -f file (exists) or:
  118.  *
  119.  * if A < B   <, >, =, <=, >=, <>, where A and B are either:
  120.  * nothing
  121.  * a string
  122.  * a value (begins w/ number)
  123.  */
  124.  
  125. #define IF_NOT 128
  126.  
  127. int
  128. do_if( char *garbage, int com )
  129. {
  130.     int result;
  131.  
  132.     switch (com) {
  133.     case 0:
  134.         if (If_stack && If_base[If_stack - 1])
  135.             If_base[If_stack++] = 1;
  136.         else {
  137.             result=evalif();
  138.             If_base[If_stack++]=(options & IF_NOT ? result : !result);
  139.         }
  140.         break;
  141.     case 1:
  142.         if (If_stack > 1 && If_base[If_stack - 2]) break;
  143.         if (If_stack) If_base[If_stack - 1] ^= 1;
  144.         break;
  145.     case 2:
  146.         if (If_stack) --If_stack;
  147.         break;
  148.     }
  149.     disable = (If_stack) ? If_base[If_stack - 1] : 0;
  150.     if (If_stack >= MAXIF) {
  151.         fprintf(stderr,"If's too deep\n");
  152.         disable = If_stack = 0;
  153.         return -1;
  154.     }
  155.     return 0;
  156. }
  157.  
  158. static int
  159. isint( char *s )
  160. {
  161.     while( ISSPACE(*s) ) ++s;
  162.     return *s=='+' || *s=='-' || *s>='0' && *s<='9';
  163. }
  164.  
  165. static int
  166. evalif( void )
  167. {
  168.     char c, *str, *left, *right, *cmp;
  169.     long num, t0, i=1;
  170.     int nac=ac-1, err;
  171.  
  172.     if(!strcmp(av[ac-1],"then")) ac--;
  173.  
  174.     switch(options & ~IF_NOT) {
  175.     case 0:
  176.         for( i=1; i<ac; i++ )
  177.             if( strlen(str=av[i])<=2 && *str && index("!<=>",*str) &&
  178.                    (!str[1] || index("<=>",str[1])))
  179.                 break;
  180.  
  181.         if ( i==ac )
  182.             return ac>1 && *av[1] && strcmp(av[1],"0");
  183.  
  184.         left =av[1];
  185.         right=av[i+1];
  186.         if( 1+1!=i )    left = compile_av(av,1,i   ,0xA0, 0);
  187.         if( i+1+1!=ac ) right= compile_av(av,i+1,ac,0xA0, 0);
  188.         cmp  = av[i];
  189.         num  = Atol(left) - Atol(right);
  190.         if( *cmp=='!' )  cmp++, options^=IF_NOT;
  191.         if( !isint(left) || !isint(right)) num=strcmp(left,right);
  192.         if( 1+1!=i )    free(left); 
  193.         if( i+1+1!=ac ) free(right);
  194.  
  195.         if (num < 0)       c='<';
  196.         else if (num > 0)  c='>';
  197.         else               c='=';
  198.         return index(cmp, c) != NULL;
  199.     case 1:
  200.         return do_rpn(NULL,i);
  201.     case 2:
  202.     case 256:
  203.         return exists(av[i]);
  204.     case 4:
  205.         t0=timeof(av[i++]);
  206.         for ( ; i<ac ; i++)
  207.             if (t0<=timeof(av[i])) return 1;
  208.         return 0;
  209.     case 8:
  210.         return AvailMem( MEMF_FAST )!=0;
  211.     case 16:
  212.         return isdir(av[i])!=0;
  213.     case 32:
  214.         return get_var(LEVEL_SET,av[i]) != 0;
  215.     case 64:
  216.         if( ac>1 )
  217.             return get_opt(av+1,&nac,-*av[1],&err);
  218.         return 0;
  219.     default:
  220.         ierror(NULL,500);
  221.     }
  222.     return 0;
  223. }
  224.  
  225. #if 0
  226. do_while( void )
  227. {
  228.     char *com=av[--ac];
  229.     int ret=0;
  230.  
  231.     while( evalif() && ret==0 && !CHECKBREAK() )
  232.         ret=execute( com );
  233.     return ret;
  234. }
  235. #endif
  236.  
  237.  
  238. int
  239. do_label( void )
  240. {
  241.     char aseek[32], *fwd;
  242.  
  243.     if (Src_stack == 0) {
  244.         ierror (NULL, 502);
  245.         return -1;
  246.     }
  247.  
  248.     sprintf (aseek, "%ld %d", Src_pos[Src_stack-1], If_stack);
  249.     set_var (LEVEL_LABEL + Src_stack - 1, av[1], aseek);
  250.  
  251.     fwd=get_var(LEVEL_SET,v_gotofwd);
  252.     if (fwd && !strcmp(av[1],fwd))
  253.         forward_goto = 0;
  254.     return 0;
  255. }
  256.  
  257. int
  258. do_goto( void )
  259. {
  260.     int new;
  261.     long pos;
  262.     char *lab;
  263.  
  264.     if (Src_stack == 0) {
  265.         ierror (NULL, 502);
  266.     } else {
  267.         lab = get_var (LEVEL_LABEL + Src_stack - 1, av[1]);
  268.         if (lab == NULL) {
  269.             forward_goto = 1;
  270.             set_var (LEVEL_SET, v_gotofwd, av[1]);
  271.             return(0);
  272.         } else {
  273.             pos = atoi(lab);
  274.             fseek (Src_base[Src_stack - 1], pos, 0);
  275.             Src_pos[Src_stack - 1] = pos;
  276.             new = atoi(next_word(lab));
  277.             for (; If_stack < new; ++If_stack)
  278.                 If_base[If_stack] = 0;
  279.             If_stack = new;
  280.         }
  281.     }
  282.     Exec_abortline = 1;
  283.     return (0);      /* Don't execute rest of this line */
  284. }
  285.  
  286.  
  287. int
  288. do_inc(char *garbage, int com)
  289. {
  290.     char *var, num[32];
  291.  
  292.     if (ac>2) com *= atoi(av[2]);
  293.     if (var = get_var (LEVEL_SET, av[1])) {
  294.         sprintf (num, "%d", atoi(var)+com);
  295.         set_var (LEVEL_SET, av[1], num);
  296.     }
  297.     return 0;
  298. }
  299.  
  300. BPTR Input(void);
  301. BPTR Output(void);
  302.  
  303. int
  304. do_input( void )
  305. {
  306.     char *str, in[256], *get, *put;
  307.     int i, quote=0;
  308.  
  309.     if( options&2 ) {
  310.         if( !isconsole(Input()) ) return 20;
  311.         setrawcon( -1, 0 );
  312.         in[1]=0;
  313.         for ( i=1; i<ac && !CHECKBREAK(); ++i) {
  314.             Read( Input(), in, 1 );
  315.             set_var (LEVEL_SET, av[i], in);
  316.         }
  317.         setrawcon( 0, 0 );
  318.         return 0;
  319.     }
  320.  
  321.     for ( i=1; i<ac; ++i)
  322.         if (fgets(in,256,stdin)) {
  323.             str=in, put=in+strlen(in)-1;
  324.             if( *put=='\n' ) *put=0;
  325.             if( !(options&1) ) {
  326.                 while( *str==' ' ) str++;
  327.                 for( put=get=str; *get; ) {
  328.                     if( *get=='\"' )
  329.                         quote=1-quote, get++;
  330.                     else if( *get==' ' && !quote ) {
  331.                         while( *get==' ' ) get++;
  332.                         if( *get ) *put++=0xA0;
  333.                     } else 
  334.                         *put++=*get++;
  335.                 }
  336.                 *put=0;
  337.             }
  338.             set_var (LEVEL_SET, av[i], str);
  339.         }
  340.     return 0;
  341. }
  342.  
  343. int
  344. do_ver( void )
  345. {
  346.     extern char shellname[];
  347.     extern char shellcompiled[];
  348.  
  349.     puts(shellname);
  350.     puts("1.00 Lattice (c) 1986 Matthew Dillon\n"
  351.          "2.05 Manx(M) versions by Steve Drew\n"
  352.          "3.00 ARP (A) versions by Carlo Borreo, Cesare Dieni\n"
  353.          "4.00 ARP 1.3 versions by Carlo Borreo, Cesare Dieni\n"
  354.          "5.00 Lattice versions by U. Dominik Mueller\n");
  355.     printf( shellcompiled );
  356.     return 0;
  357. }
  358.  
  359.  
  360. static int
  361. clinum( char *name )
  362. {
  363.     int ncli=(long)FindCLI(0L), count;
  364.     struct Task *task;
  365.     char cmd[40+1];
  366.  
  367.     if( *name>='0' && *name<='9' )
  368.         return atoi( name );
  369.  
  370.     Forbid();
  371.     for (count = 1; count <= ncli ; count++)
  372.         if (task = (struct Task *)FindCLI(count)) {
  373.             if ( !PROC(task)->pr_TaskNum || PROC(task)->pr_CLI == 0) continue;
  374.             BtoCStr(cmd,   CLI(PROC(task))->cli_CommandName, 40L);
  375.             if( !Strcmp( BaseName( cmd ), BaseName( name )))
  376.                 goto done;
  377.         }
  378.     count=-1;
  379. done:
  380.     Permit();
  381.     return count;
  382. }
  383.  
  384. int
  385. do_ps( void )
  386. {
  387.     /* this code fragment based on ps.c command by Dewi Williams */
  388.     int count;             /* loop variable         */
  389.     struct Task *task;     /* EXEC descriptor       */
  390.     char strbuf[64+1];     /* scratch for btocstr() */
  391.     char cmd[40+1], *com;  /* holds cmd name        */
  392.     long ncli,mycli,cli,i;
  393.  
  394.     char onoff[80];
  395.     memset( onoff, 0, 80 );
  396.     for( i=1; i<ac; i++ )
  397.         onoff[ 1+clinum( av[i] ) ]=1;
  398.     if( options&2 )
  399.         for( i=0; i<80; i++ )
  400.             onoff[i]=1-onoff[i];
  401.  
  402.     printf("Proc Command Name         CLI Type    Pri.  Address  Directory\n");
  403.     Forbid();
  404.  
  405.     ncli=(long)FindCLI(0L);
  406.     mycli= Myprocess->pr_TaskNum;
  407.     for (count = 1; count <= ncli ; count++)             /* or just assume 20?*/
  408.         if (task = (struct Task *)FindCLI((long)count)) {/* Sanity check      */
  409.             cli=PROC(task)->pr_TaskNum;
  410.             if( ac>1 && !onoff[1+cli] )
  411.                 continue;
  412.             if ( cli==0 || PROC(task)->pr_CLI == 0) continue; /* or complain? */
  413.                 BtoCStr(cmd,   CLI(PROC(task))->cli_CommandName, 40L);
  414.                 BtoCStr(strbuf,CLI(PROC(task))->cli_SetName    , 64L);
  415.             com=cmd;
  416.             if( !(options&1) )
  417.                 com=BaseName(com);
  418.                 printf("%c%2d  %-20.20s %-11.11s %3d  %8lx  %s\n",
  419.                 cli==mycli ? '*' : ' ',
  420.                 count,
  421.                 com,
  422.                 task->tc_Node.ln_Name,
  423.                 (signed char)task->tc_Node.ln_Pri,
  424.                 task,
  425.                 strbuf
  426.             );
  427.         }
  428.  
  429.     Permit();
  430.     return 0;
  431. }
  432.  
  433. /*
  434.  * CP [-d] [-u] file file
  435.  * CP [-d] [-u] file file file... destdir
  436.  * CP [-r][-u][-d] dir dir dir... destdir
  437.  */
  438.  
  439. #define CP_RECUR     0x1
  440. #define CP_UPDATE    0x2
  441. #define CP_NODATE    0x4
  442. #define CP_NOFLAGS   0x8
  443. #define CP_FRESH     0x10
  444. #define CP_MOVE      0x20
  445. #define CP_QUIET     0x40
  446. #define CP_OVERWRITE 0x80
  447.  
  448. static char *errstr;          /* let's be a little more informative */
  449. static int level;
  450.  
  451. int
  452. do_copy( void )
  453. {
  454.     int recur, ierr;
  455.     char *destname;
  456.     char destisdir;
  457.     FIB *fib;
  458.     int i;
  459.  
  460.     errstr = "";
  461.     ierr = 0;
  462.  
  463.     fib = (FIB *)SAllocMem((long)sizeof(FIB), MEMF_PUBLIC);
  464.  
  465.     recur    = OPT( CP_RECUR );
  466.     destname = av[ac - 1];
  467.  
  468.     if (ac < 3) {
  469.         ierr = 500;
  470.         goto done;
  471.     }
  472.     destisdir = isdir(destname);
  473.     if (ac > 3 && !destisdir) {
  474.         ierr = 507;
  475.         goto done;
  476.     }
  477.  
  478. /*
  479.  * copy set:                        reduce to:
  480.  *    file to file                     file to file
  481.  *    dir  to file (NOT ALLOWED)
  482.  *    file to dir                      dir to dir
  483.  *    dir  to dir                      dir to dir
  484.  *
  485.  */
  486.  
  487.     for ( i=1; i<ac-1 && !dobreak(); ++i) {
  488.         short srcisdir = isdir(av[i]);
  489.         if (srcisdir && has_wild && (ac >2)) /* hack to stop dir's from */
  490.             continue;                    /* getting copied if specified */
  491.                                          /* from wild expansion         */
  492.         if (srcisdir) {
  493.             BPTR srcdir, destdir;
  494.             if (!destisdir) {
  495.                 if (exists(destname)) {
  496.                     ierr = 507;    /* disallow dir to file */
  497.                     goto done;
  498.                     }
  499.                 if (destdir = CreateDir(destname)) UnLock(destdir);
  500.                 destisdir = 1;
  501.             }
  502.             if (!(destdir = Lock(destname, ACCESS_READ))) {
  503.                 ierr = 205;
  504.                 errstr = destname;
  505.                 goto done;
  506.             }
  507.             if (!(srcdir = Lock(av[i], ACCESS_READ))) {
  508.                 ierr = 205;
  509.                 errstr = av[i];
  510.                 UnLock(destdir);
  511.                 goto done;
  512.             }
  513.             ierr = copydir(srcdir, destdir, recur);
  514.             UnLock(srcdir);
  515.             UnLock(destdir);
  516.             if (ierr) break;
  517.         } else {                   /* FILE to DIR,   FILE to FILE   */
  518.             BPTR destdir, srcdir, tmp;
  519.             char *destfilename;
  520.  
  521.             srcdir = (BPTR)(Myprocess->pr_CurrentDir);
  522.  
  523.             if ((tmp = Lock(av[i], ACCESS_READ)) == NULL || !Examine(tmp,fib)) {
  524.                 if (tmp) UnLock(tmp);
  525.                 ierr = 205;
  526.                 errstr = av[i];
  527.                 goto done;
  528.             }
  529.             UnLock(tmp);
  530.             if (destisdir) {
  531.                 destdir = Lock(destname, ACCESS_READ);
  532.                 destfilename = fib->fib_FileName;
  533.             } else {
  534.                 destdir = srcdir;
  535.                 destfilename = destname;
  536.             }
  537.             ierr = copyfile(av[i], srcdir, destfilename, destdir);
  538.             if (destisdir) UnLock(destdir);
  539.             if (ierr) break;
  540.         }
  541.     }
  542.  
  543. done:
  544.  
  545.     FreeMem(fib, sizeof(FIB));
  546.     if (ierr) {
  547.         ierror(errstr, ierr);
  548.         return 20;
  549.     }
  550.     return 0;
  551. }
  552.  
  553.  
  554. static int
  555. copydir(BPTR srcdir, BPTR destdir, int recur)
  556. {
  557.     BPTR cwd;
  558.     BPTR destlock, srclock;
  559.     char *srcname;
  560.     FIB *srcfib;
  561.     int ierr=0;
  562.  
  563.     srcfib = (FIB *)SAllocMem((long)sizeof(FIB), MEMF_PUBLIC);
  564.     if( !Examine(srcdir, srcfib)) {
  565.         ierr=IoErr();
  566.         goto done;
  567.     }
  568.     srcname=srcfib->fib_FileName;
  569.  
  570.     while (ExNext(srcdir, srcfib)) {
  571.         if (CHECKBREAK())
  572.             break;
  573.         if (srcfib->fib_DirEntryType < 0) {
  574.             ierr = copyfile(srcname,srcdir,srcname,destdir);
  575.             if (ierr) break;
  576.         } else {
  577.             if ( srcfib->fib_DirEntryType!=ST_USERDIR ) {
  578.                 printf("%*s%s (Dir)....[Skipped. Is a link]\n", level * 6,
  579.                                " ",srcname);
  580.                 continue;
  581.             }
  582.             if (recur) {
  583.                 cwd = CurrentDir(srcdir);
  584.                 if (srclock = Lock(srcname, ACCESS_READ)) {
  585.                     CurrentDir(destdir);
  586.                     if (!(destlock = Lock(srcname,ACCESS_READ))) {
  587.                         destlock = CreateDir(srcname);
  588.                         printf("%*s%s (Dir)....[Created]\n", level * 6,
  589.                                " ",srcname);
  590.  
  591.                         /* UnLock and re Lock if newly created
  592.                          * for file_date() to work properly */
  593.                         if (destlock)
  594.                             UnLock(destlock);
  595.                         destlock = Lock(srcname, ACCESS_READ);
  596.                     } else
  597.                         printf("%*s%s (Dir)\n", level * 6," ",srcname);
  598.                     if (destlock) {
  599.                         level++;
  600.                         ierr = copydir(srclock, destlock, recur);
  601.                         level--;
  602.                         UnLock(destlock);
  603.                     } else
  604.                         ierr = IoErr();
  605.                     UnLock(srclock);
  606.                     if (!OPT(CP_NOFLAGS)) {
  607.                         SetProtection(srcname,srcfib->fib_Protection );
  608.                         if( *srcfib->fib_Comment )
  609.                             SetComment( srcname, srcfib->fib_Comment );
  610.                     }
  611.                 } else {
  612.                     ierr = IoErr();
  613.                 }
  614.                 CurrentDir(cwd);
  615.                 if (ierr)
  616.                     break;
  617.             }
  618.         }
  619.     }
  620. done:
  621.     FreeMem(srcfib, sizeof(FIB));
  622.     return ierr;
  623. }
  624.  
  625.  
  626.  
  627. #define COPYBUF 32768
  628.  
  629. static int
  630. copyfile(char *srcname, BPTR srcdir, char *destname, BPTR destdir)
  631. {
  632.     BPTR cwd;
  633.     BPTR f_src  = NULL;
  634.     BPTR f_dest = NULL;
  635.     long dest_mode;
  636.     long j;
  637.     int  stat,ierr=0,reset=0;
  638.     char *buf;
  639.     DPTR *dpd = NULL;
  640.     DPTR *dps;
  641.     long src_pbits;
  642.  
  643.     errstr    = "";
  644.  
  645.     if ( !(buf= (char *)AllocMem(COPYBUF, MEMF_PUBLIC|MEMF_CLEAR)))
  646.         return ERROR_NO_FREE_STORE; 
  647.  
  648.     cwd = CurrentDir(srcdir);
  649.  
  650.     if ( !(dps = dopen(srcname,&stat))) {
  651.         ierr = IoErr();
  652.         goto done; 
  653.     }
  654.     UnLock(dps->lock);
  655.     dps->lock = NULL;
  656.     src_pbits = dps->fib->fib_Protection ;
  657.     /*
  658.      * ensure src file can be read - if not, force it so (maybe)
  659.      */
  660.     if ( src_pbits & FIBF_READ) {            /* not readable */
  661.         if (!OPT(CP_OVERWRITE)) { 
  662.             errstr    = srcname; 
  663.             ierr    = ERROR_READ_PROTECTED ; 
  664.             goto done; 
  665.         }
  666.         if (!SetProtection(srcname, 0)) {    /* ----rwed */
  667.             pError(srcname); 
  668.             goto done; 
  669.         }
  670.         reset=1;
  671.     }
  672.  
  673.     if ( (f_src=Open(srcname, MODE_OLDFILE)) == NULL) { 
  674.         errstr    = srcname; 
  675.         ierr    = IoErr();
  676.         goto done; 
  677.     }
  678.  
  679.     CurrentDir(destdir);
  680.  
  681.     /*
  682.      *    if dest does not exist, open it with 'rwed' protections
  683.      *    if dest does exist, get its protections.
  684.      *    if -o option , set its protections to enable a write
  685.      *        (and reset them to original status when copied)      
  686.      *    if not -o , then error
  687.      */
  688.  
  689.     dest_mode    = MODE_NEWFILE;
  690.  
  691.     if( (dpd=dopen(destname, &stat)) != NULL) {
  692.         UnLock(dpd->lock);
  693.         dpd->lock  = NULL;
  694.  
  695.         if ( OPT(CP_OVERWRITE)) {
  696.             if (!SetProtection(destname, 0)) {    /* ----rwed */
  697.                 pError(destname);
  698.                 goto done;
  699.             }
  700.         } else if( dpd->fib->fib_Protection & FIBF_WRITE ) { /* not writeable */
  701.             errstr = destname;
  702.             ierr   = ERROR_WRITE_PROTECTED;
  703.             goto done;
  704.         }  
  705.     }
  706.  
  707.     /*
  708.      *  check if copying necessary (update, fresh)
  709.      */
  710.     if (OPT( CP_UPDATE | CP_FRESH )) {
  711.         if (!dpd && OPT(CP_FRESH)) {
  712.             if( !OPT(CP_QUIET))
  713.                 printf("%*s%s....not there\n",level*6," ",srcname);
  714.             goto done;
  715.         } else if ( dpd && dptrtosecs(dpd) >= dptrtosecs(dps) &&
  716.                      !Strcmp(dps->fib->fib_FileName, dpd->fib->fib_FileName)) {
  717.             if (!OPT(CP_QUIET))
  718.                 printf("%*s%s....not newer\n",level*6," ",srcname);
  719.             goto done;
  720.         }
  721.     }
  722.  
  723.     /*
  724.      *    now open dest file
  725.      */
  726.     if ( !(f_dest= Open(destname, dest_mode))) { 
  727.         errstr   = destname;
  728.         ierr     = IoErr();
  729.         goto done; 
  730.     }
  731.  
  732.     printf("%*s%s..",level*6," ",srcname);
  733.     fflush(stdout);
  734.  
  735.     /*
  736.      * now actually copy something :-)
  737.      */
  738.     while (j = Read(f_src, buf, COPYBUF)) {
  739.         if( dobreak() ) { 
  740.             ierr = 513;
  741.             break; 
  742.         }
  743.         if (Write(f_dest, buf, j) != j) { 
  744.             ierr = IoErr(); 
  745.             break; 
  746.         }
  747.     }
  748.     Close(f_src);
  749.     Close(f_dest);
  750.     f_src= f_dest= NULL;
  751.  
  752.     /*
  753.      *    replace original src protection bits - maybe 
  754.      */
  755.     if ( reset ) {
  756.         CurrentDir( srcdir );
  757.         if (!SetProtection(srcname, src_pbits )) {
  758.             pError(srcname);
  759.             goto done;
  760.         }
  761.         CurrentDir( destdir );
  762.     }
  763.  
  764.     if (!ierr) {
  765.         SetProtection( destname, src_pbits & ~FIBF_ARCHIVE);
  766.  
  767.         if (!OPT(CP_NODATE)) 
  768.             file_date(&dps->fib->fib_Date, destname);
  769.  
  770.         if (!OPT(CP_NOFLAGS)) {
  771.             if( *dps->fib->fib_Comment )
  772.                 SetComment( destname, dps->fib->fib_Comment );
  773.         }
  774.         if( OPT(CP_MOVE) ) {
  775.             CurrentDir(srcdir);
  776.             DeleteFile(srcname);
  777.             printf("..moved\n");
  778.         } else
  779.             printf("..copied\n");
  780.     } else {
  781.         DeleteFile(destname);
  782.     }
  783.  
  784. done:
  785.  
  786.     if ( f_src  )    Close(f_src);
  787.     if ( f_dest )    Close(f_dest);
  788.     if ( dps )       dclose(dps);
  789.     if ( dpd )       dclose(dpd);
  790.     if ( buf )       FreeMem(buf, COPYBUF);
  791.     CurrentDir(cwd);
  792.  
  793.     return ierr;
  794. }
  795.  
  796.  
  797. int
  798. do_touch( void )
  799. {
  800.     struct DateStamp ds;
  801.     int i;
  802.     DateStamp(&ds);
  803.     for (i=1; i<ac; i++)
  804.         if (file_date(&ds, av[i]))
  805.             ierror(av[i],500);
  806.         else 
  807.             clear_archive_bit( av[i] );
  808.     return 0;
  809. }
  810.  
  811.  
  812. static int
  813. file_date( struct DateStamp *date, char *name )
  814. {
  815.     long packargs[7];
  816.     struct MsgPort *task;
  817.     DPTR *tmp;
  818.     BPTR dirlock;
  819.     char *ptr;
  820.     int stat;
  821.  
  822.     if (!(task = (struct MsgPort *)DeviceProc(name))) return(1);
  823.     if (tmp = dopen(name, &stat)) {
  824.         dirlock = ParentDir(tmp->lock);
  825.         ptr=SAllocMem(65L,MEMF_PUBLIC);
  826.         CtoBStr(tmp->fib->fib_FileName,(ULONG)ptr >> 2L,64L);
  827.         dclose(tmp);
  828.         packargs[1]=dirlock;
  829.         packargs[2]=(ULONG)ptr >> 2L;
  830.         packargs[3]=(long)date;
  831.         SendPacket(ACTION_SET_DATE,packargs,task);
  832.         UnLock(dirlock);
  833.         FreeMem(ptr,65L);
  834.     }
  835.     return 0;
  836. }
  837.  
  838. int
  839. do_addbuffers( void )
  840. {
  841.     long packargs[7], i;
  842.     struct MsgPort *task;
  843.  
  844.     for( i=1; i<=ac-2; i+=2 ) {
  845.         if( i==ac-1 )
  846.             { ierror( av[i], 500 ); return 20; }
  847.         if( !(task=(struct MsgPort *)DeviceProc(av[i])))
  848.             { ierror(av[1],510); return 20; }
  849.         packargs[0]=myatoi(av[i+1],1,32767); if (atoierr) return 20;
  850.         SendPacket(ACTION_MORE_CACHE,packargs,task);
  851.     }
  852.     return 0;
  853. }
  854.  
  855. int
  856. do_relabel( void )
  857. {
  858.     long packargs[7];
  859.     struct MsgPort *task=(struct MsgPort *)DeviceProc(av[1]);
  860.     char *ptr;
  861.  
  862.     if (!task) { ierror(av[1],510); return 20; }
  863.     ptr=SAllocMem(65L,MEMF_PUBLIC);
  864.     CtoBStr(av[2],(ULONG)ptr >> 2L,64L);
  865.     packargs[0]=(ULONG)ptr >> 2L;
  866.     SendPacket(ACTION_RENAME_DISK,packargs,task);
  867.     FreeMem(ptr,65L);
  868.     Delay(10);
  869.     changedisk(task);
  870.     return 0;
  871. }
  872.  
  873. int
  874. do_diskchange( void )
  875. {
  876.     struct MsgPort *task=(struct MsgPort *)DeviceProc(av[1]);
  877.  
  878.     if (!task) { ierror(av[1],510); return 20; }
  879.     changedisk(task);
  880.     return 0;
  881. }
  882.  
  883. static void
  884. changedisk(struct MsgPort *task)
  885. {
  886.     long packargs[7];
  887.  
  888.     packargs[0]= DOSTRUE;
  889.     SendPacket(ACTION_INHIBIT,packargs,task);
  890.     packargs[0]= DOSFALSE;
  891.     SendPacket(ACTION_INHIBIT,packargs,task);
  892. }
  893.  
  894.  
  895. extern int atoierr;
  896.  
  897. static int
  898. func_array( char *fav[], int fac)
  899. {
  900.     char *ret;
  901.     if( atoierr ) return 20;
  902.     if( fac ) {
  903.         ret=compile_av( fav, 0, fac, 0xa0, 0);
  904.         set_var( LEVEL_SET, v_value, ret );
  905.         free( ret );
  906.     } else 
  907.         unset_var( LEVEL_SET, v_value );
  908.     return 0;
  909. }
  910.  
  911. static int
  912. func_int( int i )
  913. {
  914.     char buf[12];
  915.     if( atoierr ) return 20;
  916.     sprintf(buf,"%d",i);
  917.     set_var( LEVEL_SET, v_value, buf );
  918.     return 0;
  919. }
  920.  
  921. static int
  922. func_bool( int i )
  923. {
  924.     if( atoierr ) return 20;
  925.     set_var( LEVEL_SET, v_value, i ? "1" : "0" );
  926.     return 0;
  927. }
  928.  
  929. static int
  930. func_string( char *str )
  931. {
  932.     if( atoierr ) return 20;
  933.     set_var( LEVEL_SET, v_value, str ? str : "" );
  934.     return 0;
  935. }
  936.  
  937. static int
  938. commas( char *av[], int ac, int n )
  939. {
  940.     int i=0;
  941.  
  942.     while( --ac>=0 )
  943.         if( !strcmp( *av++, ",") )
  944.             i++;
  945.     if( i-=n )
  946.         fprintf( stderr, "Need %d comma%s\n", n, (n==1) ? "" : "s" );
  947.     return i;
  948. }
  949.  
  950. static int
  951. wordset( char *av[], int ac, char **(*func)(char **,int,char**,int,int*,int) )
  952. {
  953.     char **av1=av, **av2;
  954.     int  ac1=0, ac2, ret;
  955.  
  956.     if( commas( av, ac, 1 ) ) return 20;
  957.     while( strcmp( *av++, ",") ) ac1++;
  958.     av2=av, ac2=ac-ac1-1;
  959.     av=(*func)( av1, ac1, av2, ac2, &ac, 0 );
  960.     ret=func_array( av, ac );
  961.     free( av );
  962.     return ret;
  963. }
  964.  
  965. static int
  966. split_arg( char **av, int ac )
  967. {
  968.     char **arr, **old, *arg;
  969.     int i, j=1, ret;
  970.  
  971.     for( i=strlen(av[0])-1; i>=0; i-- )
  972.         if( av[0][i]==' ' )
  973.             av[0][i]=0, j++;
  974.  
  975.     arr=old=(char **)salloc( j*sizeof( char * ) );
  976.     arg=*av;
  977.     for( ; j>0; j-- ) {
  978.         *arr++=arg;
  979.         arg+=strlen(arg)+1;
  980.     }
  981.     ret=func_array( old, arr-old );
  982.     free(old);
  983.     return ret;
  984. }
  985.  
  986. static char *
  987. info_part( char **av, int ac, int n, char *buf )
  988. {
  989.     char *str;
  990.     DPTR *dp;
  991.     int  len=0, i, t;
  992.  
  993.     buf[0]=0;
  994.     while( --ac>=0 ) {
  995.         if( dp=dopen( *av++, &t) ) {
  996.             if( n==0 ) {
  997.                 for (str=buf, i=7; i>=0; i--)
  998.                     *str++= (dp->fib->fib_Protection & (1L<<i) ?
  999.                             "hspa----" : "----rwed")[7-i];
  1000.                 *str=0;
  1001.             } else
  1002.                 len+= dp->fib->fib_NumBlocks+1;
  1003.             dclose( dp );
  1004.         }
  1005.     }
  1006.     if( n ) sprintf(buf, "%d", len);
  1007.     return buf;
  1008. }
  1009.  
  1010. static struct FileRequester freq;
  1011.  
  1012. static char *
  1013. file_request(char **av, int ac, char *path)
  1014. {
  1015.     struct FileRequester fr;
  1016.     char filebuf[128];
  1017.  
  1018.     path[0]=0; filebuf[0]=0;
  1019.     fr=freq; /* clear everything */
  1020.     fr.fr_Hail="";
  1021.     if( ac>0 ) {
  1022.         fr.fr_Hail=av[0];
  1023.         if( ac>1 ) {
  1024.             strcpy( path,av[1]);
  1025.             if( ac>2 )
  1026.                 strcpy(filebuf,av[2]);
  1027.         }
  1028.     }
  1029.     fr.fr_File  = filebuf;
  1030.     fr.fr_Dir   = path;
  1031.     fr.fr_Flags2= FR2F_LongPath;
  1032.     if( !FileRequest( &fr ) )
  1033.         return NULL;
  1034.     TackOn( path,filebuf );
  1035.     return path;
  1036. }
  1037.  
  1038. int
  1039. confirm( char *title, char *file )
  1040. {
  1041.     char buf[80];
  1042.  
  1043.     buf[0]=0;
  1044.  
  1045.     if( !confirmed ) {
  1046.         fprintf(stderr,"%s %s%-16s%s [YES/no/all/done] ? ",
  1047.                         title,o_hilite,file,o_lolite);
  1048.         strupr(fgets(buf,80,stdin));
  1049.         if( *buf=='A' )
  1050.             confirmed=1;
  1051.         if( *buf=='D' || breakcheck() )
  1052.             confirmed=2;
  1053.     }
  1054.  
  1055.     if( confirmed==2 )
  1056.         return 0;
  1057.     return confirmed || *buf != 'N';
  1058. }
  1059.  
  1060. enum funcid {
  1061.     FN_STUB=1, FN_MEMBER, FN_DIRS, FN_NAMEEXT, FN_NAMEROOT, FN_FILES,
  1062.     FN_FILELEN, FN_SORTARGS, FN_UPPER, FN_WORDS, FN_ABBREV, FN_ABS,
  1063.     FN_BINTODEC, FN_CENTER, FN_COMPARE, FN_DECTOHEX, FN_DELWORD,
  1064.     FN_DELWORDS, FN_EXISTS, FN_INDEX, FN_STRCMP, FN_SUBWORDS,
  1065.     FN_WORD, FN_MIN, FN_MAX, FN_DRIVES, FN_WITHOUT, FN_UNION, FN_INTERSECT,
  1066.     FN_AVAILMEM, FN_UNIQUE, FN_RPN, FN_CONCAT, FN_SPLIT, FN_DRIVE,
  1067.     FN_FILEPROT, FN_FILEBLKS, FN_LOWER, FN_HOWMANY, FN_COMPLETE, FN_FIRST,
  1068.     FN_LAST, FN_MATCH, FN_CLINUM, FN_FREEBYTES, FN_FREEBLKS, FN_INFO,
  1069.     FN_MEGS, FN_FREESTORE, FN_CHECKPORT, FN_PICKOPTS, FN_PICKARGS,
  1070.     FN_FILEREQ, FN_VOLUME, FN_LOOKFOR, FN_APPSUFF, FN_DIRNAME, FN_AGE,
  1071.     FN_GETCLASS, FN_CONFIRM, FN_WINWIDTH, FN_WINHEIGHT, FN_WINTOP,
  1072.     FN_WINLEFT, FN_CONSOLE, FN_SORTNUM, FN_IOERROR, FN_TRIM, FN_MOUNTED,
  1073.     FN_RND, FN_DIRSTR, FN_MIX, FN_WINROWS, FN_WINCOLS, FN_SUBFILE,
  1074.     FN_SCRHEIGHT, FN_SCRWIDTH, FN_FLINES
  1075. };
  1076.  
  1077. #define MAXAV        30000        /* Max. # of arguments            */
  1078.  
  1079. struct FUNCTION {
  1080.     short id, minargs, maxargs;
  1081.     char *name;
  1082. } Function[]={
  1083. FN_ABBREV,   2, 3,     "abbrev",
  1084. FN_ABS,      1, 1,     "abs",
  1085. FN_AGE,      1, 1,     "age",
  1086. FN_APPSUFF,  2, 2,     "appsuff",
  1087. FN_PICKARGS, 0, MAXAV, "arg",
  1088. FN_AVAILMEM, 0, 1,     "availmem",
  1089. FN_STUB,     1, 1,     "basename",
  1090. FN_CENTER,   2, 2,     "center",
  1091. FN_CHECKPORT,1, 1,     "checkport",
  1092. FN_CLINUM,   1, 1,     "clinum",
  1093. FN_COMPLETE, 1, MAXAV, "complete",
  1094. FN_CONCAT,   0, MAXAV, "concat",
  1095. FN_CONFIRM,  1, MAXAV, "confirm",
  1096. FN_CONSOLE,  1, 1,     "console",
  1097. FN_DECTOHEX, 1, 1,     "dectohex",
  1098. FN_DELWORD,  1, MAXAV, "delword",
  1099. FN_DELWORDS, 2, MAXAV, "delwords",
  1100. FN_DIRNAME,  1, 1,     "dirname",
  1101. FN_DIRS,     0, MAXAV, "dirs",
  1102. FN_DIRSTR,   2, 2,     "dirstr",
  1103. FN_DRIVE,    1, 1,     "drive",
  1104. FN_DRIVES,   0, 0,     "drives",
  1105. FN_EXISTS,   1, 1,     "exists",
  1106. FN_FILEBLKS, 1, MAXAV, "fileblks",
  1107. FN_FILELEN,  0, MAXAV, "filelen",
  1108. FN_FILEPROT, 1, 1,     "fileprot",
  1109. FN_FILEREQ,  0, 3,     "filereq",
  1110. FN_FILES,    0, MAXAV, "files",
  1111. FN_FIRST,    0, MAXAV, "first",
  1112. FN_FLINES,   1, 1,     "flines",
  1113. FN_FREEBLKS, 1, 1,     "freeblks",
  1114. FN_FREEBYTES,1, 1,     "freebytes",
  1115. FN_FREESTORE,1, 1,     "freestore",
  1116. FN_STUB,     1, 1,     "getenv",
  1117. FN_GETCLASS, 1, 1,     "getclass",
  1118. FN_HOWMANY,  0, 0,     "howmany",
  1119. FN_IOERROR,  1, 1,     "ioerr",
  1120. FN_INDEX,    2, 2,     "index",
  1121. FN_INFO,     1, 1,     "info",
  1122. FN_INTERSECT,1, MAXAV, "intersect",
  1123. FN_LAST,     0, MAXAV, "last",
  1124. FN_LOOKFOR,  2, 2,     "lookfor",
  1125. FN_LOWER,    0, MAXAV, "lower",
  1126. FN_MATCH,    1, MAXAV, "match",
  1127. FN_MAX,      1, MAXAV, "max",
  1128. FN_MEGS,     1, 1,     "megs",
  1129. FN_MEMBER,   1, MAXAV, "member",
  1130. FN_MIN,      1, MAXAV, "min",
  1131. FN_MIX,      0, MAXAV, "mix",
  1132. FN_MOUNTED,  1, 1,     "mounted",
  1133. FN_NAMEEXT,  1, 1,     "nameext",
  1134. FN_NAMEROOT, 1, 1,     "nameroot",
  1135. FN_PICKOPTS, 0, MAXAV, "opt",
  1136. FN_DIRNAME,  1, 1,     "pathname",
  1137. FN_PICKARGS, 0, MAXAV, "pickargs",
  1138. FN_PICKOPTS, 0, MAXAV, "pickopts",
  1139. FN_RND,      0, 1,     "rnd",
  1140. FN_RPN,      1, MAXAV, "rpn",
  1141. FN_SCRHEIGHT,0, 0,     "scrheight",
  1142. FN_SCRWIDTH, 0, 0,     "scrwidth",
  1143. FN_SORTARGS, 0, MAXAV, "sortargs",
  1144. FN_SORTNUM,  0, MAXAV, "sortnum",
  1145. FN_SPLIT,    0, MAXAV, "split",
  1146. FN_STRCMP,   2, 2,     "strcmp",
  1147. FN_STUB,     2, 2,     "strhead",
  1148. FN_STUB,     2, 2,     "strleft",
  1149. FN_STUB,     2, 3,     "strmid",
  1150. FN_STUB,     2, 2,     "strright",
  1151. FN_STUB,     2, 2,     "strtail",
  1152. FN_SUBFILE,  3, 3,     "subfile",
  1153. FN_SUBWORDS, 2, MAXAV, "subwords",
  1154. FN_STUB,     2, 2,     "tackon",
  1155. FN_TRIM,     0, MAXAV, "trim",
  1156. FN_UNION,    1, MAXAV, "union",
  1157. FN_UNIQUE,   0, MAXAV, "unique",
  1158. FN_UPPER,    0, MAXAV, "upper",
  1159. FN_VOLUME,   1, 1,     "volume",
  1160. FN_WINCOLS,  0, 0,     "wincols",
  1161. FN_WINHEIGHT,0, 0,     "winheight",
  1162. FN_WINLEFT,  0, 0,     "winleft",
  1163. FN_WINROWS,  0, 0,     "winrows",
  1164. FN_WINTOP,   0, 0,     "wintop",
  1165. FN_WINWIDTH, 0, 0,     "winwidth",
  1166. FN_WITHOUT,  1, MAXAV, "without",
  1167. FN_WORD,     1, MAXAV, "word",
  1168. FN_WORDS,    0, MAXAV, "words",
  1169. 0,           0, NULL
  1170. };
  1171.  
  1172. extern char shellctr[];
  1173. extern int  w_width, w_height;
  1174.  
  1175. int
  1176. dofunc( int id, char **av, int ac)
  1177. {
  1178.     char **oldav=av, **get=av, buf[200], *str=buf, *t;
  1179.     int oldac=ac, i=0, j=0, n=0, n2=1, l;
  1180.     buf[0]=0;
  1181.     av[ac]=NULL;
  1182.     atoierr=0;
  1183.  
  1184.     switch( id ) {
  1185.     case FN_ABBREV:
  1186.         if( ac==3 ) i=posatoi(av[2] ); else i=strlen(av[0]);
  1187.         return func_bool( !Strncmp( av[0], av[1], i ));
  1188.     case FN_ABS:
  1189.         i=unlatoi(av[0]);
  1190.         return func_int( i>= 0 ? i : -i );
  1191.     case FN_AGE: {
  1192.         struct DateStamp ds; long time;
  1193.         DateStamp( &ds ); if( ds.ds_Days==0 ) return 99999;
  1194.         if( !(time=timeof(av[0]))) return 99999;
  1195.         return func_int( (ds.ds_Days*86400+ds.ds_Minute*60-time)/86400 ); }
  1196.     case FN_APPSUFF:
  1197.         strcpy(buf,av[0]);
  1198.         l=strlen(av[0])-strlen(av[1]);
  1199.         if( l<0 || Strcmp(av[0]+l,av[1])) {
  1200.             strcat(buf,".");
  1201.             strcat(buf,av[1]);
  1202.         }
  1203.         return func_string( buf );
  1204.     case FN_AVAILMEM:
  1205.         if( ac==1 && !Strcmp(av[0],"chip")) n=MEMF_CHIP;
  1206.         if( ac==1 && !Strcmp(av[0],"fast")) n=MEMF_FAST;
  1207.         return func_int( AvailMem( n ));
  1208.     case FN_CENTER:
  1209.         if( (n=posatoi( av[1] )) > (l=strlen(av[0])) ) i=(n-l)/2, j=n-i-l;
  1210.         sprintf( buf, "%*s%s%*s", i,"",av[0], j,"" );
  1211.         return func_string( buf );
  1212.     case FN_CHECKPORT:
  1213.         return func_bool( (int)FindPort( av[0] ) );
  1214.     case FN_GETCLASS:
  1215.         if( str=getclass(av[0]) )
  1216.             if( str=index(strncpy( buf,str,100 ),0xA0) )
  1217.                 *str=0;
  1218.         return func_string(buf);
  1219.     case FN_COMPLETE:
  1220.         for( i=1, l=strlen(av[0]); i<ac; i++ )
  1221.             if( !Strncmp( av[0], av[i], l ) )
  1222.                 { str=av[i]; break; }
  1223.         return func_string( str );
  1224.     case FN_CONCAT:
  1225.         return func_string( compile_av( av, 0, ac, ' ', 1));
  1226.     case FN_CONFIRM:
  1227.         for( i=1, get++, confirmed=0; i<ac; i++ )
  1228.             if( confirm( av[0], av[i]) )
  1229.                 *get++=av[i];
  1230.         return func_array( av+1, (get-av)-1 );
  1231.     case FN_CONSOLE:
  1232.         if( !strcmp(av[0],"STDIN")) i=isconsole(Input());
  1233.         else if( !strcmp(av[0],"STDOUT")) i=isconsole(Output());
  1234.         return func_bool(i);
  1235.     case FN_DECTOHEX:
  1236.         sprintf( buf, "%x", unlatoi( av[0] ));
  1237.         return func_string( buf );
  1238.     case FN_DELWORDS:
  1239.         n2=posatoi( av[--ac] ); if( atoierr ) return 20;
  1240.     case FN_DELWORD:
  1241.         n=posatoi( av[--ac] )-1;
  1242.         for( ; i<ac && i<n; i++ ) *av++=*get++;
  1243.         for( ; i<ac && i<n+n2; i++ ) get++;
  1244.         for( ; i<ac ; i++ ) *av++=*get++;
  1245.         return func_array( oldav, av-oldav );
  1246.     case FN_DIRNAME:
  1247.         str=av[0]+strlen(av[0])-1;
  1248.         while( str>av[0] && *str!=':' && *str!='/' ) str--;
  1249.         if( *str==':' ) str++;
  1250.         *str=0;
  1251.         return func_string(av[0]);
  1252.     case FN_DIRS:
  1253.         for( ; --ac>=0; get++ )
  1254.             if( exists( *get ) && isdir( *get ) )
  1255.                 *av++=*get;
  1256.         return func_array( oldav, av-oldav );
  1257.     case FN_DIRSTR:
  1258.         if( av=expand(oldav[1],&n2)) {
  1259.             lformat(oldav[0], buf, (FILEINFO *)av[0]-1);
  1260.             free_expand(av);
  1261.         }
  1262.         if( str=index(buf,'\n' )) *str=0;
  1263.         return func_string( buf );
  1264.     case FN_DRIVE:
  1265.         return func_string( drive_name( av[0] ) );
  1266.     case FN_DRIVES:
  1267.         get_drives( buf );
  1268.         return func_string( buf );
  1269.     case FN_EXISTS:
  1270.         return func_bool( exists( av[0] ));
  1271.     case FN_FILEBLKS:
  1272.         return func_string( info_part( av, ac, 1, buf ) );
  1273.     case FN_FILELEN:
  1274.         while( --ac>=0 )
  1275.             i+=filesize( *av++ );
  1276.         return func_int( i );
  1277.     case FN_FLINES:
  1278.         if( str=get_var(LEVEL_SET, av[0]))
  1279.             for( n=1; str=index(str,0xA0); str++, n++ ) ;
  1280.         return func_int( n );
  1281.     case FN_FILEPROT:
  1282.         return func_string( info_part( av, ac, 0, buf ) );
  1283.     case FN_FILEREQ:
  1284.         return func_string( file_request( av, ac, buf ) );
  1285.     case FN_FILES:
  1286.         for( ; --ac>=0; get++ )
  1287.             if( exists( *get ) && !isdir( *get ) )
  1288.                 *av++=*get;
  1289.         return func_array( oldav, av-oldav );
  1290.     case FN_FIRST:
  1291.         return func_string( av[0] );
  1292.     case FN_FREEBLKS:
  1293.         return func_string( oneinfo( av[0], 3 ));
  1294.     case FN_FREEBYTES:
  1295.         return func_string( oneinfo( av[0], 2 ));
  1296.     case FN_FREESTORE:
  1297.         return func_string( oneinfo( av[0], 4 ));
  1298.     case FN_HOWMANY:
  1299.         Getenv( shellctr, buf, 10);
  1300.         return func_string( buf );
  1301.     case FN_IOERROR:
  1302.         return func_string( ioerror( atoi( av[0] )));
  1303.     case FN_INDEX:
  1304.         str=strstr( av[0], av[1] );
  1305.         return func_int( str ? (str-av[0])+1 : 0 );
  1306.     case FN_INFO:
  1307.         return func_string( oneinfo( av[0], 1 ));
  1308.     case FN_INTERSECT:
  1309.         return wordset( av, ac, and );
  1310.     case FN_LAST:
  1311.         return func_string( ac ? av[ac-1] : "" );
  1312.     case FN_LOOKFOR:
  1313.         return func_string( dofind( av[0], "", buf, av[1]));
  1314.     case FN_LOWER:
  1315.         while( --ac>=0 ) strlwr( *av++ );
  1316.         return func_array( oldav, av-oldav );
  1317.     case FN_MATCH:
  1318.         { PATTERN *pat=compare_preparse( av[--ac], 0 );
  1319.         for( ; --ac>=0; get++ )
  1320.             if( compare_ok( pat, *get ) )
  1321.                 *av++=*get;
  1322.         compare_free(pat);
  1323.         return func_array( oldav, av-oldav );
  1324.         }
  1325.     case FN_MAX:
  1326.         for( n=MININT; i<ac; i++ ) 
  1327.             { if( (j=unlatoi(av[i] )) > n ) n=j; if( atoierr ) return 20; }
  1328.         return func_int( n );
  1329.     case FN_MEGS:
  1330.         return func_string( itok( atoi( av[0] )));
  1331.     case FN_MEMBER:
  1332.         for( i=1; i<ac && Strcmp(av[0],av[i]) ; i++ ) ;
  1333.         return func_bool( ac!=i );
  1334.     case FN_MIN:
  1335.         for( n=MAXINT; i<ac; i++ )
  1336.             { if( (j=unlatoi(av[i] )) < n ) n=j; if( atoierr ) return 20; }
  1337.         return func_int( n );
  1338.     case FN_MIX:
  1339.         for( ; i<ac; i++ )
  1340.             j=rand()%ac, str=av[j], av[j]=av[i], av[i]=str;
  1341.         return func_array(av,ac);
  1342.     case FN_MOUNTED:
  1343.         return func_bool((int)mounted(av[0]));
  1344.     case FN_NAMEEXT:
  1345.         return func_string( rindex(av[0],'.')?rindex(av[0],'.')+1:(char *)NULL);
  1346.     case FN_NAMEROOT:
  1347.         if( rindex(av[0],'.') ) *rindex(av[0],'.')=0;
  1348.         return func_string( av[0] );
  1349.     case FN_PICKARGS:
  1350.         while( *get && **get=='-' ) get++;
  1351.         while( *get )  *av++=*get++ ;
  1352.         return func_array( oldav, av-oldav );
  1353.     case FN_PICKOPTS:
  1354.         while( *get && **get=='-' ) *av++=*get++;
  1355.         return func_array( oldav, av-oldav );
  1356.     case FN_CLINUM:
  1357.         return func_int( clinum( av[0] ) );
  1358.     case FN_RND:
  1359.         /* if( ac>0 ) srand(atoi(av[0])); */
  1360.         return func_int( rand() );
  1361.     case FN_RPN:
  1362.         return func_int( eval_rpn( av, ac, 1 ));
  1363.     case FN_SCRHEIGHT:
  1364.         return func_int( Win ? Win->WScreen->Height: 0 );
  1365.     case FN_SCRWIDTH:
  1366.         return func_int( Win ? Win->WScreen->Width : 0 );
  1367.     case FN_SORTARGS:
  1368.         QuickSort( av, ac );
  1369.         return func_array( av, ac );
  1370.     case FN_SORTNUM:
  1371.         DirQuickSort( av, ac, numcmp, 0, 0 );
  1372.         return func_array( av, ac );
  1373.     case FN_SPLIT:
  1374.         return split_arg( av, ac );
  1375.     case FN_STRCMP:
  1376.         return func_int( strcmp( av[0], av[1] ) );
  1377.     case FN_TRIM:
  1378.         for( ; *av; av++ ) {
  1379.             for( ; **av==' '; (*av)++ ) ;
  1380.             for( str=*av+strlen(*av); str>*av && str[-1]==' '; *--str=0 ) ;
  1381.         }
  1382.         return func_array( oldav, av-oldav );
  1383.     case FN_UNION:
  1384.         return wordset( av, ac, or );
  1385.     case FN_UNIQUE:
  1386.         QuickSort( av, ac );
  1387.         while( *get )
  1388.             { *av++=*get++; while( *get && !Strcmp(*get,*(get-1)))
  1389.                 get++; }
  1390.         return func_array( oldav, av-oldav );
  1391.     case FN_UPPER:
  1392.         while( --ac>=0 ) strupr( *av++ );
  1393.         return func_array( oldav, oldac );
  1394.     case FN_VOLUME:
  1395.         return func_string( oneinfo( av[0], 5 ));
  1396.     case FN_WINCOLS:
  1397.         newwidth();
  1398.         return func_int(w_width);
  1399.     case FN_WINTOP:
  1400.         return func_int( Win ? Win->TopEdge : 0 );
  1401.     case FN_WINLEFT:
  1402.         return func_int( Win ? Win->LeftEdge: 0 );
  1403.     case FN_WINHEIGHT:
  1404.         return func_int( Win ? Win->Height : 0 );
  1405.     case FN_WINROWS:
  1406.         newwidth();
  1407.         return func_int(w_height);
  1408.     case FN_WINWIDTH:
  1409.         return func_int( Win ? Win->Width  : 0 );
  1410.     case FN_WORD:
  1411.         n2=1; goto wordf;
  1412.     case FN_SUBFILE:
  1413.         n2= posatoi( av[2] ); if( atoierr ) return 20;
  1414.         n = posatoi( av[1] ); if( atoierr ) return 20;
  1415.         if( !(str=get_var( LEVEL_SET, av[0] ))) return 20;
  1416.         for( i=1; i<n; i++ )
  1417.             str= (str=index(str,0xA0)) ? str+1 : "";
  1418.         for( t=str, i=0; i<n2; i++ )
  1419.             if(t=index(t+1,0xA0));
  1420.         if( t ) *t=0;
  1421.         set_var( LEVEL_SET, v_value, str );
  1422.         if( t ) *t=0xA0;
  1423.         return 0;
  1424.     case FN_SUBWORDS:
  1425.         n2=posatoi( av[--ac] ); if( atoierr ) return 20;
  1426.     wordf:
  1427.         n=posatoi( av[--ac] )-1; if( atoierr ) return 20;
  1428.         for( i=0; i<ac && i<n; i++ ) get++;
  1429.         for(    ; i<ac && i<n+n2; i++ ) *av++=*get++;;
  1430.         return func_array( oldav, av-oldav );
  1431.     case FN_WITHOUT:
  1432.         return wordset( av, ac, without );
  1433.     case FN_WORDS:
  1434.         return func_int( ac );
  1435.     }
  1436.     return func_string( "" );
  1437. }
  1438.