blob: 5b213dc4d95805f7373b7b6aecf2ba0f7ef6fc8b (
plain)
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
|
// Copyright (c) 2011 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.
#include <string>
#include "base/strings/string_util.h"
#include "courgette/assembly_program.h"
#include "courgette/courgette.h"
#include "courgette/streams.h"
#include "testing/gtest/include/gtest/gtest.h"
class AdjustmentMethodTest : public testing::Test {
public:
void Test1() const;
private:
void SetUp() {
}
void TearDown() {
}
// Returns one of two similar a simple programs. They differ only in the
// label assignment, so that it is possible to make them look identical.
courgette::AssemblyProgram* MakeProgram(int kind) const {
courgette::AssemblyProgram* prog =
new courgette::AssemblyProgram(courgette::EXE_WIN_32_X86);
prog->set_image_base(0x00400000);
courgette::Label* labelA = prog->FindOrMakeAbs32Label(0x00410000);
courgette::Label* labelB = prog->FindOrMakeAbs32Label(0x00410004);
EXPECT_TRUE(prog->EmitAbs32(labelA));
EXPECT_TRUE(prog->EmitAbs32(labelA));
EXPECT_TRUE(prog->EmitAbs32(labelB));
EXPECT_TRUE(prog->EmitAbs32(labelA));
EXPECT_TRUE(prog->EmitAbs32(labelA));
EXPECT_TRUE(prog->EmitAbs32(labelB));
if (kind == 0) {
labelA->index_ = 0;
labelB->index_ = 1;
} else {
labelA->index_ = 1;
labelB->index_ = 0;
}
prog->AssignRemainingIndexes();
return prog;
}
courgette::AssemblyProgram* MakeProgramA() const { return MakeProgram(0); }
courgette::AssemblyProgram* MakeProgramB() const { return MakeProgram(1); }
// Returns a string that is the serialized version of |program|.
// Deletes |program|.
std::string Serialize(courgette::AssemblyProgram *program) const {
courgette::EncodedProgram* encoded = NULL;
const courgette::Status encode_status = Encode(program, &encoded);
EXPECT_EQ(courgette::C_OK, encode_status);
DeleteAssemblyProgram(program);
courgette::SinkStreamSet sinks;
const courgette::Status write_status = WriteEncodedProgram(encoded, &sinks);
EXPECT_EQ(courgette::C_OK, write_status);
DeleteEncodedProgram(encoded);
courgette::SinkStream sink;
bool can_collect = sinks.CopyTo(&sink);
EXPECT_TRUE(can_collect);
return std::string(reinterpret_cast<const char *>(sink.Buffer()),
sink.Length());
}
};
void AdjustmentMethodTest::Test1() const {
courgette::AssemblyProgram* prog1 = MakeProgramA();
courgette::AssemblyProgram* prog2 = MakeProgramB();
std::string s1 = Serialize(prog1);
std::string s2 = Serialize(prog2);
// Don't use EXPECT_EQ because strings are unprintable.
EXPECT_FALSE(s1 == s2); // Unadjusted A and B differ.
courgette::AssemblyProgram* prog5 = MakeProgramA();
courgette::AssemblyProgram* prog6 = MakeProgramB();
courgette::Status can_adjust = Adjust(*prog5, prog6);
EXPECT_EQ(courgette::C_OK, can_adjust);
std::string s5 = Serialize(prog5);
std::string s6 = Serialize(prog6);
EXPECT_TRUE(s1 == s5); // Adjustment did not change A (prog5)
EXPECT_TRUE(s5 == s6); // Adjustment did change B into A
}
TEST_F(AdjustmentMethodTest, All) {
Test1();
}
|