NTFS文件系统详解(二)MBR\EBR基本信息

article/2025/8/27 15:36:17

NTFS文件系统详解(二)MBR\EBR基本信息

  • 一、MBR结构分析
    • 1. 第一个分区表项
    • 2. 第二个分区表项
    • 3. 第三个分区表项
    • 4. 第四个分区表项
  • 二、EBR结构分析
    • 1. 第一个分区表项
    • 2. 第二个分区表项
      • 2.1 第一个分区表项
      • 2.2 第二个分区表项
      • 2.3 第三个分区表项
  • NTFS文件系统详解系列



  经过上一篇文章对硬盘的基本结构的详细介绍后,我们再来看看MBR和EBR的结构。

一、MBR结构分析


  MBR(Main Boot Record 主引导记录区)位于整个硬盘的0磁道0柱面1扇区。不过,在总共512字节的主引导扇区中,MBR只占用了其中的446个字节,另外的64个字节交给了 DPT(Disk Partition Table硬盘分区表),最后两个字节“55,AA”是分区的结束标志。这个整体构成了硬盘的主引导扇区。
  主引导记录中包含了硬盘的一系列参数和一段引导程序。其中的硬盘引导程序的主要作用是检查分区表是否正确并且在系统硬件完成自检以后引导具有激活标志的分区上的操作系统,并将控制权交给启动程序。MBR是由分区程序(如fdisk.exe)所产生的,它不依赖任何操作系统,而且硬盘引导程序也是可以改变 的,从而实现多系统共存。

在这里插入图片描述


  其中从0x01BE到0x01FD这六十四个字节表示的是4个主分区的信息。每个主分区16个字节描述,这16反个字节的含义是:

存贮字节位内容及含义
第1字节引导标志,若值为80H表示活动分区,若值为00H表示非活动分区。
第2\3\4字节本分区的起始磁头号、扇区号、柱面号。其中:
  磁头号–第2字节;
  扇区号–第3字节的低6位;
  柱面号–第3字节高2位+第4字节8位。
第5字节分区类型符;
  00H–表示该分区未用(即没有指定);
  06H–FAT16基本分区;
  0BH–FAT32基本分区;
  05H–扩展分区;
  07H–NTFS分区;
  0FH–(LBA模式)扩展分区(83H–Linux分区等)
第6\7\8字节本分区的结束磁头号、扇区号、柱面号。其中:
  磁头号–第6字节;
  扇区号–第7字节的低6位;
  柱面号–第7字节高2位+第8字节8位。
第9\10\11\12字节本分区之前已用了的扇区数(分区起始扇区)。
第13\14\15\16字节本分区的总扇区数


  注:由于硬盘的发展,2\3\4字节的分区起始扇区所在柱面,磁头,扇区,6\7\8字节的分区终止扇区所在的柱面,磁头,扇区信息已不正确,如果硬盘稍微大点,chs逻辑硬盘地址方案中的c值有可能就不能由现在分配的位数表达了。因而,在柱面超过3FF的时候,将失效,这三个值一般为FE FF FF(16450560个扇区之后,也就是扇区位置超过7.44G時设为这个值),在MBR规范中,直接舍弃“virtual CHS values逻辑硬盘地址方案”,而采用LBA方案,更详细的分区类型符如下:

