/*
* Copyright 2017, The Android Open Source Project
*
* 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.
*/
#include "pass_queue.h"
#include "file_utils.h"
#include "spirit.h"
#include "test_utils.h"
#include "transformer.h"
#include "gtest/gtest.h"
#include <stdint.h>
namespace android {
namespace spirit {
namespace {
class MulToAddTransformer : public Transformer {
public:
Instruction *transform(IMulInst *mul) override {
auto ret = new IAddInst(mul->mResultType, mul->mOperand1, mul->mOperand2);
ret->setId(mul->getId());
return ret;
}
};
class AddToDivTransformer : public Transformer {
public:
Instruction *transform(IAddInst *add) override {
auto ret = new SDivInst(add->mResultType, add->mOperand1, add->mOperand2);
ret->setId(add->getId());
return ret;
}
};
class AddMulAfterAddTransformer : public Transformer {
public:
Instruction *transform(IAddInst *add) override {
insert(add);
auto ret = new IMulInst(add->mResultType, add, add);
ret->setId(add->getId());
return ret;
}
};
class Deleter : public Transformer {
public:
Instruction *transform(IMulInst *) override { return nullptr; }
};
class InPlaceModifyingPass : public Pass {
public:
Module *run(Module *m, int *error) override {
m->getFloatType(64);
if (error) {
*error = 0;
}
return m;
}
};
} // annonymous namespace
class PassQueueTest : public ::testing::Test {
protected:
virtual void SetUp() { mWordsGreyscale = readWords("greyscale.spv"); }
std::vector<uint32_t> mWordsGreyscale;
private:
std::vector<uint32_t> readWords(const char *testFile) {
static const std::string testDataPath(
"frameworks/rs/rsov/compiler/spirit/test_data/");
const std::string &fullPath = getAbsolutePath(testDataPath + testFile);
return readFile<uint32_t>(fullPath);
}
};
TEST_F(PassQueueTest, testMulToAdd) {
std::unique_ptr<Module> m(Deserialize<Module>(mWordsGreyscale));
ASSERT_NE(nullptr, m);
EXPECT_EQ(1, countEntity<IAddInst>(m.get()));
EXPECT_EQ(1, countEntity<IMulInst>(m.get()));
PassQueue passes;
passes.append(new MulToAddTransformer());
auto m1 = passes.run(m.get());
ASSERT_NE(nullptr, m1);
ASSERT_TRUE(m1->resolveIds());
EXPECT_EQ(2, countEntity<IAddInst>(m1));
EXPECT_EQ(0, countEntity<IMulInst>(m1));
}
TEST_F(PassQueueTest, testInPlaceModifying) {
std::unique_ptr<Module> m(Deserialize<Module>(mWordsGreyscale));
ASSERT_NE(nullptr, m);
EXPECT_EQ(1, countEntity<IAddInst>(m.get()));
EXPECT_EQ(1, countEntity<IMulInst>(m.get()));
EXPECT_EQ(1, countEntity<TypeFloatInst>(m.get()));
PassQueue passes;
passes.append(new InPlaceModifyingPass());
auto m1 = passes.run(m.get());
ASSERT_NE(nullptr, m1);
ASSERT_TRUE(m1->resolveIds());
EXPECT_EQ(1, countEntity<IAddInst>(m1));
EXPECT_EQ(1, countEntity<IMulInst>(m1));
EXPECT_EQ(2, countEntity<TypeFloatInst>(m1));
}
TEST_F(PassQueueTest, testDeletion) {
std::unique_ptr<Module> m(Deserialize<Module>(mWordsGreyscale));
ASSERT_NE(nullptr, m.get());
EXPECT_EQ(1, countEntity<IMulInst>(m.get()));
PassQueue passes;
passes.append(new Deleter());
auto m1 = passes.run(m.get());
// One of the ids from the input module is missing now.
ASSERT_EQ(nullptr, m1);
}
TEST_F(PassQueueTest, testMulToAddToDiv) {
std::unique_ptr<Module> m(Deserialize<Module>(mWordsGreyscale));
ASSERT_NE(nullptr, m);
EXPECT_EQ(1, countEntity<IAddInst>(m.get()));
EXPECT_EQ(1, countEntity<IMulInst>(m.get()));
PassQueue passes;
passes.append(new MulToAddTransformer());
passes.append(new AddToDivTransformer());
auto m1 = passes.run(m.get());
ASSERT_NE(nullptr, m1);
ASSERT_TRUE(m1->resolveIds());
EXPECT_EQ(0, countEntity<IAddInst>(m1));
EXPECT_EQ(0, countEntity<IMulInst>(m1));
EXPECT_EQ(2, countEntity<SDivInst>(m1));
}
TEST_F(PassQueueTest, testAMix) {
std::unique_ptr<Module> m(Deserialize<Module>(mWordsGreyscale));
ASSERT_NE(nullptr, m);
EXPECT_EQ(1, countEntity<IAddInst>(m.get()));
EXPECT_EQ(1, countEntity<IMulInst>(m.get()));
EXPECT_EQ(0, countEntity<SDivInst>(m.get()));
EXPECT_EQ(1, countEntity<TypeFloatInst>(m.get()));
PassQueue passes;
passes.append(new MulToAddTransformer());
passes.append(new AddToDivTransformer());
passes.append(new InPlaceModifyingPass());
std::unique_ptr<Module> m1(passes.run(m.get()));
ASSERT_NE(nullptr, m1);
ASSERT_TRUE(m1->resolveIds());
EXPECT_EQ(0, countEntity<IAddInst>(m1.get()));
EXPECT_EQ(0, countEntity<IMulInst>(m1.get()));
EXPECT_EQ(2, countEntity<SDivInst>(m1.get()));
EXPECT_EQ(2, countEntity<TypeFloatInst>(m1.get()));
}
TEST_F(PassQueueTest, testAnotherMix) {
std::unique_ptr<Module> m(Deserialize<Module>(mWordsGreyscale));
ASSERT_NE(nullptr, m);
EXPECT_EQ(1, countEntity<IAddInst>(m.get()));
EXPECT_EQ(1, countEntity<IMulInst>(m.get()));
EXPECT_EQ(0, countEntity<SDivInst>(m.get()));
EXPECT_EQ(1, countEntity<TypeFloatInst>(m.get()));
PassQueue passes;
passes.append(new InPlaceModifyingPass());
passes.append(new MulToAddTransformer());
passes.append(new AddToDivTransformer());
auto outputWords = passes.runAndSerialize(m.get());
std::unique_ptr<Module> m1(Deserialize<Module>(outputWords));
ASSERT_NE(nullptr, m1);
ASSERT_TRUE(m1->resolveIds());
EXPECT_EQ(0, countEntity<IAddInst>(m1.get()));
EXPECT_EQ(0, countEntity<IMulInst>(m1.get()));
EXPECT_EQ(2, countEntity<SDivInst>(m1.get()));
EXPECT_EQ(2, countEntity<TypeFloatInst>(m1.get()));
}
TEST_F(PassQueueTest, testMulToAddToDivFromWords) {
PassQueue passes;
passes.append(new MulToAddTransformer());
passes.append(new AddToDivTransformer());
auto outputWords = passes.run(std::move(mWordsGreyscale));
std::unique_ptr<Module> m1(Deserialize<Module>(outputWords));
ASSERT_NE(nullptr, m1);
ASSERT_TRUE(m1->resolveIds());
EXPECT_EQ(0, countEntity<IAddInst>(m1.get()));
EXPECT_EQ(0, countEntity<IMulInst>(m1.get()));
EXPECT_EQ(2, countEntity<SDivInst>(m1.get()));
}
TEST_F(PassQueueTest, testMulToAddToDivToWords) {
std::unique_ptr<Module> m(Deserialize<Module>(mWordsGreyscale));
ASSERT_NE(nullptr, m);
EXPECT_EQ(1, countEntity<IAddInst>(m.get()));
EXPECT_EQ(1, countEntity<IMulInst>(m.get()));
PassQueue passes;
passes.append(new MulToAddTransformer());
passes.append(new AddToDivTransformer());
auto outputWords = passes.runAndSerialize(m.get());
std::unique_ptr<Module> m1(Deserialize<Module>(outputWords));
ASSERT_NE(nullptr, m1);
ASSERT_TRUE(m1->resolveIds());
EXPECT_EQ(0, countEntity<IAddInst>(m1.get()));
EXPECT_EQ(0, countEntity<IMulInst>(m1.get()));
EXPECT_EQ(2, countEntity<SDivInst>(m1.get()));
}
TEST_F(PassQueueTest, testAddMulAfterAdd) {
std::unique_ptr<Module> m(Deserialize<Module>(mWordsGreyscale));
ASSERT_NE(nullptr, m);
EXPECT_EQ(1, countEntity<IAddInst>(m.get()));
EXPECT_EQ(1, countEntity<IMulInst>(m.get()));
constexpr int kNumMulToAdd = 100;
PassQueue passes;
for (int i = 0; i < kNumMulToAdd; i++) {
passes.append(new AddMulAfterAddTransformer());
}
auto outputWords = passes.runAndSerialize(m.get());
std::unique_ptr<Module> m1(Deserialize<Module>(outputWords));
ASSERT_NE(nullptr, m1);
ASSERT_TRUE(m1->resolveIds());
EXPECT_EQ(1, countEntity<IAddInst>(m1.get()));
EXPECT_EQ(1 + kNumMulToAdd, countEntity<IMulInst>(m1.get()));
}
} // namespace spirit
} // namespace android