最近排查一个USB相关的故障,由于信息安全就不多说工作上的事情了,顺路学习了MBR的相关知识,在网上找了一些资料,现在把学习心得写下来,抛砖引玉。感谢无数前辈的分享精神。

我的U盘插入linux后被识别成了sdb4,我当时很纳闷,为什么是4,没有sdb1 sdb2 sdb3,直接就sdb4 了。linux是从哪里显示的这个4.

是这样的,sdb,这个b是有linux 动态分配的,Linux那些事儿,我是SCSI硬盘中有精彩的讲解。在Kernel代码中时sd_probe函数做的事情。linux对于scsi设备,支持sda~sdzzz,因为英文字母有26个,所以支持这么多SCSI设备

    #define SD_MAX_DISKS (((26 * 26) 26 1) * 26)
因为我的板子上,有自带的sda SCSI盘,所以我插入的U盘被识别成了sdb。为啥是sdb,LINUX是怎样防止不同的SCSI赋予同一个disk_name 呢,这就是IDR做的事情。IDR integer ID management,是管理小整数分配的。不太了解的筒子请摸我and 再摸我

剩下的是,linux 为什么把我的U盘识别成了sdb4。这个分区的信息记录在了U盘的mbr,这就引出了我们本文的主角 MBR。

MBR,Master Boot Record的缩写,可以成为主引导记录或者主引导扇区。计算机开机之后,访问磁盘必须先访问这个MBR,获取到这个磁盘的相关信息,比如这个磁盘有几个分区啊,每个分区从哪开始,到哪结束,每个分区都是啥文件系统等等信息。

MBR是一个扇区,在磁盘的位置是(柱面,磁头,扇区)=(0,0,1)。MBR是怎么组织的呢?一个扇区512个字节(不一定,apple有文章表示,扇区大小不一定是512)。


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


其中扇区最后两个字节是0x55 0xaa,这也可以验证是否是标准的MBR。下面代码中有。 对了,有个工具winhex,可以看磁盘的信息,我今天在我的Window 7上看了,发现MBR信息完全不对,按照维基百科提到,分区激活状态时0x80,非激活状态时0x00,结果在winhex上显示的是0xDE,搞得哥很难受。也许是我没学会正确的使用方法。感兴趣的可以搜搜这个工具。其实我们完全可以自己解析MBR的格式。
我将我的U盘的第一个扇区的内容保存了下来,写了个程序分析这个扇区(MBR)的内容。本程序主要参考了网上的资源,风格不太一致,大家也可以看得出来。上面的表格是wiki百科的内容,光荣属于前辈,我只是整理了下自己的学习心得。

    #include
    #include
    #include
    #include
    #include

    #define PT_OFFSET 446
    #define PT_LEN 16
    #define PT_N 4
    #define CHECK_POS 510
    //Cylinder-Head-Sectoradderess type
    typedef struct
    {
    unsigned char head:8;
    unsigned char sector:6;
    unsigned short cylinder:10;
    } CHS;

    void printPT(char *buf);

    int main(int argc, char *argv[])
    {
    int fd = open(argv[1], O_RDONLY);

    if(fd < 0)
    {
    perror("open");
    exit(1);
    }

    if(lseek(fd,CHECK_POS,SEEK_SET) == -1)
    {
    fprintf(stderr,"lseek to CHECK_POS err %m\n");
    close(fd);
    return -1;
    }
    unsigned char checkbuf[16] = {0};
    if(read(fd,checkbuf,2) != 2)
    {
    fprintf(stderr,"read check info failed %m\n");
    close(fd);
    return -2;
    }

    if(checkbuf[0] != 0x55 || checkbuf[1] != 0xaa)
    {
    fprintf(stderr, "not valid mbr format\n");
    close(fd);
    return -3;
    }
    if(lseek(fd, PT_OFFSET, SEEK_SET) ==-1)
    {
    close(fd);
    perror("lseek");
    exit(1);
    }

    char buf[PT_LEN];
    int i;

    for(i = 0; i < PT_N; i )
    {
    bzero(buf, PT_LEN);
    if(read(fd, buf, PT_LEN) !=PT_LEN)
    {
    printf("can't getfull partition table[%d]\n", i);
    close(fd);
    exit(1);
    }

    if(buf[1] || buf[2] || buf[3])
    printPT(buf);
    }

    close(fd);
    return 0;
    }

    void printPT(char *buf)
    {
    switch((unsigned char)buf[0])
    {
    case 0x80:
    printf("bootable\n");
    break;
    case 0x00:
    printf("non-bootable\n");
    break;
    default:
    printf("invalid\n");
    }

    CHS chs;
    memcpy(chs, buf 1, 3);
    printf("from CHSAddr: \n");
    printf("sss\n","head","sector","cylinder");
    printf("ddd\n",chs.head,chs.sector,chs.cylinder);
    memcpy(chs, buf 5, 3);

    printf("to CHSAddr: \n");
    printf("sss\n","head","sector","cylinder");
    printf("ddd\n",chs.head,chs.sector,chs.cylinder);

    printf("partition type:%d\n",(unsigned char)buf[4]);
    printf("LENGTH:%d(sectors)\n\n", *(unsigned int*)buf[12]);
    }

    dd if=/dev/sdb of=mbr bs=512 count=1
