C++程序  |  296行  |  7.74 KB

/*
 * 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