⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 main.c

📁 UNIX模拟系统模拟得很完善!
💻 C
📖 第 1 页 / 共 4 页
字号:

		// 以下重写一个文件  

		// 释放原有文件的磁盘块  
		for( i = 0; i < inode->i_size / BLOCKSIZ + 1; i++ )
			bfree( inode->i_addr[i] );
		inode->i_size = 0;
		
		// 该文件已经打开  
		for( i = 0; i < SYSOPENFILE; i++ )
			if( sys_ofile[i].f_inode == inode )
			{
				sys_ofile[i].f_off = 0;
				for( j = 0; j < NOFILE; j++ )
				{
					if( user[user_id].u_ofile[j] == i )
						return j;
				}
			}

			// 该文件还没有打开  
		for( i = 0; i < NOFILE; i++ )
			if( user[user_id].u_ofile[i] == SYSOPENFILE + 1 )
			{
				user[user_id].u_uid = inode->i_uid;
				user[user_id].u_gid = inode->i_gid;
				for( j = 0; j < SYSOPENFILE; j++ )
					if( sys_ofile[j].f_count == 0 )
					{
						user[user_id].u_ofile[i] = j;
						sys_ofile[j].f_off = 0;
						sys_ofile[j].f_count = 1;
						sys_ofile[j].f_inode = inode;
						sys_ofile[j].f_flag = inode->i_mode;
					}
				return i;
			}
	}
	else													// 文件不存在,建立新文件  
	{
		pre_inode = iget( pre_dinodeid );
		if( !( access( pre_inode ) | WRITE ))				// 父目录不能写  
		{
			iput( pre_inode );
			textcolor(12);
			printf( "You have no access to creat\n" );
			textcolor(7);
			return -1;
		}
		inode = ialloc();
		inode->i_addr[0] = balloc();

		// 父目录为当前目录  
		if( pre_dinodeid == cur_path_inode->i_ino )
		{
			di_ith = iname( dir, end_path_name );
			strcpy( dir.direct[di_ith].d_name, end_path_name );
			dir.direct[di_ith].d_ino = inode->i_ino;
			dir.size++;
		}
		else			// 父目录不是当前目录  
		{
			tempdir = getdir( pre_dinodeid, " " );
			di_ith = iname( tempdir, end_path_name );
			strcpy( tempdir.direct[di_ith].d_name, end_path_name );
			tempdir.direct[di_ith].d_ino = inode->i_ino;
			tempdir.size++;
			putdir( pre_dinodeid, tempdir );
		}

		// 设置i节点信息  
		inode->i_mode = mode | DIREG;
		inode->i_uid = user[user_id].u_uid;
		inode->i_gid = user[user_id].u_gid;
		inode->i_size = 0;
		inode->i_number = 1;
		
		// 找空闲的系统打开表项  
		for( i = 0; i < SYSOPENFILE; i++ )
		{
			if( sys_ofile[i].f_count == 0 )
				break;
		}
		if( i == SYSOPENFILE )
		{
			textcolor(12);
			printf( "Can't creat file\n" );
			textcolor(7);
			return 0;
		}
		// 找空闲用户打开表项  
		for( j = 0; j < NOFILE; j++ )
		{
			if( user[user_id].u_ofile[j] == SYSOPENFILE + 1 ) 
				break;
		}
		if( j == NOFILE )
		{
			textcolor(12);
			printf( "Can't creat file\n" );
			textcolor(7);
			return 0;
		}
		// 写系统打开表项信息  
		user[user_id].u_ofile[j] = i;
		sys_ofile[i].f_flag = mode;
		sys_ofile[i].f_count = 1;
		sys_ofile[i].f_off = 0;
		sys_ofile[i].f_inode = inode;

		return j;		                 // 返回文件描述符  
	}
}

