libamxo  4.3.4
Object Definition Language (ODL) parsing
test_import_resolver.c
Go to the documentation of this file.
1 /****************************************************************************
2 **
3 ** SPDX-License-Identifier: BSD-2-Clause-Patent
4 **
5 ** SPDX-FileCopyrightText: Copyright (c) 2023 SoftAtHome
6 **
7 ** Redistribution and use in source and binary forms, with or without modification,
8 ** are permitted provided that the following conditions are met:
9 **
10 ** 1. Redistributions of source code must retain the above copyright notice,
11 ** this list of conditions and the following disclaimer.
12 **
13 ** 2. Redistributions in binary form must reproduce the above copyright notice,
14 ** this list of conditions and the following disclaimer in the documentation
15 ** and/or other materials provided with the distribution.
16 **
17 ** Subject to the terms and conditions of this license, each copyright holder
18 ** and contributor hereby grants to those receiving rights under this license
19 ** a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable
20 ** (except for failure to satisfy the conditions of this license) patent license
21 ** to make, have made, use, offer to sell, sell, import, and otherwise transfer
22 ** this software, where such license applies only to those patent claims, already
23 ** acquired or hereafter acquired, licensable by such copyright holder or contributor
24 ** that are necessarily infringed by:
25 **
26 ** (a) their Contribution(s) (the licensed copyrights of copyright holders and
27 ** non-copyrightable additions of contributors, in source or binary form) alone;
28 ** or
29 **
30 ** (b) combination of their Contribution(s) with the work of authorship to which
31 ** such Contribution(s) was added by such copyright holder or contributor, if,
32 ** at the time the Contribution is added, such addition causes such combination
33 ** to be necessarily infringed. The patent license shall not apply to any other
34 ** combinations which include the Contribution.
35 **
36 ** Except as expressly stated above, no rights or licenses from any copyright
37 ** holder or contributor is granted under this license, whether expressly, by
38 ** implication, estoppel or otherwise.
39 **
40 ** DISCLAIMER
41 **
42 ** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
43 ** AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
44 ** IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
45 ** ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE
46 ** LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
47 ** DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
48 ** SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
49 ** CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
50 ** OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
51 ** USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
52 **
53 ****************************************************************************/
54 #include <sys/time.h>
55 #include <sys/resource.h>
56 #include <sys/types.h>
57 #include <sys/stat.h>
58 
59 #include <string.h>
60 #include <stdlib.h>
61 #include <stdio.h>
62 #include <stdarg.h>
63 #include <stddef.h>
64 #include <setjmp.h>
65 #include <inttypes.h>
66 #include <limits.h>
67 #include <unistd.h>
68 #include <fcntl.h>
69 #include <cmocka.h>
70 #include <dlfcn.h>
71 
72 #include <amxc/amxc.h>
73 #include <amxp/amxp_signal.h>
74 #include <amxd/amxd_dm.h>
75 #include <amxd/amxd_object.h>
76 #include <amxd/amxd_parameter.h>
77 #include <amxo/amxo.h>
78 #include <amxo/amxo_hooks.h>
79 
80 #include "test_import_resolver.h"
81 
82 #include <amxc/amxc_macros.h>
83 static void check_can_invoke_functions(amxd_dm_t* dm) {
84  amxc_var_t args;
85  amxc_var_t ret;
86 
87  amxd_object_t* root = amxd_dm_get_root(dm);
88  amxd_object_t* object = amxd_object_get_child(root, "TestObject");
89 
90  amxc_var_init(&args);
91  amxc_var_init(&ret);
92  amxc_var_set_type(&args, AMXC_VAR_ID_HTABLE);
93 
94  assert_ptr_not_equal(object, NULL);
95 
96  assert_int_equal(amxd_object_invoke_function(object, "TestFunc1", &args, &ret), amxd_status_ok);
97  assert_int_equal(amxd_object_invoke_function(object, "TestFunc2", &args, &ret), amxd_status_ok);
98  assert_int_equal(amxd_object_invoke_function(object, "TestFunc3", &args, &ret), amxd_status_ok);
99  assert_int_equal(amxd_object_invoke_function(object, "TestFunc4", &args, &ret), amxd_status_ok);
100 
101  amxc_var_clean(&args);
102  amxc_var_clean(&ret);
103 }
104 
106  amxd_dm_t dm;
107  amxo_parser_t parser;
108 
109  amxd_dm_init(&dm);
110  amxo_parser_init(&parser);
111 
112  assert_int_equal(amxo_parser_parse_file(&parser, "test_valid2.odl", amxd_dm_get_root(&dm)), 0);
113  assert_int_equal(amxo_parser_get_status(&parser), amxd_status_ok);
114 
116 
117  amxd_dm_clean(&dm);
118  amxo_parser_clean(&parser);
120 }
121 
123  amxd_dm_t dm;
124  amxo_parser_t parser;
125  const char* odls[] = {
126  "%define { object Test { void testa()<!import::_test_func!>;} }",
127  "%define { object Test { void testb()<!import:test:_test_func:toomuch!>;} }",
128  NULL
129  };
130 
131  amxd_dm_init(&dm);
132  amxo_parser_init(&parser);
133 
134  assert_int_equal(amxo_resolver_import_open(&parser, "../test_plugin/test_plugin.so", "test", 0), 0);
135 
136  for(int i = 0; odls[i] != NULL; i++) {
137  assert_int_not_equal(amxo_parser_parse_string(&parser, odls[i], amxd_dm_get_root(&dm)), 0);
138  assert_int_equal(amxo_parser_get_status(&parser), amxd_status_file_not_found);
139  amxd_dm_clean(&dm);
140  }
141 
142  amxo_parser_clean(&parser);
143  amxd_dm_clean(&dm);
145 }
146 
148  amxd_dm_t dm;
149  amxo_parser_t parser;
150  const char* odls[] = {
151  "import \"../test_plugin/test_plugin.so\" RTLD_NOW RTLD_GLOBAL RTLD_NODELETE; %define { object TestObject { void TestFunc1(); } }",
152  NULL
153  };
154 
155  amxd_dm_init(&dm);
156  amxo_parser_init(&parser);
157 
158  for(int i = 0; odls[i] != NULL; i++) {
159  assert_int_equal(amxo_parser_parse_string(&parser, odls[i], amxd_dm_get_root(&dm)), 0);
160  assert_int_equal(amxo_parser_get_status(&parser), amxd_status_ok);
161  amxd_dm_clean(&dm);
162  amxo_parser_clean(&parser);
163  amxo_parser_init(&parser);
164  }
165 
166  amxo_parser_clean(&parser);
167  amxd_dm_clean(&dm);
169 }
170 
172  amxd_dm_t dm;
173  amxo_parser_t parser;
174  const char* odls[] = {
175  "import \"../test_plugin/test_plugin.so\" as test; %define { object TestObject { void TestFunc1(); } }",
176  "import \"../test_plugin/test_plugin.so\" as test2; %define { object TestObject { void TestFunc1(); } }",
177  "import \"../test_plugin/test_plugin.so\" as test3; %define { object TestObject { void TestFunc1(); } }",
178  "import \"../test_plugin/test_plugin.so\" as test; %define { object TestObject { void TestFunc1(); } }",
179  "%config { silent = false; import-dbg = true; } import \"../test_plugin/test_plugin.so\" as test; %define { object TestObject { void TestFunc1(); } }",
180  "%config { dir = \"test_plugin\"; myalias = \"test\"; } import \"../${dir}/test_plugin.so\" as \"${myalias}\"; %define { object TestObject { void TestFunc1(); } }",
181  "%config { thelib = \"test\"; } import \"../test_plugin/test_plugin.so\" as test; %define { object TestObject { void TestFunc1()<!import:${thelib}:data!>; } }",
182  "%config { odl-import = false; } import \"../test_plugin/test_plugin.so\" as test; %define { object TestObject { void TestFunc1(); } }",
183  NULL
184  };
185 
186  amxd_dm_init(&dm);
187  amxo_parser_init(&parser);
188 
189  assert_int_equal(amxo_resolver_import_open(&parser, "../test_plugin/test_plugin.so", "test", 0), 0);
190 
191  for(int i = 0; odls[i] != NULL; i++) {
192  assert_int_equal(amxo_parser_parse_string(&parser, odls[i], amxd_dm_get_root(&dm)), 0);
193  assert_int_equal(amxo_parser_get_status(&parser), amxd_status_ok);
194  amxd_dm_clean(&dm);
195  amxo_parser_clean(&parser);
196  amxo_parser_init(&parser);
197  }
198 
199  amxo_parser_clean(&parser);
200  amxd_dm_clean(&dm);
202 }
203 
205  amxd_dm_t dm;
206  amxo_parser_t parser;
207  const char* odls[] = {
208  "import \"./fake.so\" as test; %define { object TestObject { void TestFunc1(); } }",
209  "%config { silent = false; import-dbg = true; } import \"./fake.so\" as test; %define { object TestObject { void TestFunc1(); } }",
210  NULL
211  };
212 
213  amxd_dm_init(&dm);
214  amxo_parser_init(&parser);
215 
216  for(int i = 0; odls[i] != NULL; i++) {
217  assert_int_not_equal(amxo_parser_parse_string(&parser, odls[i], amxd_dm_get_root(&dm)), 0);
218  amxd_dm_clean(&dm);
219  amxo_parser_clean(&parser);
220  amxo_parser_init(&parser);
221  }
222 
223  amxo_parser_clean(&parser);
224  amxd_dm_clean(&dm);
226 }
227 
228 void test_can_call_entry_point(UNUSED void** state) {
229  amxd_dm_t dm;
230  amxo_parser_t parser;
231  amxc_var_t counter;
232 
233  const char* odl = "%define { entry-point test.test_entry_point; }";
234 
235  amxd_dm_init(&dm);
236  amxo_parser_init(&parser);
237 
238  amxc_var_init(&counter);
239  amxc_var_set(uint32_t, &counter, 0);
240  amxo_parser_set_config(&parser, "counter", &counter);
241 
242  assert_int_equal(amxo_resolver_import_open(&parser, "../test_plugin/test_plugin.so", "test", 0), 0);
243 
244  assert_int_equal(amxo_parser_parse_string(&parser, odl, amxd_dm_get_root(&dm)), 0);
245  assert_int_equal(amxo_parser_get_status(&parser), amxd_status_ok);
246  assert_int_equal(amxo_parser_invoke_entry_points(&parser, &dm, AMXO_START), 0);
247  assert_int_equal(amxo_parser_rinvoke_entry_points(&parser, &dm, AMXO_STOP), 0);
248 
249  assert_int_equal(amxc_var_constcast(uint32_t, amxo_parser_get_config(&parser, "counter")), 2);
250 
251  amxo_parser_clean(&parser);
252  amxd_dm_clean(&dm);
254 }
255 
257  amxd_dm_t dm;
258  amxo_parser_t parser;
259  amxc_var_t counter;
260 
261  const char* odl = "%define { entry-point test.test_failing_entry_point; entry-point test.test_entry_point; }";
262 
263  amxd_dm_init(&dm);
264  amxo_parser_init(&parser);
265 
266  amxc_var_init(&counter);
267  amxc_var_set(uint32_t, &counter, 0);
268  amxo_parser_set_config(&parser, "counter", &counter);
269 
270  assert_int_equal(amxo_resolver_import_open(&parser, "../test_plugin/test_plugin.so", "test", 0), 0);
271 
272  assert_int_equal(amxo_parser_parse_string(&parser, odl, amxd_dm_get_root(&dm)), 0);
273  assert_int_equal(amxo_parser_get_status(&parser), amxd_status_ok);
274  assert_int_equal(amxo_parser_invoke_entry_points(&parser, &dm, AMXO_START), 1);
275  assert_int_equal(amxo_parser_rinvoke_entry_points(&parser, &dm, AMXO_STOP), 1);
276 
277  assert_int_equal(amxc_var_constcast(uint32_t, amxo_parser_get_config(&parser, "counter")), 4);
278 
279  amxo_parser_clean(&parser);
280  amxd_dm_clean(&dm);
282 }
283 
285  amxd_dm_t dm;
286  amxo_parser_t parser;
287  amxc_var_t counter;
288 
289  const char* odl = "%define { entry-point test.test_entry_point; entry-point test.test_entry_point; }";
290 
291  amxd_dm_init(&dm);
292  amxo_parser_init(&parser);
293 
294  amxc_var_init(&counter);
295  amxc_var_set(uint32_t, &counter, 0);
296  amxo_parser_set_config(&parser, "counter", &counter);
297 
298  assert_int_equal(amxo_resolver_import_open(&parser, "../test_plugin/test_plugin.so", "test", 0), 0);
299 
300  assert_int_equal(amxo_parser_parse_string(&parser, odl, amxd_dm_get_root(&dm)), 0);
301  assert_int_equal(amxo_parser_get_status(&parser), amxd_status_ok);
302  assert_int_equal(amxo_parser_invoke_entry_points(&parser, &dm, AMXO_START), 0);
303  assert_int_equal(amxo_parser_rinvoke_entry_points(&parser, &dm, AMXO_STOP), 0);
304 
305  assert_int_equal(amxc_var_constcast(uint32_t, amxo_parser_get_config(&parser, "counter")), 2);
306 
307  amxo_parser_clean(&parser);
308  amxd_dm_clean(&dm);
310 }
311 
313  amxd_dm_t dm;
314  amxo_parser_t parser;
315 
316  const char* odl = "%define { entry-point test.not_existing; }";
317 
318  amxd_dm_init(&dm);
319  amxo_parser_init(&parser);
320 
321  assert_int_not_equal(amxo_parser_parse_string(&parser, odl, amxd_dm_get_root(&dm)), 0);
322  assert_int_equal(amxo_parser_get_status(&parser), amxd_status_file_not_found);
323 
324  assert_int_equal(amxo_resolver_import_open(&parser, "../test_plugin/test_plugin.so", "test", 0), 0);
325  assert_int_not_equal(amxo_parser_parse_string(&parser, odl, amxd_dm_get_root(&dm)), 0);
326  assert_int_equal(amxo_parser_get_status(&parser), amxd_status_function_not_found);
327 
328  amxo_parser_clean(&parser);
329  amxd_dm_clean(&dm);
331 }
332 
334  amxd_dm_t dm;
335  amxo_parser_t parser;
336 
337  amxd_dm_init(&dm);
338  amxo_parser_init(&parser);
339 
340  assert_int_not_equal(amxo_parser_invoke_entry_points(NULL, &dm, AMXO_START), 0);
341  assert_int_not_equal(amxo_parser_invoke_entry_points(&parser, NULL, AMXO_START), 0);
342  assert_int_equal(amxo_parser_invoke_entry_points(&parser, &dm, 666), 0);
343 
344  amxo_parser_clean(&parser);
345  amxd_dm_clean(&dm);
346 }
347 
349  amxd_dm_t dm;
350  amxo_parser_t parser;
351  const char* odl = "%config { import-dbg = true; } import \"NONE-EXISTING.so\" as fake;";
352 
353  amxd_dm_init(&dm);
354  amxo_parser_init(&parser);
355  assert_int_not_equal(amxo_resolver_import_open(&parser, "NONE-EXISTING.so", "test", 0), 0);
356  assert_int_not_equal(amxo_resolver_import_open(NULL, "NONE-EXISTING.so", "test", 0), 0);
357  assert_int_not_equal(amxo_resolver_import_open(&parser, "", "test", 0), 0);
358  assert_int_not_equal(amxo_resolver_import_open(&parser, NULL, "test", 0), 0);
359  assert_int_not_equal(amxo_resolver_import_open(&parser, "NONE-EXISTING.so", NULL, 0), 0);
360  assert_int_not_equal(amxo_resolver_import_open(&parser, "NONE-EXISTING.so", "", 0), 0);
361  assert_int_not_equal(amxo_resolver_import_open(&parser, "../test_plugin/test_plugin.so", "", 0), 0);
362  assert_int_not_equal(amxo_parser_parse_string(&parser, odl, amxd_dm_get_root(&dm)), 0);
363  assert_int_equal(amxo_parser_get_status(&parser), amxd_status_file_not_found);
364 
365  amxo_parser_clean(&parser);
366  amxd_dm_clean(&dm);
367 }
368 
370  amxd_dm_t dm;
371  amxo_parser_t parser;
372  const char* odls[] = {
373  "%config { silent = false; import-dbg = true; } import \"../test_plugin/test_plugin.so\" as test; %define { object TestObject { void FakeFunc(); } }",
374  NULL
375  };
376 
377  amxd_dm_init(&dm);
378  amxo_parser_init(&parser);
379 
380  for(int i = 0; odls[i] != NULL; i++) {
381  assert_int_equal(amxo_parser_parse_string(&parser, odls[i], amxd_dm_get_root(&dm)), 0);
382  amxd_dm_clean(&dm);
383  amxo_parser_clean(&parser);
384  amxo_parser_init(&parser);
385  }
386 
387  amxo_parser_clean(&parser);
388  amxd_dm_clean(&dm);
390 }
Ambiorix ODL parser header file.
include test_include odl
Definition: test_valid.odl:66
@ AMXO_START
Definition: amxo_types.h:207
@ AMXO_STOP
Definition: amxo_types.h:208
amxc_var_t * amxo_parser_get_config(amxo_parser_t *parser, const char *path)
Gets a configuration option.
int amxo_parser_set_config(amxo_parser_t *parser, const char *path, amxc_var_t *value)
Sets a configuration option.
int amxo_parser_parse_string(amxo_parser_t *parser, const char *text, amxd_object_t *object)
Parses a string containing a valid ODL part.
void amxo_parser_clean(amxo_parser_t *parser)
Cleans up the odl parser instance.
int amxo_parser_parse_file(amxo_parser_t *parser, const char *file_path, amxd_object_t *object)
Parses an odl file.
static amxd_status_t amxo_parser_get_status(amxo_parser_t *parser)
Get the status of the odl parser.
Definition: amxo.h:414
int amxo_parser_init(amxo_parser_t *parser)
Initializes a new odl parser instance.
int amxo_parser_rinvoke_entry_points(amxo_parser_t *parser, amxd_dm_t *dm, int reason)
Invokes all registered entry points in reversed order.
int amxo_parser_invoke_entry_points(amxo_parser_t *parser, amxd_dm_t *dm, int reason)
Invokes all registered entry points.
int amxo_resolver_import_open(amxo_parser_t *parser, const char *so_name, const char *alias, int flags)
Opens a shared object file (.so file)
void amxo_resolver_import_close_all(void)
Unloads all loaded shared objects.
The ODL parser structure.
Definition: amxo_types.h:245
void test_entry_point_only_added_once(UNUSED void **state)
void test_open_non_existing_file(UNUSED void **state)
void test_parsing_fails_when_entry_point_can_not_be_resolved(UNUSED void **state)
void test_import_resolver_import_dlopen_fails(UNUSED void **state)
void test_entry_point_invocation_continues_after_failing_entry_point(UNUSED void **state)
void test_import_resolver_resolves(UNUSED void **state)
static void check_can_invoke_functions(amxd_dm_t *dm)
void test_can_call_entry_point(UNUSED void **state)
void test_import_resolver_invalid_data(UNUSED void **state)
void test_import_resolver_can_specify_flags(UNUSED void **state)
void test_resolve_non_existing_function(UNUSED void **state)
void test_import_resolver_multiple_import(UNUSED void **state)
void test_entry_point_invoke_does_not_crash_with_invalid_args(UNUSED void **state)
#define UNUSED
Definition: test_issue_48.c:84