分区类型标志
----
00空,mocrosoft不允许使用。63GNU HURD or Sys
01FAT3264Novell Netware
02XENIX root65Novell Netware
03XENIX usr70Disk Secure Mult
04FAT16<32M75PC/IX
05Extended80Old Minix
06FAT1681Minix/Old Linux
07HPFS/NTFS82Linux swap
08AIX83Linux
09AIX bootable84OS/2 hidden C:
0AOS/2 Boot Manage85Linux extended
0BWin95 FAT3286NT volume set
0CWin95 FAT3287NT volume set
0BWin95 FAT1693Amoeba
0FWin95 Extended(>8GB)94Amoeba BBT
10OPUSA0IBM Thinkpad hidden
11Hidden FAT12A5BS0/386
12Compaq diagnost 116 Open BSD
16HiddenFAT16A7NextSTEP
14Hidden FAT16<32GBB7BSDI fs
17Hidden HPFS/NTFSB8BSDI swap
18AST Windows swapBESolaris boot
partition
1BHidden FAT32--
1CHidden FAT32 partition (using LBA-mode INT 13 extensions)C0DR-DOS/Novell DOS secured partition
--C1DRDOS/sec
1EHidden LBA VFAT partitionC4DRDOS/sec
24NEC DOSC6DRDOS/sec
3CPartition MagicC7Syrinx
40Venix 80286DBCP/M/CTOS
41PPC PreP BootE1DOS access
42SFSE3DOS R/0
40QNX4.xE4SpeedStor
4EQNX4.x 2nd partEBBeOS fs
4FQNX4.x 3rd partF1SpeedStor
50OnTrack DMF2DOS 3.3+ secondary partition
51OnTrack DM6 Aux--
52CP/MF4SpeedStor
53oNtRACK DM6 AuxFELAN step
54OnTrack DM6FFBBT
55EZ-Drive--
56Golden Bow--
5CPriam Edisk--
61Speed Stor--


  知道MBR的基本信息之后,我们再来看看实际的情况事怎样的.我将我电脑磁盘分了五个区,其中三个基本分区,三个扩展分区(其中一个分区格式化成FAT16)。

在这里插入图片描述


  用WinHex查看MBR的数据,可以看到446字节的主引导程序和64字节的分区表项+两字节的”55 AA”有效分区结束标志。

在这里插入图片描述

1. 第一个分区表项

在这里插入图片描述


  80 01 01 00 07 FE FF FF 3F 00 00 00 C1 FF BF 03

偏移(offset)描述
80活动分区
01 01 00磁头号1,扇区号1,柱面号0
07NTFS分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
3F 00 00 00分区起始扇区号63
C1 FF BF 03分区所占扇区数03BFFFC1(62914497),30G大小

2. 第二个分区表项

在这里插入图片描述


  00 FE FF FF 07 FE FF FF 00 00 C0 03 00 00 40 00

偏移(offset)描述
00非活动分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
07NTFS分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
00 00 C0 03分区起始扇区号62914560
00 00 40 00分区所占扇区数(4194304),2G大小

3. 第三个分区表项

在这里插入图片描述


  00 FE FF FF 07 FE FF FF 00 00 00 04 00 00 40 00

偏移(offset)描述
00非活动分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
07NTFS分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
00 00 00 04分区起始扇区号67108864
00 00 40 00分区所占扇区数(4194304),2G大小

4. 第四个分区表项

在这里插入图片描述


  00 FE FF FF 0F FE FF FF 00 00 40 04 00 00 C0 00

偏移(offset)描述
00非活动分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
0F扩展分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
00 00 40 04分区起始扇区号71303168
00 00 C0 00分区所占扇区数(12582912),6G大小

二、EBR结构分析


  通过分区表项,我们便可以得到分区所在的扇区范围,找到分区相应的扇区.但是如果是扩展分区,还需要EBR找到相应扩展分区所在的位置
  扩展分区中的每个逻辑驱动器都存在一个类似于MBR的扩展引导记录(Extended Boot Record)EBR,也有人称之为虚拟MBR或扩展MBR,意思是一样的。扩展引导记录包括一个扩展分区表和该扇区的标签。扩展引导记录将记录只包含扩展分区中每个 逻辑驱动器的第一个柱面的第一面的信息。一个逻辑驱动器中的引导扇区一般位于相对扇区32或63。
  但是,如果磁盘上没有扩展分区,那么就不会有扩展引导记 录和逻辑驱动器。
  第一个逻辑驱动器的扩展分区表中的第一项指向它自身的引导扇区。第二项指向下一个逻辑驱动器的EBR。如果不存在进一步的逻辑驱动器,第二项就不会使用,而且被记录成一系列零。如果有附加的逻辑驱动器,那么第二个逻辑驱动器的扩展分区表的第一项会指向它本身的引导扇区。第二个逻辑驱动器的扩展分区表的第二项指向下一个逻辑驱动器的EBR。扩展分区表的第三项和第四项永远都不会被使用。
  下图是扩展分区的一般结构。

