blob: e3134438204b9df8d616f584cacccfd052eae9d7 [file] [log] [blame]
// 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 <memory>
#include <string>
#include <utility>
#include <vector>
#include "base/bind.h"
#include "base/memory/ptr_util.h"
#include "courgette/assembly_program.h"
#include "courgette/courgette.h"
#include "courgette/encoded_program.h"
#include "courgette/image_utils.h"
#include "courgette/streams.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace courgette {
namespace {
class AdjustmentMethodTest : public testing::Test {
public:
void Test1() const;
private:
void SetUp() {
}
void TearDown() {
}
// Returns one of two similar simple programs. These differ only in Label
// assignment, so it is possible to make them look identical.
std::unique_ptr<AssemblyProgram> MakeProgram(int kind) const {
auto prog = base::MakeUnique<AssemblyProgram>(EXE_WIN_32_X86, 0x00400000);
RVA kRvaA = 0x00410000;
RVA kRvaB = 0x00410004;
std::vector<RVA> abs32_rvas;
abs32_rvas.push_back(kRvaA);
abs32_rvas.push_back(kRvaB);
std::vector<RVA> rel32_rvas; // Stub.
TrivialRvaVisitor abs32_visitor(abs32_rvas);
TrivialRvaVisitor rel32_visitor(rel32_rvas);
prog->PrecomputeLabels(&abs32_visitor, &rel32_visitor);
Label* labelA = prog->FindAbs32Label(kRvaA);
Label* labelB = prog->FindAbs32Label(kRvaB);
AssemblyProgram::InstructionGenerator gen = base::Bind(
[](Label* labelA, Label* labelB, AssemblyProgram* prog,
InstructionReceptor* receptor) -> CheckBool {
EXPECT_TRUE(receptor->EmitAbs32(labelA));
EXPECT_TRUE(receptor->EmitAbs32(labelA));
EXPECT_TRUE(receptor->EmitAbs32(labelB));
EXPECT_TRUE(receptor->EmitAbs32(labelA));
EXPECT_TRUE(receptor->EmitAbs32(labelA));
EXPECT_TRUE(receptor->EmitAbs32(labelB));
return true;
},
labelA, labelB);
EXPECT_TRUE(prog->GenerateInstructions(gen));
if (kind == 0) {
labelA->index_ = 0;
labelB->index_ = 1;
} else {
labelA->index_ = 1;
labelB->index_ = 0;
}
prog->AssignRemainingIndexes();
return prog;
}
std::unique_ptr<AssemblyProgram> MakeProgramA() const {
return MakeProgram(0);
}
std::unique_ptr<AssemblyProgram> MakeProgramB() const {
return MakeProgram(1);
}
// Returns a string that is the serialized version of |program|.
// Deletes |program|.
std::string Serialize(std::unique_ptr<AssemblyProgram> program) const {
std::unique_ptr<EncodedProgram> encoded;
const Status encode_status = Encode(*program, &encoded);
EXPECT_EQ(C_OK, encode_status);
program.reset();
SinkStreamSet sinks;
const Status write_status = WriteEncodedProgram(encoded.get(), &sinks);
EXPECT_EQ(C_OK, write_status);
encoded.reset();
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 {
std::unique_ptr<AssemblyProgram> prog1 = MakeProgramA();
std::unique_ptr<AssemblyProgram> prog2 = MakeProgramB();
std::string s1 = Serialize(std::move(prog1));
std::string s2 = Serialize(std::move(prog2));
// Don't use EXPECT_EQ because strings are unprintable.
EXPECT_FALSE(s1 == s2); // Unadjusted A and B differ.
std::unique_ptr<AssemblyProgram> prog5 = MakeProgramA();
std::unique_ptr<AssemblyProgram> prog6 = MakeProgramB();
Status can_adjust = Adjust(*prog5, prog6.get());
EXPECT_EQ(C_OK, can_adjust);
std::string s5 = Serialize(std::move(prog5));
std::string s6 = Serialize(std::move(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();
}
} // namespace
} // namespace courgette