// 删除文件  
void _delete( char * filename )
{
	unsigned int dinodeid;
	struct inode * inode;
	struct dir tempdir;
	int i;

	dinodeid = namei( filename );
	if(( dinodeid == NULL ) || ( dinodeid == -1 ))	// 找不到该文件  
	{
		textcolor(12);
		printf( "rm: %s: 文件不存在\n", filename );
		textcolor(7);
		return;
	}

	inode = iget( dinodeid );
	if( !( inode->i_mode & DIREG ))			// 不是文件  
	{
		textcolor(12);
		printf( "rm: %s: 不是文件\n", filename );
		textcolor(7);
		iput( inode );
		return;
	}
	if( !( access( inode ) & WRITE))		// 没有权限  
	{
		textcolor(12);
		printf( "bash: %s: 无此权限\n", filename );
		textcolor(7);
		iput( inode );
		return;
	}

	if( pre_dinodeid == cur_path_inode->i_ino )
	{
		for( i = 0; i < dir.size; i++ )
		{
			if( dir.direct[i].d_ino == dinodeid )
				break;
		}
		for( ; i < dir.size - 1; i++ )
		{
			dir.direct[i].d_ino == dir.direct[i + 1].d_ino;///////////
			dir.direct[i].d_name == dir.direct[i + 1].d_name;//////////
		}
		dir.direct[i].d_ino = 0;
		strcpy( dir.direct[i].d_name, " " );
		dir.size--;
	}
	else
	{
		tempdir = getdir( pre_dinodeid, " " );

		for( i = 0; i < tempdir.size; i++ )
		{
			if( tempdir.direct[i].d_ino == dinodeid )
				break;
		}
		for( ; i < tempdir.size - 1; i++ )
		{
			tempdir.direct[i].d_ino == tempdir.direct[i + 1].d_ino;
			tempdir.direct[i].d_name == tempdir.direct[i + 1].d_name;
		}
		tempdir.direct[i].d_ino = 0;
		strcpy( tempdir.direct[i].d_name, " " );
		tempdir.size--;

		putdir( pre_dinodeid, tempdir );
	}
	printf("文件删除成功!\n\n");

	inode->i_number--;
	iput( inode );
}

// 打开文件  
int open( char * filename, unsigned short openmode )
{
	unsigned int dinodeid;
	struct inode * inode;
	int i, j;

	// 检索目录  
	dinodeid = namei( filename );
	if( dinodeid == -1 )
	{
		textcolor(12);
		printf( "无此目录\n" );
		textcolor(7);
		return -1;
	}

	if( dinodeid == NULL )
	{
		if( openmode & O_CREAT )
			return creat( filename, openmode & ( ~ O_CREAT ));
		textcolor(12);
		printf( "文件不存在\n" );
		textcolor(7);
		return -1;
	}

	// 分配内存i节点  
	inode = iget( dinodeid );

	if( !access( inode ))
	{
		textcolor(12);
		printf( "无此权限打开文件\n" );
		textcolor(7);
		iput( inode );
		return -1;
	}

	// 分配文件表项  
	for( i = 0; i < SYSOPENFILE; i++ )
	{
		if( sys_ofile[i].f_count == 0 )
			break;
	}
	if( i == SYSOPENFILE )
	{
		textcolor(12);
		printf( "系统打开了太多文件\n" );
		textcolor(7);
		iput( inode );
		return -1;
	}

	// 分配用户文件描述表项  
	for( j = 0; j < NOFILE; j++ )
		if( user[user_id].u_ofile[j] == SYSOPENFILE + 1 )
			break;
	if( j == NOFILE )
	{
		textcolor(12);
		printf( "用户打开太多文件\n" );
		textcolor(7);
		iput( inode );
		return -1;
	}
	printf("文件打开成功!\n\n");

	sys_ofile[i].f_inode = inode;
	sys_ofile[i].f_flag = openmode;
	sys_ofile[i].f_count++;

	// 读/写指针置位  
	if( openmode & O_APPEND )
		sys_ofile[i].f_off = inode->i_size;			// 读/写指针置于文件尾  
	else
		sys_ofile[i].f_off = 0;					// 读/写指针置于文件开始位置  

	user[user_id].u_ofile[j] = i;

	// 将原来存在的文件清空  
	if( openmode & O_TRUNC )
	{
		for( i = 0; i < inode->i_size / BLOCKSIZ + 1; i++ )
			bfree( inode->i_addr[i] );
		inode->i_size = 0;
	}

	return j;
}