在这里插入图片描述


  跟上面一样,我们通过WinHex查看EBR的数据,可以看到在01BE开始处有着扩展分区的分区表项。

在这里插入图片描述

1. 第一个分区表项

在这里插入图片描述


00 FE FF FF 07 FE FF FF 40 00 00 00 C0 07 40 00

偏移(offset)描述
00非活动分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
07NTFS分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
40 00 00 00分区起始扇区号64(相对扩展分区首扇区号),实际64+71303168=71303232
C0 07 40 00分区所占扇区数(4196288),2G大小

2. 第二个分区表项


00 FE FF FF 05 FE FF FF 00 08 40 00 00 08 40 00

偏移(offset)描述
00非活动分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
05扩展分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
00 08 40 00分区起始扇区号4196352(相对本逻辑驱动器首扇区号),实际4196352+71303168=75499520
00 08 40 00分区所占扇区数(4196352),2G大小


  可以看到这里是第二个逻辑驱动器的第一个扇区,有两个分区表项。

在这里插入图片描述

2.1 第一个分区表项

在这里插入图片描述


00 FE FF FF 07 FE FF FF 40 00 00 00 C0 07 40 00

偏移(offset)描述
00非活动分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
07NTFS分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
40 00 00 00分区起始扇区号64(相对本逻辑驱动器首扇区号),实际64+75499520=75499584
C0 07 40 00分区所占扇区数(4196288),2G大小

2.2 第二个分区表项


00 FE FF FF 05 FE FF FF 00 10 80 00 00 F0 3F 00

偏移(offset)描述
00非活动分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
05扩展分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
00 10 80 00分区起始扇区号8392704(相对扩展分区首扇区号),实际8392704+71303168=79695872
00 F0 3F 00分区所占扇区数(4190208),2G大小


  可以看到,因为是最后一个扩展驱动器,所以分区项只有一项。

在这里插入图片描述

2.3 第三个分区表项

在这里插入图片描述


00 FE FF FF 0E FE FF FF 40 00 00 00 C0 EF 3F 00

偏移(offset)描述
00非活动分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
0EFAT16分区
FE FF FF超过第16450560个扇区,设为最大值,可忽视
40 00 00 00分区起始扇区号64(相对本逻辑驱动器首扇区号),实际64+79695872=79695936
C0 EF 3F 00分区所占扇区数(4190144),2G大小


  通过实际的实验,我发现,在MBR里只有四个分区项,分别保存着三个基本分区和一个扩展分区的基本信息,在扩展分区里,又分为多个扩展逻辑分区,每个分区的扩展分区表里保存着本扩展分区的实际数据开始扇区信息和下一个扩展分区的基本信息,直到最后一个扩展分区
  下面附加一段C语言代码,可用于读取任一扇区信息。

