IODA
Type.cpp
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2020-2021 UCAR
3  *
4  * This software is licensed under the terms of the Apache Licence Version 2.0
5  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
6  */
7 #include "ioda/Types/Type.h"
8 
9 #include <map>
10 
11 #include "ioda/defs.h"
12 #include "ioda/Exception.h"
13 
14 namespace ioda {
15 namespace detail {
16 /** \brief Safe char array copy.
17  \returns the number of characters actually written.
18  \param dest is the pointer to the destination. Always null terminated.
19  \param destSz is the size of the destination buller, including the trailing null character.
20  \param src is the pointer to the source. Characters from src are copied either until the
21  first null character or until srcSz. Note that null termination comes later.
22  \param srcSz is the max size of the source buffer.
23  \deprecated This function is old and should not be used!
24  **/
25 size_t COMPAT_strncpy_s(char* dest, size_t destSz, const char* src, size_t srcSz) {
26  if (!dest || !src) throw Exception("Null pointer passed to function.", ioda_Here());
27 #ifdef JEDI_USING_SECURE_STRINGS
28  strncpy_s(dest, destSz, src, srcSz);
29  return strnlen_s(dest, destSz);
30 #else
31  if (destSz == 0) throw; // jedi_throw.add("Reason", "Invalid destination size.");
32  // See https://devblogs.microsoft.com/oldnewthing/20170927-00/?p=97095
33  // Unfortunately, we can't really detect pointer range overlaps in a system-independent
34  // manner. This is why we want to use the secure string functions if at all possible.
35  if (srcSz <= destSz) {
36  strncpy(dest, src, srcSz);
37  if (dest[srcSz - 1] != '\0') throw Exception("Non-terminated null copy.", ioda_Here());
38  } else {
39  strncpy(dest, src, destSz);
40  // Additionally, throw on null-string truncation.
41 
42  // Not using strchr because of its memory-unsafe nature.
43  for (size_t i = 0; i < destSz - 1; ++i) {
44  if (i < srcSz) {
45  if (dest[i] == '\0' && src[i] == '\0') break; // First null hit. Success.
46  if (dest[i] == '\0' && src[i] != '\0')
47  throw Exception("Truncated array copy error.", ioda_Here());
48  } else
49  throw Exception("Null not reached by end of source!", ioda_Here());
50  }
51  }
52 
53  dest[destSz - 1] = 0;
54  for (size_t i = 0; i < destSz; ++i) {
55  if (dest[i] == '\0') return i;
56  }
57  throw Exception("Truncated array copy error.", ioda_Here());
58 #endif
59 }
60 
61 // template<> Type_Base<>::~Type_Base() {}
62 // template<> Type_Base<>::Type_Base(std::shared_ptr<Type_Backend> b) : backend_(b) {}
63 // template<> std::shared_ptr<Type_Backend> Type_Base<>::getBackend() const { return backend_; }
64 
65 
66 template <>
67 size_t Type_Base<>::getSize() const {
68  try {
69  if (backend_ == nullptr)
70  throw Exception("Missing backend or unimplemented backend function.", ioda_Here());
71  return backend_->getSize();
72  } catch (...) {
73  std::throw_with_nested(Exception(
74  "An exception occurred inside ioda while getting the size of a data type.", ioda_Here()));
75  }
76 }
77 
78 Type_Backend::Type_Backend() : Type_Base{nullptr, nullptr} {}
79 Type_Backend::~Type_Backend() = default;
80 
81 size_t Type_Backend::getSize() const {
82  throw Exception("This function must be implemented in the backend engine.", ioda_Here());
83 }
84 
85 } // namespace detail
86 
87 Type::Type() : Type_Base(nullptr, nullptr), as_type_index_(typeid(void)) {}
88 Type::Type(std::shared_ptr<detail::Type_Backend> b, std::type_index t)
89  : Type_Base(b, b->provider_), as_type_index_(t) {}
90 
91 Type::Type(BasicTypes typ, gsl::not_null<::ioda::detail::Type_Provider*> t)
92  : Type_Base(nullptr, t.get()), as_type_index_(typeid(void)) {
93  static const std::map<BasicTypes, std::type_index> workable_types
94  = {{BasicTypes::float_, typeid(float)}, // NOLINT: cpplint doesn't understand this!
95  {BasicTypes::double_, typeid(double)}, // NOLINT
96  {BasicTypes::ldouble_, typeid(long double)},
97  {BasicTypes::char_, typeid(char)}, // NOLINT
98  {BasicTypes::short_, typeid(short)},
99  {BasicTypes::ushort_, typeid(unsigned short)},
100  {BasicTypes::int_, typeid(int)}, // NOLINT
101  {BasicTypes::uint_, typeid(unsigned)},
102  {BasicTypes::lint_, typeid(long)},
103  {BasicTypes::ulint_, typeid(unsigned long)},
104  {BasicTypes::llint_, typeid(long long)},
105  {BasicTypes::ullint_, typeid(unsigned long long)},
106  {BasicTypes::int32_, typeid(int32_t)},
107  {BasicTypes::uint32_, typeid(uint32_t)},
108  {BasicTypes::int16_, typeid(int16_t)},
109  {BasicTypes::uint16_, typeid(uint16_t)},
110  {BasicTypes::int64_, typeid(int64_t)},
111  {BasicTypes::uint64_, typeid(uint64_t)},
112  {BasicTypes::bool_, typeid(bool)}, // NOLINT
113  {BasicTypes::str_, typeid(std::string)}};
114  as_type_index_ = workable_types.at(typ);
115 
116  if (typ == BasicTypes::undefined_) throw Exception("Bad input", ioda_Here());
117  if (typ == BasicTypes::str_)
118  *this = t->makeStringType(Types::constants::_Variable_Length, typeid(std::string));
119  else
120  *this = t->makeFundamentalType(workable_types.at(typ));
121 }
122 
123 Type::~Type() = default;
124 
125 
126 } // namespace ioda
IODA's error system.
Interfaces for ioda::Type and related classes.
The ioda exception class.
Definition: Exception.h:54
std::type_index as_type_index_
Definition: Type.h:142
virtual ~Type()
Type()
Definition: Type.cpp:87
size_t getSize() const override
Get the size of a single element of a type, in bytes.
Definition: Type.cpp:81
virtual size_t getSize() const
Get the size of a single element of a type, in bytes.
Definition: Type.cpp:67
Common preprocessor definitions used throughout IODA.
constexpr size_t _Variable_Length
Definition: Type.h:182
IODA_DL size_t COMPAT_strncpy_s(char *dest, size_t destSz, const char *src, size_t srcSz)
Safe char array copy.
Definition: Type.cpp:25
BasicTypes
Definition: Type.h:37
@ undefined_
Internal use only.
#define ioda_Here()