| //===- llvm/Support/IncludeFile.h - Ensure Linking Of Library ---*- C++ -*-===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file defines the FORCE_DEFINING_FILE_TO_BE_LINKED and DEFINE_FILE_FOR |
| // macros. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #ifndef LLVM_SYSTEM_INCLUDEFILE_H |
| #define LLVM_SYSTEM_INCLUDEFILE_H |
| |
| /// This macro is the public interface that IncludeFile.h exports. This gives |
| /// us the option to implement the "link the definition" capability in any |
| /// manner that we choose. All header files that depend on a specific .cpp |
| /// file being linked at run time should use this macro instead of the |
| /// IncludeFile class directly. |
| /// |
| /// For example, foo.h would use:<br/> |
| /// <tt>FORCE_DEFINING_FILE_TO_BE_LINKED(foo)</tt><br/> |
| /// |
| /// And, foo.cp would use:<br/> |
| /// <tt>DEFINING_FILE_FOR(foo)</tt><br/> |
| #ifdef __GNUC__ |
| // If the `used' attribute is available, use it to create a variable |
| // with an initializer that will force the linking of the defining file. |
| #define FORCE_DEFINING_FILE_TO_BE_LINKED(name) \ |
| namespace llvm { \ |
| extern const char name ## LinkVar; \ |
| __attribute__((used)) static const char *const name ## LinkObj = \ |
| &name ## LinkVar; \ |
| } |
| #else |
| // Otherwise use a constructor call. |
| #define FORCE_DEFINING_FILE_TO_BE_LINKED(name) \ |
| namespace llvm { \ |
| extern const char name ## LinkVar; \ |
| static const IncludeFile name ## LinkObj ( &name ## LinkVar ); \ |
| } |
| #endif |
| |
| /// This macro is the counterpart to FORCE_DEFINING_FILE_TO_BE_LINKED. It should |
| /// be used in a .cpp file to define the name referenced in a header file that |
| /// will cause linkage of the .cpp file. It should only be used at extern level. |
| #define DEFINING_FILE_FOR(name) \ |
| namespace llvm { const char name ## LinkVar = 0; } |
| |
| namespace llvm { |
| |
| /// This class is used in the implementation of FORCE_DEFINING_FILE_TO_BE_LINKED |
| /// macro to make sure that the implementation of a header file is included |
| /// into a tool that uses the header. This is solely |
| /// to overcome problems linking .a files and not getting the implementation |
| /// of compilation units we need. This is commonly an issue with the various |
| /// Passes but also occurs elsewhere in LLVM. We like to use .a files because |
| /// they link faster and provide the smallest executables. However, sometimes |
| /// those executables are too small, if the program doesn't reference something |
| /// that might be needed, especially by a loaded share object. This little class |
| /// helps to resolve that problem. The basic strategy is to use this class in |
| /// a header file and pass the address of a variable to the constructor. If the |
| /// variable is defined in the header file's corresponding .cpp file then all |
| /// tools/libraries that \#include the header file will require the .cpp as |
| /// well. |
| /// For example:<br/> |
| /// <tt>extern int LinkMyCodeStub;</tt><br/> |
| /// <tt>static IncludeFile LinkMyModule(&LinkMyCodeStub);</tt><br/> |
| /// @brief Class to ensure linking of corresponding object file. |
| struct IncludeFile { |
| explicit IncludeFile(const void *); |
| }; |
| |
| } |
| |
| #endif |