1

我有一个磁盘映像,其中包含使用 fuse 的标准映像。超级块包含以下内容,我有一个函数 read_superblock(*buf) 返回以下原始数据:

Bytes 0-3: Magic Number (0xC0000112)
      4-7: Block Size (1024)
     8-11: Total file system size (in blocks)
    12-15: FAT length (in blocks)
    16-19: Root Directory (block number)
  20-1023: NOT USED

我对 C 语言非常陌生,为了让我开始这个项目,我很好奇有什么简单的方法可以将它读入结构或一些变量,然后使用 printf 将它们打印到屏幕上进行调试。

我最初正在考虑做类似以下的事情,我认为我可以看到原始数据,但我认为情况并非如此。也没有结构,我试图将其作为字符串读取,这似乎也非常错误。让我从中获取数据。有没有办法让我指定结构并定义每个变量中的字节数?

char *buf;
read_superblock(*buf);
printf("%s", buf);
4

3 回答 3

6

是的,我认为你最好把它读成一个结构。包含有用数据的字段都是 32 位整数,因此您可以定义一个如下所示的结构(使用标准头文件中定义的类型stdint.h):

typedef struct SuperBlock_Struct {
  uint32_t magic_number;
  uint32_t block_size;
  uint32_t fs_size;
  uint32_t fat_length;
  uint32_t root_dir;
} SuperBlock_t;

您可以char*在调用时将结构转换为 a read_superblock,如下所示:

SuperBlock_t sb;
read_superblock((char*) &sb);

现在要打印出您的数据,您可以进行如下调用:

printf("%d %d %d %d\n",
  sb.magic_number,
  sb.block_size,
  sb.fs_size,
  sb.fat_length,
  sb.root_dir);

请注意,在使用这样的技术时,您需要注意平台的字节顺序,因为您正在读取整数数据(即,您可能需要在读取数据时交换字节)。您应该能够使用第一个字段中的幻数快速确定。

请注意,通常最好在不强制转换的情况下传递这样的结构;这使您可以利用编译器的类型检查并消除强制转换可能隐藏的潜在问题。但是,这将需要更改您的实现read_superblock以将数据直接读取到结构中。这并不难,可以使用标准的 C 运行时函数来完成fread(假设您的数据在文件中,如您的问题中所暗示的那样),如下所示:

fread(&sb.magic_number, sizeof(sb.magic_number), 1, fp);
fread(&sb.block_size, sizeof(sb.block_size), 1, fp);
...
于 2009-12-08T01:59:44.263 回答
1

这里要补充两点:

  1. 将原始数据拉入结构时,将结构设置为零填充是一个好主意,即使它完全由 32 位无符号整数组成。在 gcc 中,您可以#pragma pack(0)在结构定义之前和#pragma pack()之后执行此操作。
  2. 为了处理潜在的字节顺序问题,两个调用是ntohs()ntohl(),分别用于 16 位和 32 位值。请注意,这些从网络字节顺序交换到主机字节顺序;如果它们相同(它们不在基于 x86 的平台上),它们什么也不做。htons()您使用和从主机到网络字节顺序htonl()。但是,由于这些数据来自您的文件系统而不是网络,我不知道字节序是否是一个问题。通过将您期望的值(例如块大小)与您获得的值(以十六进制进行比较)来计算应该很容易。
于 2009-12-08T04:06:03.277 回答
0

将数据成功复制到 Emerick 提出的结构后,打印数据并不难。假设您用来保存数据的结构的实例名为 SuperBlock_t_Instance。

然后你可以像这样打印它的字段:

printf("Magic Number:\t%u\nBlock Size:\t%u\n etc", 
SuperBlock_t_Instance.magic_number, 
SuperBlock_t_Instance.block_size);
于 2009-12-08T02:11:08.350 回答