/*******************************************************************************
程序员       : enjoy5512
最后修改时间 : 2016年3月22日 20:56:41
程序说明     : 本程序,用于读取给定输入的磁盘扇区的扇区号的512字节内容,并回显到命令行界面
测试环境     : Windows XP SP2 + VC6.0
待解决问题   : 磁盘第二个扇区的信息获取不到因为测试的原因,程序只有在输入超大的扇区号才会退出
*******************************************************************************/#include <windows.h>  
#include <stdio.h>
#include <stdlib.h>/*
函数说明:这个函数用于按照指定的格式输出一个扇区的512字节的信息,字符ascll码低于32或者大于127的都用'.'来代替,其余的输出指定的字符
输入参数:char buff[] : 字符串数组
输出参数:0*/
int ShowBuff(char buff[])
{int i = 0;int j = 0;system("cls");                            //清屏system("mode con cols=80 lines=37");      //设定命令行窗口大小printf("********************************************************************************");printf("********************************************************************************");printf("\t     0  1  2  3  4  5  6  7  8  9  A  B  C  D  E  F  |\n");for(i = 0;i < 32;i++)                     //每行输出16个字节,后面跟着相应的字符{printf("%#011X:",i*16);for(j = 0;j < 16;j++){//每四位单独输出,否则会被当作八位有符号整数输出,暂无解决办法printf(" %X%X",(buff[j+i*16]&240)>>4,buff[j+i*16]&15);                                               }printf(" | ");for(j = 0;j < 16;j++){if(buff[j+i*16] > 31 && buff[j+i*16] < 127){//输出相应的字符,若对应ascll码低于32或者大于127的都用'.'来代替printf("%c",buff[j+i*16]);     }else{printf(".");}}printf("\n");}return 0;
}/*
函数说明:这个函数用于按照给定的扇区号返回相应扇区512字节信息
输入参数:HANDLE hDisk : 磁盘句柄char buff[]  : 字符串数组int n        : 扇区号DWORD* DCount: 读取的字节数
输出参数:0*/
int GetBuff(HANDLE hDisk,char buff[] , int n,DWORD* DCount)
{LARGE_INTEGER li={0}; //记得初始化int h_sector = 0;     //32位系统,用于当超出32位寻址范围时保存高地址int l_sector = 0;     //32位系统,用于当超出32位寻址范围时保存低地址l_sector = n%8388607; //取低地址部分h_sector = n/8388607; //取高地址部分li.LowPart = l_sector * 512; //我32位系统int取四位,FFFFFFFF/200H = 8388607li.HighPart = h_sector;      //高位直接传过去//如果超出寻址范围则退出,否则将指针移动到相应扇区if(-1 == SetFilePointer(hDisk,li.LowPart,&li.HighPart,FILE_BEGIN)){MessageBox(NULL,"Error sector!The program will exit!!","Error",MB_OK);exit(-1);}//读取512字节内容到buffReadFile(hDisk,  //要读的文件句柄 buff,            //存放读出内容512,                 //要读的字节数DCount,        //实际读取字节数NULL);return 0;
}/*
函数说明:这个函数用于按照给定的磁盘号,返回相应的句柄
输入参数:HANDLE hDisk : 磁盘句柄int nDisk    : 磁盘号号
输出参数:0*/
int GetHandle(HANDLE* hDisk,int nDisk)
{char szDriverBuffer[128];     //保存相应磁盘路径信息memset(szDriverBuffer,0,128); //格式化设备文件名称sprintf(szDriverBuffer,"\\\\.\\PhysicalDrive%d",nDisk);//获取设备名称*hDisk = CreateFileA(szDriverBuffer,               // 设备名称,这里指第一块硬盘,多个硬盘的自己修改就好了GENERIC_READ,                 // 指定读访问方式FILE_SHARE_READ | FILE_SHARE_WRITE,  // 共享模式为读 | 写,只读的话我总是出错..NULL,                         // NULL表示该句柄不能被子程序继承OPEN_EXISTING,                // 打开已经存在的文件,文件不存在则函数调用失败NULL,                         // 指定文件属性NULL); if (*hDisk==INVALID_HANDLE_VALUE)    //如果获取句柄失败,则退出{MessageBox(NULL,"Error open disk!","Error",MB_OK);exit(-1);}return 0;
}/*
函数说明:主函数
输入参数:0
输出参数:0*/
int main(void)
{char buff[513] = {0};  //保存512字节信息,因为字符串数组后面会加\0,所以申请513字节HANDLE hDisk;          //磁盘句柄DWORD DCount=0;        //返回读取的字节数int sector = 0;        //扇区号int nDisk = 0;         //磁盘号printf("请输入需要读取的磁盘号:");   //获取磁盘号scanf("%d",&nDisk);ShowBuff(buff);                      //初始化界面GetHandle(&hDisk,nDisk);             //获取磁盘句柄for(;;)                              //循环显示扇区信息{printf("please input the number of sector : ");  //获取扇区号scanf("%d",&sector);GetBuff(hDisk,buff,sector,&DCount);              //获取扇区信息ShowBuff(buff);                                  //显示扇区信息}CloseHandle(hDisk);                  //释放句柄return 0;
}

