背景: #EDF0F5 #FAFBE6 #FFF2E2 #FDE6E0 #F3FFE1 #DAFAF3 #EAEAEF 默认  
阅读新闻

简易软盘镜像工具的实现及操作系统编写初步

[日期:2006-11-20] 作者: [字体: ]

简易软盘镜像工具的实现及操作系统编写初步

作者:liu_sir

下载源代码


摘要

  本文给出了软盘镜像生成工具的实现,并用其把自己编译生成的引导文件制作成软盘镜像,写入软盘实现一个简单操作系统的引导。
关键词 软盘镜像工具 操作系统引导。
  为了深入的了解80X86计算机的内部原理,参照别人的程序用汇编写了几个小程序。但生成软盘镜像的工具大都是DOS版本的,于是就想自己用VC也实现一个,并用把自己的引导程序写入软盘上验证一下。

1、镜像工具的基本原理

镜像工具读取磁盘上的文件,然后安装一定的格式生成软盘镜像文件,最后把软盘镜像文以512字节扇区为单位,写入软盘上。

2、镜像工具的界面设计及功能介绍


图1 主界面

2.1 【功能1】生成软盘镜像文件
(1)要生成镜像文件,需要先选择源文件,然后点击【生成…】,如MyDiskImg.img,点击【保存】生成镜像文件。


图2 生成镜像文件

(2)然后插入一张软盘,点击写软盘,选择刚生成的镜像文件。点击【打开】,系统便开始把选择的镜像文件写入软盘中。


图3 把镜像文件写入软盘

(3)重新启动计算机,选择从软盘启动,看操作是否成功。

3、具体代码编写

3.1 生成软盘镜像代码

基本原理为对列表中的源文件依次读取,然后写入到一个img文件中,具体的代码请参考(省略了部分不关键代码):

//生成镜像文件函数
UINT FuncGenImageFile(LPVOID pDialog)
{
	//Begin 显示设置
	省略…
	//1.创建输出镜像文件
	byte* pBuf = NULL;
	DWORD count = 0,dwFileLength=0;
	CFile outfile;//镜像文件
	if(!outfile.Open(pCurDlg->m_ImgFileName,CFile::modeCreate | CFile::modeWrite))
	{
		错误提示…
			return -1;
	}
	
	//2.依次对输入文件读取,写入输出镜像文件
	for( int i = 0 ; i < pList->GetCount(); ++i )
	{
		CString sInFileName; 
		pList->GetText(i,sInFileName);
		CFile infile;
		
		if(!infile.Open(sInFileName,CFile::modeRead))
		{
			错误提示…
				return -1;
		}
		dwFileLength = infile.GetLength();
		if(i==0)
		{
			if(dwFileLength>512)
			{
				sOut.Format(_T("文件%S不是一个有效的引导区文件,请使用[功能3]裁减该文件!"),sInFileName);
				return -1;
			}
		}
		if(i==0)//对0扇区文件进行特殊处理
		{
			pBuf = new byte[BlOCKSIZE]; 
			::memset(pBuf,0,BlOCKSIZE);
			infile.Read(pBuf,dwFileLength);
			if(pBuf[510]!=0x55) pBuf[510]=0x55;
			if(pBuf[511]!=0xAA) pBuf[510]=0xAA;
			outfile.Write(pBuf,BlOCKSIZE);
			dwFileLength = BlOCKSIZE;
		}
		else
		{
			pBuf = new byte[dwFileLength]; 
			infile.Read(pBuf,dwFileLength);
			outfile.Write(pBuf,dwFileLength);
		}
		infile.Close() ;
		
		delete[] pBuf;
		count += dwFileLength;
		pProgCtrl->SetPos((int)(count*100/FLOPYBYTESIZE));
    	}
	
#ifdef ADISKMODE
    //3.补充剩余的软盘字节为0
	DWORD dwRet = FLOPYBYTESIZE-count;
	pBuf = new byte[dwRet];
	memset(pBuf,0,dwRet);
	outfile.Write(pBuf,dwRet);
	delete[] pBuf;
#endif
    outfile.Close() ;
	
	
	//end 显示设置
	省略…
		return 0;
}
3.2 写镜像文件到软盘代码
基本原理是读取生成的镜像文件,然后把其写入软盘启动器A中, 具体的代码请参考(省略了部分不关键代码):
//写镜像文件到软驱
UINT FuncWriteFlopy(LPVOID pDialog)
{
	//begin显示设置
	省略…
		int nTotalBlocks = 80*18*2;
	
	//1.打开A驱动器
	HANDLE    hFile = CreateFile(_T("\\\\.\\A:"),
		GENERIC_WRITE,
		FILE_SHARE_WRITE,
		NULL,
		OPEN_EXISTING,
		0,
		NULL);
	if(hFile==NULL)
	{
		sOut = _T("不能打开驱动器A");//…
		return -1;
	}
	
	PBYTE pBuffer = (PBYTE)malloc(BlOCKSIZE);
	if(pBuffer==NULL)
	{
		sOut = _T("开辟内存空间失败!");//…
		return -1;
	}
	memset(pBuffer,0,BlOCKSIZE);
	//2.打开镜像文件
	CFile fInFile;
	BOOL b = fInFile.Open(pCurDlg->m_ImgFileName, CFile::modeRead);
	if(!b)
	{
		sOut.Format(_T("不能打开镜像文件%s!"),pCurDlg->m_ImgFileName);
		…
			return -1;
	}
	
	//3.分块写入镜像文件
	DWORD dwImgLen = fInFile.GetLength();
	for(int i=0;i<nTotalBlocks;i++)
	{
		DWORD dwLen=0;
		fInFile.Read(pBuffer, BlOCKSIZE);
		WriteFile(hFile, pBuffer, BlOCKSIZE, &dwLen, NULL);
		
#ifndef ADISKMODE
		if((DWORD)((i+1)*BlOCKSIZE)>=dwImgLen) break;//镜像文件写入完毕
#endif
		if(dwLen!=BlOCKSIZE)
		{
			sOut.Format(_T("写入镜像文件%s失败,请检查软驱!"),pCurDlg->m_ImgFileName);
			//…
			return -1;
			
		}
		pProgCtrl->SetPos((int)(i*100/nTotalBlocks));
	}
	
	//4.关闭文件
	fInFile.Close();
	CloseHandle(hFile);
	free(pBuffer);
	
	
	//end 显示设置
	pProgCtrl->SetPos(100);
	sOut.Format(_T("镜像文件%s写入软驱成功!"),pCurDlg->m_ImgFileName);
	//…
	return 0;
}
4、操作系统编写

