1 /*
2 * Copyright (c) 2021 Nordic Semiconductor
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 /**
8 * @file
9 * @brief Test log links
10 *
11 */
12
13
14 #include <zephyr/tc_util.h>
15 #include <stdbool.h>
16 #include <zephyr/kernel.h>
17 #include <zephyr/ztest.h>
18 #include <zephyr/logging/log_backend.h>
19 #include <zephyr/logging/log_ctrl.h>
20 #include <zephyr/logging/log.h>
21 #include "mock_log_link.h"
22
23 LOG_MODULE_REGISTER(test);
24
process(const struct log_backend * const backend,union log_msg_generic * msg)25 static void process(const struct log_backend *const backend,
26 union log_msg_generic *msg)
27 {
28
29 }
30
panic(const struct log_backend * const backend)31 static void panic(const struct log_backend *const backend)
32 {
33
34 }
35
36 const struct log_backend_api log_backend_test_api = {
37 .process = process,
38 .panic = panic
39 };
40
41 LOG_BACKEND_DEFINE(backend1, log_backend_test_api, false);
42
43 LOG_BACKEND_DEFINE(backend2, log_backend_test_api, false);
44
log_setup(bool backend2_enable)45 static void log_setup(bool backend2_enable)
46 {
47 uint8_t offset = 0;
48
49 log_init();
50 (void)z_log_links_activate(0xFFFFFFFF, &offset);
51
52 log_backend_enable(&backend1, NULL, LOG_LEVEL_DBG);
53
54 if (backend2_enable) {
55 log_backend_enable(&backend2, NULL, LOG_LEVEL_INF);
56 } else {
57 log_backend_disable(&backend2);
58 }
59 }
60
61
62 struct mock_log_link_source domain1_sources[] = {
63 { .source = "abc", .clevel = 4, .rlevel = 4},
64 { .source = "xxx", .clevel = 3, .rlevel = 3},
65 { .source = "yyy", .clevel = 2, .rlevel = 2},
66 { .source = "zzz", .clevel = 4, .rlevel = 1},
67 };
68
69 struct mock_log_link_source domain2_sources[] = {
70 { .source = "abc2", .clevel = 2, .rlevel = 1},
71 { .source = "xxx2", .clevel = 2, .rlevel = 2},
72 { .source = "yyy2", .clevel = 3, .rlevel = 3},
73 { .source = "zzz2", .clevel = 4, .rlevel = 4},
74 };
75
76 struct mock_log_link_source domain3_sources[] = {
77 { .source = "abc", .clevel = 4, .rlevel = 4},
78 { .source = "xxx", .clevel = 3, .rlevel = 3},
79 { .source = "yyy", .clevel = 2, .rlevel = 2},
80 { .source = "zzz", .clevel = 2, .rlevel = 1},
81 };
82
83 struct mock_log_link_domain domain1 = {
84 .source_cnt = ARRAY_SIZE(domain1_sources),
85 .sources = domain1_sources,
86 .name = "domain1"
87 };
88
89 struct mock_log_link_domain domain2 = {
90 .source_cnt = ARRAY_SIZE(domain2_sources),
91 .sources = domain2_sources,
92 .name = "domain2"
93 };
94
95 struct mock_log_link_domain domain3 = {
96 .source_cnt = ARRAY_SIZE(domain3_sources),
97 .sources = domain3_sources,
98 .name = "domain3"
99 };
100
101 struct mock_log_link_domain *domains_a[] = {&domain1, &domain2};
102 struct mock_log_link mock_link_a = {
103 .domain_cnt = ARRAY_SIZE(domains_a),
104 .domains = domains_a
105 };
106
107 struct mock_log_link_domain *domains_b[] = {&domain3};
108 struct mock_log_link mock_link_b = {
109 .domain_cnt = ARRAY_SIZE(domains_b),
110 .domains = domains_b
111 };
112
113 extern struct log_link_api mock_log_link_api;
114
115 LOG_LINK_DEF(mock_link1, mock_log_link_api, 0, &mock_link_a);
116 LOG_LINK_DEF(mock_link2, mock_log_link_api, 0, &mock_link_b);
117
ZTEST(log_links,test_log_domain_count)118 ZTEST(log_links, test_log_domain_count)
119 {
120 uint8_t domains_cnt;
121 uint8_t exp_domains_cnt;
122
123 log_setup(false);
124
125 exp_domains_cnt = 1 + mock_link_a.domain_cnt + mock_link_b.domain_cnt;
126 domains_cnt = log_domains_count();
127 zassert_equal(domains_cnt, exp_domains_cnt,
128 "Unexpected number of domains (%d)", domains_cnt);
129 }
130
ZTEST(log_links,test_log_source_count)131 ZTEST(log_links, test_log_source_count)
132 {
133 uint8_t exp_source_cnt[] = {
134 log_const_source_id(TYPE_SECTION_END(log_const)),
135 /*link1*/
136 domains_a[0]->source_cnt,
137 domains_a[1]->source_cnt,
138 domains_b[0]->source_cnt,
139 };
140
141 log_setup(false);
142
143
144 for (uint8_t d = 0; d < log_domains_count(); d++) {
145 uint16_t source_cnt = log_src_cnt_get(d);
146
147 zassert_equal(source_cnt, exp_source_cnt[d],
148 "Unexpected source count (%d:%d)", d, source_cnt);
149 }
150 }
151
test_single_compile_level(uint8_t d,uint16_t s,uint32_t exp_level)152 static void test_single_compile_level(uint8_t d, uint16_t s, uint32_t exp_level)
153 {
154 uint32_t level = log_filter_get(NULL, d, s, false);
155
156 zassert_equal(level, exp_level,
157 "%d:%d Unexpected compiled level (%d vs %d)",
158 d, s, level, exp_level);
159 }
160
ZTEST(log_links,test_log_compiled_level_get)161 ZTEST(log_links, test_log_compiled_level_get)
162 {
163 log_setup(false);
164
165 test_single_compile_level(1, 0, domains_a[0]->sources[0].clevel);
166 test_single_compile_level(1, 1, domains_a[0]->sources[1].clevel);
167 test_single_compile_level(1, 3, domains_a[0]->sources[3].clevel);
168 test_single_compile_level(2, 2, domains_a[1]->sources[2].clevel);
169 test_single_compile_level(3, 2, domains_b[0]->sources[2].clevel);
170 }
171
test_single_runtime_level(uint8_t d,uint16_t s,uint8_t * link_level)172 static void test_single_runtime_level(uint8_t d, uint16_t s, uint8_t *link_level)
173 {
174 uint32_t level1;
175 uint32_t level2;
176
177 level1 = log_filter_get(&backend1, d, s, true);
178 level2 = log_filter_get(&backend2, d, s, true);
179 zassert_equal(level1, *link_level,
180 "%d:%d Unexpected compiled level (%d vs %d)",
181 d, s, level1, *link_level);
182 zassert_equal(level2, MIN(*link_level, LOG_LEVEL_INF),
183 "%d:%d Unexpected compiled level (%d vs %d)",
184 d, s, level2, MIN(*link_level, LOG_LEVEL_INF));
185
186 log_filter_set(&backend1, d, s, 1);
187 log_filter_set(&backend2, d, s, 2);
188 level1 = log_filter_get(&backend1, d, s, true);
189 level2 = log_filter_get(&backend2, d, s, true);
190 zassert_equal(level1, 1, "%d:%d Unexpected compiled level (%d vs %d)",
191 d, s, level1, 1);
192 zassert_equal(level2, 2, "%d:%d Unexpected compiled level (%d vs %d)",
193 d, s, level2, 2);
194
195 /* level set in link should be the max of both level set */
196 zassert_equal(*link_level, 2,
197 "%d:%d Unexpected compiled level (got:%d exp:%d)",
198 d, s, *link_level, 2);
199
200 }
201
ZTEST(log_links,test_log_runtime_level_set)202 ZTEST(log_links, test_log_runtime_level_set)
203 {
204 log_setup(true);
205
206 test_single_runtime_level(1, 0, &domains_a[0]->sources[0].rlevel);
207 test_single_runtime_level(1, 3, &domains_a[0]->sources[3].rlevel);
208 test_single_runtime_level(2, 1, &domains_a[1]->sources[1].rlevel);
209 test_single_runtime_level(3, 1, &domains_b[0]->sources[1].rlevel);
210 }
211
ZTEST(log_links,test_log_domain_name_get)212 ZTEST(log_links, test_log_domain_name_get)
213 {
214 zassert_str_equal(log_domain_name_get(0), "",
215 "Unexpected domain name");
216 zassert_equal(strcmp(log_domain_name_get(1), "domain1"), 0,
217 "Unexpected domain name (%s)", log_domain_name_get(1));
218 zassert_equal(strcmp(log_domain_name_get(2), "domain2"), 0,
219 "Unexpected domain name (%s)", log_domain_name_get(2));
220 zassert_equal(strcmp(log_domain_name_get(3), "domain3"), 0,
221 "Unexpected domain name (%s)", log_domain_name_get(3));
222 }
223
test_single_log_source_name_get(uint8_t d,uint16_t s,const char * exp_name)224 static void test_single_log_source_name_get(uint8_t d, uint16_t s,
225 const char *exp_name)
226 {
227 const char *name = log_source_name_get(d, s);
228
229 zassert_equal(strcmp(name, exp_name), 0, "%d:%d Unexpected source name",
230 d, s);
231 }
232
ZTEST(log_links,test_log_source_name_get)233 ZTEST(log_links, test_log_source_name_get)
234 {
235 log_setup(false);
236
237 test_single_log_source_name_get(1, 0, domains_a[0]->sources[0].source);
238 test_single_log_source_name_get(1, 1, domains_a[0]->sources[1].source);
239 test_single_log_source_name_get(2, 2, domains_a[1]->sources[2].source);
240 test_single_log_source_name_get(3, 3, domains_b[0]->sources[3].source);
241 }
242
243 ZTEST_SUITE(log_links, NULL, NULL, NULL, NULL, NULL);
244