https://reviews.llvm.org/D67133 While investigating some non determinism (CSE doesn't produce wrong code, it just doesn't CSE some times) in GISel CSE on an out of tree target, I realized that the core issue was that there were lots of code that mutates (setReg, setRegClass etc), but doesn't notify observers (CSE in this case but this could be any other observer). In order to make the Observer be available in various parts of code and to avoid having to thread it through various API, the MachineFunction now has the observer as field. This allows it to be easily used in helper functions such as constrainOperandRegClass. Also added some invariant verification method in CSEInfo which can catch these issues (when CSE is enabled).
49 lines
1.7 KiB
C++
49 lines
1.7 KiB
C++
//===-- lib/CodeGen/GlobalISel/GISelChangeObserver.cpp --------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file constains common code to combine machine functions at generic
|
|
// level.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/CodeGen/GlobalISel/GISelChangeObserver.h"
|
|
#include "llvm/CodeGen/MachineRegisterInfo.h"
|
|
|
|
using namespace llvm;
|
|
|
|
void GISelChangeObserver::changingAllUsesOfReg(
|
|
const MachineRegisterInfo &MRI, unsigned Reg) {
|
|
for (auto &ChangingMI : MRI.use_instructions(Reg)) {
|
|
changingInstr(ChangingMI);
|
|
ChangingAllUsesOfReg.insert(&ChangingMI);
|
|
}
|
|
}
|
|
|
|
void GISelChangeObserver::finishedChangingAllUsesOfReg() {
|
|
for (auto *ChangedMI : ChangingAllUsesOfReg)
|
|
changedInstr(*ChangedMI);
|
|
ChangingAllUsesOfReg.clear();
|
|
}
|
|
|
|
RAIIDelegateInstaller::RAIIDelegateInstaller(MachineFunction &MF,
|
|
MachineFunction::Delegate *Del)
|
|
: MF(MF), Delegate(Del) {
|
|
// Register this as the delegate for handling insertions and deletions of
|
|
// instructions.
|
|
MF.setDelegate(Del);
|
|
}
|
|
|
|
RAIIDelegateInstaller::~RAIIDelegateInstaller() { MF.resetDelegate(Delegate); }
|
|
|
|
RAIIMFObserverInstaller::RAIIMFObserverInstaller(MachineFunction &MF,
|
|
GISelChangeObserver &Observer)
|
|
: MF(MF) {
|
|
MF.setObserver(&Observer);
|
|
}
|
|
|
|
RAIIMFObserverInstaller::~RAIIMFObserverInstaller() { MF.setObserver(nullptr); }
|