-
Notifications
You must be signed in to change notification settings - Fork 310
/
Minimizer.swift
119 lines (101 loc) · 7.12 KB
/
Minimizer.swift
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
// Copyright 2019 Google LLC
//
// 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
//
// https://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.
import Foundation
/// Minimizes programs.
///
/// Executes various program reducers to shrink a program in size while retaining its special aspects. All of this
/// happens on a separate dispatch queue so the main queue stays responsive.
///
/// There are two basic principles for program minimization:
/// - Minimization must only remove program features that can be added back through mutations later on.
/// For example, variadic inputs to instructions can be removed because the OperationMutator can add them back.
/// - Minimization should generally strive to be as powerful as possible, i.e. be able to find the smallest possible programs.
/// To counter "over-minimization" (i.e. programs becoming too small, making mutations less effective), there should
/// be a configurable limit to minimization which keeps some random instructions alive, so that those are uniformly
/// distributed and not biased to a certain type of instruction or instruction sequence.
///
/// We *cannot* store any information as indices as they are invalidated as soon as any reducer moves instructions around.
/// Therefore, also every reducer needs to preserve flags when they operate on an instruction or operation.
public class Minimizer: ComponentBase {
/// DispatchQueue on which program minimization happens.
private let minimizationQueue = DispatchQueue(label: "Minimizer")
public init() {
super.init(name: "Minimizer")
}
/// Minimizes the given program while preserving its special aspects.
///
/// Minimization will not modify the given program. Instead, it produce a new Program instance.
/// Once minimization is finished, the passed block will be invoked on the fuzzer's queue with the minimized program.
func withMinimizedCopy(_ program: Program, withAspects aspects: ProgramAspects, limit minimizationLimit: Double = 0.0, block: @escaping (Program) -> ()) {
minimizationQueue.async {
let minimizedCode = self.internalMinimize(program, withAspects: aspects, limit: minimizationLimit, runningSynchronously: false)
self.fuzzer.async {
let minimizedProgram: Program
if self.fuzzer.config.enableInspection {
minimizedProgram = Program(code: minimizedCode, parent: program, contributors: program.contributors)
minimizedProgram.comments.add("Minimizing \(program.id)", at: .header)
} else {
minimizedProgram = Program(code: minimizedCode, contributors: program.contributors)
}
block(minimizedProgram)
}
}
}
/// Synchronous version of withMinimizedCopy. Should only be used for tests since it otherwise blocks the fuzzer queue.
func minimize(_ program: Program, withAspects aspects: ProgramAspects, limit minimizationLimit: Double = 0.0) -> Program {
let minimizedCode = internalMinimize(program, withAspects: aspects, limit: minimizationLimit, runningSynchronously: true)
return Program(code: minimizedCode, parent: program, contributors: program.contributors)
}
private func internalMinimize(_ program: Program, withAspects aspects: ProgramAspects, limit minimizationLimit: Double, runningSynchronously: Bool) -> Code {
assert(program.code.countIntructionsWith(flags: .notRemovable) == 0)
let helper = MinimizationHelper(for: aspects, forCode: program.code, of: fuzzer, runningOnFuzzerQueue: runningSynchronously)
helper.applyMinimizationLimit(limit: minimizationLimit)
assert(helper.code.countIntructionsWith(flags: .notRemovable) >= helper.numKeptInstructions)
var iterations = 0
repeat {
helper.didReduce = false
// Notes on reducer scheduling:
// - The ReplaceReducer should run before the InliningReducer as it changes "special" functions into plain functions, which the inlining reducer inlines.
// - The ReassignmentReducer should run right after the InliningReducer as inlining produces new Reassign instructions.
// - The VariadicInputReducer should run after the InliningReducer as it may remove function call arguments, causing the parameters to be undefined after inlining.
let reducers: [Reducer] = [GenericInstructionReducer(), BlockReducer(), SimplifyingReducer(), LoopReducer(), InliningReducer(), ReassignmentReducer(), VariadicInputReducer(), DeduplicatingReducer()]
for reducer in reducers {
reducer.reduce(with: helper)
// The reducers should not remove any instructions that we want to keep unconditionally.
// The code might have more non-removable instructions due to other analyzers marking them as non-removable
// but it should be at least more than we have seen at the start.
assert(helper.code.countIntructionsWith(flags: .notRemovable) >= helper.numKeptInstructions)
assert(helper.code.isStaticallyValid())
}
iterations += 1
guard iterations < 100 else {
// This can happen if a reducer performs a no-op change in every iteration, e.g. replacing one instruction with the same instruction. This is considered a bug since it leads to this kind of issue.
logger.error("Fixpoint iteration for program minimization did not converge after 100 iterations for program:\n\(FuzzILLifter().lift(helper.code)). Aborting minimization.")
break
}
} while helper.didReduce
// Most reducers replace instructions with NOPs instead of deleting them. Remove those NOPs now.
helper.removeNops()
assert(helper.code.countIntructionsWith(flags: .notRemovable) >= helper.numKeptInstructions)
// Post-process the sample after minimization. This step adds certain features back to the program that may have been minimized away but are typically helpful for future mutations.
// Currently we run this regardless of whether we're processing a crash or an interesting sample. If we wanted to, we could only run this for interesting samples (that will be mutated again), but its fine to also run it for crashes.
// Adding instructions will invalidate the keptInstructions array. Since we're not removing any more instructions, clear that array now.
helper.clearFlags()
let postProcessor = MinimizationPostProcessor()
postProcessor.process(with: helper)
assert(helper.code.isStaticallyValid())
assert(!helper.code.contains(where: { $0.isNop }))
return helper.finalize()
}
}