aboutsummaryrefslogtreecommitdiff
path: root/lib/Transforms/NaCl/RemoveAsmMemory.cpp
blob: 5295fe5ff8a6255a488e06c06af90a9c0905e08d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
//===- RemoveAsmMemory.cpp - Remove ``asm("":::"memory")`` ----------------===//
//
//                     The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
// This pass removes all instances of ``asm("":::"memory")``.
//
//===----------------------------------------------------------------------===//

#include "llvm/ADT/Twine.h"
#include "llvm/IR/Function.h"
#include "llvm/IR/InlineAsm.h"
#include "llvm/IR/Instructions.h"
#include "llvm/IR/Module.h"
#include "llvm/InstVisitor.h"
#include "llvm/Pass.h"
#include <string>

using namespace llvm;

namespace {
class RemoveAsmMemory : public FunctionPass {
public:
  static char ID; // Pass identification, replacement for typeid
  RemoveAsmMemory() : FunctionPass(ID) {
    initializeRemoveAsmMemoryPass(*PassRegistry::getPassRegistry());
  }

  virtual bool runOnFunction(Function &F);
};

class AsmDirectivesVisitor : public InstVisitor<AsmDirectivesVisitor> {
public:
  AsmDirectivesVisitor(Function &F)
      : F(F), C(F.getParent()->getContext()), ModifiedFunction(false) {}
  ~AsmDirectivesVisitor() {}
  bool modifiedFunction() const { return ModifiedFunction; }

  /// Only Call Instructions are ever inline assembly directives.
  void visitCallInst(CallInst &CI);

private:
  Function &F;
  LLVMContext &C;
  bool ModifiedFunction;

  AsmDirectivesVisitor() LLVM_DELETED_FUNCTION;
  AsmDirectivesVisitor(const AsmDirectivesVisitor &) LLVM_DELETED_FUNCTION;
  AsmDirectivesVisitor &operator=(
      const AsmDirectivesVisitor &) LLVM_DELETED_FUNCTION;
};
}

char RemoveAsmMemory::ID = 0;
INITIALIZE_PASS(RemoveAsmMemory, "remove-asm-memory",
                "remove all instances of ``asm(\"\":::\"memory\")``", false,
                false)

bool RemoveAsmMemory::runOnFunction(Function &F) {
  AsmDirectivesVisitor AV(F);
  AV.visit(F);
  return AV.modifiedFunction();
}

void AsmDirectivesVisitor::visitCallInst(CallInst &CI) {
  if (!CI.isInlineAsm() ||
      !cast<InlineAsm>(CI.getCalledValue())->isAsmMemory())
    return;

  // In NaCl ``asm("":::"memory")`` always comes in pairs, straddling a
  // sequentially consistent fence. Other passes rewrite this fence to
  // an equivalent stable NaCl intrinsic, meaning that this assembly can
  // be removed.
  CI.eraseFromParent();
  ModifiedFunction = true;
}

namespace llvm {
FunctionPass *createRemoveAsmMemoryPass() { return new RemoveAsmMemory(); }
}