NTFS文件系统详解系列

  • NTFS文件系统详解
  • NTFS文件系统详解(一)硬盘基本信息
  • NTFS文件系统详解(二)MBR\EBR基本信息
  • NTFS文件系统详解(三)NTFS元文件解析#)

http://chatgpt.dhexx.cn/article/8D5OXVJD.shtml

相关文章

MBR、EBR与DBR详解

demo&#xff1a;https://github.com/Hilaver/NtfsResolution/ 先看一张硬盘图片&#xff08;一个盘面&#xff09;&#xff1a; MBR 主引导记录&#xff08;MBR&#xff0c;Main Boot Record&#xff09;是位于磁盘最前边的一段引导&#xff08;Loader&#xff09;代码。它负…

MBR、EBR、DBR是什么

通常情况下可以这样描述这几个概念&#xff1a; MBR&#xff1a;为计算机启动后从可启动介质上首先装入内存并且执行的代码&#xff0c;通常用来解释分区结构。以硬盘为例&#xff0c;通常为LBA的0扇区。 EBR:自MICROSOFT推出扩展分区的概念后&#xff0c;扩展分区就沿用了基本…

NTFS文件系统详解(二)之MBR(EBR)基本信息

经过上一篇文章对硬盘的基本结构的详细介绍后,我们再来看看MBR和EBR的结构 MBR(Main Boot Record 主引导记录区)位于整个硬盘的0磁道0柱面1扇区。不过,在总共512字节的主引导扇区中,MBR只占用了其中的446个字节,另外的64个字节交给了 DPT(Disk Partition Table硬盘分区表…

EBR内容解析

原先博客放弃使用,几篇文章搬运过来 EBR(Extended Boot Record)即扩展分区引导记录.类似于主引导记录MBR.因为MBR的四条分区信息的限制,可以使用EBR方便扩展. 它的结构与MBR类似于,但是没有引导程序和磁盘签名,仅仅保留了分区表和结束标志. 下面以实例分析: 新建一个512MB虚拟…

C++中常见的几种输入字符串的方法

C中常见的几种输入字符串的方法_是赵薰薰呀的博客-CSDN博客_c输入字符串 可以用的函数 substr substr有2种用法&#xff1a; 假设&#xff1a;string s "0123456789"; string sub1 s.substr(5); //只有一个数字5表示从下标为5开始一直到结尾&#xff1a;sub1 &…

MATLAB-字符串

本节我们学习如何在MATLAB中创建一个字符串。 例如&#xff1a; my_string 搬砖工程师domi MATLAB执行上述语句&#xff0c;返回以下结果&#xff1a; my_string 搬砖工程师domi MATLAB 认为所有变量&#xff0c;数组和字符串被视为字符数组。 让我们使用命令检查上面创建…

c语言:动态输入字符串数组

考研题中经常会遇到给定一条英文句子&#xff0c;让我们对其每个单词进行操作&#xff0c;不同于对整型数据操作&#xff0c;字符串在c语言中必须存放到数组中&#xff0c;无形之中又给考题增加了难度 一般情况下&#xff0c;字符串数组都以二维数组或指针数组定义&#xff0c…

用scanf_s输入字符串

用的是vs2019。 为了安全&#xff0c;scanf_s输入字符串时需要加一个数字来限制最多读取的字符数。 错误示例&#xff1a; char* str[100]; scanf_s("%s",str);这样写编译器没有报错&#xff0c;但是运行会出错&#xff0c;debug显示 正确写法&#xff1a; char…

写一函数,求一个字符串的长度。在main函数中输入字符串,并输出其长度。

写一函数,求一个字符串的长度。在main函数中输入字符串,并输出其长度。 解题思路&#xff1a; 字符串以\0作为结尾&#xff0c;则从第一个字符开始向后移动遇到\0认为字符串结束。 答案&#xff1a; #include <stdio.h> int mystrlen(char *str) {int len 0;char *pt…

C++输入字符串的几种方式

最近有复习到 C 基础知识&#xff0c;这里总结下在 C 中输入字符串的几种方式&#xff0c;有需要的可以参考&#xff1a; 1、cin >> <1> 输入一个数字 void Cin1() {int a, b;cin >> a >> b;cout << "a b " << a b <&l…

c语言如何用循环输入字符串,C语言使用scanf连续输入字符串出现的问题

#include int main() { char a,b; scanf("%c",&a)&#xff1b; scanf("%c",&b); printf("%c,%c",a,b); } 一段看上去很简单的代码&#xff0c;即使是C语言初学者也能看懂。 可是在运行的时候却出现了问题&#xff1a; ①输入a后按下回车…

c语言用scanf输入指针字符串,c语言如何用scanf输入字符串

在C语言中&#xff0c;可以使用“scanf(“格式控制字符串”,变量内存地址)”语句输入字符串。scanf()函数的第一个参数是格式字符串&#xff0c;它指定了输入的格式&#xff0c;并按照格式说明符解析输入对应位置的信息并存储于可变参数列表中对应的指针所指位置。 本教程操作环…

C++中输入字符串的几种方法

C中输入字符串的几种方法 C的字符串输入方式有 1.cin>> 2.cin.get() 3.cin.getline() 4.getline() 5.gets()cin 第一种用法是逐个输入数字或者字符&#xff0c;中间用空格隔开。 第二种用法是读入一个字符串&#xff0c;遇到“空格”“TAB”“回车” 都会结束。例如&a…

C语言输入字符串

文章目录 1.scanf2.gets 1.scanf 输入字符串使用 %s。不能读取空格&#xff0c;遇到空格自动结束。例如&#xff0c;输入abcde&#xff0c;得到abcde&#xff1b;而输入abc de&#xff0c;只能得到abc。输入格式&#xff1a;scanf("%s",str)。其中str表示 地址值。 …

【C语言】字符串输入的三种方式

一、gets() 该函数原型为&#xff1a; # include <stdio.h> char *gets(char *str);gets() 函数的功能是从输入缓冲区中读取一个字符串存储到字符指针变量 str 所指向的内存空间 # include <stdio.h> int main(void) {char a[256] {0};gets(a);printf("%s…

字符串输入的2种常用方法详解

字符串输入的2种常用方法详解 一.scanf函数1.扫描集2.逆向使用扫描集 二.gets函数 一.scanf函数 int main() {char arr[50] { 0 };scanf("%s", arr);printf("%s", arr);return 0; }这种方法可以输入字符串&#xff0c;但是只能输入一个词&#xff0c;遇到…

字符串的输入

字符串数据作为经常需要使用的数据&#xff0c;我们需要掌握它的输入&#xff0c;下面总结了不同的输入字符串的方式 c和c的有一点点不同 ①逐个逐个字符输入 我们可以设置一个数组&#xff0c;或者数组加指针的形式&#xff0c; 通过for循环或while循环的方式&#xff0c;…

C++字符串的几种输入方法(string和字符数组)

ps:本文大部分内容参考于这篇博客&#xff0c;在这里加入了自己对这些函数在字符数组和string变量的理解和总结。 C中的输入大致有6种方法&#xff1a;cin&#xff0c;cin.get(),cin.getline(),gets(),getchar() 1,cin 用法一&#xff1a;最常用的方法&#xff0c;接收一个字…

制作黑苹果安装U盘(Clover+OC+PE)三引导

制作黑苹果安装U盘&#xff08;OCCloverPE&#xff09;三引导 1.删除U盘分区表2.写入dmg到U盘3.替换EFI(Clover引导)4.扩建U盘分区表5.转换U盘分区类型 效果如图所示&#xff0c;嫌麻烦可以关注 黑果小兵的部落阁微信公众号&#xff0c;在“安装镜像”里找"10.15.6 19G73…

MINIS FORUM U820 黑苹果引导OC或CLOVER更新教程

准备工具&#xff1a;1.OC Auxiliary Tools&#xff08;简称OCAT)&#xff0c;OC Auxiliary Tools for Mac-OC Auxiliary Tools Mac版下载 V0.6.9-PC6苹果网 即新OC引导的编辑器&#xff0c;须与OC引导版本相对应&#xff0c;否则容易出错&#xff1b;2.Clover ConfiGurator&a…