我有一个磁盘镜像,其中包含使用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);
发布于 2009-12-08 09:59:44
是的,我认为你最好把它读成一个结构。包含有用数据的字段都是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;
您可以在调用read_superblock
时将结构转换为char*
,如下所示:
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-08 12:06:03
这里有两件事要补充:
#pragma pack(0)
,在结构定义之后使用#pragma pack()
。ntohs()
和ntohl()
。请注意,这些字节顺序从网络字节顺序交换到主机字节顺序;如果它们是相同的(在基于x86的平台上不是这样),则它们什么也不做。您使用htons()
和htonl()
从一个主机转到另一个网络字节顺序。但是,由于这些数据来自您的文件系统,而不是网络,我不知道字节顺序是否是一个问题。通过比较您的期望值(例如,块大小)和您获得的十六进制值,应该很容易计算出来。发布于 2009-12-08 10:11:08
在成功地将数据复制到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);
https://stackoverflow.com/questions/1864103
复制相似问题