delphidelphi-xe4delphi-xe5

incorrect structure size when Variant Parts used in records


I have a DLL project, that provides a type definition with a Delphi "variant part" as follows:

type TValue = record 
case Kind : cardinal of 
valueShortCard : ( ValShortCard : byte ); 
valueLongReal : ( ValLongReal : double ); 
end; 

I expect that this structure would have a size of 12 bytes (4B for cardinal plus 8B for double, as this is the bigger of those two types).

But if I have an application and call the function sizeof() for the structure in the DLL, it states the size is 16 bytes.

Also, if I declare the same structure directly in the .exe application project, the sizeof() returns a correct size of 12 bytes.

If I run the application and try to get a ValLongReal value from the compiled DLL, the returned value is incorrect and it seems it is shifted by 4 bytes in the memory.

I.E. if the "double" variable in the DLL bares the value of "40 45 9a e1 47 ae 14 7b" (interpreted in byte values) , the value returned by the application after reading it from the DLL is "47 ae 14 7b 77 07 06 7d". That means there is a 4 byte overlap of correct data and the latter 4 bytes are incorrect and is read from the following space in memory.

Note: This behaviour has been observed already in Delphi XE4, we have just upgraded to XE5, but it behaves the same way. The used OS is Win7 32-bit.

thanks for your suggestions


Solution

  • Define it as such:

    type TValue = packed record 
      case Kind : cardinal of 
      valueShortCard : ( ValShortCard : byte ); 
      valueLongReal : ( ValLongReal : double ); 
      end; 
    

    Here, sizeof(TValue)=12, as you expected.

    Adding packed record force no alignment to be generated. If there is no alignment, double will be aligned on 8 bytes boundaries, so Kind will use 8 bytes instead of 4, and valueLongReal also, therefore, sizeof(TValue)=8+8=16 in your case.

    Note that under AMD/Intel, you should better align your double variables for performance reasons. On ARM, alignment is required.

    Note also that variable records alignment changed with Delphi XE3 AFAIR in respect to previous versions.