1*86b64dcbSAndroid Build Coastguard Worker /* -*- Mode: C; indent-tabs-mode:nil -*- */
2*86b64dcbSAndroid Build Coastguard Worker /*
3*86b64dcbSAndroid Build Coastguard Worker * Unit tests for libusb_set_option
4*86b64dcbSAndroid Build Coastguard Worker * Copyright © 2023 Nathan Hjelm <[email protected]>
5*86b64dcbSAndroid Build Coastguard Worker * Copyright © 2023 Google, LLC. All rights reserved.
6*86b64dcbSAndroid Build Coastguard Worker *
7*86b64dcbSAndroid Build Coastguard Worker * This library is free software; you can redistribute it and/or
8*86b64dcbSAndroid Build Coastguard Worker * modify it under the terms of the GNU Lesser General Public
9*86b64dcbSAndroid Build Coastguard Worker * License as published by the Free Software Foundation; either
10*86b64dcbSAndroid Build Coastguard Worker * version 2.1 of the License, or (at your option) any later version.
11*86b64dcbSAndroid Build Coastguard Worker *
12*86b64dcbSAndroid Build Coastguard Worker * This library is distributed in the hope that it will be useful,
13*86b64dcbSAndroid Build Coastguard Worker * but WITHOUT ANY WARRANTY; without even the implied warranty of
14*86b64dcbSAndroid Build Coastguard Worker * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15*86b64dcbSAndroid Build Coastguard Worker * Lesser General Public License for more details.
16*86b64dcbSAndroid Build Coastguard Worker *
17*86b64dcbSAndroid Build Coastguard Worker * You should have received a copy of the GNU Lesser General Public
18*86b64dcbSAndroid Build Coastguard Worker * License along with this library; if not, write to the Free Software
19*86b64dcbSAndroid Build Coastguard Worker * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
20*86b64dcbSAndroid Build Coastguard Worker */
21*86b64dcbSAndroid Build Coastguard Worker
22*86b64dcbSAndroid Build Coastguard Worker #include "config.h"
23*86b64dcbSAndroid Build Coastguard Worker
24*86b64dcbSAndroid Build Coastguard Worker #include <stdlib.h>
25*86b64dcbSAndroid Build Coastguard Worker #include <stdio.h>
26*86b64dcbSAndroid Build Coastguard Worker #include <inttypes.h>
27*86b64dcbSAndroid Build Coastguard Worker #include "libusbi.h"
28*86b64dcbSAndroid Build Coastguard Worker #include "libusb_testlib.h"
29*86b64dcbSAndroid Build Coastguard Worker
30*86b64dcbSAndroid Build Coastguard Worker #if defined(_WIN32) && !defined(__CYGWIN__)
31*86b64dcbSAndroid Build Coastguard Worker #include <winbase.h>
32*86b64dcbSAndroid Build Coastguard Worker
33*86b64dcbSAndroid Build Coastguard Worker #if defined(ENABLE_LOGGING)
unsetenv(const char * env)34*86b64dcbSAndroid Build Coastguard Worker static int unsetenv(const char *env) {
35*86b64dcbSAndroid Build Coastguard Worker return _putenv_s(env, "");
36*86b64dcbSAndroid Build Coastguard Worker }
37*86b64dcbSAndroid Build Coastguard Worker
setenv(const char * env,const char * value,int overwrite)38*86b64dcbSAndroid Build Coastguard Worker static int setenv(const char *env, const char *value, int overwrite) {
39*86b64dcbSAndroid Build Coastguard Worker if (getenv(env) && !overwrite)
40*86b64dcbSAndroid Build Coastguard Worker return 0;
41*86b64dcbSAndroid Build Coastguard Worker return _putenv_s(env, value);
42*86b64dcbSAndroid Build Coastguard Worker }
43*86b64dcbSAndroid Build Coastguard Worker #endif
44*86b64dcbSAndroid Build Coastguard Worker #endif
45*86b64dcbSAndroid Build Coastguard Worker
46*86b64dcbSAndroid Build Coastguard Worker #define LIBUSB_TEST_CLEAN_EXIT(code) \
47*86b64dcbSAndroid Build Coastguard Worker do { \
48*86b64dcbSAndroid Build Coastguard Worker if (test_ctx != NULL) { \
49*86b64dcbSAndroid Build Coastguard Worker libusb_exit(test_ctx); \
50*86b64dcbSAndroid Build Coastguard Worker } \
51*86b64dcbSAndroid Build Coastguard Worker unsetenv("LIBUSB_DEBUG"); \
52*86b64dcbSAndroid Build Coastguard Worker return (code); \
53*86b64dcbSAndroid Build Coastguard Worker } while (0)
54*86b64dcbSAndroid Build Coastguard Worker
55*86b64dcbSAndroid Build Coastguard Worker /**
56*86b64dcbSAndroid Build Coastguard Worker * Fail the test if the expression does not evaluate to LIBUSB_SUCCESS.
57*86b64dcbSAndroid Build Coastguard Worker */
58*86b64dcbSAndroid Build Coastguard Worker #define LIBUSB_TEST_RETURN_ON_ERROR(expr) \
59*86b64dcbSAndroid Build Coastguard Worker do { \
60*86b64dcbSAndroid Build Coastguard Worker int _result = (expr); \
61*86b64dcbSAndroid Build Coastguard Worker if (LIBUSB_SUCCESS != _result) { \
62*86b64dcbSAndroid Build Coastguard Worker libusb_testlib_logf("Not success (%s) at %s:%d", #expr, \
63*86b64dcbSAndroid Build Coastguard Worker __FILE__, __LINE__); \
64*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_CLEAN_EXIT(TEST_STATUS_FAILURE); \
65*86b64dcbSAndroid Build Coastguard Worker } \
66*86b64dcbSAndroid Build Coastguard Worker } while (0)
67*86b64dcbSAndroid Build Coastguard Worker
68*86b64dcbSAndroid Build Coastguard Worker /**
69*86b64dcbSAndroid Build Coastguard Worker * Use relational operator to compare two values and fail the test if the
70*86b64dcbSAndroid Build Coastguard Worker * comparison is false. Intended to compare integer or pointer types.
71*86b64dcbSAndroid Build Coastguard Worker *
72*86b64dcbSAndroid Build Coastguard Worker * Example: LIBUSB_EXPECT(==, 0, 1) -> fail, LIBUSB_EXPECT(==, 0, 0) -> ok.
73*86b64dcbSAndroid Build Coastguard Worker */
74*86b64dcbSAndroid Build Coastguard Worker #define LIBUSB_EXPECT(operator, lhs, rhs) \
75*86b64dcbSAndroid Build Coastguard Worker do { \
76*86b64dcbSAndroid Build Coastguard Worker int64_t _lhs = (int64_t)(intptr_t)(lhs), _rhs = (int64_t)(intptr_t)(rhs); \
77*86b64dcbSAndroid Build Coastguard Worker if (!(_lhs operator _rhs)) { \
78*86b64dcbSAndroid Build Coastguard Worker libusb_testlib_logf("Expected %s (%" PRId64 ") " #operator \
79*86b64dcbSAndroid Build Coastguard Worker " %s (%" PRId64 ") at %s:%d", #lhs, \
80*86b64dcbSAndroid Build Coastguard Worker (int64_t)(intptr_t)_lhs, #rhs, \
81*86b64dcbSAndroid Build Coastguard Worker (int64_t)(intptr_t)_rhs, __FILE__, \
82*86b64dcbSAndroid Build Coastguard Worker __LINE__); \
83*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_CLEAN_EXIT(TEST_STATUS_FAILURE); \
84*86b64dcbSAndroid Build Coastguard Worker } \
85*86b64dcbSAndroid Build Coastguard Worker } while (0)
86*86b64dcbSAndroid Build Coastguard Worker
87*86b64dcbSAndroid Build Coastguard Worker
test_set_log_level_basic(void)88*86b64dcbSAndroid Build Coastguard Worker static libusb_testlib_result test_set_log_level_basic(void) {
89*86b64dcbSAndroid Build Coastguard Worker #if defined(ENABLE_LOGGING) && !defined(ENABLE_DEBUG_LOGGING)
90*86b64dcbSAndroid Build Coastguard Worker libusb_context *test_ctx = NULL;
91*86b64dcbSAndroid Build Coastguard Worker
92*86b64dcbSAndroid Build Coastguard Worker /* unset LIBUSB_DEBUG if it is set */
93*86b64dcbSAndroid Build Coastguard Worker unsetenv("LIBUSB_DEBUG");
94*86b64dcbSAndroid Build Coastguard Worker
95*86b64dcbSAndroid Build Coastguard Worker /* test basic functionality */
96*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_init_context(&test_ctx, /*options=*/NULL,
97*86b64dcbSAndroid Build Coastguard Worker /*num_options=*/0));
98*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_set_option(test_ctx,
99*86b64dcbSAndroid Build Coastguard Worker LIBUSB_OPTION_LOG_LEVEL,
100*86b64dcbSAndroid Build Coastguard Worker LIBUSB_LOG_LEVEL_ERROR));
101*86b64dcbSAndroid Build Coastguard Worker LIBUSB_EXPECT(==, test_ctx->debug, LIBUSB_LOG_LEVEL_ERROR);
102*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_set_option(test_ctx,
103*86b64dcbSAndroid Build Coastguard Worker LIBUSB_OPTION_LOG_LEVEL,
104*86b64dcbSAndroid Build Coastguard Worker LIBUSB_LOG_LEVEL_NONE));
105*86b64dcbSAndroid Build Coastguard Worker LIBUSB_EXPECT(==, test_ctx->debug, LIBUSB_LOG_LEVEL_NONE);
106*86b64dcbSAndroid Build Coastguard Worker
107*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_CLEAN_EXIT(TEST_STATUS_SUCCESS);
108*86b64dcbSAndroid Build Coastguard Worker #else
109*86b64dcbSAndroid Build Coastguard Worker return TEST_STATUS_SKIP;
110*86b64dcbSAndroid Build Coastguard Worker #endif
111*86b64dcbSAndroid Build Coastguard Worker }
112*86b64dcbSAndroid Build Coastguard Worker
test_set_log_level_default(void)113*86b64dcbSAndroid Build Coastguard Worker static libusb_testlib_result test_set_log_level_default(void) {
114*86b64dcbSAndroid Build Coastguard Worker #if defined(ENABLE_LOGGING) && !defined(ENABLE_DEBUG_LOGGING)
115*86b64dcbSAndroid Build Coastguard Worker libusb_context *test_ctx = NULL;
116*86b64dcbSAndroid Build Coastguard Worker
117*86b64dcbSAndroid Build Coastguard Worker /* set the default debug level */
118*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_set_option(NULL, LIBUSB_OPTION_LOG_LEVEL,
119*86b64dcbSAndroid Build Coastguard Worker LIBUSB_LOG_LEVEL_ERROR));
120*86b64dcbSAndroid Build Coastguard Worker
121*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_init_context(&test_ctx, /*options=*/NULL,
122*86b64dcbSAndroid Build Coastguard Worker /*num_options=*/0));
123*86b64dcbSAndroid Build Coastguard Worker /* check that debug level came from the default */
124*86b64dcbSAndroid Build Coastguard Worker LIBUSB_EXPECT(==, test_ctx->debug, LIBUSB_LOG_LEVEL_ERROR);
125*86b64dcbSAndroid Build Coastguard Worker
126*86b64dcbSAndroid Build Coastguard Worker /* try to override the old log level. since this was set from the default it
127*86b64dcbSAndroid Build Coastguard Worker * should be possible to change it */
128*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_set_option(test_ctx,
129*86b64dcbSAndroid Build Coastguard Worker LIBUSB_OPTION_LOG_LEVEL,
130*86b64dcbSAndroid Build Coastguard Worker LIBUSB_LOG_LEVEL_NONE));
131*86b64dcbSAndroid Build Coastguard Worker LIBUSB_EXPECT(==, test_ctx->debug, LIBUSB_LOG_LEVEL_NONE);
132*86b64dcbSAndroid Build Coastguard Worker
133*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_CLEAN_EXIT(TEST_STATUS_SUCCESS);
134*86b64dcbSAndroid Build Coastguard Worker #else
135*86b64dcbSAndroid Build Coastguard Worker return TEST_STATUS_SKIP;
136*86b64dcbSAndroid Build Coastguard Worker #endif
137*86b64dcbSAndroid Build Coastguard Worker }
138*86b64dcbSAndroid Build Coastguard Worker
test_set_log_level_env(void)139*86b64dcbSAndroid Build Coastguard Worker static libusb_testlib_result test_set_log_level_env(void) {
140*86b64dcbSAndroid Build Coastguard Worker #if defined(ENABLE_LOGGING)
141*86b64dcbSAndroid Build Coastguard Worker libusb_context *test_ctx = NULL;
142*86b64dcbSAndroid Build Coastguard Worker
143*86b64dcbSAndroid Build Coastguard Worker /* check that libusb_set_option does not change the log level when it was set
144*86b64dcbSAndroid Build Coastguard Worker * from the environment. */
145*86b64dcbSAndroid Build Coastguard Worker setenv("LIBUSB_DEBUG", "4", /*overwrite=*/0);
146*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_init_context(&test_ctx, /*options=*/NULL,
147*86b64dcbSAndroid Build Coastguard Worker /*num_options=*/0));
148*86b64dcbSAndroid Build Coastguard Worker #ifndef ENABLE_DEBUG_LOGGING
149*86b64dcbSAndroid Build Coastguard Worker LIBUSB_EXPECT(==, test_ctx->debug, 4);
150*86b64dcbSAndroid Build Coastguard Worker #endif
151*86b64dcbSAndroid Build Coastguard Worker
152*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_set_option(test_ctx,
153*86b64dcbSAndroid Build Coastguard Worker LIBUSB_OPTION_LOG_LEVEL,
154*86b64dcbSAndroid Build Coastguard Worker LIBUSB_LOG_LEVEL_ERROR));
155*86b64dcbSAndroid Build Coastguard Worker /* environment variable should always override LIBUSB_OPTION_LOG_LEVEL if set */
156*86b64dcbSAndroid Build Coastguard Worker #ifndef ENABLE_DEBUG_LOGGING
157*86b64dcbSAndroid Build Coastguard Worker LIBUSB_EXPECT(==, test_ctx->debug, 4);
158*86b64dcbSAndroid Build Coastguard Worker #endif
159*86b64dcbSAndroid Build Coastguard Worker
160*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_CLEAN_EXIT(TEST_STATUS_SUCCESS);
161*86b64dcbSAndroid Build Coastguard Worker #else
162*86b64dcbSAndroid Build Coastguard Worker return TEST_STATUS_SKIP;
163*86b64dcbSAndroid Build Coastguard Worker #endif
164*86b64dcbSAndroid Build Coastguard Worker }
165*86b64dcbSAndroid Build Coastguard Worker
166*86b64dcbSAndroid Build Coastguard Worker
test_no_discovery(void)167*86b64dcbSAndroid Build Coastguard Worker static libusb_testlib_result test_no_discovery(void)
168*86b64dcbSAndroid Build Coastguard Worker {
169*86b64dcbSAndroid Build Coastguard Worker #if defined(__linux__)
170*86b64dcbSAndroid Build Coastguard Worker libusb_context *test_ctx;
171*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_init_context(&test_ctx, /*options=*/NULL,
172*86b64dcbSAndroid Build Coastguard Worker /*num_options=*/0));
173*86b64dcbSAndroid Build Coastguard Worker libusb_device **device_list = NULL;
174*86b64dcbSAndroid Build Coastguard Worker ssize_t num_devices = libusb_get_device_list(test_ctx, &device_list);
175*86b64dcbSAndroid Build Coastguard Worker libusb_free_device_list(device_list, /*unref_devices=*/1);
176*86b64dcbSAndroid Build Coastguard Worker libusb_exit(test_ctx);
177*86b64dcbSAndroid Build Coastguard Worker test_ctx = NULL;
178*86b64dcbSAndroid Build Coastguard Worker
179*86b64dcbSAndroid Build Coastguard Worker if (num_devices == 0) {
180*86b64dcbSAndroid Build Coastguard Worker libusb_testlib_logf("Warning: no devices found, the test will only verify that setting LIBUSB_OPTION_NO_DEVICE_DISCOVERY succeeds.");
181*86b64dcbSAndroid Build Coastguard Worker }
182*86b64dcbSAndroid Build Coastguard Worker
183*86b64dcbSAndroid Build Coastguard Worker LIBUSB_EXPECT(>=, num_devices, 0);
184*86b64dcbSAndroid Build Coastguard Worker
185*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_set_option(NULL, LIBUSB_OPTION_NO_DEVICE_DISCOVERY));
186*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_init_context(&test_ctx, /*options=*/NULL,
187*86b64dcbSAndroid Build Coastguard Worker /*num_options=*/0));
188*86b64dcbSAndroid Build Coastguard Worker device_list = NULL;
189*86b64dcbSAndroid Build Coastguard Worker num_devices = libusb_get_device_list(test_ctx, &device_list);
190*86b64dcbSAndroid Build Coastguard Worker libusb_free_device_list(device_list, /*unref_devices=*/1);
191*86b64dcbSAndroid Build Coastguard Worker
192*86b64dcbSAndroid Build Coastguard Worker LIBUSB_EXPECT(==, num_devices, 0);
193*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_CLEAN_EXIT(TEST_STATUS_SUCCESS);
194*86b64dcbSAndroid Build Coastguard Worker #else
195*86b64dcbSAndroid Build Coastguard Worker return TEST_STATUS_SKIP;
196*86b64dcbSAndroid Build Coastguard Worker #endif
197*86b64dcbSAndroid Build Coastguard Worker }
198*86b64dcbSAndroid Build Coastguard Worker
199*86b64dcbSAndroid Build Coastguard Worker #if defined(ENABLE_LOGGING) && !defined(ENABLE_DEBUG_LOGGING)
test_log_cb(libusb_context * ctx,enum libusb_log_level level,const char * str)200*86b64dcbSAndroid Build Coastguard Worker static void LIBUSB_CALL test_log_cb(libusb_context *ctx, enum libusb_log_level level,
201*86b64dcbSAndroid Build Coastguard Worker const char *str) {
202*86b64dcbSAndroid Build Coastguard Worker UNUSED(ctx);
203*86b64dcbSAndroid Build Coastguard Worker UNUSED(level);
204*86b64dcbSAndroid Build Coastguard Worker UNUSED(str);
205*86b64dcbSAndroid Build Coastguard Worker }
206*86b64dcbSAndroid Build Coastguard Worker #endif
207*86b64dcbSAndroid Build Coastguard Worker
208*86b64dcbSAndroid Build Coastguard Worker
test_set_log_cb(void)209*86b64dcbSAndroid Build Coastguard Worker static libusb_testlib_result test_set_log_cb(void)
210*86b64dcbSAndroid Build Coastguard Worker {
211*86b64dcbSAndroid Build Coastguard Worker #if defined(ENABLE_LOGGING) && !defined(ENABLE_DEBUG_LOGGING)
212*86b64dcbSAndroid Build Coastguard Worker libusb_context *test_ctx = NULL;
213*86b64dcbSAndroid Build Coastguard Worker
214*86b64dcbSAndroid Build Coastguard Worker /* set the log callback on the context */
215*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_init_context(&test_ctx, /*options=*/NULL,
216*86b64dcbSAndroid Build Coastguard Worker /*num_options=*/0));
217*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_set_option(test_ctx, LIBUSB_OPTION_LOG_CB,
218*86b64dcbSAndroid Build Coastguard Worker test_log_cb));
219*86b64dcbSAndroid Build Coastguard Worker
220*86b64dcbSAndroid Build Coastguard Worker /* check that debug level came from the default */
221*86b64dcbSAndroid Build Coastguard Worker LIBUSB_EXPECT(==, test_ctx->log_handler, test_log_cb);
222*86b64dcbSAndroid Build Coastguard Worker
223*86b64dcbSAndroid Build Coastguard Worker libusb_exit(test_ctx);
224*86b64dcbSAndroid Build Coastguard Worker test_ctx = NULL;
225*86b64dcbSAndroid Build Coastguard Worker
226*86b64dcbSAndroid Build Coastguard Worker /* set the log callback for all future contexts */
227*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_set_option(/*ctx=*/NULL, LIBUSB_OPTION_LOG_CB,
228*86b64dcbSAndroid Build Coastguard Worker test_log_cb));
229*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_RETURN_ON_ERROR(libusb_init_context(&test_ctx, /*options=*/NULL,
230*86b64dcbSAndroid Build Coastguard Worker /*num_options=*/0));
231*86b64dcbSAndroid Build Coastguard Worker LIBUSB_EXPECT(==, test_ctx->log_handler, test_log_cb);
232*86b64dcbSAndroid Build Coastguard Worker
233*86b64dcbSAndroid Build Coastguard Worker
234*86b64dcbSAndroid Build Coastguard Worker LIBUSB_TEST_CLEAN_EXIT(TEST_STATUS_SUCCESS);
235*86b64dcbSAndroid Build Coastguard Worker #else
236*86b64dcbSAndroid Build Coastguard Worker return TEST_STATUS_SKIP;
237*86b64dcbSAndroid Build Coastguard Worker #endif
238*86b64dcbSAndroid Build Coastguard Worker }
239*86b64dcbSAndroid Build Coastguard Worker
240*86b64dcbSAndroid Build Coastguard Worker static const libusb_testlib_test tests[] = {
241*86b64dcbSAndroid Build Coastguard Worker { "test_set_log_level_basic", &test_set_log_level_basic },
242*86b64dcbSAndroid Build Coastguard Worker { "test_set_log_level_env", &test_set_log_level_env },
243*86b64dcbSAndroid Build Coastguard Worker { "test_no_discovery", &test_no_discovery },
244*86b64dcbSAndroid Build Coastguard Worker /* since default options can't be unset, run this one last */
245*86b64dcbSAndroid Build Coastguard Worker { "test_set_log_level_default", &test_set_log_level_default },
246*86b64dcbSAndroid Build Coastguard Worker { "test_set_log_cb", &test_set_log_cb },
247*86b64dcbSAndroid Build Coastguard Worker LIBUSB_NULL_TEST
248*86b64dcbSAndroid Build Coastguard Worker };
249*86b64dcbSAndroid Build Coastguard Worker
main(int argc,char * argv[])250*86b64dcbSAndroid Build Coastguard Worker int main(int argc, char *argv[])
251*86b64dcbSAndroid Build Coastguard Worker {
252*86b64dcbSAndroid Build Coastguard Worker return libusb_testlib_run_tests(argc, argv, tests);
253*86b64dcbSAndroid Build Coastguard Worker }
254