mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
239 lines
8.4 KiB
C++
239 lines
8.4 KiB
C++
//===--- SILExtract.cpp - SIL function extraction utility -----------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2016 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See http://swift.org/LICENSE.txt for license information
|
|
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This utility is meant to help simplify the extraction of test cases from sil
|
|
// files by removing (currently only) functions that do not match a
|
|
// string. Eventually this should have additional capabilities like stripping
|
|
// globals, vtables, etc.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "swift/Subsystems.h"
|
|
#include "swift/Basic/Demangle.h"
|
|
#include "swift/Basic/LLVMInitialize.h"
|
|
#include "swift/Frontend/DiagnosticVerifier.h"
|
|
#include "swift/Frontend/Frontend.h"
|
|
#include "swift/Frontend/PrintingDiagnosticConsumer.h"
|
|
#include "swift/Serialization/SerializedModuleLoader.h"
|
|
#include "swift/Serialization/SerializedSILLoader.h"
|
|
#include "swift/SILOptimizer/Analysis/Analysis.h"
|
|
#include "swift/SILOptimizer/PassManager/Passes.h"
|
|
#include "swift/SILOptimizer/PassManager/PassManager.h"
|
|
#include "swift/SIL/SILBuilder.h"
|
|
#include "swift/SIL/SILUndef.h"
|
|
#include "llvm/Support/CommandLine.h"
|
|
#include "llvm/Support/FileSystem.h"
|
|
#include "llvm/Support/ManagedStatic.h"
|
|
#include "llvm/Support/MemoryBuffer.h"
|
|
#include "llvm/Support/Path.h"
|
|
#include "llvm/Support/Signals.h"
|
|
#include <cstdio>
|
|
using namespace swift;
|
|
|
|
static llvm::cl::opt<std::string>
|
|
InputFilename(llvm::cl::desc("input file"), llvm::cl::init("-"),
|
|
llvm::cl::Positional);
|
|
|
|
static llvm::cl::opt<std::string>
|
|
OutputFilename("o", llvm::cl::desc("output filename"), llvm::cl::init("-"));
|
|
|
|
|
|
static llvm::cl::opt<bool>
|
|
EmitVerboseSIL("emit-verbose-sil",
|
|
llvm::cl::desc("Emit locations during sil emission."));
|
|
|
|
static llvm::cl::opt<std::string>
|
|
FunctionName("func", llvm::cl::desc("Function name to extract."));
|
|
|
|
static llvm::cl::list<std::string>
|
|
ImportPaths("I", llvm::cl::desc("add a directory to the import search path"));
|
|
|
|
static llvm::cl::opt<std::string>
|
|
ModuleName("module-name", llvm::cl::desc("The name of the module if processing"
|
|
" a module. Necessary for processing "
|
|
"stdin."));
|
|
|
|
static llvm::cl::opt<std::string>
|
|
ModuleCachePath("module-cache-path", llvm::cl::desc("Clang module cache path"));
|
|
|
|
static llvm::cl::opt<std::string>
|
|
ResourceDir("resource-dir",
|
|
llvm::cl::desc("The directory that holds the compiler resource files"));
|
|
|
|
static llvm::cl::opt<std::string>
|
|
SDKPath("sdk", llvm::cl::desc("The path to the SDK for use with the clang "
|
|
"importer."),
|
|
llvm::cl::init(""));
|
|
|
|
static llvm::cl::opt<std::string>
|
|
Triple("target", llvm::cl::desc("target triple"));
|
|
|
|
// This function isn't referenced outside its translation unit, but it
|
|
// can't use the "static" keyword because its address is used for
|
|
// getMainExecutable (since some platforms don't support taking the
|
|
// address of main, and some platforms can't implement getMainExecutable
|
|
// without being given the address of a function in the main executable).
|
|
void anchorForGetMainExecutable() {}
|
|
|
|
void
|
|
removeUnwantedFunctions(SILModule *M, llvm::StringRef Name) {
|
|
assert(!Name.empty() && "Expected name of function we want to retain!");
|
|
assert(M && "Expected a SIL module to extract from.");
|
|
|
|
// If the function name passed is already mangled then we assume the
|
|
// user knows exactly what function they want and thus don't try
|
|
// to demangle any functions.
|
|
bool isMangled = Name.startswith("_T");
|
|
|
|
std::vector<SILFunction *> DeadFunctions;
|
|
for (auto &F : M->getFunctionList()) {
|
|
auto FnName = isMangled
|
|
? F.getName().str()
|
|
: swift::Demangle::demangleSymbolAsString(F.getName());
|
|
|
|
// A rather simple way to get at just the demangled function
|
|
// name (fully qualified i.e. Swift.String.init) without the
|
|
// argument and return types.
|
|
if (!isMangled)
|
|
FnName = FnName.substr(0, FnName.find(' '));
|
|
|
|
if (Name != FnName) {
|
|
if (F.size()) {
|
|
SILBasicBlock &BB = F.front();
|
|
|
|
SILLocation Loc = BB.back().getLoc();
|
|
BB.splitBasicBlock(BB.begin());
|
|
// Make terminator unreachable.
|
|
SILBuilder(&BB).createUnreachable(Loc);
|
|
|
|
DeadFunctions.push_back(&F);
|
|
}
|
|
}
|
|
}
|
|
// After running this pass all of the functions we will remove
|
|
// should consist only of one basic block terminated by
|
|
// UnreachableInst.
|
|
performSILDiagnoseUnreachable(M);
|
|
|
|
// Now mark all of these functions as public and remove their bodies.
|
|
for (auto &F : DeadFunctions) {
|
|
F->setLinkage(SILLinkage::PublicExternal);
|
|
F->getBlocks().clear();
|
|
}
|
|
|
|
// Remove dead functions.
|
|
SILPassManager PM(M);
|
|
PM.addDeadFunctionElimination();
|
|
PM.run();
|
|
}
|
|
|
|
int main(int argc, char **argv) {
|
|
INITIALIZE_LLVM(argc, argv);
|
|
|
|
llvm::cl::ParseCommandLineOptions(argc, argv, "Swift SIL Extractor\n");
|
|
|
|
CompilerInvocation Invocation;
|
|
|
|
Invocation.setMainExecutablePath(
|
|
llvm::sys::fs::getMainExecutable(argv[0],
|
|
reinterpret_cast<void *>(&anchorForGetMainExecutable)));
|
|
|
|
// Give the context the list of search paths to use for modules.
|
|
Invocation.setImportSearchPaths(ImportPaths);
|
|
// Set the SDK path and target if given.
|
|
if (SDKPath.getNumOccurrences() == 0) {
|
|
const char *SDKROOT = getenv("SDKROOT");
|
|
if (SDKROOT)
|
|
SDKPath = SDKROOT;
|
|
}
|
|
if (!SDKPath.empty())
|
|
Invocation.setSDKPath(SDKPath);
|
|
if (!Triple.empty())
|
|
Invocation.setTargetTriple(Triple);
|
|
if (!ResourceDir.empty())
|
|
Invocation.setRuntimeResourcePath(ResourceDir);
|
|
Invocation.getClangImporterOptions().ModuleCachePath = ModuleCachePath;
|
|
Invocation.setParseStdlib();
|
|
Invocation.getLangOptions().EnableAccessControl = false;
|
|
Invocation.getLangOptions().EnableObjCAttrRequiresFoundation = false;
|
|
|
|
// Load the input file.
|
|
llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> FileBufOrErr =
|
|
llvm::MemoryBuffer::getFileOrSTDIN(InputFilename);
|
|
if (!FileBufOrErr) {
|
|
fprintf(stderr, "Error! Failed to open file: %s\n", InputFilename.c_str());
|
|
exit(-1);
|
|
}
|
|
|
|
// If it looks like we have an AST, set the source file kind to SIL and the
|
|
// name of the module to the file's name.
|
|
Invocation.addInputBuffer(FileBufOrErr.get().get());
|
|
|
|
serialization::ExtendedValidationInfo extendedInfo;
|
|
auto result = serialization::validateSerializedAST(
|
|
FileBufOrErr.get()->getBuffer(), &extendedInfo);
|
|
bool HasSerializedAST = result.status == serialization::Status::Valid;
|
|
|
|
if (HasSerializedAST) {
|
|
const StringRef Stem = ModuleName.size() ?
|
|
StringRef(ModuleName) :
|
|
llvm::sys::path::stem(InputFilename);
|
|
Invocation.setModuleName(Stem);
|
|
Invocation.setInputKind(InputFileKind::IFK_Swift_Library);
|
|
} else {
|
|
Invocation.setModuleName("main");
|
|
Invocation.setInputKind(InputFileKind::IFK_SIL);
|
|
}
|
|
|
|
CompilerInstance CI;
|
|
PrintingDiagnosticConsumer PrintDiags;
|
|
CI.addDiagnosticConsumer(&PrintDiags);
|
|
|
|
if (CI.setup(Invocation))
|
|
return 1;
|
|
CI.performSema();
|
|
|
|
// If parsing produced an error, don't run any passes.
|
|
if (CI.getASTContext().hadError())
|
|
return 1;
|
|
|
|
// Load the SIL if we have a module. We have to do this after SILParse
|
|
// creating the unfortunate double if statement.
|
|
if (HasSerializedAST) {
|
|
assert(!CI.hasSILModule() &&
|
|
"performSema() should not create a SILModule.");
|
|
CI.setSILModule(SILModule::createEmptyModule(CI.getMainModule(),
|
|
CI.getSILOptions()));
|
|
std::unique_ptr<SerializedSILLoader> SL = SerializedSILLoader::create(
|
|
CI.getASTContext(), CI.getSILModule(), nullptr);
|
|
|
|
if (extendedInfo.isSIB())
|
|
SL->getAllForModule(CI.getMainModule()->getName(), nullptr);
|
|
else
|
|
SL->getAll();
|
|
}
|
|
|
|
if (!FunctionName.empty())
|
|
removeUnwantedFunctions(CI.getSILModule(), FunctionName);
|
|
|
|
std::error_code EC;
|
|
llvm::raw_fd_ostream OS(OutputFilename, EC, llvm::sys::fs::F_None);
|
|
if (EC) {
|
|
llvm::errs() << "while opening '" << OutputFilename << "': "
|
|
<< EC.message() << '\n';
|
|
return 1;
|
|
}
|
|
CI.getSILModule()->print(OS, EmitVerboseSIL, CI.getMainModule());
|
|
|
|
return CI.getASTContext().hadError();
|
|
}
|