@@ -362,7 +362,7 @@ def mutate(self):
362
362
return results
363
363
return self .random_fuzzing ()
364
364
365
- def app_fuzzing (self ) -> List [bytes ]:
365
+ def app_fuzzing (self ) -> List [Tuple [ bytes , str ] ]:
366
366
def byte_len () -> int :
367
367
"""
368
368
The number of bytes in the input
@@ -388,6 +388,9 @@ def byte_len() -> int:
388
388
self .fully_explored = True
389
389
return results
390
390
391
+ # Denotes the generation method for each input
392
+ # S: constraint solving; F: fuzzing; R: random generation
393
+ method = "S"
391
394
while len (results ) < MAX_SAMPLES :
392
395
try :
393
396
val = next (self .samples )
@@ -396,9 +399,11 @@ def byte_len() -> int:
396
399
break
397
400
if val is None and len (results ) < MIN_SAMPLES :
398
401
# requires constraint solving but not enough results
402
+ method = "S"
399
403
continue
400
- result = val .to_bytes (byte_len (), 'big' )
404
+ result = ( val .to_bytes (byte_len (), 'big' ), method )
401
405
results .append (result )
406
+ method = "F"
402
407
except StopIteration :
403
408
# NOTE: Insufficient results from APPFuzzing:
404
409
# Case 1: break in the outside while:
@@ -440,7 +445,7 @@ def byte_len() -> int:
440
445
return results
441
446
442
447
@staticmethod
443
- def random_fuzzing () -> List [bytes ]:
448
+ def random_fuzzing () -> List [Tuple [ bytes , str ] ]:
444
449
def random_bytes ():
445
450
LOGGER .debug ("Generating random {} bytes" .format (MAX_BYTES ))
446
451
# input_bytes = b''
@@ -449,7 +454,7 @@ def random_bytes():
449
454
# return input_bytes
450
455
# Or return end of file char?
451
456
return os .urandom (MAX_BYTES )
452
- return [random_bytes () for _ in range (MIN_SAMPLES )]
457
+ return [( random_bytes (), "R" ) for _ in range (MIN_SAMPLES )]
453
458
454
459
def add_child (self , key : str or int , new_child : 'TreeNode' ) -> None :
455
460
debug_assertion ((key == 'Simulation' ) ^ (key == new_child .addr ))
@@ -1006,12 +1011,12 @@ def simulation(node: TreeNode = None) -> List[List[int]]:
1006
1011
1007
1012
global FOUND_BUG , MSGS , INPUTS , TIMES
1008
1013
mutants = node .mutate () if node else \
1009
- [bytes ("" .join (mutant ), 'utf-8' )
1014
+ [( bytes ("" .join (mutant ), 'utf-8' ), "D" )
1010
1015
# Note: Need to make sure the first binary execution must complete successfully
1011
1016
# Otherwise (e.g. timeout) the root address will be wrong
1012
- for mutant in SEEDS ] if SEEDS else ([b'\x00 ' * MAX_BYTES ]
1013
- + [b'\x01 \x00 \x00 \x00 ' * (MAX_BYTES // 4 )]
1014
- + [b'\x0a ' ] + TreeNode .random_fuzzing ())
1017
+ for mutant in SEEDS ] if SEEDS else ([( b'\x00 ' * MAX_BYTES , "D" ) ]
1018
+ + [( b'\x01 \x00 \x00 \x00 ' * (MAX_BYTES // 4 ), "D" )]
1019
+ + [( b'\x0a ' , "D" ) ] + TreeNode .random_fuzzing ())
1015
1020
# for mutant in SEEDS] if SEEDS else [b'\x0a']
1016
1021
# for mutant in SEEDS] if SEEDS else TreeNode.random_fuzzing()
1017
1022
@@ -1032,7 +1037,7 @@ def simulation(node: TreeNode = None) -> List[List[int]]:
1032
1037
return traces
1033
1038
1034
1039
1035
- def binary_execute_parallel (input_bytes : bytes ):
1040
+ def binary_execute_parallel (input_bytes : Tuple [ bytes , str ] ):
1036
1041
"""
1037
1042
Execute the binary with an input in bytes
1038
1043
:param input_bytes: the input to feed the binary
@@ -1052,7 +1057,7 @@ def execute():
1052
1057
# 0: no timeout; 1: instrumented binary timeout; 2: uninstrumented binary timeout
1053
1058
timeout = False
1054
1059
try :
1055
- msg = instr .communicate (input_bytes , timeout = CONEX_TIMEOUT )
1060
+ msg = instr .communicate (input_bytes [ 0 ] , timeout = CONEX_TIMEOUT )
1056
1061
ret = instr .returncode
1057
1062
instr .terminate ()
1058
1063
del instr
@@ -1069,7 +1074,7 @@ def execute():
1069
1074
try :
1070
1075
uninstr = sp .Popen (UNINSTR_BIN , stdin = sp .PIPE , stdout = sp .PIPE ,
1071
1076
stderr = sp .PIPE , close_fds = True )
1072
- msg = uninstr .communicate (input_bytes , timeout = CONEX_TIMEOUT )
1077
+ msg = uninstr .communicate (input_bytes [ 0 ] , timeout = CONEX_TIMEOUT )
1073
1078
ret = uninstr .returncode
1074
1079
LOGGER .info ("Uninstrumented binary execution completed" )
1075
1080
uninstr .terminate ()
@@ -1088,16 +1093,15 @@ def execute():
1088
1093
debug_assertion (bool (report ))
1089
1094
1090
1095
report_msg , return_code , time_out = report
1091
-
1092
- completed = report != (None , None )
1096
+ completed = report != (None , None , True )
1093
1097
traced = completed and report_msg [1 ]
1094
1098
found_bug = False
1095
1099
1096
1100
if (SAVE_TESTCASES or SAVE_TESTINPUTS ) and completed :
1097
1101
curr_time = time .time () - TIME_START
1098
1102
if SAVE_TESTCASES and (not time_out or SAVE_TESTCASES_TIMEOUT ):
1099
1103
stdout = report_msg [0 ].decode ('utf-8' )
1100
- save_tests_to_file (curr_time , stdout , ("-T" if time_out else "-C" ))
1104
+ save_tests_to_file (curr_time , stdout , ("-T" if time_out else "-C" )+ ( "-" + input_bytes [ 1 ]) )
1101
1105
if SAVE_TESTINPUTS :
1102
1106
save_input_to_file (curr_time , input_bytes )
1103
1107
0 commit comments