ua_types.h 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805
  1. /*
  2. * Copyright (C) 2013-2015 the contributors as stated in the AUTHORS file
  3. *
  4. * This file is part of open62541. open62541 is free software: you can
  5. * redistribute it and/or modify it under the terms of the GNU Lesser General
  6. * Public License, version 3 (as published by the Free Software Foundation) with
  7. * a static linking exception as stated in the LICENSE file provided with
  8. * open62541.
  9. *
  10. * open62541 is distributed in the hope that it will be useful, but WITHOUT ANY
  11. * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
  12. * A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
  13. * details.
  14. */
  15. #ifndef UA_TYPES_H_
  16. #define UA_TYPES_H_
  17. #ifdef __cplusplus
  18. extern "C" {
  19. #endif
  20. #include "ua_config.h"
  21. #include "ua_constants.h"
  22. #include <stdint.h>
  23. #include <stdbool.h>
  24. /**
  25. * Data Types
  26. * ==========
  27. *
  28. * In open62541, all data types share the same basic API for creation, copying
  29. * and deletion. The header ua_types.h defines the builtin types. In addition,
  30. * we auto-generate ua_types_generated.h with additional types as well as the
  31. * following function definitions for all (builtin and generated) data types
  32. * ``T``.
  33. *
  34. * ``void T_init(T *ptr)``
  35. * Initialize the data type. This is synonymous with zeroing out the memory,
  36. * i.e. ``memset(dataptr, 0, sizeof(T))``.
  37. * ``T* T_new()``
  38. * Allocate and return the memory for the data type. The memory is already
  39. * initialized.
  40. * ``UA_StatusCode T_copy(const T *src, T *dst)``
  41. * Copy the content of the data type. Returns ``UA_STATUSCODE_GOOD`` or
  42. * ``UA_STATUSCODE_BADOUTOFMEMORY``.
  43. * ``void T_deleteMembers(T *ptr)``
  44. * Delete the dynamically allocated content of the data type, but not the data
  45. * type itself.
  46. * ``void T_delete(T *ptr)``
  47. * Delete the content of the data type and the memory for the data type itself.
  48. *
  49. * OPC UA defines 25 builtin data types. All other data types are combinations
  50. * of the 25 builtin data types. */
  51. #define UA_BUILTIN_TYPES_COUNT 25U
  52. /**
  53. * Builtin Types Part 1
  54. * --------------------
  55. *
  56. * Boolean
  57. * ^^^^^^^
  58. * A two-state logical value (true or false). */
  59. typedef bool UA_Boolean;
  60. #define UA_TRUE true
  61. #define UA_FALSE false
  62. /**
  63. * SByte
  64. * ^^^^^
  65. * An integer value between -128 and 127. */
  66. typedef int8_t UA_SByte;
  67. #define UA_SBYTE_MAX 127
  68. #define UA_SBYTE_MIN (-128)
  69. /**
  70. * Byte
  71. * ^^^^
  72. * An integer value between 0 and 256. */
  73. typedef uint8_t UA_Byte;
  74. #define UA_BYTE_MAX 256
  75. #define UA_BYTE_MIN 0
  76. /**
  77. * Int16
  78. * ^^^^^
  79. * An integer value between -32 768 and 32 767. */
  80. typedef int16_t UA_Int16;
  81. #define UA_INT16_MAX 32767
  82. #define UA_INT16_MIN (-32768)
  83. /**
  84. * UInt16
  85. * ^^^^^^
  86. * An integer value between 0 and 65 535. */
  87. typedef uint16_t UA_UInt16;
  88. #define UA_UINT16_MAX 65535
  89. #define UA_UINT16_MIN 0
  90. /**
  91. * Int32
  92. * ^^^^^
  93. * An integer value between -2 147 483 648 and 2 147 483 647. */
  94. typedef int32_t UA_Int32;
  95. #define UA_INT32_MAX 2147483647
  96. #define UA_INT32_MIN (-2147483648)
  97. /**
  98. * UInt32
  99. * ^^^^^^
  100. * An integer value between 0 and 4 294 967 295. */
  101. typedef uint32_t UA_UInt32;
  102. #define UA_UINT32_MAX 4294967295
  103. #define UA_UINT32_MIN 0
  104. /**
  105. * Int64
  106. * ^^^^^
  107. * An integer value between -10 223 372 036 854 775 808 and
  108. * 9 223 372 036 854 775 807. */
  109. typedef int64_t UA_Int64;
  110. #define UA_INT64_MAX (int64_t)9223372036854775807
  111. #define UA_INT64_MIN ((int64_t)-9223372036854775808)
  112. /**
  113. * UInt64
  114. * ^^^^^^
  115. * An integer value between 0 and 18 446 744 073 709 551 615. */
  116. typedef uint64_t UA_UInt64;
  117. #define UA_UINT64_MAX (int64_t)18446744073709551615
  118. #define UA_UINT64_MIN (int64_t)0
  119. /**
  120. * Float
  121. * ^^^^^
  122. * An IEEE single precision (32 bit) floating point value. */
  123. typedef float UA_Float;
  124. /**
  125. * Double
  126. * ^^^^^^
  127. * An IEEE double precision (64 bit) floating point value. */
  128. typedef double UA_Double;
  129. /**
  130. * .. _statuscode:
  131. *
  132. * StatusCode
  133. * ^^^^^^^^^^
  134. * A numeric identifier for a error or condition that is associated with a value
  135. * or an operation. See the section :ref:`statuscodes` for the meaning of a
  136. * specific code. */
  137. typedef uint32_t UA_StatusCode;
  138. /**
  139. * Array handling
  140. * --------------
  141. * In OPC UA, arrays can have a length of zero or more with the usual meaning.
  142. * In addition, arrays can be undefined. Then, they don't even have a length. In
  143. * the binary encoding, this is indicated by an array of length -1.
  144. *
  145. * In open62541 however, we use ``size_t`` for array lengths. An undefined array
  146. * has length 0 and the data pointer is NULL. An array of length 0 also has
  147. * length 0 but points to a sentinel memory address. */
  148. #define UA_EMPTY_ARRAY_SENTINEL ((void*)0x01)
  149. /** Forward Declaration of UA_DataType. See Section `Generic Type Handling`_
  150. for details. */
  151. struct UA_DataType;
  152. typedef struct UA_DataType UA_DataType;
  153. /** The following functions are used for handling arrays of any data type. */
  154. /* Allocates and initializes an array of variables of a specific type
  155. *
  156. * @param size The requested array length
  157. * @param type The datatype description
  158. * @return Returns the memory location of the variable or (void*)0 if no memory
  159. could be allocated */
  160. void UA_EXPORT * UA_Array_new(size_t size, const UA_DataType *type) UA_FUNC_ATTR_MALLOC;
  161. /* Allocates and copies an array
  162. *
  163. * @param src The memory location of the source array
  164. * @param size The size of the array
  165. * @param dst The location of the pointer to the new array
  166. * @param type The datatype of the array members
  167. * @return Returns UA_STATUSCODE_GOOD or UA_STATUSCODE_BADOUTOFMEMORY */
  168. UA_StatusCode UA_EXPORT
  169. UA_Array_copy(const void *src, size_t size, void **dst,
  170. const UA_DataType *type) UA_FUNC_ATTR_WARN_UNUSED_RESULT;
  171. /* Deletes an array.
  172. *
  173. * @param p The memory location of the array
  174. * @param size The size of the array
  175. * @param type The datatype of the array members */
  176. void UA_EXPORT UA_Array_delete(void *p, size_t size, const UA_DataType *type);
  177. /**
  178. * Builtin Types, Part 2
  179. * ---------------------
  180. *
  181. * String
  182. * ^^^^^^
  183. * A sequence of Unicode characters. Strings are just an array of UA_Byte. */
  184. typedef struct {
  185. size_t length; /* The length of the string */
  186. UA_Byte *data; /* The content (not null-terminated) */
  187. } UA_String;
  188. /* Copies the content on the heap. Returns a null-string when alloc fails */
  189. UA_String UA_EXPORT UA_String_fromChars(char const src[]) UA_FUNC_ATTR_WARN_UNUSED_RESULT;
  190. UA_Boolean UA_EXPORT UA_String_equal(const UA_String *s1, const UA_String *s2);
  191. UA_EXPORT extern const UA_String UA_STRING_NULL;
  192. /**
  193. * ``UA_STRING`` returns a string pointing to the preallocated char-array.
  194. * ``UA_STRING_ALLOC`` is shorthand for ``UA_String_fromChars`` and makes a copy
  195. * of the char-array. */
  196. static UA_INLINE UA_String
  197. UA_STRING(char *chars) {
  198. UA_String str; str.length = strlen(chars);
  199. str.data = (UA_Byte*)chars; return str;
  200. }
  201. #define UA_STRING_ALLOC(CHARS) UA_String_fromChars(CHARS)
  202. /**
  203. * DateTime
  204. * ^^^^^^^^
  205. * An instance in time. A DateTime value is encoded as a 64-bit signed integer
  206. * which represents the number of 100 nanosecond intervals since January 1, 1601
  207. * (UTC). */
  208. typedef int64_t UA_DateTime;
  209. /* Multiply to convert units for time difference computations */
  210. #define UA_USEC_TO_DATETIME 10LL
  211. #define UA_MSEC_TO_DATETIME (UA_USEC_TO_DATETIME * 1000LL)
  212. #define UA_SEC_TO_DATETIME (UA_MSEC_TO_DATETIME * 1000LL)
  213. /* Datetime of 1 Jan 1970 00:00 UTC */
  214. #define UA_DATETIME_UNIX_EPOCH (11644473600LL * UA_SEC_TO_DATETIME)
  215. /* The current time */
  216. UA_DateTime UA_EXPORT UA_DateTime_now(void);
  217. /* CPU clock invariant to system time changes. Use only for time diffs, not
  218. * current time */
  219. UA_DateTime UA_EXPORT UA_DateTime_nowMonotonic(void);
  220. typedef struct UA_DateTimeStruct {
  221. UA_UInt16 nanoSec;
  222. UA_UInt16 microSec;
  223. UA_UInt16 milliSec;
  224. UA_UInt16 sec;
  225. UA_UInt16 min;
  226. UA_UInt16 hour;
  227. UA_UInt16 day;
  228. UA_UInt16 month;
  229. UA_UInt16 year;
  230. } UA_DateTimeStruct;
  231. UA_DateTimeStruct UA_EXPORT UA_DateTime_toStruct(UA_DateTime t);
  232. UA_String UA_EXPORT UA_DateTime_toString(UA_DateTime t);
  233. /**
  234. * Guid
  235. * ^^^^
  236. * A 16 byte value that can be used as a globally unique identifier. */
  237. typedef struct {
  238. UA_UInt32 data1;
  239. UA_UInt16 data2;
  240. UA_UInt16 data3;
  241. UA_Byte data4[8];
  242. } UA_Guid;
  243. UA_Boolean UA_EXPORT UA_Guid_equal(const UA_Guid *g1, const UA_Guid *g2);
  244. UA_EXPORT extern const UA_Guid UA_GUID_NULL;
  245. /**
  246. * ByteString
  247. * ^^^^^^^^^^
  248. * A sequence of octets. */
  249. typedef UA_String UA_ByteString;
  250. static UA_INLINE UA_Boolean
  251. UA_ByteString_equal(const UA_ByteString *string1, const UA_ByteString *string2) {
  252. return UA_String_equal((const UA_String*)string1, (const UA_String*)string2);
  253. }
  254. /* Allocates memory of size length for the bytestring.
  255. * The content is not set to zero. */
  256. UA_StatusCode UA_EXPORT
  257. UA_ByteString_allocBuffer(UA_ByteString *bs, size_t length);
  258. UA_EXPORT extern const UA_ByteString UA_BYTESTRING_NULL;
  259. static UA_INLINE UA_ByteString
  260. UA_BYTESTRING(char *chars) {
  261. UA_ByteString str; str.length = strlen(chars);
  262. str.data = (UA_Byte*)chars; return str;
  263. }
  264. static UA_INLINE UA_ByteString
  265. UA_BYTESTRING_ALLOC(const char *chars) {
  266. UA_String str = UA_String_fromChars(chars); UA_ByteString bstr;
  267. bstr.length = str.length; bstr.data = str.data; return bstr;
  268. }
  269. /**
  270. * XmlElement
  271. * ^^^^^^^^^^
  272. * An XML element. */
  273. typedef UA_String UA_XmlElement;
  274. /**
  275. * NodeId
  276. * ^^^^^^
  277. * An identifier for a node in the address space of an OPC UA Server. */
  278. enum UA_NodeIdType {
  279. UA_NODEIDTYPE_NUMERIC = 0, /* In the binary encoding, this can also
  280. become 1 or 2 (2byte and 4byte encoding of
  281. small numeric nodeids) */
  282. UA_NODEIDTYPE_STRING = 3,
  283. UA_NODEIDTYPE_GUID = 4,
  284. UA_NODEIDTYPE_BYTESTRING = 5
  285. };
  286. typedef struct {
  287. UA_UInt16 namespaceIndex;
  288. enum UA_NodeIdType identifierType;
  289. union {
  290. UA_UInt32 numeric;
  291. UA_String string;
  292. UA_Guid guid;
  293. UA_ByteString byteString;
  294. } identifier;
  295. } UA_NodeId;
  296. UA_EXPORT extern const UA_NodeId UA_NODEID_NULL;
  297. UA_Boolean UA_EXPORT UA_NodeId_isNull(const UA_NodeId *p);
  298. UA_Boolean UA_EXPORT UA_NodeId_equal(const UA_NodeId *n1, const UA_NodeId *n2);
  299. /** The following functions are shorthand for creating NodeIds. */
  300. static UA_INLINE UA_NodeId
  301. UA_NODEID_NUMERIC(UA_UInt16 nsIndex, UA_UInt32 identifier) {
  302. UA_NodeId id; id.namespaceIndex = nsIndex;
  303. id.identifierType = UA_NODEIDTYPE_NUMERIC;
  304. id.identifier.numeric = identifier; return id;
  305. }
  306. static UA_INLINE UA_NodeId
  307. UA_NODEID_STRING(UA_UInt16 nsIndex, char *chars) {
  308. UA_NodeId id; id.namespaceIndex = nsIndex;
  309. id.identifierType = UA_NODEIDTYPE_STRING;
  310. id.identifier.string = UA_STRING(chars); return id;
  311. }
  312. static UA_INLINE UA_NodeId
  313. UA_NODEID_STRING_ALLOC(UA_UInt16 nsIndex, const char *chars) {
  314. UA_NodeId id; id.namespaceIndex = nsIndex;
  315. id.identifierType = UA_NODEIDTYPE_STRING;
  316. id.identifier.string = UA_STRING_ALLOC(chars); return id;
  317. }
  318. static UA_INLINE UA_NodeId
  319. UA_NODEID_GUID(UA_UInt16 nsIndex, UA_Guid guid) {
  320. UA_NodeId id; id.namespaceIndex = nsIndex;
  321. id.identifierType = UA_NODEIDTYPE_GUID;
  322. id.identifier.guid = guid; return id;
  323. }
  324. static UA_INLINE UA_NodeId
  325. UA_NODEID_BYTESTRING(UA_UInt16 nsIndex, char *chars) {
  326. UA_NodeId id; id.namespaceIndex = nsIndex;
  327. id.identifierType = UA_NODEIDTYPE_BYTESTRING;
  328. id.identifier.byteString = UA_BYTESTRING(chars); return id;
  329. }
  330. static UA_INLINE UA_NodeId
  331. UA_NODEID_BYTESTRING_ALLOC(UA_UInt16 nsIndex, const char *chars) {
  332. UA_NodeId id; id.namespaceIndex = nsIndex;
  333. id.identifierType = UA_NODEIDTYPE_BYTESTRING;
  334. id.identifier.byteString = UA_BYTESTRING_ALLOC(chars); return id;
  335. }
  336. /**
  337. * ExpandedNodeId
  338. * ^^^^^^^^^^^^^^
  339. * A NodeId that allows the namespace URI to be specified instead of an index. */
  340. typedef struct {
  341. UA_NodeId nodeId;
  342. UA_String namespaceUri;
  343. UA_UInt32 serverIndex;
  344. } UA_ExpandedNodeId;
  345. /** The following functions are shorthand for creating ExpandedNodeIds. */
  346. static UA_INLINE UA_ExpandedNodeId
  347. UA_EXPANDEDNODEID_NUMERIC(UA_UInt16 nsIndex, UA_UInt32 identifier) {
  348. UA_ExpandedNodeId id; id.nodeId = UA_NODEID_NUMERIC(nsIndex, identifier);
  349. id.serverIndex = 0; id.namespaceUri = UA_STRING_NULL; return id;
  350. }
  351. static UA_INLINE UA_ExpandedNodeId
  352. UA_EXPANDEDNODEID_STRING(UA_UInt16 nsIndex, char *chars) {
  353. UA_ExpandedNodeId id; id.nodeId = UA_NODEID_STRING(nsIndex, chars);
  354. id.serverIndex = 0; id.namespaceUri = UA_STRING_NULL; return id;
  355. }
  356. static UA_INLINE UA_ExpandedNodeId
  357. UA_EXPANDEDNODEID_STRING_ALLOC(UA_UInt16 nsIndex, const char *chars) {
  358. UA_ExpandedNodeId id; id.nodeId = UA_NODEID_STRING_ALLOC(nsIndex, chars);
  359. id.serverIndex = 0; id.namespaceUri = UA_STRING_NULL; return id;
  360. }
  361. static UA_INLINE UA_ExpandedNodeId
  362. UA_EXPANDEDNODEID_STRING_GUID(UA_UInt16 nsIndex, UA_Guid guid) {
  363. UA_ExpandedNodeId id; id.nodeId = UA_NODEID_GUID(nsIndex, guid);
  364. id.serverIndex = 0; id.namespaceUri = UA_STRING_NULL; return id;
  365. }
  366. static UA_INLINE UA_ExpandedNodeId
  367. UA_EXPANDEDNODEID_BYTESTRING(UA_UInt16 nsIndex, char *chars) {
  368. UA_ExpandedNodeId id; id.nodeId = UA_NODEID_BYTESTRING(nsIndex, chars);
  369. id.serverIndex = 0; id.namespaceUri = UA_STRING_NULL; return id;
  370. }
  371. static UA_INLINE UA_ExpandedNodeId
  372. UA_EXPANDEDNODEID_BYTESTRING_ALLOC(UA_UInt16 nsIndex, const char *chars) {
  373. UA_ExpandedNodeId id; id.nodeId = UA_NODEID_BYTESTRING_ALLOC(nsIndex, chars);
  374. id.serverIndex = 0; id.namespaceUri = UA_STRING_NULL; return id;
  375. }
  376. /**
  377. * QualifiedName
  378. * ^^^^^^^^^^^^^
  379. * A name qualified by a namespace. */
  380. typedef struct {
  381. UA_UInt16 namespaceIndex;
  382. UA_String name;
  383. } UA_QualifiedName;
  384. static UA_INLINE UA_Boolean
  385. UA_QualifiedName_isNull(const UA_QualifiedName *q) {
  386. return (q->namespaceIndex == 0 && q->name.length == 0);
  387. }
  388. static UA_INLINE UA_QualifiedName
  389. UA_QUALIFIEDNAME(UA_UInt16 nsIndex, char *chars) {
  390. UA_QualifiedName qn; qn.namespaceIndex = nsIndex;
  391. qn.name = UA_STRING(chars); return qn;
  392. }
  393. static UA_INLINE UA_QualifiedName
  394. UA_QUALIFIEDNAME_ALLOC(UA_UInt16 nsIndex, const char *chars) {
  395. UA_QualifiedName qn; qn.namespaceIndex = nsIndex;
  396. qn.name = UA_STRING_ALLOC(chars); return qn;
  397. }
  398. /**
  399. * LocalizedText
  400. * ^^^^^^^^^^^^^
  401. * Human readable text with an optional locale identifier. */
  402. typedef struct {
  403. UA_String locale;
  404. UA_String text;
  405. } UA_LocalizedText;
  406. static UA_INLINE UA_LocalizedText
  407. UA_LOCALIZEDTEXT(char *locale, char *text) {
  408. UA_LocalizedText lt; lt.locale = UA_STRING(locale);
  409. lt.text = UA_STRING(text); return lt;
  410. }
  411. static UA_INLINE UA_LocalizedText
  412. UA_LOCALIZEDTEXT_ALLOC(const char *locale, const char *text) {
  413. UA_LocalizedText lt; lt.locale = UA_STRING_ALLOC(locale);
  414. lt.text = UA_STRING_ALLOC(text); return lt;
  415. }
  416. /**
  417. * ExtensionObject
  418. * ^^^^^^^^^^^^^^^
  419. * ExtensionObjects may contain scalars of any data type. Even those that are
  420. * unknown to the receiver. See the Section `Generic Type Handling`_ on how
  421. * types are described. An ExtensionObject always contains the NodeId of the
  422. * Data Type. If the data cannot be decoded, we keep the encoded string and the
  423. * NodeId. */
  424. typedef struct {
  425. enum {
  426. UA_EXTENSIONOBJECT_ENCODED_NOBODY = 0,
  427. UA_EXTENSIONOBJECT_ENCODED_BYTESTRING = 1,
  428. UA_EXTENSIONOBJECT_ENCODED_XML = 2,
  429. UA_EXTENSIONOBJECT_DECODED = 3,
  430. UA_EXTENSIONOBJECT_DECODED_NODELETE = 4 /* Don't delete the content
  431. together with the
  432. ExtensionObject */
  433. } encoding;
  434. union {
  435. struct {
  436. UA_NodeId typeId; /* The nodeid of the datatype */
  437. UA_ByteString body; /* The bytestring of the encoded data */
  438. } encoded;
  439. struct {
  440. const UA_DataType *type;
  441. void *data;
  442. } decoded;
  443. } content;
  444. } UA_ExtensionObject;
  445. /**
  446. * .. _variant:
  447. *
  448. * Variant
  449. * ^^^^^^^
  450. * Variants may contain data of any type. See the Section `Generic Type
  451. * Handling`_ on how types are described. If the data is not of one of the 25
  452. * builtin types, it will be encoded as an `ExtensionObject`_ on the wire. (The
  453. * standard says that a variant is a union of the built-in types. open62541
  454. * generalizes this to any data type by transparently de- and encoding
  455. * ExtensionObjects in the background. If the decoding fails, the variant
  456. * contains the original ExtensionObject.)
  457. *
  458. * Variants can contain a single scalar or an array. For details on the handling
  459. * of arrays, see the Section `Array Handling`_. Array variants can have an
  460. * additional dimensionality (matrix, 3-tensor, ...) defined in an array of
  461. * dimension sizes. Higher rank dimensions are serialized first.
  462. *
  463. * The differentiation between variants containing a scalar, an array or no data
  464. * is as follows:
  465. *
  466. * - arrayLength == 0 && data == NULL: no existing data
  467. * - arrayLength == 0 && data == UA_EMPTY_ARRAY_SENTINEL: array of length 0
  468. * - arrayLength == 0 && data > UA_EMPTY_ARRAY_SENTINEL: scalar value
  469. * - arrayLength > 0: array of the given length */
  470. typedef struct {
  471. const UA_DataType *type; /* The data type description */
  472. enum {
  473. UA_VARIANT_DATA, /* The data has the same lifecycle as the
  474. variant */
  475. UA_VARIANT_DATA_NODELETE, /* The data is "borrowed" by the variant and
  476. shall not be deleted at the end of the
  477. variant's lifecycle. */
  478. } storageType;
  479. size_t arrayLength; /* The number of elements in the data array */
  480. void *data; /* Points to the scalar or array data */
  481. size_t arrayDimensionsSize; /* The number of dimensions the data-array has */
  482. UA_Int32 *arrayDimensions; /* The length of each dimension */
  483. } UA_Variant;
  484. /* Returns true if the variant contains a scalar value. Note that empty variants
  485. * contain an array of length -1 (undefined).
  486. *
  487. * @param v The variant
  488. * @return Does the variant contain a scalar value. */
  489. static UA_INLINE UA_Boolean
  490. UA_Variant_isScalar(const UA_Variant *v) {
  491. return (v->arrayLength == 0 && v->data > UA_EMPTY_ARRAY_SENTINEL);
  492. }
  493. /* Set the variant to a scalar value that already resides in memory. The value
  494. * takes on the lifecycle of the variant and is deleted with it.
  495. *
  496. * @param v The variant
  497. * @param p A pointer to the value data
  498. * @param type The datatype of the value in question */
  499. void UA_EXPORT
  500. UA_Variant_setScalar(UA_Variant *v, void * UA_RESTRICT p,
  501. const UA_DataType *type);
  502. /* Set the variant to a scalar value that is copied from an existing variable.
  503. * @param v The variant
  504. * @param p A pointer to the value data
  505. * @param type The datatype of the value
  506. * @return Indicates whether the operation succeeded or returns an error code */
  507. UA_StatusCode UA_EXPORT
  508. UA_Variant_setScalarCopy(UA_Variant *v, const void *p,
  509. const UA_DataType *type);
  510. /* Set the variant to an array that already resides in memory. The array takes
  511. * on the lifecycle of the variant and is deleted with it.
  512. *
  513. * @param v The variant
  514. * @param array A pointer to the array data
  515. * @param arraySize The size of the array
  516. * @param type The datatype of the array */
  517. void UA_EXPORT
  518. UA_Variant_setArray(UA_Variant *v, void * UA_RESTRICT array,
  519. size_t arraySize, const UA_DataType *type);
  520. /* Set the variant to an array that is copied from an existing array.
  521. *
  522. * @param v The variant
  523. * @param array A pointer to the array data
  524. * @param arraySize The size of the array
  525. * @param type The datatype of the array
  526. * @return Indicates whether the operation succeeded or returns an error code */
  527. UA_StatusCode UA_EXPORT
  528. UA_Variant_setArrayCopy(UA_Variant *v, const void *array,
  529. size_t arraySize, const UA_DataType *type);
  530. /**
  531. * NumericRanges are used to indicate subsets of a (multidimensional) variant
  532. * array. NumericRange has no official type structure in the standard. On the
  533. * wire, it only exists as an encoded string, such as "1:2,0:3,5". The colon
  534. * separates min/max index and the comma separates dimensions. A single value
  535. * indicates a range with a single element (min==max). */
  536. typedef struct {
  537. size_t dimensionsSize;
  538. struct UA_NumericRangeDimension {
  539. UA_UInt32 min;
  540. UA_UInt32 max;
  541. } *dimensions;
  542. } UA_NumericRange;
  543. /* Copy the variant, but use only a subset of the (multidimensional) array into
  544. * a variant. Returns an error code if the variant is not an array or if the
  545. * indicated range does not fit.
  546. *
  547. * @param src The source variant
  548. * @param dst The target variant
  549. * @param range The range of the copied data
  550. * @return Returns UA_STATUSCODE_GOOD or an error code */
  551. UA_StatusCode UA_EXPORT
  552. UA_Variant_copyRange(const UA_Variant *src, UA_Variant *dst,
  553. const UA_NumericRange range);
  554. /* Insert a range of data into an existing variant. The data array can't be
  555. * reused afterwards if it contains types without a fixed size (e.g. strings)
  556. * since the members are moved into the variant and take on its lifecycle.
  557. *
  558. * @param v The variant
  559. * @param dataArray The data array. The type must match the variant
  560. * @param dataArraySize The length of the data array. This is checked to match
  561. * the range size.
  562. * @param range The range of where the new data is inserted
  563. * @return Returns UA_STATUSCODE_GOOD or an error code */
  564. UA_StatusCode UA_EXPORT
  565. UA_Variant_setRange(UA_Variant *v, void * UA_RESTRICT array,
  566. size_t arraySize, const UA_NumericRange range);
  567. /* Deep-copy a range of data into an existing variant.
  568. *
  569. * @param v The variant
  570. * @param dataArray The data array. The type must match the variant
  571. * @param dataArraySize The length of the data array. This is checked to match
  572. * the range size.
  573. * @param range The range of where the new data is inserted
  574. * @return Returns UA_STATUSCODE_GOOD or an error code */
  575. UA_StatusCode UA_EXPORT
  576. UA_Variant_setRangeCopy(UA_Variant *v, const void *array,
  577. size_t arraySize, const UA_NumericRange range);
  578. /**
  579. * DataValue
  580. * ^^^^^^^^^
  581. * A data value with an associated status code and timestamps. */
  582. typedef struct {
  583. UA_Boolean hasValue : 1;
  584. UA_Boolean hasStatus : 1;
  585. UA_Boolean hasSourceTimestamp : 1;
  586. UA_Boolean hasServerTimestamp : 1;
  587. UA_Boolean hasSourcePicoseconds : 1;
  588. UA_Boolean hasServerPicoseconds : 1;
  589. UA_Variant value;
  590. UA_StatusCode status;
  591. UA_DateTime sourceTimestamp;
  592. UA_UInt16 sourcePicoseconds;
  593. UA_DateTime serverTimestamp;
  594. UA_UInt16 serverPicoseconds;
  595. } UA_DataValue;
  596. /**
  597. * DiagnosticInfo
  598. * ^^^^^^^^^^^^^^
  599. * A structure that contains detailed error and diagnostic information
  600. * associated with a StatusCode. */
  601. typedef struct UA_DiagnosticInfo {
  602. UA_Boolean hasSymbolicId : 1;
  603. UA_Boolean hasNamespaceUri : 1;
  604. UA_Boolean hasLocalizedText : 1;
  605. UA_Boolean hasLocale : 1;
  606. UA_Boolean hasAdditionalInfo : 1;
  607. UA_Boolean hasInnerStatusCode : 1;
  608. UA_Boolean hasInnerDiagnosticInfo : 1;
  609. UA_Int32 symbolicId;
  610. UA_Int32 namespaceUri;
  611. UA_Int32 localizedText;
  612. UA_Int32 locale;
  613. UA_String additionalInfo;
  614. UA_StatusCode innerStatusCode;
  615. struct UA_DiagnosticInfo *innerDiagnosticInfo;
  616. } UA_DiagnosticInfo;
  617. /**
  618. * Generic Type Handling
  619. * ---------------------
  620. * The builtin types can be combined to data structures. All information about a
  621. * (structured) data type is stored in a ``UA_DataType``. The array ``UA_TYPES``
  622. * contains the description of all standard-defined types and is used for
  623. * handling of generic types. */
  624. typedef struct {
  625. #ifdef UA_ENABLE_TYPENAMES
  626. const char *memberName;
  627. #endif
  628. UA_UInt16 memberTypeIndex; /* Index of the member in the array of data
  629. types */
  630. UA_Byte padding; /* How much padding is there before this
  631. member element? For arrays this is the
  632. padding before the size_t lenght member.
  633. (No padding between size_t and the
  634. following ptr.) */
  635. UA_Boolean namespaceZero : 1; /* The type of the member is defined in
  636. namespace zero. In this implementation,
  637. types from custom namespace may contain
  638. members from the same namespace or ns0
  639. only.*/
  640. UA_Boolean isArray : 1; /* The member is an array */
  641. } UA_DataTypeMember;
  642. struct UA_DataType {
  643. #ifdef UA_ENABLE_TYPENAMES
  644. const char *typeName;
  645. #endif
  646. UA_NodeId typeId; /* The nodeid of the type */
  647. UA_UInt16 memSize; /* Size of the struct in memory */
  648. UA_UInt16 typeIndex; /* Index of the type in the datatypetable */
  649. UA_Byte membersSize; /* How many members does the type have? */
  650. UA_Boolean builtin : 1; /* The type is "builtin" and has dedicated de-
  651. and encoding functions */
  652. UA_Boolean fixedSize : 1; /* The type (and its members) contains no
  653. pointers */
  654. UA_Boolean overlayable : 1; /* The type has the identical memory layout in
  655. memory and on the binary stream. */
  656. UA_DataTypeMember *members;
  657. };
  658. /** The following functions are used for generic handling of data types. */
  659. /* Allocates and initializes a variable of type dataType
  660. *
  661. * @param type The datatype description
  662. * @return Returns the memory location of the variable or (void*)0 if no
  663. * memory is available */
  664. void UA_EXPORT * UA_new(const UA_DataType *type) UA_FUNC_ATTR_MALLOC;
  665. /* Initializes a variable to default values
  666. *
  667. * @param p The memory location of the variable
  668. * @param type The datatype description */
  669. static UA_INLINE void
  670. UA_init(void *p, const UA_DataType *type) {
  671. memset(p, 0, type->memSize);
  672. }
  673. /* Copies the content of two variables. If copying fails (e.g. because no memory
  674. * was available for an array), then dst is emptied and initialized to prevent
  675. * memory leaks.
  676. *
  677. * @param src The memory location of the source variable
  678. * @param dst The memory location of the destination variable
  679. * @param type The datatype description
  680. * @return Indicates whether the operation succeeded or returns an error code */
  681. UA_StatusCode UA_EXPORT
  682. UA_copy(const void *src, void *dst, const UA_DataType *type);
  683. /* Deletes the dynamically allocated content of a variable (e.g. resets all
  684. * arrays to undefined arrays). Afterwards, the variable can be safely deleted
  685. * without causing memory leaks. But the variable is not initialized and may
  686. * contain old data that is not memory-relevant.
  687. *
  688. * @param p The memory location of the variable
  689. * @param type The datatype description of the variable */
  690. void UA_EXPORT UA_deleteMembers(void *p, const UA_DataType *type);
  691. /* Frees a variable and all of its content.
  692. *
  693. * @param p The memory location of the variable
  694. * @param type The datatype description of the variable */
  695. void UA_EXPORT UA_delete(void *p, const UA_DataType *type);
  696. /**
  697. * Random Number Generator
  698. * -----------------------
  699. * If UA_ENABLE_MULTITHREADING is defined, then the seed is stored in thread
  700. * local storage. The seed is initialized for every thread in the
  701. * server/client. */
  702. void UA_EXPORT UA_random_seed(UA_UInt64 seed);
  703. UA_UInt32 UA_EXPORT UA_UInt32_random(void); /* no cryptographic entropy */
  704. UA_Guid UA_EXPORT UA_Guid_random(void); /* no cryptographic entropy */
  705. #ifdef __cplusplus
  706. } // extern "C"
  707. #endif
  708. #endif /* UA_TYPES_H_ */