2012-01-13 06:30:30 +00:00
|
|
|
//===- MachineScheduler.cpp - Machine Instruction Scheduler ---------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// MachineScheduler schedules machine instructions after phi elimination. It
|
|
|
|
// preserves LiveIntervals so it can be invoked before register allocation.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#define DEBUG_TYPE "misched"
|
|
|
|
|
2012-04-24 20:36:19 +00:00
|
|
|
#include "RegisterClassInfo.h"
|
2012-04-24 17:56:43 +00:00
|
|
|
#include "RegisterPressure.h"
|
2012-01-13 06:30:30 +00:00
|
|
|
#include "llvm/CodeGen/LiveIntervalAnalysis.h"
|
2012-03-08 01:41:12 +00:00
|
|
|
#include "llvm/CodeGen/MachineScheduler.h"
|
2012-01-13 06:30:30 +00:00
|
|
|
#include "llvm/CodeGen/Passes.h"
|
2012-03-07 23:01:06 +00:00
|
|
|
#include "llvm/CodeGen/ScheduleDAGInstrs.h"
|
2012-01-13 06:30:30 +00:00
|
|
|
#include "llvm/Analysis/AliasAnalysis.h"
|
2012-01-14 02:17:09 +00:00
|
|
|
#include "llvm/Target/TargetInstrInfo.h"
|
2012-01-13 06:30:30 +00:00
|
|
|
#include "llvm/Support/CommandLine.h"
|
|
|
|
#include "llvm/Support/Debug.h"
|
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
#include "llvm/ADT/OwningPtr.h"
|
2012-03-14 04:00:41 +00:00
|
|
|
#include "llvm/ADT/PriorityQueue.h"
|
2012-01-13 06:30:30 +00:00
|
|
|
|
2012-01-17 06:55:07 +00:00
|
|
|
#include <queue>
|
|
|
|
|
2012-01-13 06:30:30 +00:00
|
|
|
using namespace llvm;
|
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
static cl::opt<bool> ForceTopDown("misched-topdown", cl::Hidden,
|
|
|
|
cl::desc("Force top-down list scheduling"));
|
|
|
|
static cl::opt<bool> ForceBottomUp("misched-bottomup", cl::Hidden,
|
|
|
|
cl::desc("Force bottom-up list scheduling"));
|
|
|
|
|
2012-03-07 00:18:25 +00:00
|
|
|
#ifndef NDEBUG
|
|
|
|
static cl::opt<bool> ViewMISchedDAGs("view-misched-dags", cl::Hidden,
|
|
|
|
cl::desc("Pop up a window to show MISched dags after they are processed"));
|
2012-03-19 18:38:38 +00:00
|
|
|
|
|
|
|
static cl::opt<unsigned> MISchedCutoff("misched-cutoff", cl::Hidden,
|
|
|
|
cl::desc("Stop scheduling after N instructions"), cl::init(~0U));
|
2012-03-07 00:18:25 +00:00
|
|
|
#else
|
|
|
|
static bool ViewMISchedDAGs = false;
|
|
|
|
#endif // NDEBUG
|
|
|
|
|
2012-01-14 02:17:06 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Machine Instruction Scheduling Pass and Registry
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2012-04-24 20:36:19 +00:00
|
|
|
MachineSchedContext::MachineSchedContext():
|
|
|
|
MF(0), MLI(0), MDT(0), PassConfig(0), AA(0), LIS(0) {
|
|
|
|
RegClassInfo = new RegisterClassInfo();
|
|
|
|
}
|
|
|
|
|
|
|
|
MachineSchedContext::~MachineSchedContext() {
|
|
|
|
delete RegClassInfo;
|
|
|
|
}
|
|
|
|
|
2012-01-13 06:30:30 +00:00
|
|
|
namespace {
|
2012-01-17 06:55:03 +00:00
|
|
|
/// MachineScheduler runs after coalescing and before register allocation.
|
2012-03-08 01:41:12 +00:00
|
|
|
class MachineScheduler : public MachineSchedContext,
|
|
|
|
public MachineFunctionPass {
|
2012-01-13 06:30:30 +00:00
|
|
|
public:
|
2012-01-17 06:55:03 +00:00
|
|
|
MachineScheduler();
|
2012-01-13 06:30:30 +00:00
|
|
|
|
|
|
|
virtual void getAnalysisUsage(AnalysisUsage &AU) const;
|
|
|
|
|
|
|
|
virtual void releaseMemory() {}
|
|
|
|
|
|
|
|
virtual bool runOnMachineFunction(MachineFunction&);
|
|
|
|
|
|
|
|
virtual void print(raw_ostream &O, const Module* = 0) const;
|
|
|
|
|
|
|
|
static char ID; // Class identification, replacement for typeinfo
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
2012-01-17 06:55:03 +00:00
|
|
|
char MachineScheduler::ID = 0;
|
2012-01-13 06:30:30 +00:00
|
|
|
|
2012-01-17 06:55:03 +00:00
|
|
|
char &llvm::MachineSchedulerID = MachineScheduler::ID;
|
2012-01-13 06:30:30 +00:00
|
|
|
|
2012-01-17 06:55:03 +00:00
|
|
|
INITIALIZE_PASS_BEGIN(MachineScheduler, "misched",
|
2012-01-13 06:30:30 +00:00
|
|
|
"Machine Instruction Scheduler", false, false)
|
|
|
|
INITIALIZE_AG_DEPENDENCY(AliasAnalysis)
|
|
|
|
INITIALIZE_PASS_DEPENDENCY(SlotIndexes)
|
|
|
|
INITIALIZE_PASS_DEPENDENCY(LiveIntervals)
|
2012-01-17 06:55:03 +00:00
|
|
|
INITIALIZE_PASS_END(MachineScheduler, "misched",
|
2012-01-13 06:30:30 +00:00
|
|
|
"Machine Instruction Scheduler", false, false)
|
|
|
|
|
2012-01-17 06:55:03 +00:00
|
|
|
MachineScheduler::MachineScheduler()
|
2012-03-08 01:41:12 +00:00
|
|
|
: MachineFunctionPass(ID) {
|
2012-01-17 06:55:03 +00:00
|
|
|
initializeMachineSchedulerPass(*PassRegistry::getPassRegistry());
|
2012-01-13 06:30:30 +00:00
|
|
|
}
|
|
|
|
|
2012-01-17 06:55:03 +00:00
|
|
|
void MachineScheduler::getAnalysisUsage(AnalysisUsage &AU) const {
|
2012-01-13 06:30:30 +00:00
|
|
|
AU.setPreservesCFG();
|
|
|
|
AU.addRequiredID(MachineDominatorsID);
|
|
|
|
AU.addRequired<MachineLoopInfo>();
|
|
|
|
AU.addRequired<AliasAnalysis>();
|
2012-03-09 00:52:20 +00:00
|
|
|
AU.addRequired<TargetPassConfig>();
|
2012-01-13 06:30:30 +00:00
|
|
|
AU.addRequired<SlotIndexes>();
|
|
|
|
AU.addPreserved<SlotIndexes>();
|
|
|
|
AU.addRequired<LiveIntervals>();
|
|
|
|
AU.addPreserved<LiveIntervals>();
|
|
|
|
MachineFunctionPass::getAnalysisUsage(AU);
|
|
|
|
}
|
|
|
|
|
|
|
|
MachinePassRegistry MachineSchedRegistry::Registry;
|
|
|
|
|
2012-03-09 00:52:20 +00:00
|
|
|
/// A dummy default scheduler factory indicates whether the scheduler
|
|
|
|
/// is overridden on the command line.
|
|
|
|
static ScheduleDAGInstrs *useDefaultMachineSched(MachineSchedContext *C) {
|
|
|
|
return 0;
|
|
|
|
}
|
2012-01-13 06:30:30 +00:00
|
|
|
|
|
|
|
/// MachineSchedOpt allows command line selection of the scheduler.
|
|
|
|
static cl::opt<MachineSchedRegistry::ScheduleDAGCtor, false,
|
|
|
|
RegisterPassParser<MachineSchedRegistry> >
|
|
|
|
MachineSchedOpt("misched",
|
2012-03-09 00:52:20 +00:00
|
|
|
cl::init(&useDefaultMachineSched), cl::Hidden,
|
2012-01-13 06:30:30 +00:00
|
|
|
cl::desc("Machine instruction scheduler to use"));
|
|
|
|
|
2012-03-09 00:52:20 +00:00
|
|
|
static MachineSchedRegistry
|
2012-03-14 04:00:41 +00:00
|
|
|
DefaultSchedRegistry("default", "Use the target's default scheduler choice.",
|
2012-03-09 00:52:20 +00:00
|
|
|
useDefaultMachineSched);
|
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
/// Forward declare the standard machine scheduler. This will be used as the
|
2012-03-09 00:52:20 +00:00
|
|
|
/// default scheduler if the target does not set a default.
|
2012-03-14 04:00:41 +00:00
|
|
|
static ScheduleDAGInstrs *createConvergingSched(MachineSchedContext *C);
|
2012-03-09 00:52:20 +00:00
|
|
|
|
2012-04-24 18:04:34 +00:00
|
|
|
|
|
|
|
/// Decrement this iterator until reaching the top or a non-debug instr.
|
|
|
|
static MachineBasicBlock::iterator
|
|
|
|
priorNonDebug(MachineBasicBlock::iterator I, MachineBasicBlock::iterator Beg) {
|
|
|
|
assert(I != Beg && "reached the top of the region, cannot decrement");
|
|
|
|
while (--I != Beg) {
|
|
|
|
if (!I->isDebugValue())
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return I;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// If this iterator is a debug value, increment until reaching the End or a
|
|
|
|
/// non-debug instruction.
|
|
|
|
static MachineBasicBlock::iterator
|
|
|
|
nextIfDebug(MachineBasicBlock::iterator I, MachineBasicBlock::iterator End) {
|
|
|
|
while(I != End) {
|
|
|
|
if (!I->isDebugValue())
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return I;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:00:38 +00:00
|
|
|
/// Top-level MachineScheduler pass driver.
|
|
|
|
///
|
|
|
|
/// Visit blocks in function order. Divide each block into scheduling regions
|
2012-03-14 04:00:41 +00:00
|
|
|
/// and visit them bottom-up. Visiting regions bottom-up is not required, but is
|
|
|
|
/// consistent with the DAG builder, which traverses the interior of the
|
|
|
|
/// scheduling regions bottom-up.
|
2012-03-14 04:00:38 +00:00
|
|
|
///
|
|
|
|
/// This design avoids exposing scheduling boundaries to the DAG builder,
|
2012-03-14 04:00:41 +00:00
|
|
|
/// simplifying the DAG builder's support for "special" target instructions.
|
|
|
|
/// At the same time the design allows target schedulers to operate across
|
2012-03-14 04:00:38 +00:00
|
|
|
/// scheduling boundaries, for example to bundle the boudary instructions
|
|
|
|
/// without reordering them. This creates complexity, because the target
|
|
|
|
/// scheduler must update the RegionBegin and RegionEnd positions cached by
|
|
|
|
/// ScheduleDAGInstrs whenever adding or removing instructions. A much simpler
|
|
|
|
/// design would be to split blocks at scheduling boundaries, but LLVM has a
|
|
|
|
/// general bias against block splitting purely for implementation simplicity.
|
2012-03-08 01:41:12 +00:00
|
|
|
bool MachineScheduler::runOnMachineFunction(MachineFunction &mf) {
|
|
|
|
// Initialize the context of the pass.
|
|
|
|
MF = &mf;
|
|
|
|
MLI = &getAnalysis<MachineLoopInfo>();
|
|
|
|
MDT = &getAnalysis<MachineDominatorTree>();
|
2012-03-09 00:52:20 +00:00
|
|
|
PassConfig = &getAnalysis<TargetPassConfig>();
|
2012-03-08 01:41:12 +00:00
|
|
|
AA = &getAnalysis<AliasAnalysis>();
|
|
|
|
|
|
|
|
LIS = &getAnalysis<LiveIntervals>();
|
|
|
|
const TargetInstrInfo *TII = MF->getTarget().getInstrInfo();
|
|
|
|
|
2012-04-24 20:36:19 +00:00
|
|
|
RegClassInfo->runOnMachineFunction(*MF);
|
2012-04-24 17:56:43 +00:00
|
|
|
|
2012-03-08 01:41:12 +00:00
|
|
|
// Select the scheduler, or set the default.
|
2012-03-09 00:52:20 +00:00
|
|
|
MachineSchedRegistry::ScheduleDAGCtor Ctor = MachineSchedOpt;
|
|
|
|
if (Ctor == useDefaultMachineSched) {
|
|
|
|
// Get the default scheduler set by the target.
|
|
|
|
Ctor = MachineSchedRegistry::getDefault();
|
|
|
|
if (!Ctor) {
|
2012-03-14 04:00:41 +00:00
|
|
|
Ctor = createConvergingSched;
|
2012-03-09 00:52:20 +00:00
|
|
|
MachineSchedRegistry::setDefault(Ctor);
|
|
|
|
}
|
2012-03-08 01:41:12 +00:00
|
|
|
}
|
|
|
|
// Instantiate the selected scheduler.
|
|
|
|
OwningPtr<ScheduleDAGInstrs> Scheduler(Ctor(this));
|
|
|
|
|
|
|
|
// Visit all machine basic blocks.
|
2012-04-24 17:56:43 +00:00
|
|
|
//
|
|
|
|
// TODO: Visit blocks in global postorder or postorder within the bottom-up
|
|
|
|
// loop tree. Then we can optionally compute global RegPressure.
|
2012-03-08 01:41:12 +00:00
|
|
|
for (MachineFunction::iterator MBB = MF->begin(), MBBEnd = MF->end();
|
|
|
|
MBB != MBBEnd; ++MBB) {
|
|
|
|
|
2012-03-09 08:02:51 +00:00
|
|
|
Scheduler->startBlock(MBB);
|
|
|
|
|
2012-03-08 01:41:12 +00:00
|
|
|
// Break the block into scheduling regions [I, RegionEnd), and schedule each
|
2012-03-09 22:34:56 +00:00
|
|
|
// region as soon as it is discovered. RegionEnd points the the scheduling
|
|
|
|
// boundary at the bottom of the region. The DAG does not include RegionEnd,
|
|
|
|
// but the region does (i.e. the next RegionEnd is above the previous
|
|
|
|
// RegionBegin). If the current block has no terminator then RegionEnd ==
|
|
|
|
// MBB->end() for the bottom region.
|
|
|
|
//
|
|
|
|
// The Scheduler may insert instructions during either schedule() or
|
|
|
|
// exitRegion(), even for empty regions. So the local iterators 'I' and
|
|
|
|
// 'RegionEnd' are invalid across these calls.
|
2012-03-08 01:41:12 +00:00
|
|
|
unsigned RemainingCount = MBB->size();
|
2012-03-09 03:46:39 +00:00
|
|
|
for(MachineBasicBlock::iterator RegionEnd = MBB->end();
|
2012-03-09 22:34:56 +00:00
|
|
|
RegionEnd != MBB->begin(); RegionEnd = Scheduler->begin()) {
|
2012-04-24 17:56:43 +00:00
|
|
|
|
2012-03-09 08:02:51 +00:00
|
|
|
// Avoid decrementing RegionEnd for blocks with no terminator.
|
|
|
|
if (RegionEnd != MBB->end()
|
|
|
|
|| TII->isSchedulingBoundary(llvm::prior(RegionEnd), MBB, *MF)) {
|
|
|
|
--RegionEnd;
|
|
|
|
// Count the boundary instruction.
|
|
|
|
--RemainingCount;
|
|
|
|
}
|
|
|
|
|
2012-03-08 01:41:12 +00:00
|
|
|
// The next region starts above the previous region. Look backward in the
|
|
|
|
// instruction stream until we find the nearest boundary.
|
|
|
|
MachineBasicBlock::iterator I = RegionEnd;
|
2012-03-09 03:46:39 +00:00
|
|
|
for(;I != MBB->begin(); --I, --RemainingCount) {
|
2012-03-08 01:41:12 +00:00
|
|
|
if (TII->isSchedulingBoundary(llvm::prior(I), MBB, *MF))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Notify the scheduler of the region, even if we may skip scheduling
|
|
|
|
// it. Perhaps it still needs to be bundled.
|
|
|
|
Scheduler->enterRegion(MBB, I, RegionEnd, RemainingCount);
|
|
|
|
|
|
|
|
// Skip empty scheduling regions (0 or 1 schedulable instructions).
|
|
|
|
if (I == RegionEnd || I == llvm::prior(RegionEnd)) {
|
|
|
|
// Close the current region. Bundle the terminator if needed.
|
2012-03-09 22:34:56 +00:00
|
|
|
// This invalidates 'RegionEnd' and 'I'.
|
2012-03-08 01:41:12 +00:00
|
|
|
Scheduler->exitRegion();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
DEBUG(dbgs() << "MachineScheduling " << MF->getFunction()->getName()
|
|
|
|
<< ":BB#" << MBB->getNumber() << "\n From: " << *I << " To: ";
|
|
|
|
if (RegionEnd != MBB->end()) dbgs() << *RegionEnd;
|
|
|
|
else dbgs() << "End";
|
|
|
|
dbgs() << " Remaining: " << RemainingCount << "\n");
|
|
|
|
|
2012-03-09 03:46:42 +00:00
|
|
|
// Schedule a region: possibly reorder instructions.
|
2012-03-09 22:34:56 +00:00
|
|
|
// This invalidates 'RegionEnd' and 'I'.
|
2012-03-08 01:41:12 +00:00
|
|
|
Scheduler->schedule();
|
2012-03-09 03:46:42 +00:00
|
|
|
|
|
|
|
// Close the current region.
|
2012-03-08 01:41:12 +00:00
|
|
|
Scheduler->exitRegion();
|
|
|
|
|
|
|
|
// Scheduling has invalidated the current iterator 'I'. Ask the
|
|
|
|
// scheduler for the top of it's scheduled region.
|
|
|
|
RegionEnd = Scheduler->begin();
|
|
|
|
}
|
|
|
|
assert(RemainingCount == 0 && "Instruction count mismatch!");
|
|
|
|
Scheduler->finishBlock();
|
|
|
|
}
|
2012-04-01 07:24:23 +00:00
|
|
|
Scheduler->finalizeSchedule();
|
2012-03-21 04:12:12 +00:00
|
|
|
DEBUG(LIS->print(dbgs()));
|
2012-03-08 01:41:12 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MachineScheduler::print(raw_ostream &O, const Module* m) const {
|
|
|
|
// unimplemented
|
|
|
|
}
|
|
|
|
|
2012-01-14 02:17:06 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
2012-03-14 04:00:41 +00:00
|
|
|
// MachineSchedStrategy - Interface to a machine scheduling algorithm.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
class ScheduleDAGMI;
|
|
|
|
|
|
|
|
/// MachineSchedStrategy - Interface used by ScheduleDAGMI to drive the selected
|
|
|
|
/// scheduling algorithm.
|
|
|
|
///
|
|
|
|
/// If this works well and targets wish to reuse ScheduleDAGMI, we may expose it
|
|
|
|
/// in ScheduleDAGInstrs.h
|
|
|
|
class MachineSchedStrategy {
|
|
|
|
public:
|
|
|
|
virtual ~MachineSchedStrategy() {}
|
|
|
|
|
|
|
|
/// Initialize the strategy after building the DAG for a new region.
|
|
|
|
virtual void initialize(ScheduleDAGMI *DAG) = 0;
|
|
|
|
|
|
|
|
/// Pick the next node to schedule, or return NULL. Set IsTopNode to true to
|
|
|
|
/// schedule the node at the top of the unscheduled region. Otherwise it will
|
|
|
|
/// be scheduled at the bottom.
|
|
|
|
virtual SUnit *pickNode(bool &IsTopNode) = 0;
|
|
|
|
|
|
|
|
/// When all predecessor dependencies have been resolved, free this node for
|
|
|
|
/// top-down scheduling.
|
|
|
|
virtual void releaseTopNode(SUnit *SU) = 0;
|
|
|
|
/// When all successor dependencies have been resolved, free this node for
|
|
|
|
/// bottom-up scheduling.
|
|
|
|
virtual void releaseBottomNode(SUnit *SU) = 0;
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// ScheduleDAGMI - Base class for MachineInstr scheduling with LiveIntervals
|
|
|
|
// preservation.
|
|
|
|
//===----------------------------------------------------------------------===//
|
2012-01-14 02:17:06 +00:00
|
|
|
|
|
|
|
namespace {
|
2012-03-14 04:00:41 +00:00
|
|
|
/// ScheduleDAGMI is an implementation of ScheduleDAGInstrs that schedules
|
2012-01-14 02:17:06 +00:00
|
|
|
/// machine instructions while updating LiveIntervals.
|
2012-03-14 04:00:41 +00:00
|
|
|
class ScheduleDAGMI : public ScheduleDAGInstrs {
|
2012-03-08 01:41:12 +00:00
|
|
|
AliasAnalysis *AA;
|
2012-04-24 17:56:43 +00:00
|
|
|
RegisterClassInfo *RegClassInfo;
|
2012-03-14 04:00:41 +00:00
|
|
|
MachineSchedStrategy *SchedImpl;
|
|
|
|
|
2012-04-24 17:56:43 +00:00
|
|
|
// Register pressure in this region computed by buildSchedGraph.
|
|
|
|
IntervalPressure RegPressure;
|
|
|
|
RegPressureTracker RPTracker;
|
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
/// The top of the unscheduled zone.
|
|
|
|
MachineBasicBlock::iterator CurrentTop;
|
|
|
|
|
|
|
|
/// The bottom of the unscheduled zone.
|
|
|
|
MachineBasicBlock::iterator CurrentBottom;
|
2012-03-19 18:38:38 +00:00
|
|
|
|
|
|
|
/// The number of instructions scheduled so far. Used to cut off the
|
|
|
|
/// scheduler at the point determined by misched-cutoff.
|
|
|
|
unsigned NumInstrsScheduled;
|
2012-01-14 02:17:06 +00:00
|
|
|
public:
|
2012-03-14 04:00:41 +00:00
|
|
|
ScheduleDAGMI(MachineSchedContext *C, MachineSchedStrategy *S):
|
2012-03-08 01:41:12 +00:00
|
|
|
ScheduleDAGInstrs(*C->MF, *C->MLI, *C->MDT, /*IsPostRA=*/false, C->LIS),
|
2012-04-24 20:36:19 +00:00
|
|
|
AA(C->AA), RegClassInfo(C->RegClassInfo), SchedImpl(S),
|
2012-04-24 17:56:43 +00:00
|
|
|
RPTracker(RegPressure), CurrentTop(), CurrentBottom(),
|
2012-03-19 18:38:38 +00:00
|
|
|
NumInstrsScheduled(0) {}
|
2012-01-17 06:55:07 +00:00
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
~ScheduleDAGMI() {
|
|
|
|
delete SchedImpl;
|
|
|
|
}
|
2012-01-17 06:55:07 +00:00
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
MachineBasicBlock::iterator top() const { return CurrentTop; }
|
|
|
|
MachineBasicBlock::iterator bottom() const { return CurrentBottom; }
|
2012-01-17 06:55:07 +00:00
|
|
|
|
2012-04-24 17:56:43 +00:00
|
|
|
/// Implement the ScheduleDAGInstrs interface for handling the next scheduling
|
|
|
|
/// region. This covers all instructions in a block, while schedule() may only
|
|
|
|
/// cover a subset.
|
|
|
|
void enterRegion(MachineBasicBlock *bb,
|
|
|
|
MachineBasicBlock::iterator begin,
|
|
|
|
MachineBasicBlock::iterator end,
|
|
|
|
unsigned endcount);
|
|
|
|
|
|
|
|
/// Implement ScheduleDAGInstrs interface for scheduling a sequence of
|
|
|
|
/// reorderable instructions.
|
2012-03-14 04:00:41 +00:00
|
|
|
void schedule();
|
2012-01-17 06:55:07 +00:00
|
|
|
|
|
|
|
protected:
|
2012-03-14 04:00:41 +00:00
|
|
|
void moveInstruction(MachineInstr *MI, MachineBasicBlock::iterator InsertPos);
|
2012-03-21 04:12:07 +00:00
|
|
|
bool checkSchedLimit();
|
2012-03-14 04:00:41 +00:00
|
|
|
|
2012-01-17 06:55:07 +00:00
|
|
|
void releaseSucc(SUnit *SU, SDep *SuccEdge);
|
|
|
|
void releaseSuccessors(SUnit *SU);
|
2012-03-14 04:00:41 +00:00
|
|
|
void releasePred(SUnit *SU, SDep *PredEdge);
|
|
|
|
void releasePredecessors(SUnit *SU);
|
2012-04-24 18:04:37 +00:00
|
|
|
|
|
|
|
void placeDebugValues();
|
2012-01-14 02:17:06 +00:00
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
2012-01-17 06:55:07 +00:00
|
|
|
/// ReleaseSucc - Decrement the NumPredsLeft count of a successor. When
|
|
|
|
/// NumPredsLeft reaches zero, release the successor node.
|
2012-03-14 04:00:41 +00:00
|
|
|
void ScheduleDAGMI::releaseSucc(SUnit *SU, SDep *SuccEdge) {
|
2012-01-17 06:55:07 +00:00
|
|
|
SUnit *SuccSU = SuccEdge->getSUnit();
|
|
|
|
|
|
|
|
#ifndef NDEBUG
|
|
|
|
if (SuccSU->NumPredsLeft == 0) {
|
|
|
|
dbgs() << "*** Scheduling failed! ***\n";
|
|
|
|
SuccSU->dump(this);
|
|
|
|
dbgs() << " has been released too many times!\n";
|
|
|
|
llvm_unreachable(0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
--SuccSU->NumPredsLeft;
|
|
|
|
if (SuccSU->NumPredsLeft == 0 && SuccSU != &ExitSU)
|
2012-03-14 04:00:41 +00:00
|
|
|
SchedImpl->releaseTopNode(SuccSU);
|
2012-01-17 06:55:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// releaseSuccessors - Call releaseSucc on each of SU's successors.
|
2012-03-14 04:00:41 +00:00
|
|
|
void ScheduleDAGMI::releaseSuccessors(SUnit *SU) {
|
2012-01-17 06:55:07 +00:00
|
|
|
for (SUnit::succ_iterator I = SU->Succs.begin(), E = SU->Succs.end();
|
|
|
|
I != E; ++I) {
|
|
|
|
releaseSucc(SU, &*I);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
/// ReleasePred - Decrement the NumSuccsLeft count of a predecessor. When
|
|
|
|
/// NumSuccsLeft reaches zero, release the predecessor node.
|
|
|
|
void ScheduleDAGMI::releasePred(SUnit *SU, SDep *PredEdge) {
|
|
|
|
SUnit *PredSU = PredEdge->getSUnit();
|
|
|
|
|
|
|
|
#ifndef NDEBUG
|
|
|
|
if (PredSU->NumSuccsLeft == 0) {
|
|
|
|
dbgs() << "*** Scheduling failed! ***\n";
|
|
|
|
PredSU->dump(this);
|
|
|
|
dbgs() << " has been released too many times!\n";
|
|
|
|
llvm_unreachable(0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
--PredSU->NumSuccsLeft;
|
|
|
|
if (PredSU->NumSuccsLeft == 0 && PredSU != &EntrySU)
|
|
|
|
SchedImpl->releaseBottomNode(PredSU);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// releasePredecessors - Call releasePred on each of SU's predecessors.
|
|
|
|
void ScheduleDAGMI::releasePredecessors(SUnit *SU) {
|
|
|
|
for (SUnit::pred_iterator I = SU->Preds.begin(), E = SU->Preds.end();
|
|
|
|
I != E; ++I) {
|
|
|
|
releasePred(SU, &*I);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ScheduleDAGMI::moveInstruction(MachineInstr *MI,
|
|
|
|
MachineBasicBlock::iterator InsertPos) {
|
2012-03-21 04:12:10 +00:00
|
|
|
// Fix RegionBegin if the first instruction moves down.
|
|
|
|
if (&*RegionBegin == MI)
|
|
|
|
RegionBegin = llvm::next(RegionBegin);
|
2012-03-14 04:00:41 +00:00
|
|
|
BB->splice(InsertPos, BB, MI);
|
|
|
|
LIS->handleMove(MI);
|
2012-03-21 04:12:10 +00:00
|
|
|
// Fix RegionBegin if another instruction moves above the first instruction.
|
2012-03-14 04:00:41 +00:00
|
|
|
if (RegionBegin == InsertPos)
|
|
|
|
RegionBegin = MI;
|
|
|
|
}
|
|
|
|
|
2012-03-21 04:12:07 +00:00
|
|
|
bool ScheduleDAGMI::checkSchedLimit() {
|
|
|
|
#ifndef NDEBUG
|
|
|
|
if (NumInstrsScheduled == MISchedCutoff && MISchedCutoff != ~0U) {
|
|
|
|
CurrentTop = CurrentBottom;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
++NumInstrsScheduled;
|
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-04-24 17:56:43 +00:00
|
|
|
/// enterRegion - Called back from MachineScheduler::runOnMachineFunction after
|
|
|
|
/// crossing a scheduling boundary. [begin, end) includes all instructions in
|
|
|
|
/// the region, including the boundary itself and single-instruction regions
|
|
|
|
/// that don't get scheduled.
|
|
|
|
void ScheduleDAGMI::enterRegion(MachineBasicBlock *bb,
|
|
|
|
MachineBasicBlock::iterator begin,
|
|
|
|
MachineBasicBlock::iterator end,
|
|
|
|
unsigned endcount)
|
|
|
|
{
|
|
|
|
ScheduleDAGInstrs::enterRegion(bb, begin, end, endcount);
|
|
|
|
// Setup the register pressure tracker to begin tracking at the end of this
|
|
|
|
// region.
|
|
|
|
RPTracker.init(&MF, RegClassInfo, LIS, BB, end);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
/// schedule - Called back from MachineScheduler::runOnMachineFunction
|
2012-04-24 17:56:43 +00:00
|
|
|
/// after setting up the current scheduling region. [RegionBegin, RegionEnd)
|
|
|
|
/// only includes instructions that have DAG nodes, not scheduling boundaries.
|
2012-03-14 04:00:41 +00:00
|
|
|
void ScheduleDAGMI::schedule() {
|
2012-04-24 17:56:43 +00:00
|
|
|
while(RPTracker.getPos() != RegionEnd) {
|
|
|
|
bool Moved = RPTracker.recede();
|
|
|
|
assert(Moved && "Regpressure tracker cannot find RegionEnd"); (void)Moved;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build the DAG.
|
|
|
|
buildSchedGraph(AA, &RPTracker);
|
2012-01-17 06:55:07 +00:00
|
|
|
|
|
|
|
DEBUG(dbgs() << "********** MI Scheduling **********\n");
|
|
|
|
DEBUG(for (unsigned su = 0, e = SUnits.size(); su != e; ++su)
|
|
|
|
SUnits[su].dumpAll(this));
|
|
|
|
|
2012-03-07 00:18:25 +00:00
|
|
|
if (ViewMISchedDAGs) viewGraph();
|
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
SchedImpl->initialize(this);
|
|
|
|
|
|
|
|
// Release edges from the special Entry node or to the special Exit node.
|
2012-01-17 06:55:07 +00:00
|
|
|
releaseSuccessors(&EntrySU);
|
2012-03-14 04:00:41 +00:00
|
|
|
releasePredecessors(&ExitSU);
|
2012-01-17 06:55:07 +00:00
|
|
|
|
|
|
|
// Release all DAG roots for scheduling.
|
|
|
|
for (std::vector<SUnit>::iterator I = SUnits.begin(), E = SUnits.end();
|
|
|
|
I != E; ++I) {
|
2012-03-14 04:00:41 +00:00
|
|
|
// A SUnit is ready to top schedule if it has no predecessors.
|
2012-01-17 06:55:07 +00:00
|
|
|
if (I->Preds.empty())
|
2012-03-14 04:00:41 +00:00
|
|
|
SchedImpl->releaseTopNode(&(*I));
|
|
|
|
// A SUnit is ready to bottom schedule if it has no successors.
|
|
|
|
if (I->Succs.empty())
|
|
|
|
SchedImpl->releaseBottomNode(&(*I));
|
2012-01-17 06:55:07 +00:00
|
|
|
}
|
|
|
|
|
2012-04-24 18:04:34 +00:00
|
|
|
CurrentTop = nextIfDebug(RegionBegin, RegionEnd);
|
2012-03-14 04:00:41 +00:00
|
|
|
CurrentBottom = RegionEnd;
|
|
|
|
bool IsTopNode = false;
|
|
|
|
while (SUnit *SU = SchedImpl->pickNode(IsTopNode)) {
|
|
|
|
DEBUG(dbgs() << "*** " << (IsTopNode ? "Top" : "Bottom")
|
|
|
|
<< " Scheduling Instruction:\n"; SU->dump(this));
|
2012-03-21 04:12:07 +00:00
|
|
|
if (!checkSchedLimit())
|
|
|
|
break;
|
2012-01-17 06:55:07 +00:00
|
|
|
|
|
|
|
// Move the instruction to its new location in the instruction stream.
|
|
|
|
MachineInstr *MI = SU->getInstr();
|
2012-03-14 04:00:41 +00:00
|
|
|
|
|
|
|
if (IsTopNode) {
|
|
|
|
assert(SU->isTopReady() && "node still has unscheduled dependencies");
|
|
|
|
if (&*CurrentTop == MI)
|
2012-04-24 18:04:34 +00:00
|
|
|
CurrentTop = nextIfDebug(++CurrentTop, CurrentBottom);
|
2012-03-14 04:00:41 +00:00
|
|
|
else
|
|
|
|
moveInstruction(MI, CurrentTop);
|
|
|
|
// Release dependent instructions for scheduling.
|
|
|
|
releaseSuccessors(SU);
|
|
|
|
}
|
2012-01-17 06:55:07 +00:00
|
|
|
else {
|
2012-03-14 04:00:41 +00:00
|
|
|
assert(SU->isBottomReady() && "node still has unscheduled dependencies");
|
2012-04-24 18:04:34 +00:00
|
|
|
MachineBasicBlock::iterator priorII =
|
|
|
|
priorNonDebug(CurrentBottom, CurrentTop);
|
|
|
|
if (&*priorII == MI)
|
|
|
|
CurrentBottom = priorII;
|
2012-03-14 04:00:41 +00:00
|
|
|
else {
|
2012-03-21 04:12:10 +00:00
|
|
|
if (&*CurrentTop == MI)
|
2012-04-24 18:04:34 +00:00
|
|
|
CurrentTop = nextIfDebug(++CurrentTop, CurrentBottom);
|
2012-03-14 04:00:41 +00:00
|
|
|
moveInstruction(MI, CurrentBottom);
|
|
|
|
CurrentBottom = MI;
|
|
|
|
}
|
|
|
|
// Release dependent instructions for scheduling.
|
|
|
|
releasePredecessors(SU);
|
2012-01-17 06:55:07 +00:00
|
|
|
}
|
2012-03-14 04:00:41 +00:00
|
|
|
SU->isScheduled = true;
|
2012-01-17 06:55:07 +00:00
|
|
|
}
|
2012-03-14 04:00:41 +00:00
|
|
|
assert(CurrentTop == CurrentBottom && "Nonempty unscheduled zone.");
|
2012-04-24 18:04:37 +00:00
|
|
|
|
|
|
|
placeDebugValues();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Reinsert any remaining debug_values, just like the PostRA scheduler.
|
|
|
|
void ScheduleDAGMI::placeDebugValues() {
|
|
|
|
// If first instruction was a DBG_VALUE then put it back.
|
|
|
|
if (FirstDbgValue) {
|
|
|
|
BB->splice(RegionBegin, BB, FirstDbgValue);
|
|
|
|
RegionBegin = FirstDbgValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (std::vector<std::pair<MachineInstr *, MachineInstr *> >::iterator
|
|
|
|
DI = DbgValues.end(), DE = DbgValues.begin(); DI != DE; --DI) {
|
|
|
|
std::pair<MachineInstr *, MachineInstr *> P = *prior(DI);
|
|
|
|
MachineInstr *DbgValue = P.first;
|
|
|
|
MachineBasicBlock::iterator OrigPrevMI = P.second;
|
|
|
|
BB->splice(++OrigPrevMI, BB, DbgValue);
|
|
|
|
if (OrigPrevMI == llvm::prior(RegionEnd))
|
|
|
|
RegionEnd = DbgValue;
|
|
|
|
}
|
|
|
|
DbgValues.clear();
|
|
|
|
FirstDbgValue = NULL;
|
2012-01-17 06:55:07 +00:00
|
|
|
}
|
|
|
|
|
2012-01-17 06:55:03 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
2012-03-14 04:00:41 +00:00
|
|
|
// ConvergingScheduler - Implementation of the standard MachineSchedStrategy.
|
2012-01-17 06:55:03 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
namespace {
|
2012-03-14 04:00:41 +00:00
|
|
|
/// ConvergingScheduler shrinks the unscheduled zone using heuristics to balance
|
|
|
|
/// the schedule.
|
|
|
|
class ConvergingScheduler : public MachineSchedStrategy {
|
|
|
|
ScheduleDAGMI *DAG;
|
|
|
|
|
|
|
|
unsigned NumTopReady;
|
|
|
|
unsigned NumBottomReady;
|
|
|
|
|
2012-01-17 06:55:03 +00:00
|
|
|
public:
|
2012-03-14 04:00:41 +00:00
|
|
|
virtual void initialize(ScheduleDAGMI *dag) {
|
|
|
|
DAG = dag;
|
2012-01-17 06:55:03 +00:00
|
|
|
|
2012-03-14 11:26:37 +00:00
|
|
|
assert((!ForceTopDown || !ForceBottomUp) &&
|
2012-03-14 04:00:41 +00:00
|
|
|
"-misched-topdown incompatible with -misched-bottomup");
|
|
|
|
}
|
2012-01-17 06:55:03 +00:00
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
virtual SUnit *pickNode(bool &IsTopNode) {
|
|
|
|
if (DAG->top() == DAG->bottom())
|
|
|
|
return NULL;
|
2012-01-17 06:55:03 +00:00
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
// As an initial placeholder heuristic, schedule in the direction that has
|
|
|
|
// the fewest choices.
|
|
|
|
SUnit *SU;
|
|
|
|
if (ForceTopDown || (!ForceBottomUp && NumTopReady <= NumBottomReady)) {
|
|
|
|
SU = DAG->getSUnit(DAG->top());
|
|
|
|
IsTopNode = true;
|
|
|
|
}
|
|
|
|
else {
|
2012-04-24 18:04:34 +00:00
|
|
|
SU = DAG->getSUnit(priorNonDebug(DAG->bottom(), DAG->top()));
|
2012-03-14 04:00:41 +00:00
|
|
|
IsTopNode = false;
|
|
|
|
}
|
|
|
|
if (SU->isTopReady()) {
|
|
|
|
assert(NumTopReady > 0 && "bad ready count");
|
|
|
|
--NumTopReady;
|
|
|
|
}
|
|
|
|
if (SU->isBottomReady()) {
|
|
|
|
assert(NumBottomReady > 0 && "bad ready count");
|
|
|
|
--NumBottomReady;
|
|
|
|
}
|
|
|
|
return SU;
|
|
|
|
}
|
2012-01-17 06:55:03 +00:00
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
virtual void releaseTopNode(SUnit *SU) {
|
|
|
|
++NumTopReady;
|
|
|
|
}
|
|
|
|
virtual void releaseBottomNode(SUnit *SU) {
|
|
|
|
++NumBottomReady;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
} // namespace
|
2012-01-17 06:55:03 +00:00
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
/// Create the standard converging machine scheduler. This will be used as the
|
|
|
|
/// default scheduler if the target does not set a default.
|
|
|
|
static ScheduleDAGInstrs *createConvergingSched(MachineSchedContext *C) {
|
2012-03-14 11:26:37 +00:00
|
|
|
assert((!ForceTopDown || !ForceBottomUp) &&
|
2012-03-14 04:00:41 +00:00
|
|
|
"-misched-topdown incompatible with -misched-bottomup");
|
|
|
|
return new ScheduleDAGMI(C, new ConvergingScheduler());
|
2012-01-17 06:55:03 +00:00
|
|
|
}
|
2012-03-14 04:00:41 +00:00
|
|
|
static MachineSchedRegistry
|
|
|
|
ConvergingSchedRegistry("converge", "Standard converging scheduler.",
|
|
|
|
createConvergingSched);
|
2012-01-17 06:55:03 +00:00
|
|
|
|
2012-01-14 02:17:06 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Machine Instruction Shuffler for Correctness Testing
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2012-01-13 06:30:30 +00:00
|
|
|
#ifndef NDEBUG
|
|
|
|
namespace {
|
2012-03-14 04:00:41 +00:00
|
|
|
/// Apply a less-than relation on the node order, which corresponds to the
|
|
|
|
/// instruction order prior to scheduling. IsReverse implements greater-than.
|
|
|
|
template<bool IsReverse>
|
|
|
|
struct SUnitOrder {
|
2012-01-17 06:55:07 +00:00
|
|
|
bool operator()(SUnit *A, SUnit *B) const {
|
2012-03-14 04:00:41 +00:00
|
|
|
if (IsReverse)
|
|
|
|
return A->NodeNum > B->NodeNum;
|
|
|
|
else
|
|
|
|
return A->NodeNum < B->NodeNum;
|
2012-01-17 06:55:07 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-01-13 06:30:30 +00:00
|
|
|
/// Reorder instructions as much as possible.
|
2012-03-14 04:00:41 +00:00
|
|
|
class InstructionShuffler : public MachineSchedStrategy {
|
|
|
|
bool IsAlternating;
|
|
|
|
bool IsTopDown;
|
|
|
|
|
|
|
|
// Using a less-than relation (SUnitOrder<false>) for the TopQ priority
|
|
|
|
// gives nodes with a higher number higher priority causing the latest
|
|
|
|
// instructions to be scheduled first.
|
|
|
|
PriorityQueue<SUnit*, std::vector<SUnit*>, SUnitOrder<false> >
|
|
|
|
TopQ;
|
|
|
|
// When scheduling bottom-up, use greater-than as the queue priority.
|
|
|
|
PriorityQueue<SUnit*, std::vector<SUnit*>, SUnitOrder<true> >
|
|
|
|
BottomQ;
|
2012-01-13 06:30:30 +00:00
|
|
|
public:
|
2012-03-14 04:00:41 +00:00
|
|
|
InstructionShuffler(bool alternate, bool topdown)
|
|
|
|
: IsAlternating(alternate), IsTopDown(topdown) {}
|
2012-01-13 06:30:30 +00:00
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
virtual void initialize(ScheduleDAGMI *) {
|
|
|
|
TopQ.clear();
|
|
|
|
BottomQ.clear();
|
|
|
|
}
|
2012-01-17 06:55:07 +00:00
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
/// Implement MachineSchedStrategy interface.
|
|
|
|
/// -----------------------------------------
|
|
|
|
|
|
|
|
virtual SUnit *pickNode(bool &IsTopNode) {
|
|
|
|
SUnit *SU;
|
|
|
|
if (IsTopDown) {
|
|
|
|
do {
|
|
|
|
if (TopQ.empty()) return NULL;
|
|
|
|
SU = TopQ.top();
|
|
|
|
TopQ.pop();
|
|
|
|
} while (SU->isScheduled);
|
|
|
|
IsTopNode = true;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
do {
|
|
|
|
if (BottomQ.empty()) return NULL;
|
|
|
|
SU = BottomQ.top();
|
|
|
|
BottomQ.pop();
|
|
|
|
} while (SU->isScheduled);
|
|
|
|
IsTopNode = false;
|
|
|
|
}
|
|
|
|
if (IsAlternating)
|
|
|
|
IsTopDown = !IsTopDown;
|
2012-01-17 06:55:07 +00:00
|
|
|
return SU;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:00:41 +00:00
|
|
|
virtual void releaseTopNode(SUnit *SU) {
|
|
|
|
TopQ.push(SU);
|
|
|
|
}
|
|
|
|
virtual void releaseBottomNode(SUnit *SU) {
|
|
|
|
BottomQ.push(SU);
|
2012-01-13 06:30:30 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
2012-03-08 01:41:12 +00:00
|
|
|
static ScheduleDAGInstrs *createInstructionShuffler(MachineSchedContext *C) {
|
2012-03-14 04:00:41 +00:00
|
|
|
bool Alternate = !ForceTopDown && !ForceBottomUp;
|
|
|
|
bool TopDown = !ForceBottomUp;
|
2012-03-14 11:26:37 +00:00
|
|
|
assert((TopDown || !ForceTopDown) &&
|
2012-03-14 04:00:41 +00:00
|
|
|
"-misched-topdown incompatible with -misched-bottomup");
|
|
|
|
return new ScheduleDAGMI(C, new InstructionShuffler(Alternate, TopDown));
|
2012-01-13 06:30:30 +00:00
|
|
|
}
|
2012-03-14 04:00:41 +00:00
|
|
|
static MachineSchedRegistry ShufflerRegistry(
|
|
|
|
"shuffle", "Shuffle machine instructions alternating directions",
|
|
|
|
createInstructionShuffler);
|
2012-01-13 06:30:30 +00:00
|
|
|
#endif // !NDEBUG
|