void close( int cfd )
{
	int sys_ip;
	
	// 获得指向文件表项的指针sys_ip  
	sys_ip = user[user_id].u_ofile[cfd];

	// 释放内存i节点  
	iput( sys_ofile[sys_ip].f_inode );

	// 文件表项引用计数减1  
	sys_ofile[sys_ip].f_count--;

	// 清空用户打开表  
	user[user_id].u_ofile[cfd] = SYSOPENFILE + 1;
}

// 读文件  
int read( int fh, char * buf, unsigned int nbyte )
{
	unsigned long off;
	int block, block_off, i, sys_ip;
	struct inode * inode;

	sys_ip = user[user_id].u_ofile[fh];
	inode = sys_ofile[sys_ip].f_inode;

	if( !( sys_ofile[sys_ip].f_flag & O_RDONLY ))
	{
		textcolor(12);
		printf( "The file is not opened for read\n" );
		textcolor(7);
		return -1;
	}

	off = sys_ofile[sys_ip].f_off;
	block = off / BLOCKSIZ;					// 从inode->i_addr[block]号盘块读起  
	block_off = off % BLOCKSIZ;				// block_off是在block号盘块中的偏移量  

	// 读写指针在文件末尾,读出字数为0  
	if( off == inode->i_size )
		return 0;

	// 读文件不用跨越盘块,即在一盘块中就可读出nbyte字  
	if( block_off + nbyte < BLOCKSIZ )
	{
		fseek( fd, DATASTART + inode->i_addr[block] * BLOCKSIZ + block_off, SEEK_SET );
		fread( buf, 1, nbyte, fd );
		return nbyte;
	}

	// 读文件跨越盘块,即所读的文件有多个盘块  

	// 先读block盘块  
	fseek( fd, DATASTART + inode->i_addr[block] * BLOCKSIZ + block_off, SEEK_SET );
	fread( buf, 1, BLOCKSIZ - block_off, fd );
	buf += BLOCKSIZ - block_off;

	block++;
	// 所读的文件没有到末尾  
	if( nbyte + off < inode->i_size )
	{
		for( i = 0; i<inode->i_size / BLOCKSIZ;i++)
		{
			fseek(fd,inode->i_addr[i + block] * BLOCKSIZ, SEEK_SET );
			fread( buf, 1, BLOCKSIZ, fd );
			buf += BLOCKSIZ;
		}
		fseek( fd, DATASTART + inode->i_addr[i + block] * BLOCKSIZ, SEEK_SET );
		fread( buf, 1, ( nbyte - block_off ) % BLOCKSIZ, fd );
		
		// 置文件读/写指针位置  
		sys_ofile[sys_ip].f_off += nbyte;

		return nbyte;
	}
	
	// 所读的文件到末尾  
	for( i = block; i < inode->i_size / BLOCKSIZ; i++ )
	{
		fseek( fd, DATASTART + inode->i_addr[i] * BLOCKSIZ, SEEK_SET );
		fread( buf, 1, BLOCKSIZ, fd );
		buf += BLOCKSIZ;
	}
	fseek( fd, DATASTART + inode->i_addr[i] * BLOCKSIZ, SEEK_SET );
	fread( buf, 1, inode->i_size % BLOCKSIZ, fd );
	sys_ofile[sys_ip].f_off = inode->i_size;
	return ( inode->i_size - off );
}

