@@ -46,6 +46,7 @@ const NUMBER_OF_RETRIES: usize = 5;
46
46
struct Config {
47
47
verbose : bool ,
48
48
sequential : bool ,
49
+ batch : bool ,
49
50
bind : SocketAddr ,
50
51
}
51
52
@@ -54,6 +55,7 @@ impl Config {
54
55
Config {
55
56
verbose : false ,
56
57
sequential : false ,
58
+ batch : false ,
57
59
bind : if cfg ! ( target_os = "android" ) || cfg ! ( windows) {
58
60
( [ 0 , 0 , 0 , 0 ] , 12345 ) . into ( )
59
61
} else {
@@ -75,6 +77,7 @@ impl Config {
75
77
}
76
78
"--bind" => next_is_bind = true ,
77
79
"--sequential" => config. sequential = true ,
80
+ "--batch" => config. batch = true ,
78
81
"--verbose" | "-v" => config. verbose = true ,
79
82
"--help" | "-h" => {
80
83
show_help ( ) ;
@@ -100,6 +103,7 @@ fn show_help() {
100
103
OPTIONS:
101
104
--bind <IP>:<PORT> Specify IP address and port to listen for requests, e.g. "0.0.0.0:12345"
102
105
--sequential Run only one test at a time
106
+ --batch Send stdout and stderr in batch instead of streaming
103
107
-v, --verbose Show status messages
104
108
-h, --help Show this help screen
105
109
"# ,
@@ -280,22 +284,30 @@ fn handle_run(socket: TcpStream, work: &Path, tmp: &Path, lock: &Mutex<()>, conf
280
284
// Some tests assume RUST_TEST_TMPDIR exists
281
285
cmd. env ( "RUST_TEST_TMPDIR" , tmp. to_owned ( ) ) ;
282
286
283
- // Spawn the child and ferry over stdout/stderr to the socket in a framed
284
- // fashion (poor man's style)
285
- let mut child =
286
- t ! ( cmd. stdin( Stdio :: null( ) ) . stdout( Stdio :: piped( ) ) . stderr( Stdio :: piped( ) ) . spawn( ) ) ;
287
- drop ( lock) ;
288
- let mut stdout = child. stdout . take ( ) . unwrap ( ) ;
289
- let mut stderr = child. stderr . take ( ) . unwrap ( ) ;
290
287
let socket = Arc :: new ( Mutex :: new ( reader. into_inner ( ) ) ) ;
291
- let socket2 = socket. clone ( ) ;
292
- let thread = thread:: spawn ( move || my_copy ( & mut stdout, 0 , & * socket2) ) ;
293
- my_copy ( & mut stderr, 1 , & * socket) ;
294
- thread. join ( ) . unwrap ( ) ;
295
288
296
- // Finally send over the exit status.
297
- let status = t ! ( child. wait( ) ) ;
289
+ let status = if config. batch {
290
+ let child =
291
+ t ! ( cmd. stdin( Stdio :: null( ) ) . stdout( Stdio :: piped( ) ) . stderr( Stdio :: piped( ) ) . output( ) ) ;
292
+ batch_copy ( & child. stdout , 0 , & * socket) ;
293
+ batch_copy ( & child. stderr , 1 , & * socket) ;
294
+ child. status
295
+ } else {
296
+ // Spawn the child and ferry over stdout/stderr to the socket in a framed
297
+ // fashion (poor man's style)
298
+ let mut child =
299
+ t ! ( cmd. stdin( Stdio :: null( ) ) . stdout( Stdio :: piped( ) ) . stderr( Stdio :: piped( ) ) . spawn( ) ) ;
300
+ drop ( lock) ;
301
+ let mut stdout = child. stdout . take ( ) . unwrap ( ) ;
302
+ let mut stderr = child. stderr . take ( ) . unwrap ( ) ;
303
+ let socket2 = socket. clone ( ) ;
304
+ let thread = thread:: spawn ( move || my_copy ( & mut stdout, 0 , & * socket2) ) ;
305
+ my_copy ( & mut stderr, 1 , & * socket) ;
306
+ thread. join ( ) . unwrap ( ) ;
307
+ t ! ( child. wait( ) )
308
+ } ;
298
309
310
+ // Finally send over the exit status.
299
311
let ( which, code) = get_status_code ( & status) ;
300
312
301
313
t ! ( socket. lock( ) . unwrap( ) . write_all( & [
@@ -368,6 +380,17 @@ fn my_copy(src: &mut dyn Read, which: u8, dst: &Mutex<dyn Write>) {
368
380
}
369
381
}
370
382
383
+ fn batch_copy ( buf : & [ u8 ] , which : u8 , dst : & Mutex < dyn Write > ) {
384
+ let n = buf. len ( ) ;
385
+ let mut dst = dst. lock ( ) . unwrap ( ) ;
386
+ t ! ( dst. write_all( & [ which, ( n >> 24 ) as u8 , ( n >> 16 ) as u8 , ( n >> 8 ) as u8 , ( n >> 0 ) as u8 , ] ) ) ;
387
+ if n > 0 {
388
+ t ! ( dst. write_all( buf) ) ;
389
+ // Marking buf finished
390
+ t ! ( dst. write_all( & [ which, 0 , 0 , 0 , 0 , ] ) ) ;
391
+ }
392
+ }
393
+
371
394
fn read_u32 ( r : & mut dyn Read ) -> u32 {
372
395
let mut len = [ 0 ; 4 ] ;
373
396
t ! ( r. read_exact( & mut len) ) ;
0 commit comments