1e1051a39Sopenharmony_ci/*
2e1051a39Sopenharmony_ci * Copyright 2016-2022 The OpenSSL Project Authors. All Rights Reserved.
3e1051a39Sopenharmony_ci *
4e1051a39Sopenharmony_ci * Licensed under the Apache License 2.0 (the "License").  You may not use
5e1051a39Sopenharmony_ci * this file except in compliance with the License.  You can obtain a copy
6e1051a39Sopenharmony_ci * in the file LICENSE in the source distribution or at
7e1051a39Sopenharmony_ci * https://www.openssl.org/source/license.html
8e1051a39Sopenharmony_ci */
9e1051a39Sopenharmony_ci
10e1051a39Sopenharmony_ci#include "../testutil.h"
11e1051a39Sopenharmony_ci#include "output.h"
12e1051a39Sopenharmony_ci#include "tu_local.h"
13e1051a39Sopenharmony_ci
14e1051a39Sopenharmony_ci#include <string.h>
15e1051a39Sopenharmony_ci#include <assert.h>
16e1051a39Sopenharmony_ci
17e1051a39Sopenharmony_ci#include "internal/nelem.h"
18e1051a39Sopenharmony_ci#include <openssl/bio.h>
19e1051a39Sopenharmony_ci
20e1051a39Sopenharmony_ci#include "platform.h"            /* From libapps */
21e1051a39Sopenharmony_ci
22e1051a39Sopenharmony_ci#if defined(_WIN32) && !defined(__BORLANDC__)
23e1051a39Sopenharmony_ci# define strdup _strdup
24e1051a39Sopenharmony_ci#endif
25e1051a39Sopenharmony_ci
26e1051a39Sopenharmony_ci
27e1051a39Sopenharmony_ci/*
28e1051a39Sopenharmony_ci * Declares the structures needed to register each test case function.
29e1051a39Sopenharmony_ci */
30e1051a39Sopenharmony_citypedef struct test_info {
31e1051a39Sopenharmony_ci    const char *test_case_name;
32e1051a39Sopenharmony_ci    int (*test_fn) (void);
33e1051a39Sopenharmony_ci    int (*param_test_fn)(int idx);
34e1051a39Sopenharmony_ci    int num;
35e1051a39Sopenharmony_ci
36e1051a39Sopenharmony_ci    /* flags */
37e1051a39Sopenharmony_ci    int subtest:1;
38e1051a39Sopenharmony_ci} TEST_INFO;
39e1051a39Sopenharmony_ci
40e1051a39Sopenharmony_cistatic TEST_INFO all_tests[1024];
41e1051a39Sopenharmony_cistatic int num_tests = 0;
42e1051a39Sopenharmony_cistatic int show_list = 0;
43e1051a39Sopenharmony_cistatic int single_test = -1;
44e1051a39Sopenharmony_cistatic int single_iter = -1;
45e1051a39Sopenharmony_cistatic int level = 0;
46e1051a39Sopenharmony_cistatic int seed = 0;
47e1051a39Sopenharmony_cistatic int rand_order = 0;
48e1051a39Sopenharmony_ci
49e1051a39Sopenharmony_ci/*
50e1051a39Sopenharmony_ci * A parameterised test runs a loop of test cases.
51e1051a39Sopenharmony_ci * |num_test_cases| counts the total number of non-subtest test cases
52e1051a39Sopenharmony_ci * across all tests.
53e1051a39Sopenharmony_ci */
54e1051a39Sopenharmony_cistatic int num_test_cases = 0;
55e1051a39Sopenharmony_ci
56e1051a39Sopenharmony_cistatic int process_shared_options(void);
57e1051a39Sopenharmony_ci
58e1051a39Sopenharmony_ci
59e1051a39Sopenharmony_civoid add_test(const char *test_case_name, int (*test_fn) (void))
60e1051a39Sopenharmony_ci{
61e1051a39Sopenharmony_ci    assert(num_tests != OSSL_NELEM(all_tests));
62e1051a39Sopenharmony_ci    all_tests[num_tests].test_case_name = test_case_name;
63e1051a39Sopenharmony_ci    all_tests[num_tests].test_fn = test_fn;
64e1051a39Sopenharmony_ci    all_tests[num_tests].num = -1;
65e1051a39Sopenharmony_ci    ++num_tests;
66e1051a39Sopenharmony_ci    ++num_test_cases;
67e1051a39Sopenharmony_ci}
68e1051a39Sopenharmony_ci
69e1051a39Sopenharmony_civoid add_all_tests(const char *test_case_name, int(*test_fn)(int idx),
70e1051a39Sopenharmony_ci                   int num, int subtest)
71e1051a39Sopenharmony_ci{
72e1051a39Sopenharmony_ci    assert(num_tests != OSSL_NELEM(all_tests));
73e1051a39Sopenharmony_ci    all_tests[num_tests].test_case_name = test_case_name;
74e1051a39Sopenharmony_ci    all_tests[num_tests].param_test_fn = test_fn;
75e1051a39Sopenharmony_ci    all_tests[num_tests].num = num;
76e1051a39Sopenharmony_ci    all_tests[num_tests].subtest = subtest;
77e1051a39Sopenharmony_ci    ++num_tests;
78e1051a39Sopenharmony_ci    if (subtest)
79e1051a39Sopenharmony_ci        ++num_test_cases;
80e1051a39Sopenharmony_ci    else
81e1051a39Sopenharmony_ci        num_test_cases += num;
82e1051a39Sopenharmony_ci}
83e1051a39Sopenharmony_ci
84e1051a39Sopenharmony_cistatic int gcd(int a, int b)
85e1051a39Sopenharmony_ci{
86e1051a39Sopenharmony_ci    while (b != 0) {
87e1051a39Sopenharmony_ci        int t = b;
88e1051a39Sopenharmony_ci        b = a % b;
89e1051a39Sopenharmony_ci        a = t;
90e1051a39Sopenharmony_ci    }
91e1051a39Sopenharmony_ci    return a;
92e1051a39Sopenharmony_ci}
93e1051a39Sopenharmony_ci
94e1051a39Sopenharmony_cistatic void set_seed(int s)
95e1051a39Sopenharmony_ci{
96e1051a39Sopenharmony_ci    seed = s;
97e1051a39Sopenharmony_ci    if (seed <= 0)
98e1051a39Sopenharmony_ci        seed = (int)time(NULL);
99e1051a39Sopenharmony_ci    test_random_seed(seed);
100e1051a39Sopenharmony_ci}
101e1051a39Sopenharmony_ci
102e1051a39Sopenharmony_ci
103e1051a39Sopenharmony_ciint setup_test_framework(int argc, char *argv[])
104e1051a39Sopenharmony_ci{
105e1051a39Sopenharmony_ci    char *test_seed = getenv("OPENSSL_TEST_RAND_ORDER");
106e1051a39Sopenharmony_ci    char *TAP_levels = getenv("HARNESS_OSSL_LEVEL");
107e1051a39Sopenharmony_ci
108e1051a39Sopenharmony_ci    if (TAP_levels != NULL)
109e1051a39Sopenharmony_ci        level = 4 * atoi(TAP_levels);
110e1051a39Sopenharmony_ci    test_adjust_streams_tap_level(level);
111e1051a39Sopenharmony_ci    if (test_seed != NULL) {
112e1051a39Sopenharmony_ci        rand_order = 1;
113e1051a39Sopenharmony_ci        set_seed(atoi(test_seed));
114e1051a39Sopenharmony_ci    } else {
115e1051a39Sopenharmony_ci        set_seed(0);
116e1051a39Sopenharmony_ci    }
117e1051a39Sopenharmony_ci
118e1051a39Sopenharmony_ci#if defined(OPENSSL_SYS_VMS) && defined(__DECC)
119e1051a39Sopenharmony_ci    argv = copy_argv(&argc, argv);
120e1051a39Sopenharmony_ci#elif defined(_WIN32)
121e1051a39Sopenharmony_ci    /*
122e1051a39Sopenharmony_ci     * Replace argv[] with UTF-8 encoded strings.
123e1051a39Sopenharmony_ci     */
124e1051a39Sopenharmony_ci    win32_utf8argv(&argc, &argv);
125e1051a39Sopenharmony_ci#endif
126e1051a39Sopenharmony_ci
127e1051a39Sopenharmony_ci    if (!opt_init(argc, argv, test_get_options()))
128e1051a39Sopenharmony_ci        return 0;
129e1051a39Sopenharmony_ci    return 1;
130e1051a39Sopenharmony_ci}
131e1051a39Sopenharmony_ci
132e1051a39Sopenharmony_ci
133e1051a39Sopenharmony_ci/*
134e1051a39Sopenharmony_ci * This can only be called after setup() has run, since num_tests and
135e1051a39Sopenharmony_ci * all_tests[] are setup at this point
136e1051a39Sopenharmony_ci */
137e1051a39Sopenharmony_cistatic int check_single_test_params(char *name, char *testname, char *itname)
138e1051a39Sopenharmony_ci{
139e1051a39Sopenharmony_ci    if (name != NULL) {
140e1051a39Sopenharmony_ci        int i;
141e1051a39Sopenharmony_ci        for (i = 0; i < num_tests; ++i) {
142e1051a39Sopenharmony_ci            if (strcmp(name, all_tests[i].test_case_name) == 0) {
143e1051a39Sopenharmony_ci                single_test = 1 + i;
144e1051a39Sopenharmony_ci                break;
145e1051a39Sopenharmony_ci            }
146e1051a39Sopenharmony_ci        }
147e1051a39Sopenharmony_ci        if (i >= num_tests)
148e1051a39Sopenharmony_ci            single_test = atoi(name);
149e1051a39Sopenharmony_ci    }
150e1051a39Sopenharmony_ci
151e1051a39Sopenharmony_ci
152e1051a39Sopenharmony_ci    /* if only iteration is specified, assume we want the first test */
153e1051a39Sopenharmony_ci    if (single_test == -1 && single_iter != -1)
154e1051a39Sopenharmony_ci        single_test = 1;
155e1051a39Sopenharmony_ci
156e1051a39Sopenharmony_ci    if (single_test != -1) {
157e1051a39Sopenharmony_ci        if (single_test < 1 || single_test > num_tests) {
158e1051a39Sopenharmony_ci            test_printf_stderr("Invalid -%s value "
159e1051a39Sopenharmony_ci                               "(Value must be a valid test name OR a value between %d..%d)\n",
160e1051a39Sopenharmony_ci                               testname, 1, num_tests);
161e1051a39Sopenharmony_ci            return 0;
162e1051a39Sopenharmony_ci        }
163e1051a39Sopenharmony_ci    }
164e1051a39Sopenharmony_ci    if (single_iter != -1) {
165e1051a39Sopenharmony_ci        if (all_tests[single_test - 1].num == -1) {
166e1051a39Sopenharmony_ci            test_printf_stderr("-%s option is not valid for test %d:%s\n",
167e1051a39Sopenharmony_ci                               itname,
168e1051a39Sopenharmony_ci                               single_test,
169e1051a39Sopenharmony_ci                               all_tests[single_test - 1].test_case_name);
170e1051a39Sopenharmony_ci            return 0;
171e1051a39Sopenharmony_ci        } else if (single_iter < 1
172e1051a39Sopenharmony_ci                   || single_iter > all_tests[single_test - 1].num) {
173e1051a39Sopenharmony_ci            test_printf_stderr("Invalid -%s value for test %d:%s\t"
174e1051a39Sopenharmony_ci                               "(Value must be in the range %d..%d)\n",
175e1051a39Sopenharmony_ci                               itname, single_test,
176e1051a39Sopenharmony_ci                               all_tests[single_test - 1].test_case_name,
177e1051a39Sopenharmony_ci                               1, all_tests[single_test - 1].num);
178e1051a39Sopenharmony_ci            return 0;
179e1051a39Sopenharmony_ci        }
180e1051a39Sopenharmony_ci    }
181e1051a39Sopenharmony_ci    return 1;
182e1051a39Sopenharmony_ci}
183e1051a39Sopenharmony_ci
184e1051a39Sopenharmony_cistatic int process_shared_options(void)
185e1051a39Sopenharmony_ci{
186e1051a39Sopenharmony_ci    OPTION_CHOICE_DEFAULT o;
187e1051a39Sopenharmony_ci    int value;
188e1051a39Sopenharmony_ci    int ret = -1;
189e1051a39Sopenharmony_ci    char *flag_test = "";
190e1051a39Sopenharmony_ci    char *flag_iter = "";
191e1051a39Sopenharmony_ci    char *testname = NULL;
192e1051a39Sopenharmony_ci
193e1051a39Sopenharmony_ci    opt_begin();
194e1051a39Sopenharmony_ci    while ((o = opt_next()) != OPT_EOF) {
195e1051a39Sopenharmony_ci        switch (o) {
196e1051a39Sopenharmony_ci        /* Ignore any test options at this level */
197e1051a39Sopenharmony_ci        default:
198e1051a39Sopenharmony_ci            break;
199e1051a39Sopenharmony_ci        case OPT_ERR:
200e1051a39Sopenharmony_ci            return ret;
201e1051a39Sopenharmony_ci        case OPT_TEST_HELP:
202e1051a39Sopenharmony_ci            opt_help(test_get_options());
203e1051a39Sopenharmony_ci            return 0;
204e1051a39Sopenharmony_ci        case OPT_TEST_LIST:
205e1051a39Sopenharmony_ci            show_list = 1;
206e1051a39Sopenharmony_ci            break;
207e1051a39Sopenharmony_ci        case OPT_TEST_SINGLE:
208e1051a39Sopenharmony_ci            flag_test = opt_flag();
209e1051a39Sopenharmony_ci            testname = opt_arg();
210e1051a39Sopenharmony_ci            break;
211e1051a39Sopenharmony_ci        case OPT_TEST_ITERATION:
212e1051a39Sopenharmony_ci            flag_iter = opt_flag();
213e1051a39Sopenharmony_ci            if (!opt_int(opt_arg(), &single_iter))
214e1051a39Sopenharmony_ci                goto end;
215e1051a39Sopenharmony_ci            break;
216e1051a39Sopenharmony_ci        case OPT_TEST_INDENT:
217e1051a39Sopenharmony_ci            if (!opt_int(opt_arg(), &value))
218e1051a39Sopenharmony_ci                goto end;
219e1051a39Sopenharmony_ci            level = 4 * value;
220e1051a39Sopenharmony_ci            test_adjust_streams_tap_level(level);
221e1051a39Sopenharmony_ci            break;
222e1051a39Sopenharmony_ci        case OPT_TEST_SEED:
223e1051a39Sopenharmony_ci            if (!opt_int(opt_arg(), &value))
224e1051a39Sopenharmony_ci                goto end;
225e1051a39Sopenharmony_ci            set_seed(value);
226e1051a39Sopenharmony_ci            break;
227e1051a39Sopenharmony_ci        }
228e1051a39Sopenharmony_ci    }
229e1051a39Sopenharmony_ci    if (!check_single_test_params(testname, flag_test, flag_iter))
230e1051a39Sopenharmony_ci        goto end;
231e1051a39Sopenharmony_ci    ret = 1;
232e1051a39Sopenharmony_ciend:
233e1051a39Sopenharmony_ci    return ret;
234e1051a39Sopenharmony_ci}
235e1051a39Sopenharmony_ci
236e1051a39Sopenharmony_ci
237e1051a39Sopenharmony_ciint pulldown_test_framework(int ret)
238e1051a39Sopenharmony_ci{
239e1051a39Sopenharmony_ci    set_test_title(NULL);
240e1051a39Sopenharmony_ci    return ret;
241e1051a39Sopenharmony_ci}
242e1051a39Sopenharmony_ci
243e1051a39Sopenharmony_cistatic void finalize(int success)
244e1051a39Sopenharmony_ci{
245e1051a39Sopenharmony_ci    if (success)
246e1051a39Sopenharmony_ci        ERR_clear_error();
247e1051a39Sopenharmony_ci    else
248e1051a39Sopenharmony_ci        ERR_print_errors_cb(openssl_error_cb, NULL);
249e1051a39Sopenharmony_ci}
250e1051a39Sopenharmony_ci
251e1051a39Sopenharmony_cistatic char *test_title = NULL;
252e1051a39Sopenharmony_ci
253e1051a39Sopenharmony_civoid set_test_title(const char *title)
254e1051a39Sopenharmony_ci{
255e1051a39Sopenharmony_ci    free(test_title);
256e1051a39Sopenharmony_ci    test_title = title == NULL ? NULL : strdup(title);
257e1051a39Sopenharmony_ci}
258e1051a39Sopenharmony_ci
259e1051a39Sopenharmony_ciPRINTF_FORMAT(2, 3) static void test_verdict(int verdict,
260e1051a39Sopenharmony_ci                                             const char *description, ...)
261e1051a39Sopenharmony_ci{
262e1051a39Sopenharmony_ci    va_list ap;
263e1051a39Sopenharmony_ci
264e1051a39Sopenharmony_ci    test_flush_stdout();
265e1051a39Sopenharmony_ci    test_flush_stderr();
266e1051a39Sopenharmony_ci
267e1051a39Sopenharmony_ci    if (verdict == 0 && seed != 0)
268e1051a39Sopenharmony_ci        test_printf_tapout("# OPENSSL_TEST_RAND_ORDER=%d\n", seed);
269e1051a39Sopenharmony_ci    test_printf_tapout("%s ", verdict != 0 ? "ok" : "not ok");
270e1051a39Sopenharmony_ci    va_start(ap, description);
271e1051a39Sopenharmony_ci    test_vprintf_tapout(description, ap);
272e1051a39Sopenharmony_ci    va_end(ap);
273e1051a39Sopenharmony_ci    if (verdict == TEST_SKIP_CODE)
274e1051a39Sopenharmony_ci        test_printf_tapout(" # skipped");
275e1051a39Sopenharmony_ci    test_printf_tapout("\n");
276e1051a39Sopenharmony_ci    test_flush_tapout();
277e1051a39Sopenharmony_ci}
278e1051a39Sopenharmony_ci
279e1051a39Sopenharmony_ciint run_tests(const char *test_prog_name)
280e1051a39Sopenharmony_ci{
281e1051a39Sopenharmony_ci    int num_failed = 0;
282e1051a39Sopenharmony_ci    int verdict = 1;
283e1051a39Sopenharmony_ci    int ii, i, jj, j, jstep;
284e1051a39Sopenharmony_ci    int test_case_count = 0;
285e1051a39Sopenharmony_ci    int subtest_case_count = 0;
286e1051a39Sopenharmony_ci    int permute[OSSL_NELEM(all_tests)];
287e1051a39Sopenharmony_ci
288e1051a39Sopenharmony_ci    i = process_shared_options();
289e1051a39Sopenharmony_ci    if (i == 0)
290e1051a39Sopenharmony_ci        return EXIT_SUCCESS;
291e1051a39Sopenharmony_ci    if (i == -1)
292e1051a39Sopenharmony_ci        return EXIT_FAILURE;
293e1051a39Sopenharmony_ci
294e1051a39Sopenharmony_ci    if (num_tests < 1) {
295e1051a39Sopenharmony_ci        test_printf_tapout("1..0 # Skipped: %s\n", test_prog_name);
296e1051a39Sopenharmony_ci    } else if (show_list == 0 && single_test == -1) {
297e1051a39Sopenharmony_ci        if (level > 0) {
298e1051a39Sopenharmony_ci            test_printf_stdout("Subtest: %s\n", test_prog_name);
299e1051a39Sopenharmony_ci            test_flush_stdout();
300e1051a39Sopenharmony_ci        }
301e1051a39Sopenharmony_ci        test_printf_tapout("1..%d\n", num_test_cases);
302e1051a39Sopenharmony_ci    }
303e1051a39Sopenharmony_ci
304e1051a39Sopenharmony_ci    test_flush_tapout();
305e1051a39Sopenharmony_ci
306e1051a39Sopenharmony_ci    for (i = 0; i < num_tests; i++)
307e1051a39Sopenharmony_ci        permute[i] = i;
308e1051a39Sopenharmony_ci    if (rand_order != 0)
309e1051a39Sopenharmony_ci        for (i = num_tests - 1; i >= 1; i--) {
310e1051a39Sopenharmony_ci            j = test_random() % (1 + i);
311e1051a39Sopenharmony_ci            ii = permute[j];
312e1051a39Sopenharmony_ci            permute[j] = permute[i];
313e1051a39Sopenharmony_ci            permute[i] = ii;
314e1051a39Sopenharmony_ci        }
315e1051a39Sopenharmony_ci
316e1051a39Sopenharmony_ci    for (ii = 0; ii != num_tests; ++ii) {
317e1051a39Sopenharmony_ci        i = permute[ii];
318e1051a39Sopenharmony_ci
319e1051a39Sopenharmony_ci        if (single_test != -1 && ((i+1) != single_test)) {
320e1051a39Sopenharmony_ci            continue;
321e1051a39Sopenharmony_ci        }
322e1051a39Sopenharmony_ci        else if (show_list) {
323e1051a39Sopenharmony_ci            if (all_tests[i].num != -1) {
324e1051a39Sopenharmony_ci                test_printf_tapout("%d - %s (%d..%d)\n", ii + 1,
325e1051a39Sopenharmony_ci                                   all_tests[i].test_case_name, 1,
326e1051a39Sopenharmony_ci                                   all_tests[i].num);
327e1051a39Sopenharmony_ci            } else {
328e1051a39Sopenharmony_ci                test_printf_tapout("%d - %s\n", ii + 1,
329e1051a39Sopenharmony_ci                                   all_tests[i].test_case_name);
330e1051a39Sopenharmony_ci            }
331e1051a39Sopenharmony_ci            test_flush_tapout();
332e1051a39Sopenharmony_ci        } else if (all_tests[i].num == -1) {
333e1051a39Sopenharmony_ci            set_test_title(all_tests[i].test_case_name);
334e1051a39Sopenharmony_ci            ERR_clear_error();
335e1051a39Sopenharmony_ci            verdict = all_tests[i].test_fn();
336e1051a39Sopenharmony_ci            finalize(verdict != 0);
337e1051a39Sopenharmony_ci            test_verdict(verdict, "%d - %s", test_case_count + 1, test_title);
338e1051a39Sopenharmony_ci            if (verdict == 0)
339e1051a39Sopenharmony_ci                num_failed++;
340e1051a39Sopenharmony_ci            test_case_count++;
341e1051a39Sopenharmony_ci        } else {
342e1051a39Sopenharmony_ci            verdict = TEST_SKIP_CODE;
343e1051a39Sopenharmony_ci            set_test_title(all_tests[i].test_case_name);
344e1051a39Sopenharmony_ci            if (all_tests[i].subtest) {
345e1051a39Sopenharmony_ci                level += 4;
346e1051a39Sopenharmony_ci                test_adjust_streams_tap_level(level);
347e1051a39Sopenharmony_ci                if (single_iter == -1) {
348e1051a39Sopenharmony_ci                    test_printf_stdout("Subtest: %s\n", test_title);
349e1051a39Sopenharmony_ci                    test_printf_tapout("%d..%d\n", 1, all_tests[i].num);
350e1051a39Sopenharmony_ci                    test_flush_stdout();
351e1051a39Sopenharmony_ci                    test_flush_tapout();
352e1051a39Sopenharmony_ci                }
353e1051a39Sopenharmony_ci            }
354e1051a39Sopenharmony_ci
355e1051a39Sopenharmony_ci            j = -1;
356e1051a39Sopenharmony_ci            if (rand_order == 0 || all_tests[i].num < 3)
357e1051a39Sopenharmony_ci                jstep = 1;
358e1051a39Sopenharmony_ci            else
359e1051a39Sopenharmony_ci                do
360e1051a39Sopenharmony_ci                    jstep = test_random() % all_tests[i].num;
361e1051a39Sopenharmony_ci                while (jstep == 0 || gcd(all_tests[i].num, jstep) != 1);
362e1051a39Sopenharmony_ci
363e1051a39Sopenharmony_ci            for (jj = 0; jj < all_tests[i].num; jj++) {
364e1051a39Sopenharmony_ci                int v;
365e1051a39Sopenharmony_ci
366e1051a39Sopenharmony_ci                j = (j + jstep) % all_tests[i].num;
367e1051a39Sopenharmony_ci                if (single_iter != -1 && ((jj + 1) != single_iter))
368e1051a39Sopenharmony_ci                    continue;
369e1051a39Sopenharmony_ci                ERR_clear_error();
370e1051a39Sopenharmony_ci                v = all_tests[i].param_test_fn(j);
371e1051a39Sopenharmony_ci
372e1051a39Sopenharmony_ci                if (v == 0) {
373e1051a39Sopenharmony_ci                    verdict = 0;
374e1051a39Sopenharmony_ci                } else if (v != TEST_SKIP_CODE && verdict != 0) {
375e1051a39Sopenharmony_ci                    verdict = 1;
376e1051a39Sopenharmony_ci                }
377e1051a39Sopenharmony_ci
378e1051a39Sopenharmony_ci                finalize(v != 0);
379e1051a39Sopenharmony_ci
380e1051a39Sopenharmony_ci                if (all_tests[i].subtest)
381e1051a39Sopenharmony_ci                    test_verdict(v, "%d - iteration %d",
382e1051a39Sopenharmony_ci                                 subtest_case_count + 1, j + 1);
383e1051a39Sopenharmony_ci                else
384e1051a39Sopenharmony_ci                    test_verdict(v, "%d - %s - iteration %d",
385e1051a39Sopenharmony_ci                                 test_case_count + subtest_case_count + 1,
386e1051a39Sopenharmony_ci                                 test_title, j + 1);
387e1051a39Sopenharmony_ci                subtest_case_count++;
388e1051a39Sopenharmony_ci            }
389e1051a39Sopenharmony_ci
390e1051a39Sopenharmony_ci            if (all_tests[i].subtest) {
391e1051a39Sopenharmony_ci                level -= 4;
392e1051a39Sopenharmony_ci                test_adjust_streams_tap_level(level);
393e1051a39Sopenharmony_ci            }
394e1051a39Sopenharmony_ci            if (verdict == 0)
395e1051a39Sopenharmony_ci                ++num_failed;
396e1051a39Sopenharmony_ci            if (all_tests[i].num == -1 || all_tests[i].subtest)
397e1051a39Sopenharmony_ci                test_verdict(verdict, "%d - %s", test_case_count + 1,
398e1051a39Sopenharmony_ci                             all_tests[i].test_case_name);
399e1051a39Sopenharmony_ci            test_case_count++;
400e1051a39Sopenharmony_ci        }
401e1051a39Sopenharmony_ci    }
402e1051a39Sopenharmony_ci    if (num_failed != 0)
403e1051a39Sopenharmony_ci        return EXIT_FAILURE;
404e1051a39Sopenharmony_ci    return EXIT_SUCCESS;
405e1051a39Sopenharmony_ci}
406e1051a39Sopenharmony_ci
407e1051a39Sopenharmony_ci/*
408e1051a39Sopenharmony_ci * Glue an array of strings together and return it as an allocated string.
409e1051a39Sopenharmony_ci * Optionally return the whole length of this string in |out_len|
410e1051a39Sopenharmony_ci */
411e1051a39Sopenharmony_cichar *glue_strings(const char *list[], size_t *out_len)
412e1051a39Sopenharmony_ci{
413e1051a39Sopenharmony_ci    size_t len = 0;
414e1051a39Sopenharmony_ci    char *p, *ret;
415e1051a39Sopenharmony_ci    int i;
416e1051a39Sopenharmony_ci
417e1051a39Sopenharmony_ci    for (i = 0; list[i] != NULL; i++)
418e1051a39Sopenharmony_ci        len += strlen(list[i]);
419e1051a39Sopenharmony_ci
420e1051a39Sopenharmony_ci    if (out_len != NULL)
421e1051a39Sopenharmony_ci        *out_len = len;
422e1051a39Sopenharmony_ci
423e1051a39Sopenharmony_ci    if (!TEST_ptr(ret = p = OPENSSL_malloc(len + 1)))
424e1051a39Sopenharmony_ci        return NULL;
425e1051a39Sopenharmony_ci
426e1051a39Sopenharmony_ci    for (i = 0; list[i] != NULL; i++)
427e1051a39Sopenharmony_ci        p += strlen(strcpy(p, list[i]));
428e1051a39Sopenharmony_ci
429e1051a39Sopenharmony_ci    return ret;
430e1051a39Sopenharmony_ci}
431e1051a39Sopenharmony_ci
432e1051a39Sopenharmony_cichar *test_mk_file_path(const char *dir, const char *file)
433e1051a39Sopenharmony_ci{
434e1051a39Sopenharmony_ci# ifndef OPENSSL_SYS_VMS
435e1051a39Sopenharmony_ci    const char *sep = "/";
436e1051a39Sopenharmony_ci# else
437e1051a39Sopenharmony_ci    const char *sep = "";
438e1051a39Sopenharmony_ci    char *dir_end;
439e1051a39Sopenharmony_ci    char dir_end_sep;
440e1051a39Sopenharmony_ci# endif
441e1051a39Sopenharmony_ci    size_t dirlen = dir != NULL ? strlen(dir) : 0;
442e1051a39Sopenharmony_ci    size_t len = dirlen + strlen(sep) + strlen(file) + 1;
443e1051a39Sopenharmony_ci    char *full_file = OPENSSL_zalloc(len);
444e1051a39Sopenharmony_ci
445e1051a39Sopenharmony_ci    if (full_file != NULL) {
446e1051a39Sopenharmony_ci        if (dir != NULL && dirlen > 0) {
447e1051a39Sopenharmony_ci            OPENSSL_strlcpy(full_file, dir, len);
448e1051a39Sopenharmony_ci# ifdef OPENSSL_SYS_VMS
449e1051a39Sopenharmony_ci            /*
450e1051a39Sopenharmony_ci             * If |file| contains a directory spec, we need to do some
451e1051a39Sopenharmony_ci             * careful merging.
452e1051a39Sopenharmony_ci             * "vol:[dir.dir]" + "[.certs]sm2-root.crt" should become
453e1051a39Sopenharmony_ci             * "vol:[dir.dir.certs]sm2-root.crt"
454e1051a39Sopenharmony_ci             */
455e1051a39Sopenharmony_ci            dir_end = &full_file[strlen(full_file) - 1];
456e1051a39Sopenharmony_ci            dir_end_sep = *dir_end;
457e1051a39Sopenharmony_ci            if ((dir_end_sep == ']' || dir_end_sep == '>')
458e1051a39Sopenharmony_ci                && (file[0] == '[' || file[0] == '<')) {
459e1051a39Sopenharmony_ci                file++;
460e1051a39Sopenharmony_ci                if (file[0] == '.')
461e1051a39Sopenharmony_ci                    *dir_end = '\0';
462e1051a39Sopenharmony_ci                else
463e1051a39Sopenharmony_ci                    *dir_end = '.';
464e1051a39Sopenharmony_ci            }
465e1051a39Sopenharmony_ci#else
466e1051a39Sopenharmony_ci            OPENSSL_strlcat(full_file, sep, len);
467e1051a39Sopenharmony_ci#endif
468e1051a39Sopenharmony_ci        }
469e1051a39Sopenharmony_ci        OPENSSL_strlcat(full_file, file, len);
470e1051a39Sopenharmony_ci    }
471e1051a39Sopenharmony_ci
472e1051a39Sopenharmony_ci    return full_file;
473e1051a39Sopenharmony_ci}
474