Skip to content

Commit 28b2eb4

Browse files
committed
address some clang tidy complaints in tests
1 parent 8559b66 commit 28b2eb4

File tree

3 files changed

+71
-105
lines changed

3 files changed

+71
-105
lines changed

test/.clang-tidy

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,2 @@
1-
Checks: '*,-altera-*,-fuchsia-*,-llvmlibc-*-namespace,-cppcoreguidelines-pro-bounds-pointer-arithmetic,-llvmlibc-restrict-system-libc-headers,-portability-restrict-system-includes,-cppcoreguidelines-pro-type-cstyle-cast,-cppcoreguidelines-pro-type-vararg,-cppcoreguidelines-avoid-magic-numbers,-hicpp-vararg,-readability-magic-numbers,-concurrency-mt-unsafe,-hicpp-no-array-decay,-cppcoreguidelines-pro-bounds-array-to-pointer-decay,-cppcoreguidelines-avoid-do-while,-cppcoreguidelines-avoid-non-const-global-variables,-misc-use-anonymous-namespace,-misc-const-correctness,-readability-identifier-length,-cppcoreguidelines-pro-type-member-init,-readability-function-cognitive-complexity,-cert-err58-cpp,-hicpp-member-init-readability-simplify-boolean-expr,-modernize-use-trailing-return-type'
1+
Checks: '*,-altera-*,-fuchsia-*,-llvmlibc-*-namespace,-cppcoreguidelines-pro-bounds-pointer-arithmetic,-llvmlibc-restrict-system-libc-headers,-portability-restrict-system-includes,-cppcoreguidelines-pro-type-cstyle-cast,-cppcoreguidelines-pro-type-vararg,-cppcoreguidelines-avoid-magic-numbers,-hicpp-vararg,-readability-magic-numbers,-concurrency-mt-unsafe,-hicpp-no-array-decay,-cppcoreguidelines-pro-bounds-array-to-pointer-decay,-cppcoreguidelines-avoid-do-while,-cppcoreguidelines-avoid-non-const-global-variables,-misc-use-anonymous-namespace,-misc-const-correctness,-readability-identifier-length,-cppcoreguidelines-pro-type-member-init,-readability-function-cognitive-complexity,-cert-err58-cpp,-hicpp-member-init-readability-simplify-boolean-expr,-modernize-use-trailing-return-type,-hicpp-member-init,-cert-msc50-cpp,-cert-msc30-c'
22
HeaderFilterRegex: 'lib_unit_tests.hpp'

test/src/lib_unit_tests.cpp

+60-95
Original file line numberDiff line numberDiff line change
@@ -99,45 +99,39 @@ auto inner_strdup(const char* str, size_t size) -> char* {
9999
return ptr;
100100
}
101101

