-
Notifications
You must be signed in to change notification settings - Fork 0
/
patcher_x86_32.h
88 lines (71 loc) · 2.7 KB
/
patcher_x86_32.h
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
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef COURGETTE_PATCHER_X86_32_H_
#define COURGETTE_PATCHER_X86_32_H_
#include <stdint.h>
#include "base/logging.h"
#include "base/macros.h"
#include "courgette/courgette_flow.h"
#include "courgette/ensemble.h"
#include "courgette/region.h"
#include "courgette/streams.h"
namespace courgette {
// PatcherX86_32 is the universal patcher for all executables. The executable
// type is determined by the program detector.
class PatcherX86_32 : public TransformationPatcher {
public:
explicit PatcherX86_32(const Region& region)
: ensemble_region_(region),
base_offset_(0),
base_length_(0) {
}
Status Init(SourceStream* parameter_stream) {
if (!parameter_stream->ReadVarint32(&base_offset_))
return C_BAD_TRANSFORM;
if (!parameter_stream->ReadVarint32(&base_length_))
return C_BAD_TRANSFORM;
if (base_offset_ > ensemble_region_.length())
return C_BAD_TRANSFORM;
if (base_length_ > ensemble_region_.length() - base_offset_)
return C_BAD_TRANSFORM;
return C_OK;
}
Status PredictTransformParameters(SinkStreamSet* predicted_parameters) {
// No code needed to write an 'empty' predicted parameter set.
return C_OK;
}
Status Transform(SourceStreamSet* corrected_parameters,
SinkStreamSet* transformed_element) {
if (!corrected_parameters->Empty())
return C_GENERAL_ERROR; // Don't expect any corrected parameters.
CourgetteFlow flow;
RegionBuffer only_buffer(
Region(ensemble_region_.start() + base_offset_, base_length_));
flow.ReadDisassemblerFromBuffer(flow.ONLY, only_buffer);
flow.CreateAssemblyProgramFromDisassembler(flow.ONLY, false);
flow.CreateEncodedProgramFromDisassemblerAndAssemblyProgram(flow.ONLY);
flow.DestroyAssemblyProgram(flow.ONLY);
flow.DestroyDisassembler(flow.ONLY);
flow.WriteSinkStreamSetFromEncodedProgram(flow.ONLY, transformed_element);
if (flow.failed())
LOG(ERROR) << flow.message();
return flow.status();
}
Status Reform(SourceStreamSet* transformed_element,
SinkStream* reformed_element) {
CourgetteFlow flow;
flow.ReadEncodedProgramFromSourceStreamSet(flow.ONLY, transformed_element);
flow.WriteExecutableFromEncodedProgram(flow.ONLY, reformed_element);
if (flow.failed())
LOG(ERROR) << flow.message();
return flow.status();
}
private:
Region ensemble_region_;
uint32_t base_offset_;
uint32_t base_length_;
DISALLOW_COPY_AND_ASSIGN(PatcherX86_32);
};
} // namespace courgette
#endif // COURGETTE_PATCHER_X86_32_H_