4 * Babeltrace CTF IR - Utilities
6 * Copyright 2015 Jérémie Galarneau <jeremie.galarneau@efficios.com>
8 * Author: Jérémie Galarneau <jeremie.galarneau@efficios.com>
10 * Permission is hereby granted, free of charge, to any person obtaining a copy
11 * of this software and associated documentation files (the "Software"), to deal
12 * in the Software without restriction, including without limitation the rights
13 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
14 * copies of the Software, and to permit persons to whom the Software is
15 * furnished to do so, subject to the following conditions:
17 * The above copyright notice and this permission notice shall be included in
18 * all copies or substantial portions of the Software.
20 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
21 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
23 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
24 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
25 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
29 #define BT_LOG_TAG "CTF-IR-UTILS"
30 #include <babeltrace/lib-logging-internal.h>
35 #include <babeltrace/ctf-ir/utils.h>
36 #include <babeltrace/ctf-ir/field-types.h>
37 #include <babeltrace/ref.h>
40 const char * const reserved_keywords_str
[] = {"align", "callsite",
41 "const", "char", "clock", "double", "enum", "env", "event",
42 "floating_point", "float", "integer", "int", "long", "short", "signed",
43 "stream", "string", "struct", "trace", "typealias", "typedef",
44 "unsigned", "variant", "void" "_Bool", "_Complex", "_Imaginary"};
46 static GHashTable
*reserved_keywords_set
;
50 void try_init_reserved_keywords(void)
53 const size_t reserved_keywords_count
=
54 sizeof(reserved_keywords_str
) / sizeof(char *);
56 if (reserved_keywords_set
) {
60 reserved_keywords_set
= g_hash_table_new(g_direct_hash
, g_direct_equal
);
61 assert(reserved_keywords_set
);
63 for (i
= 0; i
< reserved_keywords_count
; i
++) {
64 gpointer quark
= GINT_TO_POINTER(g_quark_from_string(
65 reserved_keywords_str
[i
]));
67 g_hash_table_insert(reserved_keywords_set
, quark
, quark
);
73 static __attribute__((destructor
))
74 void trace_finalize(void)
76 if (reserved_keywords_set
) {
77 g_hash_table_destroy(reserved_keywords_set
);
81 int bt_validate_identifier(const char *input_string
)
85 char *save_ptr
, *token
;
88 BT_LOGV_STR("Invalid parameter: input string is NULL.");
93 try_init_reserved_keywords();
95 if (input_string
[0] == '\0') {
100 string
= strdup(input_string
);
102 BT_LOGE("strdup() failed.");
107 token
= strtok_r(string
, " ", &save_ptr
);
109 if (g_hash_table_lookup_extended(reserved_keywords_set
,
110 GINT_TO_POINTER(g_quark_from_string(token
)),
116 token
= strtok_r(NULL
, " ", &save_ptr
);
123 bt_bool
bt_identifier_is_valid(const char *identifier
)
125 return bt_validate_identifier(identifier
) == 0;
129 int bt_validate_single_clock_class(struct bt_field_type
*field_type
,
130 struct bt_clock_class
**expected_clock_class
)
134 assert(expected_clock_class
);
136 switch (bt_field_type_get_type_id(field_type
)) {
137 case BT_FIELD_TYPE_ID_INTEGER
:
139 struct bt_clock_class
*mapped_clock_class
=
140 bt_field_type_integer_get_mapped_clock_class(field_type
);
142 if (!mapped_clock_class
) {
146 if (!*expected_clock_class
) {
147 /* Move reference to output parameter */
148 *expected_clock_class
= mapped_clock_class
;
149 mapped_clock_class
= NULL
;
150 BT_LOGV("Setting expected clock class: "
151 "expected-clock-class-addr=%p",
152 *expected_clock_class
);
154 if (mapped_clock_class
!= *expected_clock_class
) {
155 BT_LOGW("Integer field type is not mapped to "
156 "the expected clock class: "
157 "mapped-clock-class-addr=%p, "
158 "expected-clock-class-addr=%p",
160 *expected_clock_class
);
161 bt_put(mapped_clock_class
);
167 bt_put(mapped_clock_class
);
170 case BT_FIELD_TYPE_ID_ENUM
:
171 case BT_FIELD_TYPE_ID_ARRAY
:
172 case BT_FIELD_TYPE_ID_SEQUENCE
:
174 struct bt_field_type
*subtype
= NULL
;
176 switch (bt_field_type_get_type_id(field_type
)) {
177 case BT_FIELD_TYPE_ID_ENUM
:
178 subtype
= bt_field_type_enumeration_get_container_type(
181 case BT_FIELD_TYPE_ID_ARRAY
:
182 subtype
= bt_field_type_array_get_element_type(
185 case BT_FIELD_TYPE_ID_SEQUENCE
:
186 subtype
= bt_field_type_sequence_get_element_type(
190 BT_LOGF("Unexpected field type ID: id=%d",
191 bt_field_type_get_type_id(field_type
));
196 ret
= bt_validate_single_clock_class(subtype
,
197 expected_clock_class
);
201 case BT_FIELD_TYPE_ID_STRUCT
:
204 int64_t count
= bt_field_type_structure_get_field_count(
207 for (i
= 0; i
< count
; i
++) {
209 struct bt_field_type
*member_type
;
211 ret
= bt_field_type_structure_get_field_by_index(
212 field_type
, &name
, &member_type
, i
);
214 ret
= bt_validate_single_clock_class(member_type
,
215 expected_clock_class
);
218 BT_LOGW("Structure field type's field's type "
219 "is not recursively mapped to the "
220 "expected clock class: "
221 "field-ft-addr=%p, field-name=\"%s\"",
229 case BT_FIELD_TYPE_ID_VARIANT
:
232 int64_t count
= bt_field_type_variant_get_field_count(
235 for (i
= 0; i
< count
; i
++) {
237 struct bt_field_type
*member_type
;
239 ret
= bt_field_type_variant_get_field_by_index(
240 field_type
, &name
, &member_type
, i
);
242 ret
= bt_validate_single_clock_class(member_type
,
243 expected_clock_class
);
246 BT_LOGW("Variant field type's field's type "
247 "is not recursively mapped to the "
248 "expected clock class: "
249 "field-ft-addr=%p, field-name=\"%s\"",