102+
struct Child {
103+
pid_t pid;
104+
explicit Child(pid_t pid): pid(pid){}
105+
Child(const Child& other) = delete;
106+
Child(const Child&& other) = delete;
107+
Child& operator=(Child&& other) = delete;
108+
Child& operator=(Child& other) = delete;
109+
~Child() {
110+
kill(this->pid, SIGTERM); // signal to child to be done
111+
}
112+
};
102113
/*
103114
This is a bit irresponsible as there's no checking but this is only for
104115
testing, you have to pass the path to the executable, and then all the
105116
arguments, argv[0] (usually but not always the name of the executable), all
106117
arguments must be strings
107118
*/
108-
auto varargs_spawn(const char* executable, ...) -> pid_t {
109-
va_list val;
110-
const char** args = nullptr;
111-
int argc = 1; // room for terminating NULL
112-
119+
template <typename... Ts>
120+
auto spawn(const char* executable, Ts... args) -> Child {
113121
// Determine number of variadic arguments
114-
va_start(val, executable);
115-
while (va_arg(val, const char*) != nullptr) {
116-
argc++;
117-
}
118-
va_end(val);
119-
// cr_log_info("%d args to spawn:\n", argc);
120-
// Allocate args, put references to command / variadic arguments + NULL in
121-
// args because last val is NULL, and we malloc'd an extra pointer, we're
122-
// good
123-
args = static_cast<const char**>(malloc(argc * sizeof(char*)));
124-
va_start(val, executable);
125-
for (int i = 0; i < argc; i++) {
126-
args[i] = va_arg(val, const char*);
127-
// cr_log_info("%s\n", args[i]);
128-
}
129-
va_end(val);
122+
const uint size = sizeof...(args) + 1;
123+
std::array<const char*, size> array = {args..., NULL};
130124

131125
#pragma clang diagnostic push
132126
#pragma clang diagnostic ignored \
133127
"-Wincompatible-pointer-types-discards-qualifiers"
134-
return array_spawn(executable, const_cast<char* const*>(args));
128+
return array_spawn(executable, const_cast<char* const*>(array.data()));
135129
#pragma clang diagnostic pop
136130

137131
_Exit(EXIT_FAILURE);
138132
}
139133

140-
auto array_spawn(const char* executable, char* const* argv) -> pid_t {
134+
auto array_spawn(const char* executable, char* const* argv) -> Child {
141135
sigset_t emptymask = 0;
142136
sigemptyset(&emptymask);
143137
struct sigaction act {};
@@ -151,7 +145,7 @@ auto array_spawn(const char* executable, char* const* argv) -> pid_t {
151145
if (pid != 0) {
152146
sigsuspend(&emptymask); // wait for child to be ready
153147
cr_assert_neq(pid, -1);
154-
return pid;
148+
return Child(pid);
155149
}
156150

157151
#pragma clang diagnostic push
@@ -163,20 +157,6 @@ perror("child");
163157
_Exit(EXIT_FAILURE);
164158
}
165159

166-
auto get_stdout() -> std::string {
167-
(void)fflush(stdout);
168-
FILE* f_stdout = cr_get_redirected_stdout();
169-
auto const size = static_cast<const size_t>(1024 * 8);
170-
char buffer[size];
171-
char* head = buffer;
172-
size_t read = 0;
173-
do {
174-
read = fread(head, 1, size - (head - buffer), f_stdout);
175-
head += read;
176-
} while (read > 0 && (head - buffer) < size);
177-
return { buffer, static_cast<size_t>(head - buffer) };
178-
}
179-
180160
ParameterizedTestParameters(argv_argc, successful) {
181161
const size_t nb_params = 7;
182162

@@ -215,30 +195,28 @@ ParameterizedTestParameters(argv_argc, successful) {
215195
}
216196

217197
ParameterizedTest(test_case* param, argv_argc, successful) {
218-
cleanup(kill_pid) pid_t const pid =
219-
array_spawn("bin/child", (char* const*)param->argv);
198+
Child const child =
199+
array_spawn("bin/child", param->argv.data());
220200

221201
cr_assert_no_throw(
222202
{
223-
const Getargv::ArgvArgc results(pid);
203+
const Getargv::ArgvArgc results(child.pid);
224204
cr_assert_eq(results.size(), param->argc);
225205
cr_assert_eq(results.empty(), param->argc == 0);
226206
for (int i = 0; i < param->argc; i++) {
227-
cr_expect_str_eq(results[i], param->argv[i], "#%d: actual='%s' expected='%s'", param->argc, results[i], param->argv[i]);
228-
207+
cr_expect_str_eq(results[i], param->argv.at(i), "#%d: actual='%s' expected='%s'", param->argc, results[i], param->argv.at(i));
229208
const int index = (i + 1) * -1;
230209
const std::string actual = results[index];
231-
const std::string expected = param->argv[(int)param->argc + index];
210+
const std::string expected = param->argv.at((int)param->argc + index);
232211
cr_expect_eq(actual, expected, "actual='%s' expected='%s'", actual.c_str(), expected.c_str());
233212
}
234213
size_t index = 0;
235214
for (auto arg : results) {
236-
cr_expect_str_eq(arg, param->argv[index], "#%d: actual='%s' expected='%s'", param->argc, arg, param->argv[index]);
215+
cr_expect_str_eq(arg, param->argv.at(index), "#%d: actual='%s' expected='%s'", param->argc, arg, param->argv.at(index));
237216
index++;
238217
}
239218
},
240219
std::system_error);
241-
kill(pid, SIGTERM); // signal to child to be done
242220
}
243221

244222
ParameterizedTestParameters(print_argv_of_pid, successful) {
@@ -276,28 +254,30 @@ ParameterizedTestParameters(print_argv_of_pid, successful) {
276254
}
277255

278256
ParameterizedTest(test_case* param, print_argv_of_pid, successful, .init = cr_redirect_stdout) {
279-
cleanup(kill_pid) pid_t const pid =
280-
array_spawn("bin/child", (char* const*)param->argv);
257+
Child const child =
258+
array_spawn("bin/child", param->argv.data());
259+
260+
cr_assert_no_throw(Getargv::Argv(child.pid).print(), std::system_error);
281261

282-
cr_assert_no_throw(Getargv::Argv(pid).print(), std::system_error);
283-
std::string const actual = get_stdout();
284262
std::string expected;
285263
for (int i = 0; i < param->argc; i++) {
286-
const auto* arg = param->argv[i];
264+
const auto* arg = param->argv.at(i);
287265
expected += arg;
288266
expected += "\0"s;
289267
}
290268

291-
cr_assert_eq(actual, expected, "actual: '%.*s'[%ld] != expected: '%.*s'[%ld]", (int)actual.size(), actual.c_str(), actual.size(), (int)expected.size(), expected.c_str(), expected.size());
269+
cr_assert_stdout_eq_str(expected.c_str());
292270
}
293271

294272
Test(print_argv_of_pid, failure) {
295273
std::string const empty;
296274
const char* argv[] = { empty.c_str(), nullptr };
297-
cleanup(kill_pid) pid_t const pid = array_spawn("bin/child", (char* const*)argv);
275+
Child const child = array_spawn("bin/child", (char* const*)argv);
298276

277+
// NOLINTBEGIN(cppcoreguidelines-owning-memory)
299278
(void)fclose(stdout);
300-
cr_assert_throw(Getargv::Argv(pid).print(), std::system_error);
279+
// NOLINTEND(cppcoreguidelines-owning-memory)
280+
cr_assert_throw(Getargv::Argv(child.pid).print(), std::system_error);
301281
}
302282

303283
ParameterizedTestParameters(argv_of_pid_empty, correct) {
@@ -319,18 +299,17 @@ ParameterizedTestParameters(argv_of_pid_empty, correct) {
319299
}
320300

321301
ParameterizedTest(test_case* param, argv_of_pid_empty, correct) {
322-
cleanup(kill_pid) pid_t const pid =
323-
array_spawn("bin/child", (char* const*)param->argv);
302+
Child const child = array_spawn("bin/child", param->argv.data());
324303

325-
cr_assert_eq(Getargv::Argv::as_bytes(pid).empty(), param->argc == 0, "empty() was wrong, should be %s", param->argc == 0 ? "true" : "false");
304+
cr_assert_eq(Getargv::Argv::as_bytes(child.pid).empty(), param->argc == 0, "empty() was wrong, should be %s", param->argc == 0 ? "true" : "false");
326305
}
327306

328307
Test(argv_of_pid_indexing, works) {
329-
const std::string expected = "abcdefghijklmnopqrstuvwxyz";
330-
const char* argv[] = { expected.c_str(), nullptr };
331-
cleanup(kill_pid) pid_t const pid = array_spawn("bin/child", (char* const*)argv);
308+
const std::string expected = "abcdefghijklmnopqrstuvwxyz";
309+
const char* argv[] = { expected.c_str(), nullptr };
310+
Child const child = array_spawn("bin/child", (char* const*)argv);
332311

333-
const Getargv::Argv args(pid);
312+
const Getargv::Argv args(child.pid);
334313
for (int i = 0; i < expected.size(); i++) {
335314
cr_assert_eq(args[i], argv[0][i]);
336315
}
@@ -340,45 +319,45 @@ Test(argv_of_pid_indexing, works) {
340319
}
341320

342321
Test(argv_of_pid_indexing, failure) {
343-
const char* argv[] = { "", nullptr };
344-
cleanup(kill_pid) pid_t const pid = array_spawn("bin/child", (char* const*)argv);
322+
const char* argv[] = { "", nullptr };
323+
Child const child = array_spawn("bin/child", (char* const*)argv);
345324

346-
const Getargv::Argv args(pid);
325+
const Getargv::Argv args(child.pid);
347326
cr_assert_throw(args[100000], std::out_of_range);
348327
}
349328
Test(argv_argc_of_pid_indexing, failure) {
350-
const char* argv[] = { "", nullptr };
351-
cleanup(kill_pid) pid_t const pid = array_spawn("bin/child", (char* const*)argv);
329+
const char* argv[] = { "", nullptr };
330+
Child const child = array_spawn("bin/child", (char* const*)argv);
352331

353-
const Getargv::ArgvArgc args(pid);
332+
const Getargv::ArgvArgc args(child.pid);
354333
cr_assert_throw(args[100000], std::out_of_range);
355334
}
356335

357336
Test(argv_as_string, works) {
358-
const std::string expected = "abcdefghijklmnopqrstuvwxyz";
359-
const char* argv[] = { expected.c_str(), nullptr };
360-
cleanup(kill_pid) pid_t const pid = array_spawn("bin/child", (char* const*)argv);
337+
const std::string expected = "abcdefghijklmnopqrstuvwxyz";
338+
const char* argv[] = { expected.c_str(), nullptr };
339+
Child const child = array_spawn("bin/child", (char* const*)argv);
361340

362-
const std::string actual = Getargv::Argv::as_string(pid);
341+
const std::string actual = Getargv::Argv::as_string(child.pid);
363342
cr_assert_eq(actual, expected);
364343
}
365344

366345
Test(argv_argc, to_string_array) {
367-
cleanup(kill_pid) pid_t const pid = spawn("bin/child", "bin/child");
346+
Child child = spawn("bin/child", "bin/child");
368347

369348
cr_assert_no_throw(
370349
{
371-
const Getargv::ArgvArgc results(pid);
350+
const Getargv::ArgvArgc results(child.pid);
372351
const std::vector<std::string> array = results.to_string_array();
373352
},
374353
std::system_error,
375354
"error thrown");
376355
}
377356

378357
Test(argv_argc, as_string_array) {
379-
cleanup(kill_pid) pid_t const pid = spawn("bin/child", "bin/child");
358+
Child const child = spawn("bin/child", "bin/child");
380359

381-
cr_assert_no_throw(Getargv::ArgvArgc::as_string_array(pid), std::system_error, "error thrown");
360+
cr_assert_no_throw(Getargv::ArgvArgc::as_string_array(child.pid), std::system_error, "error thrown");
382361
}
383362

384363
Test(argv, convert_from_ffi_type) {
@@ -409,54 +388,47 @@ Test(argv_argc, not_copyable) {
409388

410389
Test(argv, simple) {
411390
const std::string expected = "bin/child\0"s;
412-
cleanup(kill_pid) pid_t const pid = spawn(expected.c_str(), expected.c_str());
391+
Child const child = spawn(expected.c_str(), expected.c_str());
413392

414393
cr_expect_no_throw(
415394
{
416-
const Getargv::Argv proc_ptrs(pid, 0, true);
395+
const Getargv::Argv proc_ptrs(child.pid, 0, true);
417396
cr_assert_eq(proc_ptrs.size(), expected.size());
418397
const std::string actual(proc_ptrs.begin(), proc_ptrs.end());
419398
cr_assert_eq(actual, expected);
420399
},
421400
std::system_error,
422401
"Argv constructor threw an exception when it shouldn't have");
423-
424-
kill(pid, SIGTERM); // signal to child to be done
425402
}
426403

427404
Test(argv, nuls_false) {
428405
const std::string expected = "one\0two\0three\0"s;
429406

430-
cleanup(kill_pid) pid_t const pid = spawn("bin/child", "one", "two", "three");
407+
Child child = spawn("bin/child", "one", "two", "three");
431408

432409
cr_expect_no_throw(
433410
{
434-
const Getargv::Argv proc_ptrs(pid, 0, false);
411+
const Getargv::Argv proc_ptrs(child.pid, 0, false);
435412
cr_assert_eq(proc_ptrs.size(), expected.size());
436413
const std::string actual(proc_ptrs.begin(), proc_ptrs.end());
437414
cr_assert_eq(actual, expected);
438415
},
439416
std::system_error,
440417
"Argv constructor threw an exception when it shouldn't have");
441-
442-
kill(pid, SIGTERM); // signal to child to be done
443418
}
444419

445420
Test(argv, nuls_true) {
446421
const std::string expected = "bin/tests --verbose 2 -j1\0"s;
447-
448-
cleanup(kill_pid) pid_t const pid =
449-
spawn("bin/child", "bin/tests", "--verbose", "2", "-j1");
422+
Child child = spawn("bin/child", "bin/tests", "--verbose", "2", "-j1");
450423

451424
cr_expect_no_throw(
452425
{
453-
const Getargv::Argv proc_ptrs(pid, 0, true);
426+
const Getargv::Argv proc_ptrs(child.pid, 0, true);
454427
cr_assert_eq(proc_ptrs.size(), expected.size());
455428
const std::string actual(proc_ptrs.begin(), proc_ptrs.end());
456429
cr_assert_eq(actual, expected);
457430
},
458431
std::system_error);
459-
kill(pid, SIGTERM); // signal to child to be done
460432
}
461433

462434
Test(argv, skip_one) {
@@ -519,9 +491,6 @@ void free_strings(struct criterion_test_params* crp) {
519491

520492
void free_argv_argc_test_case(struct criterion_test_params* crp) {
521493
auto* params = static_cast<test_case*>(crp->params);
522-
for (size_t i = 0; i < crp->length; ++i) {
523-
cr_free(params[i].argv);
524-
}
525494
cr_free(params);
526495
}
527496

@@ -531,7 +500,3 @@ void initialize_argv_argc_test_case(test_case* ptr) {
531500
index = nullptr;
532501
}
533502
}
534-
535-
void kill_pid(const pid_t* pid) {
536-
kill(*pid, SIGTERM); // signal to child to be done
537-
}

test/src/lib_unit_tests.hpp

+10-9
Original file line numberDiff line numberDiff line change
@@ -7,8 +7,8 @@ namespace Getargv::ffi {
77
using errno_t = ::errno_t;
88
} // namespace Getargv::ffi
99

10-
#include "../../src/argv.cpp"
11-
#include "../../src/argvargc.cpp"
10+
#include "../../src/argv.cpp" // NOLINT(bugprone-suspicious-include)
11+
#include "../../src/argvargc.cpp" // NOLINT(bugprone-suspicious-include)
1212

1313
#include <criterion/criterion.h>
1414
#include <criterion/hooks.h>
@@ -24,23 +24,24 @@ namespace Getargv::ffi {
2424
#include <cstdio>
2525
#include <cstdlib>
2626

27+
struct Child;
28+
2729
auto numPlaces(int n) -> int;
2830
auto randUpTo(int n) -> int;
2931
void redirect_all_std();
3032
auto inner_strdup(const char* str, size_t size) -> char*;
3133
auto read_file(FILE* file) -> std::string;
32-
auto inner_spawn(const char* executable, ...) -> pid_t;
34+
auto inner_spawn(const char* executable, ...) -> Child;
3335
void sig_handler(int sig);
34-
auto array_spawn(const char* executable, char* const* argv) -> pid_t;
35-
auto varargs_spawn(const char* executable, ...) -> pid_t;
36+
auto array_spawn(const char* executable, char* const* argv) -> Child;
3637

37-
#define cr_strdup(str_arg) inner_strdup(str_arg, sizeof(str_arg))
38-
#define spawn(...) varargs_spawn(__VA_ARGS__, NULL)
39-
#define cleanup(callback) __attribute__((__cleanup__(callback)))
38+
constexpr char* cr_strdup(const char* str_arg) {
39+
return inner_strdup(str_arg, sizeof(str_arg));
40+
}
4041

4142
struct get_argv_and_argc_of_pid_test_case {
4243
uint argc;
43-
const char* argv[5];
44+
std::array<char*,5> argv;
4445
};
4546
using test_case = struct get_argv_and_argc_of_pid_test_case;
4647

0 commit comments

Comments
 (0)