-
Notifications
You must be signed in to change notification settings - Fork 60
/
Copy pathengine_input.py
205 lines (171 loc) · 7.88 KB
/
engine_input.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
# Copyright 2022 Fuzz Introspector Authors
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""Analysis for creating input consumed by a fuzzer, e.g. a dictionary"""
import json
import logging
import os
from typing import (List, Dict)
from fuzz_introspector import (analysis, constants, html_helpers, json_report,
utils)
from fuzz_introspector.analyses import calltree_analysis as cta
from fuzz_introspector.datatypes import (
project_profile,
fuzzer_profile,
)
logger = logging.getLogger(name=__name__)
class EngineInput(analysis.AnalysisInterface):
"""Generates content that can be used by fuzz engines."""
name: str = "FuzzEngineInputAnalysis"
def __init__(self) -> None:
self.display_html = False
self.json_string_result = "[]"
@classmethod
def get_name(cls):
return cls.name
def get_json_string_result(self):
return self.json_string_result
def set_json_string_result(self, json_string):
self.json_string_result = json_string
def analysis_func(self,
table_of_contents: html_helpers.HtmlTableOfContents,
tables: List[str],
proj_profile: project_profile.MergedProjectProfile,
profiles: List[fuzzer_profile.FuzzerProfile],
basefolder: str, coverage_url: str,
conclusions: List[html_helpers.HTMLConclusion],
out_dir) -> str:
logger.info('- Running analysis %s', self.get_name())
if not self.display_html:
# Overwrite the table of contents variable, to avoid displaying
# the html in the resulting report.
table_of_contents = html_helpers.HtmlTableOfContents()
html_string = ""
html_string += "<div class=\"report-box\">"
html_string += html_helpers.html_add_header_with_link(
"Fuzz engine guidance", html_helpers.HTML_HEADING.H1,
table_of_contents)
html_string += "<div class=\"collapsible\">"
html_string += "<p>This sections provides heuristics that can be used as input " \
"to a fuzz engine when running a given fuzz target. The current " \
"focus is on providing input that is usable by libFuzzer.</p>"
for prof in profiles:
logger.info('Generating input for %s', prof.identifier)
html_string += html_helpers.html_add_header_with_link(
prof.fuzzer_source_file, html_helpers.HTML_HEADING.H2,
table_of_contents)
# Create dictionary section
html_string += self.get_dictionary_section(prof, table_of_contents,
out_dir)
html_string += "<br>"
# Create focus function section
html_string += self.get_fuzzer_focus_function_section(
prof,
table_of_contents,
)
html_string += "</div>" # .collapsible
html_string += "</div>" # report-box
logger.info('- Completed analysis %s', self.get_name())
if not self.display_html:
html_string = ""
return html_string
def get_dictionary(self, profile: fuzzer_profile.FuzzerProfile,
out_dir) -> str:
"""Extracts a fuzzer dictionary"""
kn = 0
dictionary_content = ""
dictionary: Dict[str, str] = {}
if profile.functions_reached_by_fuzzer is None:
return ""
for fn in profile.functions_reached_by_fuzzer:
try:
fp = profile.all_class_functions[fn]
except Exception as e:
logger.debug(e)
continue
for const in fp.constants_touched:
dictionary_content += f"k{kn}=\"{const}\"\n"
dictionary[f"k{kn}"] = const
kn += 1
self.set_json_string_result(json.dumps(dictionary))
json_report.add_analysis_json_str_as_dict_to_report(
self.get_name(), self.get_json_string_result(), out_dir)
return dictionary_content
def get_dictionary_section(
self, profile: fuzzer_profile.FuzzerProfile,
table_of_contents: html_helpers.HtmlTableOfContents,
out_dir) -> str:
"""
Returns a HTML string with dictionary content, and adds the section
link to the table_of_contents.
"""
html_string = html_helpers.html_add_header_with_link(
"Dictionary", html_helpers.HTML_HEADING.H3, table_of_contents)
html_string += "<p>Use this with the libFuzzer -dict=DICT.file flag</p>"
html_string += "<pre><code class='language-clike'>"
html_string += self.get_dictionary(profile, out_dir)
html_string += "</code></pre>"
return html_string
def get_fuzzer_focus_function_section(
self, profile: fuzzer_profile.FuzzerProfile,
table_of_contents: html_helpers.HtmlTableOfContents) -> str:
"""Returns HTML string with fuzzer focus function"""
html_string = html_helpers.html_add_header_with_link(
"Fuzzer function priority", html_helpers.HTML_HEADING.H3,
table_of_contents)
calltree_analysis = cta.FuzzCalltreeAnalysis()
fuzz_blockers = calltree_analysis.get_fuzz_blockers(
profile, max_blockers_to_extract=10)
if len(fuzz_blockers) == 0:
logger.info("Found no fuzz blockers and thus no focus function")
return ""
# Only succeed if we can get the name of the function in which the
# fuzz blocker callsite resides.
focus_functions = []
for fuzz_blocker in fuzz_blockers:
ffname = fuzz_blocker.src_function_name
if ffname is not None and ffname not in focus_functions:
if profile.target_lang == "rust":
focus_functions.append(utils.demangle_rust_func(ffname))
else:
focus_functions.append(utils.demangle_cpp_func(ffname))
logger.info('Found focus function: %s',
fuzz_blocker.src_function_name)
if len(focus_functions) == 0:
return ""
self.add_to_json_file(constants.ENGINE_INPUT_FILE, profile.identifier,
"focus-functions", focus_functions)
html_string += (
f"<p>Use one of these functions as input to libfuzzer with flag: "
f"-focus_function name </p>"
f"<pre><code class='language-clike'>"
f"-focus_function={focus_functions}"
f"</code></pre><br>")
return html_string
def add_to_json_file(self, json_file_path: str, fuzzer_name: str, key: str,
val: List[str]) -> None:
"""Add key to json dictionary in `json_file_path`."""
# Create file if it does not exist
if not os.path.isfile(json_file_path):
json_data = {}
else:
json_fd = open(json_file_path)
json_data = json.load(json_fd)
json_fd.close()
if 'fuzzers' not in json_data:
json_data['fuzzers'] = {}
if fuzzer_name not in json_data['fuzzers']:
json_data['fuzzers'][fuzzer_name] = {}
json_data['fuzzers'][fuzzer_name][key] = val
with open(json_file_path, 'w') as json_file:
json.dump(json_data, json_file)