TGA 文件 Header 在 Windows 上不正确

TGA File Header is not correct on Windows

我正在尝试读取我在 Paint.net 中创建的文件的 TGA header。好像哪里不对劲。如果我使用规范中的 header 结构,如下所示:

typedef struct {
    CHAR  idlength;
    CHAR  colourmaptype;
    CHAR  datatypecode;
    WORD colourmaporigin;
    WORD colourmaplength;
    CHAR  colourmapdepth;
    WORD x_origin;
    WORD y_origin;
    WORD width;
    WORD height;
    CHAR  bitsperpixel;
    CHAR  imagedescriptor;
} TGAHEADER;

我明白了:

Data size: 0
Color Map type: 0
Data Type code: 2
Bits per-pixel: 0
Size: 501 x 2080

这是错误的,因为我的图像是 501x501,每像素 32 位。但是,如果我从结构中注释掉两个字节,f.e。这个 colourmaporigin,我明白了:

Data size: 0
Color Map type: 0
Data Type code: 2
Bits per-pixel: 32
Size: 501 x 501

这是正确的。我正在阅读我在这种文件格式上找到的所有内容。它从来没有说这些字段中的任何一个都是可选的或其他东西。

我怎么会得到这样的结果?

读取数据的代码如下:

void Image::readTGA()
{
    TGAHEADER fileHeader;

    std::ifstream fileHandle(fileName, std::ios::binary);
    if (fileHandle.is_open())
    {
        fileHandle.read((char*)(&fileHeader), sizeof(TGAHEADER));
        fileHandle.close();
    }
    else
    {
        std::cout << "An error occured when opening a file." << std::endl;
    }
}

我正在使用 VS2015,面向 x86 平台。

这是一个填充问题。使用 Visual Studio,您可以使用 #pragma pack(1) 编译器指令禁用结构的任何填充。

演示

#include<stdio.h>
#include<windows.h>

// Default packing of structure with padding

typedef struct {
  CHAR  idlength;
  CHAR  colourmaptype;
  CHAR  datatypecode;
  WORD colourmaporigin;
  WORD colourmaplength;
  CHAR  colourmapdepth;
  WORD x_origin;
  WORD y_origin;
  WORD width;
  WORD height;
  CHAR  bitsperpixel;
  CHAR  imagedescriptor;
} TGAHEADER;


#pragma pack(1) // structure fields are aligned to byte boundary (no padding)

typedef struct {
  CHAR  idlength;
  CHAR  colourmaptype;
  CHAR  datatypecode;
  WORD colourmaporigin;
  WORD colourmaplength;
  CHAR  colourmapdepth;
  WORD x_origin;
  WORD y_origin;
  WORD width;
  WORD height;
  CHAR  bitsperpixel;
  CHAR  imagedescriptor;
} TGAHEADER_PACKED;

int main()
{
  printf("Offset of field bitsperpixel in TGAHEADER structure %d\n", offsetof(TGAHEADER, bitsperpixel));
  printf("Offset of field bitsperpixel in packed TGAHEADER structure %d\n", offsetof(TGAHEADER_PACKED, bitsperpixel));
}

输出:

Offset of field bitsperpixel in TGAHEADER structure 18
Offset of field bitsperpixel in packed TGAHEADER structure 16