[PATCH] Add a default constructor to work around Visual Studio compiler bug

Yung, Douglas douglas_yung at playstation.sony.com
Wed Jan 28 09:46:44 PST 2015

Hi, I would like to submit the following patch to include/llvm/CodeGen/TargetLoweringObjectFileImpl.h which works around a bug in the Visual Studio compiler.
The bug is that the compiler does not initialize a field in the class when it creates a default constructor when it should be zero-initialized. The result is that when the field is later accessed, the value is whatever value happens to be stored there, instead of the proper zero value (false in this case). The explanation of the issue is the following:

Quoting The C++11 standard section [class.base.init]p8 that "In a non-delegating constructor, if a given non-static data member or base class is not designated by a mem-initializer-id (including the case where there is no mem-initializer-list because the constructor has no ctor-initializer) and the entity is not a virtual base class of an abstract class (10.4), then

- if the entity is a non-static data member that has a brace-or-equal-initializer, the entity is initialized as specified in 8.5;
- otherwise, if the entity is a variant member (9.5), no initialization is performed;
- otherwise, the entity is default-initialized "

The lack of default-initialization is what is causing this problem if we use the Visual Studio compiler:
At CodeGen stage, the implicit default constructor for class TargetLoweringObjectFileELF is called. Class 'TargetLoweringObjectFileELF' has a field called 'UseInitArray'. If that field is set to true, then the compiler will emit global constructors/destructors in the .init_array section. Otherwise, it will emit those symbols in the .ctors sections.

The lack of a user defined default constructor for class TargetLoweringObjectFileELF means that the compiler will have to implicitly generate a default constructor with no initializer and empty body. Then, still according to the standard ( [class.base.init]p8 ) it should default-initialize field 'UseInitArray'.

The Visual Studio compiler does not perform the field initialization. Hence the intermittent problem (depending on the garbage value read from memory we either think we shall emit those symbols on the .ctors section or .init_array section).

To illustrate this issue, I have attached a sample file mytest2.cpp, and the assembly generated for it by the MSVC18 (Visual Studio 2013). The source defines 3 different classes A, B and C. All are identical except for the following characteristics:

*         Class A does not define a default constructor and uses one generated automatically by the compiler

*         Class B defines an empty default constructor

*         Class C defines an empty default constructor which initializes the Boolean field to false

If you examine the assembly generated by the compiler, you can see that the constructors generated for A and B area identical, and C is similar, but it contains an initialization of the Boolean field to false which A and B are both missing.

This patch works around the issue by adding a default constructor to the class which explicitly initializes the field to false (the zero value).

The patch also includes an LLVM test which checks that a .init_array is not generated for a non-linux ELF target.

Douglas Yung
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.llvm.org/pipermail/llvm-commits/attachments/20150128/c61c6120/attachment.html>
-------------- next part --------------
A non-text attachment was scrubbed...
Name: 17307b.patch
Type: application/octet-stream
Size: 2141 bytes
Desc: 17307b.patch
URL: <http://lists.llvm.org/pipermail/llvm-commits/attachments/20150128/c61c6120/attachment.obj>
-------------- next part --------------
A non-text attachment was scrubbed...
Name: mytest2.zip
Type: application/x-zip-compressed
Size: 2820 bytes
Desc: mytest2.zip
URL: <http://lists.llvm.org/pipermail/llvm-commits/attachments/20150128/c61c6120/attachment.bin>

More information about the llvm-commits mailing list