/* check.c - Check and repair a PC/MS-DOS file system */ /* Written 1993 by Werner Almesberger */ /* FAT32, VFAT, Atari format support, and various fixes additions May 1998 * by Roman Hodek */ #include #include #include #include #include #include "common.h" #include "dosfsck.h" #include "io.h" #include "fat.h" #include "file.h" #include "lfn.h" #include "check.h" static DOS_FILE *root; /* get start field of a dir entry */ #define FSTART(p,fs) \ ((unsigned long)CF_LE_W(p->dir_ent.start) | \ (fs->fat_bits == 32 ? CF_LE_W(p->dir_ent.starthi) << 16 : 0)) #define MODIFY(p,i,v) \ do { \ if (p->offset) { \ p->dir_ent.i = v; \ fs_write(p->offset+offsetof(DIR_ENT,i), \ sizeof(p->dir_ent.i),&p->dir_ent.i); \ } \ } while(0) #define MODIFY_START(p,v,fs) \ do { \ unsigned long __v = (v); \ if (!p->offset) { \ /* writing to fake entry for FAT32 root dir */ \ if (!__v) die("Oops, deleting FAT32 root dir!"); \ fs->root_cluster = __v; \ p->dir_ent.start = CT_LE_W(__v&0xffff); \ p->dir_ent.starthi = CT_LE_W(__v>>16); \ __v = CT_LE_L(__v); \ fs_write((loff_t)offsetof(struct boot_sector,root_cluster), \ sizeof(((struct boot_sector *)0)->root_cluster), \ &__v); \ } \ else { \ MODIFY(p,start,CT_LE_W((__v)&0xffff)); \ if (fs->fat_bits == 32) \ MODIFY(p,starthi,CT_LE_W((__v)>>16)); \ } \ } while(0) loff_t alloc_rootdir_entry(DOS_FS *fs, DIR_ENT *de, const char *pattern) { static int curr_num = 0; loff_t offset; if (fs->root_cluster) { DIR_ENT d2; int i = 0, got = 0; unsigned long clu_num, prev = 0; loff_t offset2; clu_num = fs->root_cluster; offset = cluster_start(fs,clu_num); while (clu_num > 0 && clu_num != -1) { fs_read(offset,sizeof(DIR_ENT),&d2); if (IS_FREE(d2.name) && d2.attr != VFAT_LN_ATTR) { got = 1; break; } i += sizeof(DIR_ENT); offset += sizeof(DIR_ENT); if ((i % fs->cluster_size) == 0) { prev = clu_num; if ((clu_num = next_cluster(fs,clu_num)) == 0 || clu_num == -1) break; offset = cluster_start(fs,clu_num); } } if (!got) { /* no free slot, need to extend root dir: alloc next free cluster * after previous one */ if (!prev) die("Root directory has no cluster allocated!"); for (clu_num = prev+1; clu_num != prev; clu_num++) { if (clu_num >= fs->clusters+2) clu_num = 2; if (!fs->fat[clu_num].value) break; } if (clu_num == prev) die("Root directory full and no free cluster"); set_fat(fs,prev,clu_num); set_fat(fs,clu_num,-1); set_owner(fs, clu_num, get_owner(fs, fs->root_cluster)); /* clear new cluster */ memset( &d2, 0, sizeof(d2) ); offset = cluster_start(fs,clu_num); for( i = 0; i < fs->cluster_size; i += sizeof(DIR_ENT) ) fs_write( offset+i, sizeof(d2), &d2 ); } memset(de,0,sizeof(DIR_ENT)); while (1) { sprintf(de->name,pattern,curr_num); clu_num = fs->root_cluster; i = 0; offset2 = cluster_start(fs,clu_num); while (clu_num > 0 && clu_num != -1) { fs_read(offset2,sizeof(DIR_ENT),&d2); if (offset2 != offset && !strncmp(d2.name,de->name,MSDOS_NAME)) break; i += sizeof(DIR_ENT); offset2 += sizeof(DIR_ENT); if ((i % fs->cluster_size) == 0) { if ((clu_num = next_cluster(fs,clu_num)) == 0 || clu_num == -1) break; offset2 = cluster_start(fs,clu_num); } } if (clu_num == 0 || clu_num == -1) break; if (++curr_num >= 10000) die("Unable to create unique name"); } } else { DIR_ENT *root; int next_free = 0, scan; root = alloc(fs->root_entries*sizeof(DIR_ENT)); fs_read(fs->root_start,fs->root_entries*sizeof(DIR_ENT),root); while (next_free < fs->root_entries) if (IS_FREE(root[next_free].name) && root[next_free].attr != VFAT_LN_ATTR) break; else next_free++; if (next_free == fs->root_entries) die("Root directory is full."); offset = fs->root_start+next_free*sizeof(DIR_ENT); memset(de,0,sizeof(DIR_ENT)); while (1) { sprintf(de->name,pattern,curr_num); for (scan = 0; scan < fs->root_entries; scan++) if (scan != next_free && !strncmp(root[scan].name,de->name,MSDOS_NAME)) break; if (scan == fs->root_entries) break; if (++curr_num >= 10000) die("Unable to create unique name"); } free(root); } ++n_files; return offset; } static char *path_name(DOS_FILE *file) { static char path[PATH_MAX*2]; if (!file) *path = 0; else { if (strlen(path_name(file->parent)) > PATH_MAX) die("Path name too long."); if (strcmp(path,"/") != 0) strcat(path,"/"); strcpy(strrchr(path,0),file->lfn?file->lfn:file_name(file->dir_ent.name)); } return path; } static int day_n[] = { 0,31,59,90,120,151,181,212,243,273,304,334,0,0,0,0 }; /* JanFebMarApr May Jun Jul Aug Sep Oct Nov Dec */ /* Convert a MS-DOS time/date pair to a UNIX date (seconds since 1 1 70). */ time_t date_dos2unix(unsigned short time,unsigned short date) { int month,year; time_t secs; month = ((date >> 5) & 15)-1; year = date >> 9; secs = (time & 31)*2+60*((time >> 5) & 63)+(time >> 11)*3600+86400* ((date & 31)-1+day_n[month]+(year/4)+year*365-((year & 3) == 0 && month < 2 ? 1 : 0)+3653); /* days since 1.1.70 plus 80's leap day */ return secs; } static char *file_stat(DOS_FILE *file) { static char temp[100]; struct tm *tm; char tmp[100]; time_t date; date = date_dos2unix(CF_LE_W(file->dir_ent.time),CF_LE_W(file-> dir_ent.date)); tm = localtime(&date); strftime(tmp,99,"%H:%M:%S %b %d %Y",tm); sprintf(temp," Size %u bytes, date %s",CF_LE_L(file->dir_ent.size),tmp); return temp; } static int bad_name(unsigned char *name) { int i, spc, suspicious = 0; char *bad_chars = atari_format ? "*?\\/:" : "*?<>|\"\\/:"; /* Do not complain about (and auto-correct) the extended attribute files * of OS/2. */ if (strncmp(name,"EA DATA SF",11) == 0 || strncmp(name,"WP ROOT SF",11) == 0) return 0; for (i = 0; i < 8; i++) { if (name[i] < ' ' || name[i] == 0x7f) return 1; if (name[i] > 0x7f) ++suspicious; if (strchr(bad_chars,name[i])) return 1; } for (i = 8; i < 11; i++) { if (name[i] < ' ' || name[i] == 0x7f) return 1; if (name[i] > 0x7f) ++suspicious; if (strchr(bad_chars,name[i])) return 1; } spc = 0; for (i = 0; i < 8; i++) { if (name[i] == ' ') spc = 1; else if (spc) /* non-space after a space not allowed, space terminates the name * part */ return 1; } spc = 0; for (i = 8; i < 11; i++) { if (name[i] == ' ') spc = 1; else if (spc) /* non-space after a space not allowed, space terminates the name * part */ return 1; } /* Under GEMDOS, chars >= 128 are never allowed. */ if (atari_format && suspicious) return 1; /* Only complain about too much suspicious chars in interactive mode, * never correct them automatically. The chars are all basically ok, so we * shouldn't auto-correct such names. */ if (interactive && suspicious > 6) return 1; return 0; } static void drop_file(DOS_FS *fs,DOS_FILE *file) { unsigned long cluster; MODIFY(file,name[0],DELETED_FLAG); for (cluster = FSTART(file,fs); cluster > 0 && cluster < fs->clusters+2; cluster = next_cluster(fs,cluster)) set_owner(fs,cluster,NULL); --n_files; } static void truncate_file(DOS_FS *fs,DOS_FILE *file,unsigned long clusters) { int deleting; unsigned long walk,next,prev; walk = FSTART(file,fs); prev = 0; if ((deleting = !clusters)) MODIFY_START(file,0,fs); while (walk > 0 && walk != -1) { next = next_cluster(fs,walk); if (deleting) set_fat(fs,walk,0); else if ((deleting = !--clusters)) set_fat(fs,walk,-1); prev = walk; walk = next; } } static void auto_rename(DOS_FILE *file) { DOS_FILE *first,*walk; int number; if (!file->offset) return; /* cannot rename FAT32 root dir */ first = file->parent ? file->parent->first : root; number = 0; while (1) { sprintf(file->dir_ent.name,"FSCK%04d",number); strncpy(file->dir_ent.ext,"REN",3); for (walk = first; walk; walk = walk->next) if (walk != file && !strncmp(walk->dir_ent.name,file->dir_ent. name,MSDOS_NAME)) break; if (!walk) { fs_write(file->offset,MSDOS_NAME,file->dir_ent.name); return; } number++; } die("Can't generate a unique name."); } static void rename_file(DOS_FILE *file) { unsigned char name[46]; unsigned char *walk,*here; if (!file->offset) { printf( "Cannot rename FAT32 root dir\n" ); return; /* cannot rename FAT32 root dir */ } while (1) { printf("New name: "); fflush(stdout); if (fgets(name,45,stdin)) { if ((here = strchr(name,'\n'))) *here = 0; for (walk = strrchr(name,0); walk >= name && (*walk == ' ' || *walk == '\t'); walk--); walk[1] = 0; for (walk = name; *walk == ' ' || *walk == '\t'; walk++); if (file_cvt(walk,file->dir_ent.name)) { fs_write(file->offset,MSDOS_NAME,file->dir_ent.name); return; } } } } static int handle_dot(DOS_FS *fs,DOS_FILE *file,int dots) { char *name; name = strncmp(file->dir_ent.name,MSDOS_DOT,MSDOS_NAME) ? ".." : "."; if (!(file->dir_ent.attr & ATTR_DIR)) { printf("%s\n Is a non-directory.\n",path_name(file)); if (interactive) printf("1) Drop it\n2) Auto-rename\n3) Rename\n" "4) Convert to directory\n"); else printf(" Auto-renaming it.\n"); switch (interactive ? get_key("1234","?") : '2') { case '1': drop_file(fs,file); return 1; case '2': auto_rename(file); printf(" Renamed to %s\n",file_name(file->dir_ent.name)); return 0; case '3': rename_file(file); return 0; case '4': MODIFY(file,size,CT_LE_L(0)); MODIFY(file,attr,file->dir_ent.attr | ATTR_DIR); break; } } if (!dots) { printf("Root contains directory \"%s\". Dropping it.\n",name); drop_file(fs,file); return 1; } return 0; } static int check_file(DOS_FS *fs,DOS_FILE *file) { DOS_FILE *owner; int restart; unsigned long expect,curr,this,clusters,prev,walk,clusters2; if (file->dir_ent.attr & ATTR_DIR) { if (CF_LE_L(file->dir_ent.size)) { printf("%s\n Directory has non-zero size. Fixing it.\n", path_name(file)); MODIFY(file,size,CT_LE_L(0)); } if (file->parent && !strncmp(file->dir_ent.name,MSDOS_DOT,MSDOS_NAME)) { expect = FSTART(file->parent,fs); if (FSTART(file,fs) != expect) { printf("%s\n Start (%ld) does not point to parent (%ld)\n", path_name(file),FSTART(file,fs),expect); MODIFY_START(file,expect,fs); } return 0; } if (file->parent && !strncmp(file->dir_ent.name,MSDOS_DOTDOT, MSDOS_NAME)) { expect = file->parent->parent ? FSTART(file->parent->parent,fs):0; if (fs->root_cluster && expect == fs->root_cluster) expect = 0; if (FSTART(file,fs) != expect) { printf("%s\n Start (%lu) does not point to .. (%lu)\n", path_name(file),FSTART(file,fs),expect); MODIFY_START(file,expect,fs); } return 0; } if (FSTART(file,fs)==0){ printf ("%s\n Start does point to root directory. Deleting dir. \n", path_name(file)); MODIFY(file,name[0],DELETED_FLAG); return 0; } } if (FSTART(file,fs) >= fs->clusters+2) { printf("%s\n Start cluster beyond limit (%lu > %lu). Truncating file.\n", path_name(file),FSTART(file,fs),fs->clusters+1); if (!file->offset) die( "Bad FAT32 root directory! (bad start cluster)\n" ); MODIFY_START(file,0,fs); } clusters = prev = 0; for (curr = FSTART(file,fs) ? FSTART(file,fs) : -1; curr != -1; curr = next_cluster(fs,curr)) { if (!fs->fat[curr].value || bad_cluster(fs,curr)) { printf("%s\n Contains a %s cluster (%lu). Assuming EOF.\n", path_name(file),fs->fat[curr].value ? "bad" : "free",curr); if (prev) set_fat(fs,prev,-1); else if (!file->offset) die( "FAT32 root dir starts with a bad cluster!" ); else MODIFY_START(file,0,fs); break; } if (!(file->dir_ent.attr & ATTR_DIR) && CF_LE_L(file->dir_ent.size) <= clusters*fs->cluster_size) { printf("%s\n File size is %u bytes, cluster chain length is > %lu " "bytes.\n Truncating file to %u bytes.\n",path_name(file), CF_LE_L(file->dir_ent.size),clusters*fs->cluster_size, CF_LE_L(file->dir_ent.size)); truncate_file(fs,file,clusters); break; } if ((owner = get_owner(fs,curr))) { int do_trunc = 0; printf("%s and\n",path_name(owner)); printf("%s\n share clusters.\n",path_name(file)); clusters2 = 0; for (walk = FSTART(owner,fs); walk > 0 && walk != -1; walk = next_cluster(fs,walk)) if (walk == curr) break; else clusters2++; restart = file->dir_ent.attr & ATTR_DIR; if (!owner->offset) { printf( " Truncating second to %lu bytes because first " "is FAT32 root dir.\n", clusters2*fs->cluster_size ); do_trunc = 2; } else if (!file->offset) { printf( " Truncating first to %lu bytes because second " "is FAT32 root dir.\n", clusters*fs->cluster_size ); do_trunc = 1; } else if (interactive) printf("1) Truncate first to %lu bytes%s\n" "2) Truncate second to %lu bytes\n",clusters*fs->cluster_size, restart ? " and restart" : "",clusters2*fs->cluster_size); else printf(" Truncating second to %lu bytes.\n",clusters2* fs->cluster_size); if (do_trunc != 2 && (do_trunc == 1 || (interactive && get_key("12","?") == '1'))) { prev = 0; clusters = 0; for (this = FSTART(owner,fs); this > 0 && this != -1; this = next_cluster(fs,this)) { if (this == curr) { if (prev) set_fat(fs,prev,-1); else MODIFY_START(owner,0,fs); MODIFY(owner,size,CT_LE_L(clusters*fs->cluster_size)); if (restart) return 1; while (this > 0 && this != -1) { set_owner(fs,this,NULL); this = next_cluster(fs,this); } break; } clusters++; prev = this; } if (this != curr) die("Internal error: didn't find cluster %d in chain" " starting at %d",curr,FSTART(owner,fs)); } else { if (prev) set_fat(fs,prev,-1); else MODIFY_START(file,0,fs); break; } } set_owner(fs,curr,file); clusters++; prev = curr; } if (!(file->dir_ent.attr & ATTR_DIR) && CF_LE_L(file->dir_ent.size) > clusters*fs->cluster_size) { printf("%s\n File size is %u bytes, cluster chain length is %lu bytes." "\n Truncating file to %lu bytes.\n",path_name(file),CF_LE_L(file-> dir_ent.size),clusters*fs->cluster_size,clusters*fs->cluster_size); MODIFY(file,size,CT_LE_L(clusters*fs->cluster_size)); } return 0; } static int check_files(DOS_FS *fs,DOS_FILE *start) { while (start) { if (check_file(fs,start)) return 1; start = start->next; } return 0; } static int check_dir(DOS_FS *fs,DOS_FILE **root,int dots) { DOS_FILE *parent,**walk,**scan; int dot,dotdot,skip,redo; int good,bad; if (!*root) return 0; parent = (*root)->parent; good = bad = 0; for (walk = root; *walk; walk = &(*walk)->next) if (bad_name((*walk)->dir_ent.name)) bad++; else good++; if (*root && parent && good+bad > 4 && bad > good/2) { printf("%s\n Has a large number of bad entries. (%d/%d)\n", path_name(parent),bad,good+bad); if (!dots) printf( " Not dropping root directory.\n" ); else if (!interactive) printf(" Not dropping it in auto-mode.\n"); else if (get_key("yn","Drop directory ? (y/n)") == 'y') { truncate_file(fs,parent,0); MODIFY(parent,name[0],DELETED_FLAG); /* buglet: deleted directory stays in the list. */ return 1; } } dot = dotdot = redo = 0; walk = root; while (*walk) { if (!strncmp((*walk)->dir_ent.name,MSDOS_DOT,MSDOS_NAME) || !strncmp((*walk)->dir_ent.name,MSDOS_DOTDOT,MSDOS_NAME)) { if (handle_dot(fs,*walk,dots)) { *walk = (*walk)->next; continue; } if (!strncmp((*walk)->dir_ent.name,MSDOS_DOT,MSDOS_NAME)) dot++; else dotdot++; } if (!((*walk)->dir_ent.attr & ATTR_VOLUME) && bad_name((*walk)->dir_ent.name)) { printf("%s\n Bad file name.\n",path_name(*walk)); if (interactive) printf("1) Drop file\n2) Rename file\n3) Auto-rename\n" "4) Keep it\n"); else printf(" Auto-renaming it.\n"); switch (interactive ? get_key("1234","?") : '3') { case '1': drop_file(fs,*walk); walk = &(*walk)->next; continue; case '2': rename_file(*walk); redo = 1; break; case '3': auto_rename(*walk); printf(" Renamed to %s\n",file_name((*walk)->dir_ent. name)); break; case '4': break; } } /* don't check for duplicates of the volume label */ if (!((*walk)->dir_ent.attr & ATTR_VOLUME)) { scan = &(*walk)->next; skip = 0; while (*scan && !skip) { if (!((*scan)->dir_ent.attr & ATTR_VOLUME) && !strncmp((*walk)->dir_ent.name,(*scan)->dir_ent.name,MSDOS_NAME)) { printf("%s\n Duplicate directory entry.\n First %s\n", path_name(*walk),file_stat(*walk)); printf(" Second %s\n",file_stat(*scan)); if (interactive) printf("1) Drop first\n2) Drop second\n3) Rename first\n" "4) Rename second\n5) Auto-rename first\n" "6) Auto-rename second\n"); else printf(" Auto-renaming second.\n"); switch (interactive ? get_key("123456","?") : '6') { case '1': drop_file(fs,*walk); *walk = (*walk)->next; skip = 1; break; case '2': drop_file(fs,*scan); *scan = (*scan)->next; continue; case '3': rename_file(*walk); printf(" Renamed to %s\n",path_name(*walk)); redo = 1; break; case '4': rename_file(*scan); printf(" Renamed to %s\n",path_name(*walk)); redo = 1; break; case '5': auto_rename(*walk); printf(" Renamed to %s\n",file_name((*walk)->dir_ent. name)); break; case '6': auto_rename(*scan); printf(" Renamed to %s\n",file_name((*scan)->dir_ent. name)); break; } } scan = &(*scan)->next; } if (skip) continue; } if (!redo) walk = &(*walk)->next; else { walk = root; dot = dotdot = redo = 0; } } if (dots && !dot) printf("%s\n \".\" is missing. Can't fix this yet.\n", path_name(parent)); if (dots && !dotdot) printf("%s\n \"..\" is missing. Can't fix this yet.\n", path_name(parent)); return 0; } static void test_file(DOS_FS *fs,DOS_FILE *file,int read_test) { DOS_FILE *owner; unsigned long walk,prev,clusters,next_clu; prev = clusters = 0; for (walk = FSTART(file,fs); walk > 0 && walk < fs->clusters+2; walk = next_clu) { next_clu = next_cluster(fs,walk); if ((owner = get_owner(fs,walk))) { if (owner == file) { printf("%s\n Circular cluster chain. Truncating to %lu " "cluster%s.\n",path_name(file),clusters,clusters == 1 ? "" : "s"); if (prev) set_fat(fs,prev,-1); else if (!file->offset) die( "Bad FAT32 root directory! (bad start cluster)\n" ); else MODIFY_START(file,0,fs); } break; } if (bad_cluster(fs,walk)) break; if (read_test) { if (fs_test(cluster_start(fs,walk),fs->cluster_size)) { prev = walk; clusters++; } else { printf("%s\n Cluster %lu (%lu) is unreadable. Skipping it.\n", path_name(file),clusters,walk); if (prev) set_fat(fs,prev,next_cluster(fs,walk)); else MODIFY_START(file,next_cluster(fs,walk),fs); set_fat(fs,walk,-2); } } set_owner(fs,walk,file); } for (walk = FSTART(file,fs); walk > 0 && walk < fs->clusters+2; walk = next_cluster(fs,walk)) if (bad_cluster(fs,walk)) break; else if (get_owner(fs,walk) == file) set_owner(fs,walk,NULL); else break; } static void undelete(DOS_FS *fs,DOS_FILE *file) { unsigned long clusters,left,prev,walk; clusters = left = (CF_LE_L(file->dir_ent.size)+fs->cluster_size-1)/ fs->cluster_size; prev = 0; for (walk = FSTART(file,fs); left && walk >= 2 && walk < fs->clusters+2 && !fs->fat[walk].value; walk++) { left--; if (prev) set_fat(fs,prev,walk); prev = walk; } if (prev) set_fat(fs,prev,-1); else MODIFY_START(file,0,fs); if (left) printf("Warning: Did only undelete %lu of %lu cluster%s.\n",clusters-left, clusters,clusters == 1 ? "" : "s"); } static void new_dir( void ) { lfn_reset(); } static void add_file(DOS_FS *fs,DOS_FILE ***chain,DOS_FILE *parent, loff_t offset,FDSC **cp) { DOS_FILE *new; DIR_ENT de; FD_TYPE type; if (offset) fs_read(offset,sizeof(DIR_ENT),&de); else { memcpy(de.name," ",MSDOS_NAME); de.attr = ATTR_DIR; de.size = de.time = de.date = 0; de.start = CT_LE_W(fs->root_cluster & 0xffff); de.starthi = CT_LE_W((fs->root_cluster >> 16) & 0xffff); } if ((type = file_type(cp,de.name)) != fdt_none) { if (type == fdt_undelete && (de.attr & ATTR_DIR)) die("Can't undelete directories."); file_modify(cp,de.name); fs_write(offset,1,&de); } if (IS_FREE(de.name)) { lfn_check_orphaned(); return; } if (de.attr == VFAT_LN_ATTR) { lfn_add_slot(&de,offset); return; } new = qalloc(&mem_queue,sizeof(DOS_FILE)); new->lfn = lfn_get(&de); new->offset = offset; memcpy(&new->dir_ent,&de,sizeof(de)); new->next = new->first = NULL; new->parent = parent; if (type == fdt_undelete) undelete(fs,new); **chain = new; *chain = &new->next; if (list) { printf("Checking file %s",path_name(new)); if (new->lfn) printf(" (%s)", file_name(new->dir_ent.name) ); printf("\n"); } if (offset && strncmp(de.name,MSDOS_DOT,MSDOS_NAME) != 0 && strncmp(de.name,MSDOS_DOTDOT,MSDOS_NAME) != 0) ++n_files; test_file(fs,new,test); } static int subdirs(DOS_FS *fs,DOS_FILE *parent,FDSC **cp); static int scan_dir(DOS_FS *fs,DOS_FILE *this,FDSC **cp) { DOS_FILE **chain; int i; unsigned long clu_num; chain = &this->first; i = 0; clu_num = FSTART(this,fs); new_dir(); while (clu_num > 0 && clu_num != -1) { add_file(fs,&chain,this,cluster_start(fs,clu_num)+(i % fs-> cluster_size),cp); i += sizeof(DIR_ENT); if (!(i % fs->cluster_size)) if ((clu_num = next_cluster(fs,clu_num)) == 0 || clu_num == -1) break; } lfn_check_orphaned(); if (check_dir(fs,&this->first,this->offset)) return 0; if (check_files(fs,this->first)) return 1; return subdirs(fs,this,cp); } static int subdirs(DOS_FS *fs,DOS_FILE *parent,FDSC **cp) { DOS_FILE *walk; for (walk = parent ? parent->first : root; walk; walk = walk->next) if (walk->dir_ent.attr & ATTR_DIR) if (strncmp(walk->dir_ent.name,MSDOS_DOT,MSDOS_NAME) && strncmp(walk->dir_ent.name,MSDOS_DOTDOT,MSDOS_NAME)) if (scan_dir(fs,walk,file_cd(cp,walk->dir_ent.name))) return 1; return 0; } int scan_root(DOS_FS *fs) { DOS_FILE **chain; int i; root = NULL; chain = &root; new_dir(); if (fs->root_cluster) { add_file(fs,&chain,NULL,0,&fp_root); } else { for (i = 0; i < fs->root_entries; i++) add_file(fs,&chain,NULL,fs->root_start+i*sizeof(DIR_ENT),&fp_root); } lfn_check_orphaned(); (void) check_dir(fs,&root,0); if (check_files(fs,root)) return 1; return subdirs(fs,NULL,&fp_root); } /* Local Variables: */ /* tab-width: 8 */ /* End: */