file.c

来自「linux dosfs 工具,可以移植到嵌入式系统下检查存储状态下的磁盘状况,修」· C语言 代码 · 共 243 行

C
243
字号
/* file.c  -  Additional file attributes *//* Written 1993 by Werner Almesberger *//* FAT32, VFAT, Atari format support, and various fixes additions May 1998 * by Roman Hodek <Roman.Hodek@informatik.uni-erlangen.de> */#include <stdlib.h>#include <stdio.h>#include <string.h>#include <ctype.h>#include <unistd.h>#define _LINUX_STAT_H		/* hack to avoid inclusion of <linux/stat.h> */#define _LINUX_STRING_H_	/* hack to avoid inclusion of <linux/string.h>*/#define _LINUX_FS_H             /* hack to avoid inclusion of <linux/fs.h> */#include <linux/msdos_fs.h>#include "common.h"#include "file.h"FDSC *fp_root = NULL;static void put_char(char **p,unsigned char c){    if ((c >= ' ' && c < 0x7f) || c >= 0xa0) *(*p)++ = c;    else {	*(*p)++ = '\\';	*(*p)++ = '0'+(c >> 6);	*(*p)++ = '0'+((c >> 3) & 7);	*(*p)++ = '0'+(c & 7);    }}char *file_name(unsigned char *fixed){    static char path[MSDOS_NAME*4+2];    char *p;    int i,j;    p = path;    for (i = j =  0; i < 8; i++)	if (fixed[i] != ' ') {	    while (j++ < i) *p++ = ' ';	    put_char(&p,fixed[i]);	}    if (strncmp(fixed+8,"   ",3)) {	*p++ = '.';	for (i = j =  0; i < 3; i++)	    if (fixed[i+8] != ' ') {		while (j++ < i) *p++ = ' ';		put_char(&p,fixed[i+8]);	    }    }    *p = 0;    return path;}int file_cvt(unsigned char *name,unsigned char *fixed){    unsigned char c;    int size,ext,cnt;    size = 8;    ext = 0;    while (*name) {	c = *name;	if (c < ' ' || c > 0x7e || strchr("*?<>|\"/",c)) {	    printf("Invalid character in name. Use \\ooo for special "	      "characters.\n");	    return 0;	}	if (c == '.') {	    if (ext) {		printf("Duplicate dots in name.\n");		return 0;	    }	    while (size--) *fixed++ = ' ';	    size = 3;	    ext = 1;	    name++;	    continue;	}	if (c == '\\') {	    c = 0;	    for (cnt = 3; cnt; cnt--) {		if (*name < '0' || *name > '7') {		    printf("Invalid octal character.\n");		    return 0;		}		c = c*8+*name++-'0';	    }	    if (cnt < 4) {		printf("Expected three octal digits.\n");		return 0;	    }	    name += 3;	}	if (islower(c)) c = toupper(c);	if (size) {	    *fixed++ = c;	    size--;	}	name++;    }    if (*name || size == 8) return 0;    if (!ext) {	while (size--) *fixed++ = ' ';	size = 3;    }    while (size--) *fixed++ = ' ';    return 1;}void file_add(char *path,FD_TYPE type){    FDSC **current,*walk;    char name[MSDOS_NAME];    char *here;    current = &fp_root;    if (*path != '/') die("%s: Absolute path required.",path);    path++;    while (1) {	if ((here = strchr(path,'/'))) *here = 0;	if (!file_cvt(path,name)) exit(2);	for (walk = *current; walk; walk = walk->next)	    if (!here && (!strncmp(name,walk->name,MSDOS_NAME) || (type ==	      fdt_undelete && !strncmp(name+1,walk->name+1,MSDOS_NAME-1))))		die("Ambiguous name: \"%s\"",path);	    else if (here && !strncmp(name,walk->name,MSDOS_NAME)) break;	if (!walk) {	    walk = alloc(sizeof(FDSC));	    strncpy(walk->name,name,MSDOS_NAME);	    walk->type = here ? fdt_none : type;	    walk->first = NULL;	    walk->next = *current;	    *current = walk;	}	current = &walk->first;	if (!here) break;	*here = '/';	path = here+1;    }}FDSC **file_cd(FDSC **curr,char *fixed){    FDSC **walk;    if (!curr || !*curr) return NULL;    for (walk = curr; *walk; walk = &(*walk)->next)	if (!strncmp((*walk)->name,fixed,MSDOS_NAME) && (*walk)->first)	    return &(*walk)->first;    return NULL;}static FDSC **file_find(FDSC **dir,char *fixed){    if (!dir || !*dir) return NULL;    if (*(unsigned char *) fixed == DELETED_FLAG) {	while (*dir) {	    if (!strncmp((*dir)->name+1,fixed+1,MSDOS_NAME-1) && !(*dir)->first)		return dir;	    dir = &(*dir)->next;	}	return NULL;    }    while (*dir) {	if (!strncmp((*dir)->name,fixed,MSDOS_NAME) && !(*dir)->first)	    return dir;	dir = &(*dir)->next;    }    return NULL;}FD_TYPE file_type(FDSC **curr,char *fixed){    FDSC **this;    if ((this = file_find(curr,fixed))) return (*this)->type;    return fdt_none;}void file_modify(FDSC **curr,char *fixed){    FDSC **this,*next;    if (!(this = file_find(curr,fixed)))	die("Internal error: file_find failed");    switch ((*this)->type) {	case fdt_drop:	    printf("Dropping %s\n",file_name(fixed));	    *(unsigned char *) fixed = DELETED_FLAG;	    break;	case fdt_undelete:	    *fixed = *(*this)->name;	    printf("Undeleting %s\n",file_name(fixed));	    break;	default:	    die("Internal error: file_modify");    }    next = (*this)->next;    free(*this);    *this = next;}static void report_unused(FDSC *this){    FDSC *next;    while (this) {	next = this->next;	if (this->first) report_unused(this->first);	else if (this->type != fdt_none)		printf("Warning: did not %s file %s\n",this->type == fdt_drop ?		  "drop" : "undelete",file_name(this->name));	free(this);	this = next;    }}void file_unused(void){    report_unused(fp_root);}/* Local Variables: *//* tab-width: 8     *//* End:             */

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?