blob: 6f3742ed039bd9310564d7b57817f754c5e7c6bf [file] [log] [blame]
Tom Stellard5cbb53c2014-11-03 19:49:05 +00001//===-- AMDGPUAlwaysInlinePass.cpp - Promote Allocas ----------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10/// \file
11/// This pass marks all internal functions as always_inline and creates
Alfred Huangf9b521f2017-06-15 23:02:55 +000012/// duplicates of all other functions and marks the duplicates as always_inline.
Tom Stellard5cbb53c2014-11-03 19:49:05 +000013//
14//===----------------------------------------------------------------------===//
15
16#include "AMDGPU.h"
17#include "llvm/IR/Module.h"
18#include "llvm/Transforms/Utils/Cloning.h"
19
20using namespace llvm;
21
22namespace {
23
24class AMDGPUAlwaysInline : public ModulePass {
Stanislav Mekhanoshin89653df2017-03-30 20:16:02 +000025 bool GlobalOpt;
26
Tom Stellard5cbb53c2014-11-03 19:49:05 +000027public:
Matt Arsenault746e0652017-06-02 18:02:42 +000028 static char ID;
29
30 AMDGPUAlwaysInline(bool GlobalOpt = false) :
31 ModulePass(ID), GlobalOpt(GlobalOpt) { }
Tom Stellard5cbb53c2014-11-03 19:49:05 +000032 bool runOnModule(Module &M) override;
Mehdi Amini117296c2016-10-01 02:56:57 +000033 StringRef getPassName() const override { return "AMDGPU Always Inline Pass"; }
Tom Stellard5cbb53c2014-11-03 19:49:05 +000034};
35
36} // End anonymous namespace
37
Matt Arsenault746e0652017-06-02 18:02:42 +000038INITIALIZE_PASS(AMDGPUAlwaysInline, "amdgpu-always-inline",
39 "AMDGPU Inline All Functions", false, false)
40
Tom Stellard5cbb53c2014-11-03 19:49:05 +000041char AMDGPUAlwaysInline::ID = 0;
42
43bool AMDGPUAlwaysInline::runOnModule(Module &M) {
Nikolay Haustoveba80892016-08-31 11:18:33 +000044 std::vector<GlobalAlias*> AliasesToRemove;
Matt Arsenaultca95d442015-07-13 19:08:36 +000045 std::vector<Function *> FuncsToClone;
Tom Stellard5cbb53c2014-11-03 19:49:05 +000046
Nikolay Haustoveba80892016-08-31 11:18:33 +000047 for (GlobalAlias &A : M.aliases()) {
48 if (Function* F = dyn_cast<Function>(A.getAliasee())) {
49 A.replaceAllUsesWith(F);
50 AliasesToRemove.push_back(&A);
51 }
52 }
53
Stanislav Mekhanoshin89653df2017-03-30 20:16:02 +000054 if (GlobalOpt) {
55 for (GlobalAlias* A : AliasesToRemove) {
56 A->eraseFromParent();
57 }
Nikolay Haustoveba80892016-08-31 11:18:33 +000058 }
59
Matt Arsenaultca95d442015-07-13 19:08:36 +000060 for (Function &F : M) {
Matt Arsenaultdeaef8e2015-04-22 17:10:44 +000061 if (!F.hasLocalLinkage() && !F.isDeclaration() && !F.use_empty() &&
62 !F.hasFnAttribute(Attribute::NoInline))
Tom Stellard5cbb53c2014-11-03 19:49:05 +000063 FuncsToClone.push_back(&F);
64 }
65
66 for (Function *F : FuncsToClone) {
67 ValueToValueMapTy VMap;
Peter Collingbournedba99562016-05-10 20:23:24 +000068 Function *NewFunc = CloneFunction(F, VMap);
Tom Stellard5cbb53c2014-11-03 19:49:05 +000069 NewFunc->setLinkage(GlobalValue::InternalLinkage);
Tom Stellard5cbb53c2014-11-03 19:49:05 +000070 F->replaceAllUsesWith(NewFunc);
71 }
72
Matt Arsenaultca95d442015-07-13 19:08:36 +000073 for (Function &F : M) {
Matt Arsenaultdeaef8e2015-04-22 17:10:44 +000074 if (F.hasLocalLinkage() && !F.hasFnAttribute(Attribute::NoInline)) {
Tom Stellard5cbb53c2014-11-03 19:49:05 +000075 F.addFnAttr(Attribute::AlwaysInline);
76 }
77 }
78 return false;
79}
80
Stanislav Mekhanoshin89653df2017-03-30 20:16:02 +000081ModulePass *llvm::createAMDGPUAlwaysInlinePass(bool GlobalOpt) {
82 return new AMDGPUAlwaysInline(GlobalOpt);
Tom Stellard5cbb53c2014-11-03 19:49:05 +000083}