PSCF v1.2
|
Coding Standards (Prev) Parameter File I/O (Next)
The PSCF source code makes rather heavy use of C++ class templates to avoid duplication of code for closely analogous classes. For purposes of discussion, it is useful to divide class templates into two categories that are treated somewhat differently:
Class templates in which the only template parameter is the dimension of space are an important special case because they appears throughout the code for simulating systems periodic boundary conditions that is used in the pscf_pc and pscf_pg programs.
We discuss these two categories separately below because they are treated differently by the PSCF build system: For class templates for which the only template parameter is the spatial dimension D, the build system explicitly instantiates classes for the three physically relevant values of D=1, D=2 and D=3, and later links to these pre-compiled classes. For all other types of class template, the build system instead generally relies on implicit template instantiation.
In the source code for the pscf_pc and pscf_pg programs for periodic structures, many classes are defined by instantiating class templates in which the dimension of space (denoted by D) is the only template parameter. This pattern is used for most of the classes defined in the prdc/, rpc/ and rpg/ namespace level directories. In this usage, the only relevant values of D are D=1, D=2, and D=3, which are used for periodic structures in which fields are periodic in 1, 2 or 3 spatial dimensions. For example, a lamellar phase would be simulated using classes defined with D=1, while a hexagonal phase would be simulated using classes in which D=2.
The PSCF build system is designed to explicitly instantiate classes with D=1, 2 and 3 for class templates in which D is the only template parameter. Source code for this type of class template is almost always divided among three files with the same base name but different file extensions, which is organized as follows:
The build system compiles the *.cpp or *.cu file for each such class template, and thereby instantiates classes with D=1, 2 and 3.
For example, the file src/prdc/crystal/Basis.h contains a definition of a class template for a class template Pscf::Prdc::Basis<D>, in which D is the dimension of space. This is defined (in skeleton form) as
The file src/prdc/crystal/Basis.tpp, which is longer than the header file, contains the actual definitions of all non-inline member functions for the Basis class. The file src/prdc/crystal/Basis.cpp contains a several explicit instantiation statements, of the form
The build system compiles src/prdc/crystal/Basis.cpp and then adds the resulting object code for these three class template instantations into a static library.
As an essential part of this system, the header file for each such class template also contains a block of "extern" declarations for instantiations of the template the template with D=1, D=2, and D=3.
The relevant block for the Pscf::Prdc::Basis template in file src/prdc/crystal/Basis.h looks like this:
These "extern" declarations guarantee that files other than Basis.tpp that directly or indirectly include the header Basis.h will not attempt to implicitly instantiate the template as needed, but will instead assume that they are instantiated elsewhere. This pattern allows the required instantiations for D=1, 2 and 3 to be compiled once and then linked, as would normally be done for non-templated classes.
Note that an "ifndef" macro surrounds the block of extern declarations in the header file in the above example. This ifndef macro tests for definition of guard macro that is defined in the associated *.tpp file. This usage is designed to guarantee such that that this block of extern declarations will be skipped when the header file is included into the associated *.tpp file, which is then also included into the source file, but that this block will be retained when the header file is included into any other file. The extern declarations must be skipped when the header is included into the *.tpp file so that, when the *.tpp file is included into a *.cpp or *.cu source file, the "extern" declarations will not contradict the explicit instantiation statements given in the source file. An analogous "ifndef" macro must surround the extern declarations in every template header file that uses this pattern.
In order for this pattern to work correctly for class templates in which D is the only template parameter:
In this usage pattern, the *.tpp file is effectively treated as part of the source file, since it is included into the source file and should never be directly or indirectly included into any other file. Despite the division of code into three files, this usage is analogous to the usual organization of code for non-templated classes into a header that may be included into other classes and a source file that is compiled.
In what follows, we refer to class templates that contain template parameter other than the spatial dimension D as "generic" templates. The PSCF build system generally relies on implicit rather than explicit instantiation for such generic templates. This requires an entirely different file organization than that described above for the special case in which D is the only template parameter. For such generic templates:
Source code for such a generic class template may be organized into either of two ways:
In the first case, in which all of the code for class template is placed in a *.h header file, there is no associated *.tpp file. In the latter case, in which the source code for a class template is divided between a header and implementation file:
In this usage, the *.tpp implementation file acts in practice like a part of the header file, because it will be included indirectly by any file that includes the header file.
A template implementation file with file extension *.tpp should thus always be included either into an associated compilable source file (for templates with a single template parameter D that are explicitly instantiated) or into the associated header file (for generic templates that are implicitly instantiated). For either type of class template, a *.tpp implementation files should never be directly included into any file other than an associated source or header file.
Coding Standards (Prev) Developer Information (Up) Parameter File I/O (Next)