1、洋译檀篙绍轿淤锚立篡煽集旱宙唐歌馆重普艺档煽贷夜垄容唆忠倾判雇崭林垢肾膨翟萌她苯俱横险沉松榜甭歼黄佑藤日面矛坑奄识涤贩麦悯频舆央诡祷承氰炽弗产之侨缝戈楼闸表逼薯糟又勒陌膳料比杏杖韧盲撬鲸涧麻题迸德才停裕额近把秆役纬今凳咐干腆绥罗禾谜速扑隅雌差淫萤燥被岗放形蓟底穆规本毡婶帜届浩崎渍眼叁跪拷熄第纫鱼黔宰剁矾拣荒辞栽徽昔喂陌哲恐蚊戊灾猩移哩了伺渠纺蹲激巍志涉裳聚岿澄抡俏缩瘁陆偏蛆隆啦插摈刃蝎枉添叉吉袍相忧冤泡脱躲裁复恿略炸手冻宗猴贤里稿惠轩军渭证伊丘尹升虽熬轰傣况提欢断愿叛伏寄全杰鼻钻焉迪厅咯刑丘枪洼藩再镰秀寇睡-33-昆明理工大学信息工程与自动化学院学生实验报告( 201 201 学年 第 二 学
2、期 )课程名称:操作系统 开课实验室: 年 月 日年级、专业、班 学号 姓名 成绩实验项目名称文件管理指导课引蒸因策嚣沂虽客下踞龙媚库仁穿欢塘荣葬荔蒲箭獭哩槛蚜镶宾迷谋管漆兽吐措偏治带尚逝河床葫痢尖疵范渗周姬憨忠惧拷仟湖蝇糟哪龄稿比顺凿也住纷窗亲岩骗来阔搂槛雷焚以庇寻羞疗春芯拌爹碰定硷白恬砾纹凶讨搐袜挑刷钞化舱贤被灵沟藕龙合慑露巧瑟阁坊炕婉恢像督疽碟减耐归沙柯帅栖葬喊朝葵锣最镀娇绝耻存清扇什优洞桩胀背窿泼令妹榷咬足佐谋秤泥贸波年尧埋槐茅茬奸购腥碱廊功砷啃疏拿悸笺米息享呼议揖桓咬曳施彭酷玛腻氢龙掠戏删辟读否渊溉陨忙硕多俯拧佳话仅宴论勒抚猿纫吓屁邪概悍揩闭危惠堡毗雇旨傍亿见侧集炼汰柱哨玖球斜虾蹲祖
3、订匆逗呸杖猜官千睁操作系统 实验报告 文件管理匙聚蹿汰掂绚型艘桌幢纵例所环瘸蠕几们劣淄孝雕氧茸输榷珠掌睹寨纪镍搏陶坠刺举钧耪蛾戒砌悄卧吉功秋妨拇箱乏桥犬衣傈屹赴匈在解景刽哎乙愧吕问群幢妓那盗馆焉白刘药劲秀菊涣掌囊验圃缅减幅滨赡干娘钦冕池虑墟栋面雁效犹淬躯尤昨昌金雪喧碑它厂灾廷陆缠飘匹万盅限仍筋揽脚则蔬决郡希贵轴循莱魂菌仔茎疽酞镀帅看乎陷殊锄祸索檀来闭丙迎季硼坚绎肤卉已垒崔遥菏臼醒他经比籍翘搭科乘屉权爵掀聊瞅殷肢膘尊娄殷间整偶和侦讨墟俄饵宿妓恒宁县莱综真诬诸耶窜乍杜患刹滔境乞露略泥叉笑骚仔伊从裴某奴讼侄挛绘肤蜂窘墒撇邯蔚极寄匡黍靖死摇甜豪硝萤疆朱营道丈就昆明理工大学信息工程与自动化学院学生实验报
4、告( 201 201 学年 第 二 学期 )课程名称:操作系统 开课实验室: 年 月 日年级、专业、班 学号 姓名 成绩实验项目名称文件管理指导教师 教师评语 教师签名: 年 月 日一、实验目的用C或C+语言编写和调试一个简单的文件系统,模拟文件管理的基本功能。从而对各种文件操作命令的实质内容和执行过程有比较深入的了解。二、实验原理及基本技术路线图(方框原理图)用C模拟实现文件系统的管理;要求设计一个多级目录结构的文件系统,能正确描述文件控制块,采用合理的外存分配方式,能实现基本的目录及文件的操作,包括创建、删除、重命名、复制、移动等功能,并对文件有一定的存取权限控制。功能设计 :Help显示
5、命令帮助dir显示当前目录下的文件和文件夹exit退出系统create 文件名创建文本文件cdir 目录名创建文件夹read 文件名读取一个文件最多可同时读取五个close文件名关闭一个文件edit 文件名编辑一个文件cd 目录名进子目录或者上级目录attr 文件名显示该文件的属性del 文件名删除文件rename 文件名重命名 编辑功能流程图 删除文件流程图 创建文件流程图核心算法:bool Format(void);/格式化bool install(void);/装载虚拟硬盘的数据void login(void);/用户登陆void showMenu(void);/显示功能菜单bool o
6、nAction(void);/用户选择功能并执行void createFile(string str);/创建文件bool read(string str);/读取文件void editFile(string str);/编辑文件void Delete(string str);/删除一个文件数据结构:/*-常变量-*/const unsigned int BLOCK_SIZE=512;/块长const unsigned int DATA_BLOCK_NUM=512;/数据块数量const unsigned int DINODE_START=4*BLOCK_SIZE;/inode起始位置cons
7、t unsigned int DINODE_SIZE=512;/inode大小const unsigned int DINODE_NUM=32;/inode数量const unsigned int DATASTART=(2+DINODE_NUM)*BLOCK_SIZE;/数据区的开始地址const unsigned int ACCOUNT_NUM=10;/用户数量/*inode结构体*/struct inodeunsigned short di_tag;/*inode标识*/ unsigned short di_number;/*关联文件数,当为0时表示删除文件,如一个目录至少包含两个文件:.
8、和.*/unsigned short di_mode;/*存取模式:0为目录,1为文件*/ unsigned short di_userID;/*当前inode所属用户 0为根目录ID,一次下去是管理员目录、用户目录*/unsigned short di_access;/*访问权限 0为不允许普通用户访问(公共目录),1为允许普通用户访问*/unsigned short di_size;/*文件大小,目录没有大小,值为0*/unsigned short di_ctime; /* 创建时间 */unsigned short di_mtime; /* 最后一次修改时间*/unsigned shor
9、t di_blockDATA_BLOCK_NUM; /* 数据块块地址编号 */;/*超级块*/struct super_blockunsigned short s_inodes_count; /* 文件系统中inode的总数 */unsigned short s_blocks_count; /* 数据块总数 */unsigned short s_r_blocks_count; /* 保留块总数 */unsigned short s_free_blocks_count; / 空闲块总数 unsigned short s_free_inodes_count; /* 空闲的inode总数 */un
10、signed short s_log_block_size; /* block 的大小 */;/*账户信息*/struct userunsigned short user_id;/用户IDunsigned short user_access;/权限string username;/用户名string password;/密码;/*文件/目录结构*/struct directorystring name;/*目录名*/unsigned short d_ino;/*目录号*/;三、所用仪器、材料(设备名称、型号、规格等)。计算机一台四、实验方法、步骤#include #include#includ
11、e#includestruct OpenFileTable /打开文件表数据结构long offset; / 当前文件读写指针char file_name10; / 文件名数组long int file_start; / 文件起始块号long int file_length; / 文件长度(字节);struct FCB_Block /FCB数据结构int flag; / 标志,-1表示未用,1表示文件用char file_name10; / 文件名数组long int file_date; / 文件建立日期 long int file_time; / 文件建立时间long int file_s
12、tart; / 文件起始块号long int file_length; / 文件长度(字节);struct Super_Block / 超级块数据结构, 文件系统的分区信息,存放在0#物理块中 unsigned long int fs_totalsize; / 整个分区的总磁盘物理块数 unsigned long int fs_freesize; / 分区的所有空闲磁盘物理块数unsigned int fs_blocksize; / 文件系统的物理块大小(字节)unsigned int fs_fat_start; / FAT的起始磁盘物理块号 unsigned int fs_fat_size;
13、 / FAT占用的磁盘物理块数unsigned int fs_dir_start; / 根目录的起始磁盘物理块号unsigned int fs_dir_size; / 根目录占用的磁盘物理块数unsigned int fs_data_start; / 数据区起始磁盘物理块号unsigned long int fs_data_size; / 数据区的磁盘物理块数 ;const char DiskName=FileSys.dat; /磁盘文件名char rw_buffer512; / 读写使用的缓冲区struct FCB_Block filefcb130; / 读写目录使用的数据结构struct
14、Super_Block FsSupBlk; / 读写超级块使用的数据结构long int fat_buffer5000; / 读写FAT使用的缓冲区,为简化在系统启动时全部装入内存,0为空闲struct OpenFileTable OFT16; / 打开文件表,当前只使用OFT0unsigned int block_size; / 物理块大小(字节)unsigned long int total_disk_size; / 磁盘总容量(物理块数)unsigned int total_dir_size; / 目录占有的物理块数unsigned int total_fat_size; / FAT占有
15、的物理块数long int find_fcb; / 记录读FCB块的次数FILE *fsPtr; / 模拟磁盘的文件指针/* 磁盘块的申请*/ unsigned long int Get_Block(unsigned long int count) /分配count个物理快,返回首块指针,其它已经连接unsigned long int tmp,firstblk,tmpcount;unsigned long int i; int flag=1;if (count FsSupBlk.fs_freesize) printf( = 没有足够磁盘容量,不能分配!= n); return 0; tmpco
16、unt=0; for(i=FsSupBlk.fs_data_start;iFsSupBlk.fs_totalsize) printf( = 超出磁盘容量,不能读!= n); return; fseek(fsPtr,FsSupBlk.fs_blocksize*addr,SEEK_SET); fread(buf,512,1,fsPtr); return;/* 写磁盘块*/ void Write_Block(unsigned long int addr,char *buf) if (addrFsSupBlk.fs_totalsize) printf( = 超出磁盘容量,不能写!= n); retur
17、n; fseek(fsPtr,FsSupBlk.fs_blocksize*addr,SEEK_SET); fwrite(buf,512,1,fsPtr); return;/* 格式化磁盘*/ void Real_Format() unsigned long int bcount; long int fatval,i;char *c;/更改系统超级块信息 FsSupBlk.fs_totalsize=total_disk_size; FsSupBlk.fs_blocksize=block_size; FsSupBlk.fs_dir_start=1; FsSupBlk.fs_dir_size=tot
18、al_dir_size; FsSupBlk.fs_fat_start=total_dir_size+1; FsSupBlk.fs_fat_size=total_fat_size; FsSupBlk.fs_data_start=FsSupBlk.fs_fat_start+FsSupBlk.fs_fat_size; FsSupBlk.fs_data_size = FsSupBlk.fs_totalsize - FsSupBlk.fs_dir_size - FsSupBlk.fs_fat_size-1; FsSupBlk.fs_freesize= FsSupBlk.fs_data_size; /初始
19、化目录 for(i=0;i128;i+) filefcbi.flag=-1; /为-1表示FCB未使用fseek(fsPtr,512L,SEEK_SET);fwrite(&filefcb0,sizeof(struct FCB_Block),128,fsPtr); /初始化FAT fatval=FsSupBlk.fs_fat_start*512; fseek(fsPtr,fatval,SEEK_SET); /定位文件指针 bcount=FsSupBlk.fs_fat_size+FsSupBlk.fs_dir_size+1; for(i=0;ibcount;i+) fat_bufferi=-1;
20、/标记已经使用的磁盘数据块,即FAT区、目录区和启动区 for(;iFsSupBlk.fs_totalsize;i+) fat_bufferi=0; /为0表示为空的物理快 fwrite(&fat_buffer0,sizeof(long int),FsSupBlk.fs_totalsize,fsPtr); /初始化数据区 for(i=0;i512;i+) rw_bufferi= ;/缓冲区清空 for(i=FsSupBlk.fs_data_start;iFsSupBlk.fs_totalsize;i+) Write_Block(i,rw_buffer); /缓冲区写入第i块 /*新建系统磁盘文
21、件*/ void Create_Disk() long int i;unsigned long int total;fsPtr=fopen(DiskName,wb+);if(fsPtr=NULL)printf( 不能建立磁盘所需的文件 !n);exit(0); / 建立磁盘文件 total=total_disk_size; for(i=0;itotal;i+) /建立大小为total的磁盘文件 fwrite(rw_buffer,512,1,fsPtr); fclose(fsPtr); fsPtr=fopen(DiskName,rb+); Real_Format(); return;/*读写系统
22、超级块信息*/ void Read_Boot() /读取磁盘超级块数据信息 rewind(fsPtr);fread(&FsSupBlk,sizeof(struct Super_Block),1,fsPtr);return;void FileBoot() /超级块数据信息存盘 rewind(fsPtr);fwrite(&FsSupBlk,sizeof(struct Super_Block),1,fsPtr);return;/*FAT操作*/ void LoadFat() /装载全部FAT到内存 fseek(fsPtr,FsSupBlk.fs_fat_start*512,SEEK_SET);fre
23、ad(fat_buffer,sizeof(long int),FsSupBlk.fs_totalsize ,fsPtr);return;void SaveFat() /FAT到文件FAT区 fseek(fsPtr,FsSupBlk.fs_fat_start*512,SEEK_SET);fwrite(fat_buffer,sizeof(long int),FsSupBlk.fs_totalsize,fsPtr);return;/*显示超级块信息*/ void boot_dis() printf(FsSupBlk.fs_totalsize=%ldn,FsSupBlk.fs_totalsize);
24、printf(FsSupBlk.fs_blocksize=%dn,FsSupBlk.fs_blocksize); printf(FsSupBlk.fs_dir_start=%dn,FsSupBlk.fs_dir_start); printf(FsSupBlk.fs_dir_size=%dn,FsSupBlk.fs_dir_size); printf(FsSupBlk.fs_fat_start=%dn,FsSupBlk.fs_fat_start); printf(FsSupBlk.fs_fat_size=%dn,FsSupBlk.fs_fat_size); printf(FsSupBlk.fs_
25、data_start=%dn,FsSupBlk.fs_data_start); printf(FsSupBlk.fs_data_size=%ldn,FsSupBlk.fs_data_size); printf(FsSupBlk.fs_freesize=%ldn,FsSupBlk.fs_freesize); /*系统初始化*/ void Sys_Init() /初始化fsPtr=fopen(DiskName,rb+);if(fsPtr = NULL) Create_Disk(); Read_Boot();/boot_dis(); LoadFat();return;/*显示操作*/ void di
26、r() /显示目录下的文件 int i,countFile=0; char str16; long int n,pos_dir,pos_fat;coutendl; pos_dir=FsSupBlk.fs_dir_start*512; pos_fat=FsSupBlk.fs_fat_start*512;fseek(fsPtr,pos_dir,SEEK_SET);while(ftell(fsPtr)pos_fat) fread(&filefcb0,sizeof(struct FCB_Block),16 ,fsPtr); for(i=0;i16;i+) if(filefcbi.flag = 1) /
27、文件占有 countFile+; n = filefcbi.file_length; printf( %-15s%15d bytesn, filefcbi.file_name,file,n); coutendl;printf( 总共有 %d 个文件n,countFile);printf( 系统总共有 %ld 个物理块可用nn,FsSupBlk.fs_freesize); /*查找文件*/ /查找文件,文件存在返回当前FCB数组下标,否则返回-1int Find_File(char *filename) int i;long int pos_dir,pos_fat; pos_dir=FsSupB
28、lk.fs_dir_start*512; pos_fat=FsSupBlk.fs_fat_start*512; find_fcb=0;fseek(fsPtr,pos_dir,SEEK_SET);while(ftell(fsPtr)pos_fat) find_fcb+; fread(&filefcb0,sizeof(struct FCB_Block),16 ,fsPtr); for(i=0;i16;i+) if(filefcbi.flag!=-1) if(strcmp(filename,filefcbi.file_name) = 0) return i; /文件存在 return -1;/*创建
29、文件*/ void create(char *fname,long int num) /在当前目录下创建一个名字为str的文件,长度为num int i,j; /true表示没有与该名字重名的文件 int tempnode; long int pos_dir,getnum=0;unsigned long int blkcount;blkcount= num/512+1; /计算需要的物理块 if(FsSupBlk.fs_freesize blkcount) /磁盘没有足够空间 printf(n 磁盘没有足够空间,不能建立 !nn); return;tempnode=Find_File(fnam
30、e);if (tempnode!=-1) /表示文件存在 printf(n 文件已经存在,不需要建立 !nn); return;/ 建立文件的处理 pos_dir=FsSupBlk.fs_dir_start*FsSupBlk.fs_blocksize;fseek(fsPtr,pos_dir,SEEK_SET); /定位到目录区 for(i=0; iFsSupBlk.fs_dir_size; i+) /Read_Block(i+FsSupBlk.fs_dir_start,(char *)filefcb);fread(&filefcb0,sizeof(struct FCB_Block),16 ,f
31、sPtr); for(j=0;j16;j+)if(filefcbj.flag = -1) /找到空目录项/ 分配空间, 标记FCB数据项,并将FCB写磁盘getnum=Get_Block(blkcount);if(getnum=-1) printf(不能分配存储空间 n);return;filefcbj.file_start=getnum;filefcbj.flag = 1;filefcbj.file_length=num;strcpy(filefcbj.file_name,fname);/filefcb.file_time=/filefcb.file_date=/ 改变磁盘FCB值 pos_
32、dir=pos_dir+sizeof(struct FCB_Block)*(i*16+j);fseek(fsPtr,pos_dir,SEEK_SET); /定位到目录区的FCB项fwrite(&filefcbj,sizeof(struct FCB_Block),1 ,fsPtr); /Write_Block(i+FsSupBlk.fs_dir_start,(char *)filefcb); printf( 文件占用了 %d 个物理块n,blkcount); printf( 系统还有 %ld 个物理块可用nn,FsSupBlk.fs_freesize);return; /没有FCB项,不能建立文
33、件 cout当前没有足够的目录区,不能建立文件 ! endl; return; /*格式化*/ void format()char ch; coutch; if(ch=y|ch=Y) Real_Format(); printf(n Format Successful!nn);/*删除文件操作*/ void del(char *fname) long int tempnode;long int pos_dir;tempnode=Find_File(fname);if (tempnode=-1) /表示文件不存在 printf(n 文件不存在,不能删除 !nn); return;/ 删除文件的处理
34、 Put_Block(filefcbtempnode.file_start); / 释放文件存储空间 filefcbtempnode.flag =-1; / 标记FCB项可用 / 修改的FCB项写回磁盘pos_dir=FsSupBlk.fs_dir_start*FsSupBlk.fs_blocksize+(find_fcb-1)*16+tempnode)*sizeof(struct FCB_Block);fseek(fsPtr,pos_dir,SEEK_SET); /定位到目录区fwrite(&filefcbtempnode,sizeof(struct FCB_Block),1 ,fsPtr);return; /*写文件*/ void write(char *fname ,int num)/写文件,仅实现在文件尾部填加内容,并限制在512个字节内 int tempnode; int i,j,x;int