首先看一下我的U盘的MBR的庐山真面目:

    [root@localhost mbr]# cat mbr |od -tx1 -Ax000000 fa 31 c0 8e d8 8e c0 8e d0 bc 00 7c fb fc 89 e6000010 bf 00 06 b9 00 01 f3 a5 ea dc 06 00 00 10 00 01000020 00 00 7c 00 00 00 00 00 00 00 00 00 00 80 3f 00000030 ff 00 cd 03 1e 0e 1f 3a 16 10 00 74 06 1f ea 36000040 e7 00 f0 3d fb 54 75 05 8c d8 fb eb 1d 80 fc 08000050 75 1b e8 81 00 8a 36 13 00 fe ce 8b 0e 15 00 86000060 cd c0 e1 06 0a 0e 11 00 31 c0 f8 eb 65 80 fc 02000070 72 cb 80 fc 04 77 c6 60 80 cc 40 50 be 00 00 c7000080 04 10 00 30 e4 89 44 02 89 5c 04 8c 44 06 66 31000090 c0 66 89 44 0c 88 f0 f6 26 11 00 88 cf 88 eb c00000a0 ef 06 81 e1 3f 00 01 c8 48 89 c7 a1 13 00 f7 260000b0 11 00 f7 e3 01 f8 81 d2 00 00 89 44 08 89 54 0a0000c0 58 30 c0 8a 16 10 00 e8 0c 00 88 26 03 00 61 a10000d0 02 00 1f ca 02 00 9c ff 1e 22 00 c3 80 fa 8f 7f0000e0 04 88 16 2d 06 be 87 07 e8 8d 00 be be 07 31 c00000f0 b9 04 00 f6 04 80 74 03 40 89 f5 81 c6 10 00 e2000100 f2 48 74 02 cd 18 bf 05 00 be 1d 06 c7 44 02 01000110 00 66 8b 46 08 66 89 44 08 b8 00 42 8a 16 2d 06000120 cd 13 73 0d 4f 74 49 30 e4 8a 16 2d 06 cd 13 eb000130 d8 a1 fe 7d 3d 55 aa 75 37 fa 66 a1 4c 00 66 a3000140 3f 06 be 13 04 8b 04 48 89 04 c1 e0 06 8e c0 31000150 ff be 1d 06 b9 60 00 fc f3 a5 c7 06 4c 00 17 00000160 a3 4e 00 fb 8a 16 2d 06 89 ee fa ea 00 7c 00 00000170 be aa 07 e8 02 00 eb fe ac 20 c0 74 09 b4 0e bb000180 07 00 cd 10 eb f2 c3 53 74 61 72 74 20 62 6f 6f000190 74 69 6e 67 20 66 72 6f 6d 20 55 53 42 20 64 650001a0 76 69 63 65 2e 2e 2e 0d 0a 00 42 6f 6f 74 20 660001b0 61 69 6c 65 64 00 00 00 ea eb d4 ca 00 00 00 000001c0 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00*0001e0 00 00 00 00 00 00 00 00 00 00 00 00 00 00 80 010001f0 01 00 0b fe ff cc 3f 00 00 00 81 b9 ee 00 55 aa
用写的程序可以分析出蓝色部分,分区的结果。可以看到各分区的filetype,起始位置,结束位置等信息。

执行结果为:



参考文献:
1 维基百科
2 Linux那些事儿3 MBR分区表实验
光荣属于前辈。