@@ -99,45 +99,39 @@ auto inner_strdup(const char* str, size_t size) -> char* {
99
99
return ptr;
100
100
}
101
101
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
+ };
102
113
/*
103
114
This is a bit irresponsible as there's no checking but this is only for
104
115
testing, you have to pass the path to the executable, and then all the
105
116
arguments, argv[0] (usually but not always the name of the executable), all
106
117
arguments must be strings
107
118
*/
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 {
113
121
// 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 };
130
124
131
125
#pragma clang diagnostic push
132
126
#pragma clang diagnostic ignored \
133
127
" -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 () ));
135
129
#pragma clang diagnostic pop
136
130
137
131
_Exit (EXIT_FAILURE);
138
132
}
139
133
140
- auto array_spawn (const char * executable, char * const * argv) -> pid_t {
134
+ auto array_spawn (const char * executable, char * const * argv) -> Child {
141
135
sigset_t emptymask = 0 ;
142
136
sigemptyset (&emptymask);
143
137
struct sigaction act {};
@@ -151,7 +145,7 @@ auto array_spawn(const char* executable, char* const* argv) -> pid_t {
151
145
if (pid != 0 ) {
152
146
sigsuspend (&emptymask); // wait for child to be ready
153
147
cr_assert_neq (pid, -1 );
154
- return pid;
148
+ return Child ( pid) ;
155
149
}
156
150
157
151
#pragma clang diagnostic push
@@ -163,20 +157,6 @@ perror("child");
163
157
_Exit (EXIT_FAILURE);
164
158
}
165
159
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
-
180
160
ParameterizedTestParameters (argv_argc, successful) {
181
161
const size_t nb_params = 7 ;
182
162
@@ -215,30 +195,28 @@ ParameterizedTestParameters(argv_argc, successful) {
215
195
}
216
196
217
197
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 () );
220
200
221
201
cr_assert_no_throw (
222
202
{
223
- const Getargv::ArgvArgc results (pid);
203
+ const Getargv::ArgvArgc results (child. pid );
224
204
cr_assert_eq (results.size (), param->argc );
225
205
cr_assert_eq (results.empty (), param->argc == 0 );
226
206
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));
229
208
const int index = (i + 1 ) * -1 ;
230
209
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 ) ;
232
211
cr_expect_eq (actual, expected, " actual='%s' expected='%s'" , actual.c_str (), expected.c_str ());
233
212
}
234
213
size_t index = 0 ;
235
214
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 ) );
237
216
index ++;
238
217
}
239
218
},
240
219
std::system_error);
241
- kill (pid, SIGTERM); // signal to child to be done
242
220
}
243
221
244
222
ParameterizedTestParameters (print_argv_of_pid, successful) {
@@ -276,28 +254,30 @@ ParameterizedTestParameters(print_argv_of_pid, successful) {
276
254
}
277
255
278
256
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);
281
261
282
- cr_assert_no_throw (Getargv::Argv (pid).print (), std::system_error);
283
- std::string const actual = get_stdout ();
284
262
std::string expected;
285
263
for (int i = 0 ; i < param->argc ; i++) {
286
- const auto * arg = param->argv [i] ;
264
+ const auto * arg = param->argv . at (i) ;
287
265
expected += arg;
288
266
expected += " \0 " s;
289
267
}
290
268
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 ());
292
270
}
293
271
294
272
Test (print_argv_of_pid, failure) {
295
273
std::string const empty;
296
274
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);
298
276
277
+ // NOLINTBEGIN(cppcoreguidelines-owning-memory)
299
278
(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);
301
281
}
302
282
303
283
ParameterizedTestParameters (argv_of_pid_empty, correct) {
@@ -319,18 +299,17 @@ ParameterizedTestParameters(argv_of_pid_empty, correct) {
319
299
}
320
300
321
301
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 ());
324
303
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" );
326
305
}
327
306
328
307
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);
332
311
333
- const Getargv::Argv args (pid);
312
+ const Getargv::Argv args (child. pid );
334
313
for (int i = 0 ; i < expected.size (); i++) {
335
314
cr_assert_eq (args[i], argv[0 ][i]);
336
315
}
@@ -340,45 +319,45 @@ Test(argv_of_pid_indexing, works) {
340
319
}
341
320
342
321
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);
345
324
346
- const Getargv::Argv args (pid);
325
+ const Getargv::Argv args (child. pid );
347
326
cr_assert_throw (args[100000 ], std::out_of_range);
348
327
}
349
328
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);
352
331
353
- const Getargv::ArgvArgc args (pid);
332
+ const Getargv::ArgvArgc args (child. pid );
354
333
cr_assert_throw (args[100000 ], std::out_of_range);
355
334
}
356
335
357
336
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);
361
340
362
- const std::string actual = Getargv::Argv::as_string (pid);
341
+ const std::string actual = Getargv::Argv::as_string (child. pid );
363
342
cr_assert_eq (actual, expected);
364
343
}
365
344
366
345
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" );
368
347
369
348
cr_assert_no_throw (
370
349
{
371
- const Getargv::ArgvArgc results (pid);
350
+ const Getargv::ArgvArgc results (child. pid );
372
351
const std::vector<std::string> array = results.to_string_array ();
373
352
},
374
353
std::system_error,
375
354
" error thrown" );
376
355
}
377
356
378
357
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" );
380
359
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" );
382
361
}
383
362
384
363
Test (argv, convert_from_ffi_type) {
@@ -409,54 +388,47 @@ Test(argv_argc, not_copyable) {
409
388
410
389
Test (argv, simple) {
411
390
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 ());
413
392
414
393
cr_expect_no_throw (
415
394
{
416
- const Getargv::Argv proc_ptrs (pid, 0 , true );
395
+ const Getargv::Argv proc_ptrs (child. pid , 0 , true );
417
396
cr_assert_eq (proc_ptrs.size (), expected.size ());
418
397
const std::string actual (proc_ptrs.begin (), proc_ptrs.end ());
419
398
cr_assert_eq (actual, expected);
420
399
},
421
400
std::system_error,
422
401
" Argv constructor threw an exception when it shouldn't have" );
423
-
424
- kill (pid, SIGTERM); // signal to child to be done
425
402
}
426
403
427
404
Test (argv, nuls_false) {
428
405
const std::string expected = " one\0 two\0 three\0 " s;
429
406
430
- cleanup (kill_pid) pid_t const pid = spawn (" bin/child" , " one" , " two" , " three" );
407
+ Child child = spawn (" bin/child" , " one" , " two" , " three" );
431
408
432
409
cr_expect_no_throw (
433
410
{
434
- const Getargv::Argv proc_ptrs (pid, 0 , false );
411
+ const Getargv::Argv proc_ptrs (child. pid , 0 , false );
435
412
cr_assert_eq (proc_ptrs.size (), expected.size ());
436
413
const std::string actual (proc_ptrs.begin (), proc_ptrs.end ());
437
414
cr_assert_eq (actual, expected);
438
415
},
439
416
std::system_error,
440
417
" Argv constructor threw an exception when it shouldn't have" );
441
-
442
- kill (pid, SIGTERM); // signal to child to be done
443
418
}
444
419
445
420
Test (argv, nuls_true) {
446
421
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" );
450
423
451
424
cr_expect_no_throw (
452
425
{
453
- const Getargv::Argv proc_ptrs (pid, 0 , true );
426
+ const Getargv::Argv proc_ptrs (child. pid , 0 , true );
454
427
cr_assert_eq (proc_ptrs.size (), expected.size ());
455
428
const std::string actual (proc_ptrs.begin (), proc_ptrs.end ());
456
429
cr_assert_eq (actual, expected);
457
430
},
458
431
std::system_error);
459
- kill (pid, SIGTERM); // signal to child to be done
460
432
}
461
433
462
434
Test (argv, skip_one) {
@@ -519,9 +491,6 @@ void free_strings(struct criterion_test_params* crp) {
519
491
520
492
void free_argv_argc_test_case (struct criterion_test_params * crp) {
521
493
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
- }
525
494
cr_free (params);
526
495
}
527
496
@@ -531,7 +500,3 @@ void initialize_argv_argc_test_case(test_case* ptr) {
531
500
index = nullptr ;
532
501
}
533
502
}
534
-
535
- void kill_pid (const pid_t * pid) {
536
- kill (*pid, SIGTERM); // signal to child to be done
537
- }
0 commit comments