mirror of
https://github.com/TREX-CoE/trexio.git
synced 2024-11-04 21:24:08 +01:00
312 lines
16 KiB
Org Mode
312 lines
16 KiB
Org Mode
|
#+TITLE: The TREXIO library
|
||
|
#+STARTUP: latexpreview
|
||
|
#+SETUPFILE: ./theme.setup
|
||
|
|
||
|
|
||
|
* Format specification
|
||
|
|
||
|
#+BEGIN_EXPORT html
|
||
|
</td>
|
||
|
<td>
|
||
|
<img src="trex_specs.png" alt="TREX in a library"
|
||
|
align="right" width="300" vspace="20" hspace="20" />
|
||
|
</td></tr>
|
||
|
</table>
|
||
|
#+END_EXPORT
|
||
|
#
|
||
|
The TREXIO format is designed to store all the necessary information
|
||
|
to represent a wave function.
|
||
|
One notable feature of TREXIO is that it is self-contained, meaning
|
||
|
that all the parameters needed to recreate the wave function are
|
||
|
explicitly stored within the file, eliminating the need for external
|
||
|
databases. For example, instead of storing the name of a basis set
|
||
|
(such as cc-pVDZ), the actual basis set parameters used in the
|
||
|
calculation are stored.
|
||
|
|
||
|
** Organization of the data
|
||
|
|
||
|
The data in TREXIO are organized into *groups*, each containing
|
||
|
multiple *attributes* defined by their *type* and *dimensions*. Each
|
||
|
attribute within a group corresponds to a single scalar or array
|
||
|
variable in a code. In what follows, the notation
|
||
|
~<group>.<attribute>~ will be used to identify an attribute within a
|
||
|
group. For example, ~nucleus.charge~ refers to the
|
||
|
~charge~ attribute in the ~nucleus~ group. It is an array of type
|
||
|
~float~ with dimensions ~nucleus.num~, the attribute describing the
|
||
|
number of nuclei.
|
||
|
|
||
|
** Data types
|
||
|
|
||
|
So that TREXIO can be used in any language, we use a limited number
|
||
|
of data types. The main data types are ~int~ for integers,
|
||
|
~float~ for floating-point values, and ~str~ for
|
||
|
character strings. For complex numbers, their real and imaginary
|
||
|
parts are stored as ~float~. To minimize the risk of integer
|
||
|
overflow and accuracy loss, numerical data types are stored using
|
||
|
64-bit representations by default. However, in specific cases where
|
||
|
integers are bounded (such as orbital indices in four-index
|
||
|
integrals), the smallest possible representation is used to reduce the
|
||
|
file size. The API handles any necessary type conversions.
|
||
|
|
||
|
There are also two types derived from ~int~: ~dim~ and ~index~.
|
||
|
~dim~ is used for dimensioning variables, which are positive integers
|
||
|
used to specify the dimensions of an array. In the previous example,
|
||
|
~nucleus.num~ is a dimensioning variable that specifies the
|
||
|
dimensions of the ~nucleus.charge~ array. ~index~ is used for
|
||
|
integers that correspond to array indices, because some languages
|
||
|
(such as C or Python) use zero-based indexing, while others (such as
|
||
|
Fortran) use one-based indexing. For convenience, values of the
|
||
|
~index~ type are shifted by one when TREXIO is used in one-based
|
||
|
languages to be consistent with the semantics of the language.
|
||
|
You may also encounter some ~dim readonly~ variables. It means
|
||
|
that the value is automatically computed and written by the TREXIO
|
||
|
library, thus it is read-only and cannot be (over)written by the
|
||
|
user.
|
||
|
|
||
|
Arrays can be stored in either dense or sparse formats. If the
|
||
|
sparse format is selected, the data is stored in coordinate format.
|
||
|
For example, the element ~A(i,j,k,l)~ is stored as a quadruplet of
|
||
|
integers $(i,j,k,l)$ along with the corresponding value. Typically,
|
||
|
two-dimensional arrays are stored as dense arrays, while arrays with
|
||
|
higher dimensions are stored in sparse format.
|
||
|
For sparse data structures the data can be too large to fit in memory
|
||
|
and the data needs to be fetched using multiple function calls to
|
||
|
perform I/O on buffers. For more information on how to read/write
|
||
|
sparse data structures, see the [[./examples.html][examples]].
|
||
|
|
||
|
For the Configuration Interaction (CI) and Configuration State
|
||
|
Function (CSF) groups, the ~buffered~ data type is introduced, which
|
||
|
allows similar incremental I/O as for ~sparse~ data but without the
|
||
|
need to write indices of the sparse values.
|
||
|
|
||
|
For determinant lists (integer bit fields), the ~special~ attribute
|
||
|
is present in the type. This means that the source code is not
|
||
|
produced by the generator, but hand-written.
|
||
|
|
||
|
Some data may be complex. In that case, the real part should be stored
|
||
|
in the variable, and the imaginary part will be stored in the variable
|
||
|
with the same name suffixed by ~_im~.
|
||
|
|
||
|
* The TREXIO library
|
||
|
|
||
|
#+BEGIN_EXPORT html
|
||
|
</td>
|
||
|
<td>
|
||
|
<img src="trex_lib.png" alt="TREX in a library"
|
||
|
align="left" width="300" vspace="20" hspace="20" />
|
||
|
</td></tr>
|
||
|
</table>
|
||
|
#+END_EXPORT
|
||
|
|
||
|
The TREXIO library is written is the C language, and is licensed under
|
||
|
the open-source 3-clause BSD license to allow for use in all types of
|
||
|
quantum chemistry software, whether commercial or not.
|
||
|
|
||
|
The design of the library is divided into two main sections: the
|
||
|
front-end and the back-end. The front-end serves as the interface
|
||
|
between users and the library, while the back-end acts as the
|
||
|
interface between the library and the physical storage.
|
||
|
|
||
|
** The front-end
|
||
|
|
||
|
By using the TREXIO library, users can store and extract data in a
|
||
|
consistent and organized manner. The library provides a user-friendly
|
||
|
API, including functions for reading, writing, and checking for the
|
||
|
existence of data. The functions follow the pattern
|
||
|
~trexio_[has|read|write]_<group>_<attribute>~, where the
|
||
|
group and attribute specify the particular data being accessed. It
|
||
|
also includes an error handling mechanism, in which each function call
|
||
|
returns an exit code of type ~trexio_exit_code~, explaining
|
||
|
the type of error.
|
||
|
This can be used to catch exceptions and improve debugging in the
|
||
|
upstream user application.
|
||
|
|
||
|
To ensure the consistency of the data, the attributes can only be
|
||
|
written if all the other attributes on which they explicitly depend
|
||
|
have been written. For example, as the ~nucleus.coord~ array is
|
||
|
dimensioned by the number of nuclei ~nucleus.num~, the ~nucleus.coord~
|
||
|
attribute can only be written after ~nucleus.num~. However, the
|
||
|
library is not aware of non-explicit dependencies, such as the
|
||
|
relation between the electron repulsion integrals (ERIs) and MO
|
||
|
coefficients. A complete control of the consistency of the data is
|
||
|
therefore impossible, so the attributes were chosen to be by default
|
||
|
/immutable/. By only allowing data to be written only once, the
|
||
|
risk of modifying data in a way that creates inconsistencies is
|
||
|
reduced. For example, if the ERIs have already been written, it would
|
||
|
be inconsistent to later modify the MO coefficients. To allow for
|
||
|
flexibility, the library also allows for the use of an /unsafe/
|
||
|
mode, in which data can be overwritten. However, this mode carries
|
||
|
the risk of producing inconsistent files, and the ~metadata~ group's
|
||
|
~unsafe~ attribute is set to ~1~ to indicate that the file has
|
||
|
potentially been modified in a dangerous way. This attribute can be
|
||
|
manually reset to ~0~ if the user is confident that the modifications
|
||
|
made are safe.
|
||
|
|
||
|
** The back-end
|
||
|
|
||
|
At present, TREXIO supports two back-ends: one relying only on the
|
||
|
C standard library to produce plain text files (the so-called /text/
|
||
|
back-end), and one relying on the HDF5 library.
|
||
|
|
||
|
With the text back-end, the TREXIO "file" is a directory containing
|
||
|
multiple text files, one for each group. This back end is intended
|
||
|
to be used in development environments, as it gives access to the
|
||
|
user to the standard tools such as ~diff~ and ~grep~.
|
||
|
In addition, text files are better adapted than binary files for
|
||
|
version control systems such as Git, so this format can be also
|
||
|
used for storing reference data for unit tests.
|
||
|
|
||
|
HDF5 is a binary file format and library for storing and managing
|
||
|
large amounts of data in a hierarchical structure. It allows users
|
||
|
to manipulate data in a way similar to how files and directories
|
||
|
are manipulated within the file system. The HDF5 library provides
|
||
|
optimal performance through its memory mapping mechanism and
|
||
|
supports advanced features such as serial and parallel I/O,
|
||
|
chunking, and compression filters. However, HDF5 files are in
|
||
|
binary format, which requires additional tools such as ~h5dump~ to
|
||
|
view them in a human-readable format. It is widely used in
|
||
|
scientific and engineering applications, and is known for its high
|
||
|
performance and ability to handle large data sets efficiently.
|
||
|
|
||
|
The TREXIO HDF5 back-end is the recommended choice for production
|
||
|
environments, as it provides high I/O performance. Furthermore,
|
||
|
all data is stored in a single file, making it especially suitable
|
||
|
for parallel file systems like Lustre. These file systems are
|
||
|
optimized for large, sequential I/O operations and are not
|
||
|
well-suited for small, random I/O operations. When multiple small
|
||
|
files are used, the file system may become overwhelmed with
|
||
|
metadata operations like creating, deleting, or modifying files,
|
||
|
which can adversely affect performance.
|
||
|
|
||
|
In a benchmarking program designed to compare the two back-ends of
|
||
|
the library, the HDF5 back-end was found to be significantly faster
|
||
|
than the text back-end. The program wrote a wave function made up
|
||
|
of 100 million Slater determinants and measured the time taken to
|
||
|
write the Slater determinants and CI coefficients. The HDF5
|
||
|
back-end achieved a speed of $10.4\times10^6$ Slater determinants
|
||
|
per second and a data transfer rate of 406 MB/s, while the text
|
||
|
back-end had a speed of $1.1\times10^6$ determinants per second and
|
||
|
a transfer rate of 69 MB/s. These results were obtained on a DELL
|
||
|
960 GB mix-use solid-state drive (SSD). The HDF5 back-end was able
|
||
|
to achieve a performance level close to the peak performance of the
|
||
|
SSD, while the text back-end's performance was limited by the speed
|
||
|
of the CPU for performing binary to ASCII conversions.
|
||
|
|
||
|
In addition to the HDF5 and text back-ends, it is also possible to
|
||
|
introduce new back-ends to the library. For example, a back-end
|
||
|
could be created to support object storage systems, such as those
|
||
|
used in cloud-based applications or for archiving in open data
|
||
|
repositories.
|
||
|
|
||
|
** Supported languages
|
||
|
|
||
|
One of the main benefits of using C as the interface for a library is
|
||
|
that it is easy to use from other programming languages. Many
|
||
|
programming languages, such as Python or Julia, provide built-in
|
||
|
support for calling C functions, which means that it is relatively
|
||
|
straightforward to write a wrapper that allows a library written in C
|
||
|
to be called from another language.
|
||
|
In general, libraries with a C interface are the easiest to use from
|
||
|
other programming languages, because C is widely supported and has a
|
||
|
simple, stable application binary interface (ABI). Other languages,
|
||
|
such as Fortran and C++, may have more complex ABIs and may
|
||
|
require more work to interface with them.
|
||
|
|
||
|
TREXIO has been employed in codes developed in various programming
|
||
|
languages, including C, C++, Fortran, Python, OCaml, and Julia. While
|
||
|
Julia is designed to enable the use of C functions without the need
|
||
|
for additional manual interfacing, the TREXIO C header file was
|
||
|
automatically integrated into Julia programs using the
|
||
|
~CBindings.jl~ package.
|
||
|
In contrast, specific bindings have been provided for Fortran, Python,
|
||
|
and OCaml to simplify the user experience.
|
||
|
|
||
|
In particular, the binding for Fortran is not distributed as multiple
|
||
|
compiled Fortran module files (~.mod~), but instead as a single
|
||
|
Fortran source file (~.F90~). The distribution of the source file
|
||
|
instead of the compiled module has multiple benefits. It ensures that
|
||
|
the TREXIO module is always compiled with the same compiler as the
|
||
|
client code, avoiding the compatibility problem of ~.mod~ files
|
||
|
between different compiler versions and vendors. The single-file
|
||
|
model requires very little changes in the build system of the user's
|
||
|
codes, and it facilitates the search for the interface of a particular
|
||
|
function. In addition, advanced text editors can parse the TREXIO
|
||
|
interface to propose interactive auto-completion of the TREXIO
|
||
|
function names to the developers.
|
||
|
|
||
|
Finally, the Python module, partly generated with SWIG and fully
|
||
|
compatible with NumPy, allows Python users to interact with the
|
||
|
library in a more intuitive and user-friendly way. Using the Python
|
||
|
interface is likely the easiest way to begin using TREXIO and
|
||
|
understanding its features. In order to help users get started with
|
||
|
TREXIO and understand its functionality, tutorials in Jupyter
|
||
|
notebooks are available on GitHub
|
||
|
(https://github.com/TREX-CoE/trexio-tutorials), and can be executed
|
||
|
via the Binder platform.
|
||
|
|
||
|
|
||
|
** Source code generation and documentation
|
||
|
|
||
|
Source code generation is a valuable technique that can significantly
|
||
|
improve the efficiency and consistency of software development. By
|
||
|
using templates to generate code automatically, developers can avoid
|
||
|
manual coding and reduce the risk of errors or inconsistencies. This
|
||
|
approach is particularly useful when a large number of functions
|
||
|
follow similar patterns, as in the case of the TREXIO library, where
|
||
|
functions are named according to the pattern
|
||
|
~trexio_[has|read|write]_<group>_<attribute>~.
|
||
|
By generating these functions from the format specification using
|
||
|
templates, the developers can ensure that the resulting code follows a
|
||
|
consistent structure and is free from errors or inconsistencies.
|
||
|
|
||
|
The description of the format is written in a text file in the Org
|
||
|
format. Org is a structured plain text format, containing information
|
||
|
expressed in a lightweight markup language similar to the popular
|
||
|
Markdown language. While Org was introduced as a mode of the GNU
|
||
|
Emacs text editor, its basic functionalities have been implemented in
|
||
|
most text editors such as Vim, Atom or VS Code.
|
||
|
|
||
|
There are multiple benefits in using the Org format. The first
|
||
|
benefit is that the Org syntax is easy to learn and allows for the
|
||
|
insertion of equations in \LaTeX{} syntax. Additionally, Org files
|
||
|
can be easily converted to HyperText Markup Language (HTML) or
|
||
|
Portable Document Format (PDF) for generating documentation. The
|
||
|
second benefit is that GNU Emacs is a programmable text editor and
|
||
|
code blocks in Org files can be executed interactively, similar to
|
||
|
Jupyter notebooks. These code blocks can also manipulate data defined
|
||
|
in tables and this feature is used to automatically transform tables
|
||
|
describing groups and attributes in the documentation into a
|
||
|
JavaScript Object Notation (JSON) file.
|
||
|
This JSON file is then used by a Python script to generate the needed
|
||
|
functions in C language, as well as header files and some files
|
||
|
required for the Fortran, Python, and OCaml interfaces.
|
||
|
|
||
|
With this approach, contributions to the development of the TREXIO
|
||
|
library can be made simply by adding new tables to the Org file, which
|
||
|
can be submitted as /pull requests/ on the project's GitHub
|
||
|
repository (https://github.com/trex-coe/trexio). Overall, this
|
||
|
process allows for a more efficient and consistent development process
|
||
|
and enables contributions from a wider range of individuals,
|
||
|
regardless of their programming skills.
|
||
|
|
||
|
** Availability
|
||
|
|
||
|
The TREXIO library is designed to be portable and easy to install
|
||
|
on a wide range of systems. It follows the C99 standard to ensure
|
||
|
compatibility with older systems, and can be configured with either
|
||
|
the GNU Autotools or the CMake build systems. The only external
|
||
|
dependency is the HDF5 library, which is widely available on HPC
|
||
|
platforms and as packages on major Linux distributions. Note that
|
||
|
it is possible to disable the HDF5 back-end at configuration time,
|
||
|
allowing TREXIO to operate only with the text back-end and have
|
||
|
zero external dependencies. This can be useful for users who may
|
||
|
not be able to install HDF5 on certain systems.
|
||
|
|
||
|
TREXIO is distributed as a tarball containing the source code,
|
||
|
generated code, documentation, and Fortran interface. It is also
|
||
|
available as a binary ~.deb~ package for Debian-based Linux
|
||
|
distributions and as packages for Guix, Spack and Conda. The Python
|
||
|
module can be found in the PyPI repository, the OCaml binding is
|
||
|
available in the official OPAM repository, and the ~.deb~ packages
|
||
|
are available in Ubuntu 23.04.
|