home *** CD-ROM | disk | FTP | other *** search
- /**
- *
- * ZAP Delete a subdirectory and all its contents
- *
- * This program deletes all files from the specified
- * directory and then removes the directory. If a directory
- * is encountered in the subdirectory, it is processed and
- * removed as well. ZAP is invoked with the command
- *
- * zap directory
- *
- * where "directory" is any directory on the system.
- * ZAP reports disk space allocation and the total number
- * of files deleted. ZAP removes an entire directory, so you
- * can easily remove a large number of files rapidly . . . use
- * caution when using ZAP.
- *
- * If a DOS critical error occurs (such as disk drive door open)
- * the error will be reported and the user will be asked to
- * correct the condition and press a key to continue.
- *
- * Version 3.0 (C)Copyright Blaise Computing Inc. 1983, 1984, 1986
- *
- * Version 3.02 March 23, 1987
- * Corrected test for 0xff in write_str() by casting 0xff
- * to char.
- * Prevented overflow of report[] array in crit_err().
- * Removed call to itoa() (which is not supported by
- * Lattice C versions 3.00-3.11.)
- *
- **/
-
- #include <ctype.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
-
- #include <bfile.h>
- #include <bdirect.h>
- #include <bisr.h>
- #include <bprogctl.h>
- #include <bquery.h>
-
- #if MSC300
- #include <malloc.h>
- #include <memory.h>
- #endif
-
- #define OK_EXIT 0 /* Exit codes */
- #define BAD_USAGE 101
- #define DIR_NOT_FOUND 102
- #define INSUFF_MEMORY 103
- #define ILL_FILENAME 104
-
- #define STACKSIZE 3000 /* Items for the critical */
- #define NUMSTACKS 2 /* error ISR. */
- ISRCTRL crit_blk = {0};
- char crit_stk[STACKSIZE*NUMSTACKS] = {0};
-
- int deldir(char *);
- void crit_err(ALLREG *,ISRCTRL *,ISRMSG *);
- void write_str(char *);
- void wait_key(void);
-
- void main(argc,argv)
- int argc;
- char *argv[];
- {
- int dfiles;
- char *pdirectory,pdir[MAX_FLEN];
- int last;
- unsigned aclus,tclus,bsec,sclus;
- FSPEC finfo;
- long freebytes;
- char old_dir[MAX_FLEN];
- int old_drive;
-
- /* Install DOS critical error handler. */
-
- isinstal(0x24,crit_err,"ZAP3.02CRIT ERR",
- &crit_blk,crit_stk,STACKSIZE,NUMSTACKS);
-
- /* Display welcome message and check for directory */
-
- printf("ZAP Subdirectory Removal\n");
- printf("Version 3.02");
- printf(" (C)Copyright Blaise Computing Inc. 1983, 1984, 1986\n\n");
-
- if (argc != 2)
- {
- printf("ZAP requires a subdirectory specification ");
- printf("and is invoked by:\n");
- printf("\n zap directory\n");
- pcexit(BAD_USAGE);
- }
-
- pdirectory = argv[1];
- /* Check for valid directory */
- /* name. */
- if (flnorm(pdirectory,pdir,&last))
- {
- printf("Illegal directory name: <%s>\n",pdir);
- pcexit(ILL_FILENAME);
- }
- if (pdir[last] != '\0') /* If not root directory, */
- /* make sure it exists. */
- if (drsfirst(pdir,AT_DIR,&finfo))
- {
- printf("Directory <%s> could not be found.\n",pdir);
- pcexit(DIR_NOT_FOUND);
- }
-
- /* Switch to target drive */
-
- old_drive = drretdrv();
- drsetdrv(toupper((int) pdir[0]) - 'A');
-
- /* Save initial directory on target drive */
-
- drcurdir(0,old_dir);
-
- /* Report disk space utilization */
-
- freebytes = qydisksp(0,&aclus,&tclus,&bsec,&sclus);
- printf("Total Disk Space : %8ld\n",
- (long)tclus * (long)sclus * (long)bsec);
- printf("Free Space : %8ld\n",freebytes);
-
- /* Process the directory */
-
- dfiles = deldir(pdir);
-
- /* Report overall results */
-
- freebytes = qydisksp(0,&aclus,&tclus,&bsec,&sclus);
- printf("Files Deleted : %8d\n", dfiles);
- printf("Free Space : %8ld\n",freebytes);
-
- /* Restore initial directory on target drive (if it's still */
- /* there). */
-
- drchdir(old_dir);
-
- /* Restore original drive. */
-
- drsetdrv(old_drive);
-
- pcexit(OK_EXIT);
- }
-
- /**
- *
- * Name deldir -- Delete all files and remove the directory
- *
- * Synopsis dfiles = deldir(pdir);
- *
- * int dfiles The number of files deleted
- * char *pdir The directory name
- *
- * Description This function deletes all files in the specified
- * directory. If a subdirectory is encountered, it is
- * processed by calling deldir() recursively.
- *
- * Method The DOS communication area which is used by DRSFIRST and
- * DRSNEXT is saved when calling deldir() recursively,
- * because the area is statically defined by DRSFIRST.
- *
- * Returns dfiles The total number (in all sub-
- * directories) of files deleted.
- *
- **/
-
- int deldir(pdir)
- char *pdir;
- {
- int ercode,dfiles;
- FSPEC info;
- char dirname[MAX_FLEN];
- int last;
-
- char loc_area[sizeof(b_drbuffer)];
-
- if (ercode = drchdir(pdir)) /* Down to subdirectory */
- {
- printf("Error %d changing to directory %s\n",ercode,pdir);
- pcexit(ercode);
- }
- printf("\nProcessing %s\n",pdir);
-
- dfiles = 0;
- /* Begin the search */
- for (ercode = drsfirst("*.*",
- AT_RDONLY+AT_HIDDEN+AT_SYSTEM+AT_DIR,
- &info);
- ercode == 0;
- ercode = drsnext(&info))
- {
- if (info.fname[0] != '.')
- if (info.fattr == AT_DIR) /* Another directory */
- {
- memcpy(loc_area,b_drbuffer,sizeof(b_drbuffer));
- dfiles += deldir(info.fname);
- memcpy(b_drbuffer,loc_area,sizeof(b_drbuffer));
- }
- else
- {
- if (info.fattr & AT_RDONLY) /* Must reset attribute */
- /* before deleting. */
- flsetatr(info.fname,AT_GENERAL);
-
- if (ercode = fldelete(info.fname))
- printf(" Error %d deleting file %s\n",
- ercode,
- info.fname);
- else
- {
- printf(" %s\n",info.fname);
- dfiles++;
- }
- }
- }
-
- if ( 0 == flnorm(pdir,dirname,&last)
- && '\0' != dirname[last])
- { /* This wasn't the root */
- drchdir("..");
- ercode = drrmdir(pdir); /* Now remove the directory */
- }
- else
- ercode = 0;
-
- if (ercode == 0)
- printf(" %s processing complete.\n\n",pdir);
- else
- printf(" Error %d removing directory %s\n\n",ercode,pdir);
-
- return(dfiles);
- }
-
- /**
- *
- * Name crit_err -- Handle DOS critical error
- *
- * Synopsis (To be called only from ISR dispatcher.)
- *
- * crit_err(pregs,pisrblk,pmsg);
- *
- * ALLREG *pregs Pointer to structure containing
- * register values in effect when
- * interrupt occurred.
- * ISRCTRL *pisrblk Pointer to control block for this
- * ISR.
- * ISRMSG *pmsg Pointer to structure containing
- * messages between this ISR and the
- * dispatcher.
- *
- * Description This function analyzes the information provided by DOS
- * in the machine registers, reports the error and the
- * device on which it occurs, and awaits a keystroke. It
- * returns to DOS in such a way as to request another
- * retry.
- *
- * Returns pregs->ax.hl.l Value 1, thereby requesting a retry.
- *
- **/
-
- void crit_err(pregs,pisrblk,pmsg)
- ALLREG *pregs;
- ISRCTRL *pisrblk;
- ISRMSG *pmsg;
- {
- ADS dev_header_ads;
- DEV_HEADER dh;
- ADS dh_ads;
- char report[100];
- int bad_fat;
-
- static char *message[] =
- {
- "Attempt to write on write-protected diskette", /* 0x00 */
- "Unknown unit", /* 0x01 */
- "Drive not ready", /* 0x02 */
- "Unknown command", /* 0x03 */
- "Data error (CRC)", /* 0x04 */
- "Bad request structure length", /* 0x05 */
- "Seek error", /* 0x06 */
- "Unknown media type", /* 0x07 */
- "Sector not found", /* 0x08 */
- "Printer out of paper", /* 0x09 */
- "Write fault", /* 0x0a */
- "Read fault", /* 0x0b */
- "General failure", /* 0x0c */
- "Unknown error 13", /* 0x0d */
- "Unknown error 14", /* 0x0e */
- "Invalid disk change" /* 0x0f */
- };
- static char *area[4] = /* Area on disk in which error */
- { /* may occur. */
- "DOS area (system files)",
- "file allocation table",
- "directory",
- "data area"
- };
-
- bad_fat = 0;
-
- if (pregs->ax.hl.h & 0x80)
- { /* Character device or FAT */
- /* error. */
- dev_header_ads.r = pregs->si;
- dev_header_ads.s = pregs->bp;
- utabsptr((char *) &dh,&dh_ads);
- utslmove(&dev_header_ads,&dh_ads,sizeof(dh));
- if (dh.attr & 0x8000)
- sprintf(report,
- "\015\012Critical error on device \"%-.8s\":",
- dh.name);
- else
- {
- bad_fat = 1;
- sprintf(report,
- "\015\012Bad FAT on drive %c:",
- pregs->ax.hl.l + 'A');
- }
- }
- else /* Disk error. */
- sprintf(report,
- "\015\012Critical error while %s %s on drive %c:",
- (pregs->ax.hl.h & 1) ? "writing" : "reading",
- area[(pregs->ax.hl.h & 6) >> 1],
- pregs->ax.hl.l + 'A');
- write_str(report);
- write_str("\015\012");
- write_str(message[utlobyte(pregs->di)]);
- write_str("\015\012");
-
- write_str(report);
-
-
- if (bad_fat)
- {
- write_str("\015\012TURN THE COMPUTER OFF!\007\015\012");
- while (1) /* Hang the computer in an */
- atoi(report); /* infinite loop. */
- }
- else
- {
- write_str("\015\012Correct condition and press any key.\015\012");
- wait_key();
- pregs->ax.hl.l = 1; /* Retry. */
- }
- }
-
- /**
- *
- * Name write_str -- Write string to DOS console using only DOS
- * functions 1-12. Ignore CTRL-BREAK.
- *
- * Synopsis write_str(pstring);
- *
- * char *pstring Pointer to beginning of character
- * array containing message to write.
- * The string must be terminated by NUL
- * ('\0') as usual.
- *
- * Description This function writes a string to the DOS console using
- * DOS functions 1-12 only. CTRL-BREAK and CTRL-C are
- * ignored.
- *
- * Returns (None. Function return type is void.)
- *
- **/
-
- void write_str(pstring)
- char *pstring;
- {
- DOSREG regs;
- char ch;
-
- regs.ax = 0x0600;
- while (ch = *pstring++)
- {
- if (ch != (char) 0xff) /* Added (char) 8/28/86. */
- {
- regs.dx = utbyword(0,ch);
- dos(®s);
- }
- }
- }
-
- /**
- *
- * Name wait_key -- Await keystroke and discard it.
- * (Use DOS functions 1-12 only.)
- *
- * Synopsis wait_key();
- *
- * Description This function discards any pending keystrokes, waits for
- * one more keystroke, and discards it. If the new
- * keystroke is CTRL-BREAK or CTRL-C, the program is
- * aborted.
- *
- * Returns (None. Function return type is void.)
- *
- **/
-
- void wait_key()
- {
- DOSREG regs;
-
- regs.ax = 0x0c08; /* Flush keyboard & await one key. */
- dos(®s);
-
- if (utlobyte(regs.ax) == 0) /* If IBM PC extended character, */
- { /* read & discard the second */
- regs.ax = 0x0600; /* byte of the keystroke. */
- regs.dx = 0x00ff;
- dos(®s);
- }
- }