// 写文件  
int write( int fh, char * buf, unsigned int nbyte )
{
	unsigned long off;
	int block, block_off, i, sys_ip;
	struct inode * inode;
	char * temp_buf;

	sys_ip = user[user_id].u_ofile[fh];
	inode = sys_ofile[sys_ip].f_inode;

	if( !( sys_ofile[sys_ip].f_flag & O_WRONLY ))
	{
		printf( "The file is not opened for write\n" );
		return -1;
	}

	off = sys_ofile[sys_ip].f_off;

	block = off / BLOCKSIZ;					// 从inode->i_addr[block]号盘块起开始写  
	block_off = off % BLOCKSIZ;				// block_off是在block号盘块中的偏移量  

	// 写文件不用跨越盘块,即在一盘块中就可写入nbyte字  
	if( block_off + nbyte < BLOCKSIZ )
	{
		fseek( fd, DATASTART + inode->i_addr[block] * BLOCKSIZ + block_off, SEEK_SET );
		fwrite( buf, 1, nbyte, fd );
		if( inode->i_size < nbyte )
			inode->i_size = nbyte;
		return nbyte;
	}

	// 写文件跨越盘块,即所写的文件有多个盘块  

	// 先往block盘块写入  
	fseek( fd, DATASTART + inode->i_addr[block] * BLOCKSIZ + block_off, SEEK_SET );
	fwrite( temp_buf, 1, BLOCKSIZ - block_off, fd );

	buf += BLOCKSIZ - block_off;

	// 所写的文件没有到末尾  
	if( nbyte + off < inode->i_size )
	{
		for( i = 0;i<inode->i_size / BLOCKSIZ;i++)
		{
			fseek(fd,inode->i_addr[i + block] * BLOCKSIZ, SEEK_SET );
			fwrite( buf, 1, BLOCKSIZ, fd );
			buf += BLOCKSIZ;
		}
		fseek( fd, DATASTART + inode->i_addr[i + block] * BLOCKSIZ, SEEK_SET );
		fwrite( buf, 1, ( nbyte - block_off ) % BLOCKSIZ, fd );
		
		// 置文件读/写指针位置  
		sys_ofile[sys_ip].f_off += nbyte;

		if( inode->i_size < nbyte )
			inode->i_size = nbyte;

		return nbyte;
	}
	// 所读的文件到末尾  

	// 写到文件尾  
	for( i = block; i < inode->i_size / BLOCKSIZ; i++ )
	{
		fseek( fd, DATASTART + inode->i_addr[i] * BLOCKSIZ, SEEK_SET );
		fwrite( buf, 1, BLOCKSIZ, fd );
		buf += BLOCKSIZ;
	}

	// 申请空间再写  
	for( i = 0;i<inode->i_size / BLOCKSIZ;i++)//; )
	{
		inode->i_addr[i] = balloc();
		fseek( fd, DATASTART + inode->i_addr[i] * BLOCKSIZ, SEEK_SET );
		fwrite( temp_buf, 1, BLOCKSIZ, fd );
		buf += BLOCKSIZ;
	}
	fseek( fd, DATASTART + inode->i_addr[i] * BLOCKSIZ, SEEK_SET );
	fwrite( buf, 1, ( nbyte + off ) % BLOCKSIZ, fd );
	inode->i_size = nbyte + off;
	sys_ofile[sys_ip].f_off = inode->i_size;

	if( inode->i_size < nbyte )
			inode->i_size = nbyte;

	return nbyte;

}

// 文件编辑器
void edlin( char * filename )
{	char buf[BLOCKSIZ];
	int i, fh;
	textcolor(12);
	if(( fh = open( filename, DEFAULTMODE | O_CREAT )) == -1)
		printf( "Can't creat file %s\n", filename );
	textcolor(7);
	printf( "请输入文件内容...按Ctrl+x后按回车结束\n\n" );
	for( i = 0; i < BLOCKSIZ; i++ )					// 按Ctrl+x后按回车结束  
	{
		buf[i] = getchar();
		if( buf[i] == 24 )
			break;
	}
	buf[i] = '\0';

	write( fh, buf, strlen( buf ));
	close( fh );
	buf[i] = getchar();
}

// 显示文件内容  
void list( char * filename )
{
	struct inode * inode;
	char * buf;
	int fh, dinodeid;

	dinodeid = namei( filename );
	inode = iget( dinodeid );
	buf = ( char * )malloc( inode->i_size );
	if(( fh = open( filename, O_RDONLY )) != -1 )
	{
		read( fh, buf, inode->i_size );
		printf( "%s\n", buf );
		close( fh );
	}
}

⌨️ 快捷键说明

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