/* Direct.c v1.2 */ /* This is part of ODS2 written by Paul Nankervis, email address: Paulnank@au1.ibm.com ODS2 is distributed freely for all members of the VMS community to use. However all derived works must maintain comments in their source to acknowledge the contibution of the original author. */ /* This module does all directory file handling - mostly lookups of filenames in directory files... */ /* This version takes relative version from last seen directory record - this is no good if there are multiple directory records for a file! */ #include #include #include #include #include "ssdef.h" #include "descrip.h" #include "fibdef.h" #include "access.h" #include "direct.h" #define DEBUGx on int directlookups = 0; int directsearches = 0; int directdels = 0; int directchecks = 0; int directmatches = 0; /* directshow - to print directory statistics */ void directshow(void) { printf("DIRECTSHOW Lookups: %d Searches: %d Deletes: %d Checks: %d Matches: %d\n", directlookups,directsearches,directdels,directchecks,directmatches); } /* namecheck - take a name specification and return name length without the version number, an integer version number, and a wildcard flag */ unsigned namecheck(char *str,int len,int *retlen,int *retver,int *wildflag) { int wildcard = 0; char *spcbeg = str; register int dots = 0; register char *spc = spcbeg; register char *spcend = spc + len; directchecks++; while (spc < spcend) { register char ch = *spc++; if (ch == '.') { if ((spc - spcbeg) > 40) return SS$_BADFILENAME; spcbeg = spc; if (dots++ > 1) break; } else { if (ch == ';') { break; } else { if (ch == '*' || ch == '%') { wildcard = 1; } else { if (ch == '[' || ch == ']' || ch == ':' || !isprint(ch)) return SS$_BADFILENAME; } } } } if ((spc - spcbeg) > 40) return SS$_BADFILENAME; *retlen = spc - str - 1; dots = 0; if (spc < spcend) { register char ch = *spc; if (ch == '*') { if (++spc < spcend) return SS$_BADFILENAME; dots = 32768; wildcard = 1; } else { register int sign = 1; if (ch == '-') { spc++; sign = -1; } while (spc < spcend) { ch = *spc++; if (!isdigit(ch)) return SS$_BADFILENAME; dots = dots * 10 + (ch - '0'); } dots *= sign; } } *retver = dots; *wildflag = wildcard; #ifdef DEBUG printf("Namecheck %d %d %d\n",*retlen,*retver,*wildflag); #endif return SS$_NORMAL; } #define MAT_LT 0 #define MAT_EQ 1 #define MAT_GT 2 #define MAT_NE 3 /* namematch - compare a name specification with a directory entry and determine if there is a match, too big, too small... */ int namematch(char *spec,int speclen,char *entry,int entrylen) { int percent = 0; register char *spc = spec,*ent = entry; register char *spcend = spc + speclen,*entend = ent + entrylen; directmatches++; /* See how much matches without wildcards... */ if (spc < spcend && ent < entend) { register int count = speclen; if (entrylen < count) count = entrylen; do { register char sch = *spc,ech = *ent; if (sch != ech) if (toupper(sch) != toupper(ech)) if (sch == '%') { percent = 1; } else { break; } spc++; ent++; } while (--count > 0); } #ifdef DEBUG printf("Namematch %3s %d %3s %d\n",spec,speclen,entry,entrylen); #endif /* Mismatch - return result unless wildcard... */ if (spc >= spcend) { if (ent >= entend) { return MAT_EQ; } else { if (percent) { return MAT_NE; } else { return MAT_GT; /* Entry longer than search spec */ } } } else { register int offset = 0; if (*spc != '*') { if (percent) return MAT_NE; if (ent < entend) if (toupper(*ent) > toupper(*spc)) return MAT_GT; return MAT_LT; } /* See if we can find a match with wildcards */ spc++; if (spc < spcend) { do { if (spc >= spcend) { if (ent >= entend) break; spc -= offset; ent -= offset - 1; offset = 0; } else { register char sch = toupper(*spc); if (sch == '*') { offset = 0; spc++; if (spc >= spcend) break; } else { if (ent < entend) { register char ech = toupper(*ent); if (sch == ech || sch == '%') { offset++; spc++; ent++; } else { spc -= offset; ent -= offset - 1; offset = 0; } } else { return MAT_NE; } } } } while (1); } } return MAT_EQ; } unsigned freesize(char *buffer) { struct dir$rec *dr = (struct dir$rec *) buffer; do { register char *nr = (char *) dr + dr->dir$size + 2; if (nr >= buffer + 512) break; dr = (struct dir$rec *) nr; } while (1); return (char *) dr - buffer - 2; } unsigned insrec(void) { printf("Insert directory record\n"); return 0; } unsigned insent(struct FCB *fcb,struct VIOC *vioc,unsigned curblk, struct dir$rec *dr,struct dir$ent *de, char *buffer,unsigned eofblk) { printf("Insert directory entry\n"); if (freesize(buffer) >= sizeof(struct dir$ent)) { char *ne = (char *) de + sizeof(struct dir$ent); memcpy(de,ne,512 - (ne - buffer)); dr->dir$size -= sizeof(struct dir$ent); } return 0; } /* delent - delete a directory entry */ unsigned delent(struct FCB *fcb,struct VIOC *vioc,unsigned curblk, struct dir$rec *dr,struct dir$ent *de, char *buffer,unsigned modmask,unsigned eofblk) { unsigned sts = 1; unsigned ent; directdels++; ent = (dr->dir$size - sizeof(struct dir$rec) - dr->dir$namecount + 3) / sizeof(struct dir$ent); printf("DELENT ent = %d %d %d\n",ent,curblk,eofblk); if (ent > 1) { char *ne = (char *) de + sizeof(struct dir$ent); memcpy(de,ne,512 - (ne - buffer)); dr->dir$size -= sizeof(struct dir$ent); } else { char *nr = (char *) dr + dr->dir$size + 2; if (eofblk == 1 || (char *) dr > buffer || (nr <= buffer + 510 && (unsigned short) *nr < 512)) { memcpy(dr,nr,512 - (nr - buffer)); } else { printf("DELENT shrinking file size %d %d\n",curblk,eofblk); while (curblk < eofblk) { char *nxtbuffer; struct VIOC *nxtvioc; unsigned nxtmodmask; sts = accesschunk(fcb,++curblk,&nxtvioc,&nxtbuffer,NULL,1,&nxtmodmask); if ((sts & 1) == 0) break; memcpy(buffer,nxtbuffer,512); sts = deaccesschunk(vioc,modmask,1); if ((sts & 1) == 0) break; buffer = nxtbuffer; vioc = nxtvioc; modmask = nxtmodmask; } if (sts & 1) { fcb->head->fh2$w_recattr.fat$l_efblk[0] = eofblk >> 16; fcb->head->fh2$w_recattr.fat$l_efblk[1] = (eofblk & 0xffff); eofblk--; } } } { unsigned retsts = deaccesschunk(vioc,modmask,1); if (sts & 1) sts = retsts; return sts; } } /* retent - return information about a directory entry */ unsigned retent(struct FCB *fcb,struct VIOC *vioc,unsigned curblk, struct dir$rec *dr,struct dir$ent *de,struct fibdef *fib, unsigned short *reslen,struct dsc$descriptor *resdsc, int wildcard) { register int scale = 10; register int version = de->dir$version; register int length = dr->dir$namecount; register char *ptr = resdsc->dsc$a_pointer; memcpy(ptr,dr->dir$name,length); while (version >= scale) scale *= 10; ptr += length++; *ptr++ = ';'; do { scale /= 10; *ptr++ = version / scale + '0'; version %= scale; length++; } while (scale > 1); *reslen = length; memcpy(&fib->fib$w_fid_num,&de->dir$fid,sizeof(struct fiddef)); if (fib->fib$b_fid_rvn == 0) fib->fib$b_fid_rvn = fcb->rvn; if (wildcard || (fib->fib$w_nmctl & FIB$M_WILD)) { fib->fib$l_wcc = curblk; } else { fib->fib$l_wcc = 0; } return deaccesschunk(vioc,0,1); } /* searchent - search for a directory entry */ unsigned searchent(struct FCB * fcb, struct dsc$descriptor * fibdsc,struct dsc$descriptor * filedsc, unsigned short *reslen,struct dsc$descriptor * resdsc,unsigned eofblk,unsigned action) { register unsigned sts,curblk; struct VIOC *vioc = NULL; unsigned modmask; char *searchspec,*buffer; int searchlen,version,wildcard,wcc_flag; struct fibdef *fib = (struct fibdef *) fibdsc->dsc$a_pointer; directlookups++; /* 1) Generate start block (wcc gives start point) 2) Search for start 3) Scan until found or too big or end */ curblk = fib->fib$l_wcc; if (curblk != 0) { searchspec = resdsc->dsc$a_pointer; sts = namecheck(searchspec,*reslen,&searchlen,&version,&wildcard); if (action || wildcard) return SS$_BADFILENAME; wcc_flag = 1; } else { searchspec = filedsc->dsc$a_pointer; sts = namecheck(searchspec,filedsc->dsc$w_length,&searchlen,&version,&wildcard); if ((action && wildcard) || (action > 1 && version < 0)) return SS$_BADFILENAME; wcc_flag = 0; } if ((sts & 1) == 0) return sts; /* Identify starting block...*/ if (*searchspec == '*' || *searchspec == '%') { curblk = 1; } else { register unsigned loblk = 1,hiblk = eofblk; if (curblk < 1 || curblk > eofblk) curblk = (eofblk + 1) / 2; while (loblk < hiblk) { register int cmp; register unsigned newblk; register struct dir$rec *dr; directsearches++; sts = accesschunk(fcb,curblk,&vioc,&buffer,NULL,action ? 1 : 0,&modmask); if ((sts & 1) == 0) return sts; dr = (struct dir$rec *) buffer; if (dr->dir$size > 510) { cmp = MAT_GT; } else { cmp = namematch(searchspec,searchlen,dr->dir$name,dr->dir$namecount); if (cmp == MAT_EQ) { if (wildcard || version < 1 || version > 32767) { cmp = MAT_NE; /* no match - want to find start */ } else { register struct dir$ent *de = (struct dir$ent *) (dr->dir$name + ((dr->dir$namecount + 1) & ~1)); if (de->dir$version < version) { cmp = MAT_GT; /* too far... */ } else { if (de->dir$version > version) { cmp = MAT_LT; /* further ahead... */ } } } } } #ifdef DEBUG printf("Direct %6.6s %d %6.6s %d (%d<%d<%d)-> %d\n", searchspec,searchlen,dr->dir$name,dr->dir$namecount, loblk,curblk,hiblk,cmp); #endif switch (cmp) { case MAT_LT: if (curblk == fib->fib$l_wcc) { newblk = hiblk = loblk = curblk; } else { loblk = curblk; newblk = (loblk + hiblk + 1) / 2; } break; case MAT_GT: case MAT_NE: newblk = (loblk + curblk) / 2; hiblk = curblk - 1; break; default: newblk = hiblk = loblk = curblk; } if (newblk != curblk) { sts = deaccesschunk(vioc,0,1); if ((sts & 1) == 0) return sts; vioc = NULL; curblk = newblk; } } } /* Now to read sequentially to find entry... */ while ((sts & 1) && curblk <= eofblk) { register struct dir$rec *dr; if (vioc == NULL) { sts = accesschunk(fcb,curblk,&vioc,&buffer,NULL,action ? 1 : 0,&modmask); if ((sts & 1) == 0) return sts; } dr = (struct dir$rec *) buffer; do { register int cmp; register char *nr = (char *) dr + dr->dir$size + 2; if (nr >= buffer + 512) break; cmp = namematch(searchspec,searchlen,dr->dir$name,dr->dir$namecount); #ifdef DEBUF printf("Direct %6.6s %d %6.6s %d -> %d\n", searchspec,searchlen,dr->dir$name,dr->dir$namecount,cmp); #endif if (cmp == MAT_GT) { if (wcc_flag) { wcc_flag = 0; searchspec = filedsc->dsc$a_pointer; sts = namecheck(searchspec,filedsc->dsc$w_length,&searchlen,&version,&wildcard); if ((sts & 1) == 0) break; } else { curblk = eofblk; /* give up */ break; } } else { if (cmp == MAT_EQ) { register int relver = 0; register struct dir$ent *de = (struct dir$ent *) (dr->dir$name + ((dr->dir$namecount + 1) & ~1)); while ((char *) de < nr) { if (version >= de->dir$version || (version < 1 && version >= relver)) { cmp = MAT_GT; if (version > 32767 || version == relver || version == de->dir$version) cmp = MAT_EQ; if (wcc_flag) { wcc_flag = 0; searchspec = filedsc->dsc$a_pointer; sts = namecheck(searchspec,filedsc->dsc$w_length,&searchlen,&version,&wildcard); if ((sts & 1) == 0) break; if (namematch(searchspec,searchlen,dr->dir$name, dr->dir$namecount) != MAT_EQ) { break; } if (cmp == MAT_EQ) { cmp = MAT_LT; } else { cmp = MAT_LT; if (version >= de->dir$version || (version < 1 && version >= relver)) { cmp = MAT_GT; if (version > 32767 || version == relver || version == de->dir$version) cmp = MAT_EQ; } } } if (cmp == MAT_EQ) { switch (action) { case 0: return retent(fcb,vioc,curblk,dr,de,fib,reslen,resdsc,wildcard); case 1: return delent(fcb,vioc,curblk,dr,de,buffer,modmask,eofblk); default: sts = SS$_DUPFILENAME; de = (struct dir$ent *) nr; } } else { if (cmp == MAT_GT) break; } } relver--; de++; } if ((sts & 1) == 0) break; if (action == 2) { return insent(fcb,vioc,curblk,dr,de,buffer,eofblk); } } dr = (struct dir$rec *) nr; } } while (1); { register unsigned dests = deaccesschunk(vioc,0,1); vioc = NULL; if (sts & 1) sts = dests; } curblk++; } if (action == 2) { return insrec(); } if (sts & 1) { fib->fib$l_wcc = 0; if (wcc_flag || wildcard) { sts = SS$_NOMOREFILES; } else { sts = SS$_NOSUCHFILE; } } return sts; } /* direct - this routine handles all directory manipulations:- action 0 - find directory entry 1 - delete entry 2 - create an entry */ unsigned direct(struct VCB *vcb,struct dsc$descriptor * fibdsc, struct dsc$descriptor *filedsc,unsigned short *reslen, struct dsc$descriptor *resdsc,unsigned action) { struct FCB *fcb; register unsigned sts,eofblk; register struct fibdef *fib = (struct fibdef *) fibdsc->dsc$a_pointer; sts = accessfile(vcb,(struct fiddef *) & fib->fib$w_did_num,&fcb,action); if (sts & 1) { if (fcb->head->fh2$l_filechar & FH2$M_DIRECTORY) { eofblk = swapw(fcb->head->fh2$w_recattr.fat$l_efblk); if (fcb->head->fh2$w_recattr.fat$w_ffbyte == 0) --eofblk; sts = searchent(fcb,fibdsc,filedsc,reslen,resdsc,eofblk,action); } else { sts = SS$_BADIRECTORY; } { register unsigned dests = deaccessfile(fcb); if (sts & 1) sts = dests; } } return sts; }