mirror of
https://github.com/apple/swift.git
synced 2025-12-21 12:14:44 +01:00
235 lines
8.4 KiB
C++
235 lines
8.4 KiB
C++
//===-------------------------- TestRunner.cpp ----------------------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2022 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See https://swift.org/LICENSE.txt for license information
|
|
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file defines test::TestRunner, the pass responsible for running tests,
|
|
// specifically test::FunctionTest and test::ModuleTest (maybe someday).
|
|
//
|
|
// To see more about writing your own tests, see include/swift/SIL/Test.h.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "swift/AST/Type.h"
|
|
#include "swift/Basic/TaggedUnion.h"
|
|
#include "swift/SIL/FieldSensitivePrunedLiveness.h"
|
|
#include "swift/SIL/MemAccessUtils.h"
|
|
#include "swift/SIL/OSSALifetimeCompletion.h"
|
|
#include "swift/SIL/OwnershipLiveness.h"
|
|
#include "swift/SIL/OwnershipUtils.h"
|
|
#include "swift/SIL/ParseTestSpecification.h"
|
|
#include "swift/SIL/PrunedLiveness.h"
|
|
#include "swift/SIL/SILArgumentArrayRef.h"
|
|
#include "swift/SIL/SILBasicBlock.h"
|
|
#include "swift/SIL/SILBridging.h"
|
|
#include "swift/SIL/SILFunction.h"
|
|
#include "swift/SIL/SILInstruction.h"
|
|
#include "swift/SIL/ScopedAddressUtils.h"
|
|
#include "swift/SIL/Test.h"
|
|
#include "swift/SILOptimizer/Analysis/BasicCalleeAnalysis.h"
|
|
#include "swift/SILOptimizer/PassManager/Passes.h"
|
|
#include "swift/SILOptimizer/PassManager/Transforms.h"
|
|
#include "swift/SILOptimizer/Transforms/SimplifyCFG.h"
|
|
#include "swift/SILOptimizer/Utils/CanonicalizeBorrowScope.h"
|
|
#include "swift/SILOptimizer/Utils/CanonicalizeOSSALifetime.h"
|
|
#include "swift/SILOptimizer/Utils/InstOptUtils.h"
|
|
#include "swift/SILOptimizer/Utils/InstructionDeleter.h"
|
|
#include "llvm/ADT/StringRef.h"
|
|
#include "llvm/Support/Debug.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
#include <iterator>
|
|
#include <memory>
|
|
|
|
using namespace swift;
|
|
using namespace swift::test;
|
|
|
|
namespace swift::test {
|
|
class TestRunner : public SILFunctionTransform {
|
|
void printTestLifetime(bool begin, unsigned testIndex, unsigned testCount,
|
|
StringRef name, ArrayRef<StringRef> components);
|
|
void runTest(StringRef name, Arguments &arguments);
|
|
void run() override;
|
|
struct FunctionTestDependenciesImpl final
|
|
: public test::FunctionTest::Dependencies {
|
|
TestRunner *pass;
|
|
SILFunction *function;
|
|
FunctionTestDependenciesImpl(TestRunner *pass, SILFunction *function)
|
|
: pass(pass), function(function) {}
|
|
DominanceInfo *getDominanceInfo() override {
|
|
auto *dominanceAnalysis = pass->getAnalysis<DominanceAnalysis>();
|
|
return dominanceAnalysis->get(function);
|
|
}
|
|
SILPassManager *getPassManager() override { return pass->getPassManager(); }
|
|
~FunctionTestDependenciesImpl() {}
|
|
};
|
|
};
|
|
|
|
void TestRunner::printTestLifetime(bool begin, unsigned testIndex,
|
|
unsigned testCount, StringRef name,
|
|
ArrayRef<StringRef> components) {
|
|
StringRef word = begin ? "\nbegin" : "end";
|
|
llvm::errs() << word << " running test " << testIndex + 1 << " of "
|
|
<< testCount << " on " << getFunction()->getName() << ": "
|
|
<< name << " with: ";
|
|
for (unsigned long index = 0, size = components.size(); index < size;
|
|
++index) {
|
|
auto componentString = components[index].trim();
|
|
if (componentString.empty())
|
|
continue;
|
|
|
|
llvm::errs() << componentString;
|
|
if (index != size - 1) {
|
|
llvm::errs() << ", ";
|
|
}
|
|
}
|
|
llvm::errs() << "\n";
|
|
}
|
|
|
|
void TestRunner::runTest(StringRef name, Arguments &arguments) {
|
|
auto *test = FunctionTest::get(name);
|
|
if (!test) {
|
|
llvm::errs() << "No test named: " << name << "\n";
|
|
assert(false && "Invalid test name");
|
|
}
|
|
auto *function = getFunction();
|
|
FunctionTestDependenciesImpl dependencies(this, function);
|
|
test->run(*function, arguments, *this, dependencies);
|
|
}
|
|
|
|
void TestRunner::run() {
|
|
llvm::SmallVector<UnparsedSpecification, 2> testSpecifications;
|
|
getTestSpecifications(getFunction(), testSpecifications);
|
|
Arguments arguments;
|
|
SmallVector<StringRef, 4> components;
|
|
for (unsigned long index = 0, size = testSpecifications.size(); index < size;
|
|
++index) {
|
|
components.clear();
|
|
arguments.clear();
|
|
auto testSpecification = testSpecifications[index];
|
|
test::parseTestArgumentsFromSpecification(getFunction(), testSpecification,
|
|
arguments, components);
|
|
auto name = arguments.takeString();
|
|
ArrayRef<StringRef> argumentStrings = components;
|
|
argumentStrings = argumentStrings.drop_front();
|
|
printTestLifetime(/*begin=*/true, /*index=*/index, /*size=*/size, name,
|
|
argumentStrings);
|
|
runTest(name, arguments);
|
|
printTestLifetime(/*begin=*/false, /*index=*/index, /*size=*/size, name,
|
|
argumentStrings);
|
|
}
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// MARK: General Unit Tests
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Arguments: NONE
|
|
// Dumps:
|
|
// - the function
|
|
static FunctionTest DumpFunctionTest("dump-function",
|
|
[](auto &function, auto &, auto &) {
|
|
function.dump();
|
|
});
|
|
|
|
// Arguments: NONE
|
|
// Dumps: the index of the self argument of the current function
|
|
static FunctionTest FunctionGetSelfArgumentIndex(
|
|
"function-get-self-argument-index", [](auto &function, auto &, auto &) {
|
|
auto index = BridgedFunction{&function}.getSelfArgumentIndex();
|
|
llvm::errs() << "self argument index = " << index << "\n";
|
|
});
|
|
|
|
// Arguments:
|
|
// - string: list of characters, each of which specifies subsequent arguments
|
|
// - A: (block) argument
|
|
// - F: function
|
|
// - B: block
|
|
// - I: instruction
|
|
// - V: value
|
|
// - O: operand
|
|
// - b: boolean
|
|
// - u: unsigned
|
|
// - s: string
|
|
// - ...
|
|
// - an argument of the type specified in the initial string
|
|
// - ...
|
|
// Dumps:
|
|
// - for each argument (after the initial string)
|
|
// - its type
|
|
// - something to identify the instance (mostly this means calling dump)
|
|
static FunctionTest TestSpecificationTest(
|
|
"test-specification-parsing",
|
|
[](auto &function, auto &arguments, auto &test) {
|
|
auto expectedFields = arguments.takeString();
|
|
for (auto expectedField : expectedFields) {
|
|
switch (expectedField) {
|
|
case 'A': {
|
|
auto *argument = arguments.takeBlockArgument();
|
|
llvm::errs() << "argument:\n";
|
|
argument->dump();
|
|
break;
|
|
}
|
|
case 'F': {
|
|
auto *function = arguments.takeFunction();
|
|
llvm::errs() << "function: " << function->getName() << "\n";
|
|
break;
|
|
}
|
|
case 'B': {
|
|
auto *block = arguments.takeBlock();
|
|
llvm::errs() << "block:\n";
|
|
block->dump();
|
|
break;
|
|
}
|
|
case 'I': {
|
|
auto *instruction = arguments.takeInstruction();
|
|
llvm::errs() << "instruction: ";
|
|
instruction->dump();
|
|
break;
|
|
}
|
|
case 'V': {
|
|
auto value = arguments.takeValue();
|
|
llvm::errs() << "value: ";
|
|
value->dump();
|
|
break;
|
|
}
|
|
case 'O': {
|
|
auto *operand = arguments.takeOperand();
|
|
llvm::errs() << "operand: ";
|
|
operand->print(llvm::errs());
|
|
break;
|
|
}
|
|
case 'u': {
|
|
auto u = arguments.takeUInt();
|
|
llvm::errs() << "uint: " << u << "\n";
|
|
break;
|
|
}
|
|
case 'b': {
|
|
auto b = arguments.takeBool();
|
|
llvm::errs() << "bool: " << b << "\n";
|
|
break;
|
|
}
|
|
case 's': {
|
|
auto s = arguments.takeString();
|
|
llvm::errs() << "string: " << s << "\n";
|
|
break;
|
|
}
|
|
default:
|
|
llvm_unreachable("unknown field type was expected?!");
|
|
}
|
|
}
|
|
});
|
|
} // namespace swift::test
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// Top Level Entry Point
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
SILTransform *swift::createUnitTestRunner() { return new TestRunner(); }
|