4.1 PC机启动基本原理

(1) 开启电源后, 机器就会开始执行ROM BIOS的一系列系统测试动作,包括检查RAM,keyboard,显示器,软硬磁盘等等。
(2)执行完BIOS的系统测试之后,紧接着控制权会转移给ROM中的启动程序(ROM bootstrap routine);这个程序会将磁盘上的第0轨第0扇区(叫boot sector或MBR ,系统的引导程序就放在此处)读入内存中,并放到自0x07C0:0x0000开始的512个字节处;然后处理机将跳到此处开始执行这一引导程序;也即装入MBR中的引导程序后, CS:IP = 0x07C0:0x0000 。
注意: 如果这个扇区的最后两个字节是"55 AA",那么这就是一个引导扇区。如果最后两个字节不是"55 AA",那么BIOS 就检查下一个磁盘驱动器。

4.2 基本编译工具选择

NASM:编写系统级汇编代码的绝好的工具,而且分别支持Windows和Linux系统。本来原来用TASM编译的,可发现在编写的读取硬盘扇区的汇编代码裁剪去512字节的DOS头后居然不能还运行,郁闷了好多天,都没有解决,而用NASM编译出来可以的,所以决定改用NASM。

4.3 引导代码编写

文件boot.asm,具体代码及注释
 org 07c00h   ; 程序会被加载到7c00处,所以需要这一句
 mov ax, cs
 mov ds, ax
 mov es, ax
 Call DispStr   ; 调用显示字符串例程
 jmp $   ; 无限循环
DispStr:
 mov ax, BootMessage
 mov bp, ax   ; ES:BP = 串地址
 mov cx, 16   ; CX = 串长度
 mov ax, 01301h  ; AH = 13h,  AL = 01h
 mov bx, 000ch  ; 页号为0(BH = 0) 黑底红字(BL = 0Ch,高亮)
 mov dl, 0
 int 10h   ; int 10h
 ret
BootMessage:   db    " Hello, OS world!"
 times 510-($-$$) db   0 ;填充剩下的空间,使生成的二进制代码恰好为512字节
 dw  0aa55h   ; 引导扇区需要以55AA结束
4.4 用NASM编译
nasm boot.asm -o boot.bin

4.5 写入镜像文件及运行

按照2.1节所示的步骤把boot.Bin生成镜像文件,然后再写入软盘。下图是我再VMWare下的运行结果:


图4 系统运行显示

4.6 系统的其它功能实现

读取硬盘扇区,BIOS中断向量替换,实现文件系统,实虚模式转换及磁盘文件加载由于自己水平有限,不能详细介绍,网上有许多高手已经实现得很好了。如果感兴趣,可以参考相应的网址。详细请见:http://www.oldlinux.org

结语

  本文中的部分代码来自网上及林永君老师的课件,在此表示感谢。当然,单纯做一个操作系统来说,其现实的意义已不是太大。但对于更好的了解PC机的运行机制及操作系统的中断向量,系统调动及实模式及虚模式还是有很大的意义的。另外,也可以为Linux或者Windows下的驱动及系统开发打下一个良好的基础。

阅读:
打印
相关新闻       相关关键词: