IODA
source_location_compat.h
Go to the documentation of this file.
1 #pragma once
2 /*
3  * (C) Copyright 2020-2021 UCAR
4  *
5  * This software is licensed under the terms of the Apache Licence Version 2.0
6  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
7  */
8 /*! @file source_location_compat.h
9 * @brief Implements ioda::source_location.
10 */
11 
12 #include <cstdint>
13 
14 #if __cplusplus >= 202002L
15 # include <source_location>
16 #endif
17 
18 namespace ioda {
19 namespace detail {
20 /// @brief Compatibility functions
21 namespace compat {
22 /// @brief This namespace is imported into the top-level ioda namespace if the
23 /// C++ language standard version is less than C++20.
24 namespace source_location {
25 /*! @brief This is a placeholder implementation for std::source_location, which
26 * is part of C++20. We backport this to older compilers as needed.
27 */
29  /// @brief Determine current source location.
30  /// @note Does not quite match the standard since we really need compiler support
31  /// to implement this outside of a macro. For now, ioda users should use ioda_Here instead.
32  /// @see ioda_Here for the standard implementation.
33  inline static source_location current() noexcept { return source_location(); }
34  /// @brief Default constructor. Implementation-defined behavior.
35  source_location() noexcept : line_(0), column_(0), file_name_(""), function_name_("") {}
36 
37  source_location(uint_least32_t line, uint_least32_t column, const char* file, const char* func)
38  : line_(line), column_(column), file_name_(file), function_name_(func) {}
39 
40  // source location field access
41  uint_least32_t line() const noexcept { return line_; }
42  uint_least32_t column() const noexcept { return column_; }
43  const char* file_name() const noexcept { return file_name_; }
44  const char* function_name() const noexcept { return function_name_; }
45 
46 private:
47  uint_least32_t line_;
48  uint_least32_t column_;
49  const char* file_name_;
50  const char* function_name_;
51 };
52 } // namespace source_location
53 } // namespace compat
54 } // namespace detail
55 
56 // __func__ is a more standard option, but this ignores function parameters and only
57 // displays the function name. Bad for distinguishing overrides.
58 #if defined(_MSC_FULL_VER)
59 # define IODA_FUNCSIG __FUNCSIG__
60 #else
61 // gcc, clang, and Apple clang all define this.
62 # define IODA_FUNCSIG __PRETTY_FUNCTION__
63 #endif
64 
65 /*! @def ioda_Here is a macro to determine the current location in source code.
66 * @details
67 * This macro is still needed in pre-C++20 compilers to accurately and portably
68 * determine source filenames, functions, and line numbers.
69 *
70 * Until C++20-conformant compilers are available, ioda_Here needs to be passed in
71 * every ioda Exception to get accurate line number information.
72 *
73 * gcc, clang, and Intel do have nonstandard builtin intrinsic functions to do this,
74 * but let's aim for maximum portability. A feature test macro in CMake could help
75 * detect where these intrinsics are available: __builtin_LINE, __builtin_FUNCTION,
76 * __builtin_FILE.
77 */
78 
79 // NOLINTs because cpplint complains about "using" declarations outside of functions,
80 // particularly in headers.
81 
82 #if __cplusplus >= 202002L
83 using std::source_location; // NOLINT
84 # define ioda_Here() ::std::source_location::current()
85 #else
86 using detail::compat::source_location::source_location; // NOLINT
87 # define ioda_Here() ::ioda::source_location(__LINE__, 0, __FILE__, IODA_FUNCSIG)
88 #endif
89 
90 } // namespace ioda
This is a placeholder implementation for std::source_location, which is part of C++20.
static source_location current() noexcept
Determine current source location.
source_location(uint_least32_t line, uint_least32_t column, const char *file, const char *func)
source_location() noexcept
Default constructor. Implementation-defined behavior.