[LLVMdev] VLIWPacketizerList: failing to schedule terminators
Andrew Trick
atrick at apple.com
Thu Mar 29 15:51:10 PDT 2012
On Mar 29, 2012, at 1:18 PM, Tom Stellard <thomas.stellard at amd.com> wrote:
> On Thu, Mar 29, 2012 at 02:57:27PM -0500, Sergei Larin wrote:
>> Tom,
>>
>> I do not have your call stack, but packetizer calls
>> ScheduleDAGInstrs::buildSchedGraph to create dependency model. If this is
>> the first time you use the new MI sched infrastructure (like your target has
>> not implemented misched yet) there might be some work needed to implement
>> couple target hooks. isSchedulingBoundary is one of them. Also try to
>> disable that assert and see what happens. It sounds strange, but in some
>> cases that assert is overly restrictive.
>
> Thanks, I disabled that assert, and I got a little farther. I hit
> some more assertion failures finalizeBundle, because the
> instructions were still using virtual registers. I'm not sure why
> finalizeBundle doesn't currently handle virtual registers, maybe the
> implementation isn't complete yet?
>
> If I remove those asserts in finalizeBundle, I can get the program
> to compile, and it has the VLIW bundles.
>
> -Tom
>
>>
>> Andy,
>>
>> Those are just my guesses, but the issue looks very similar to our first
>> experience with sched DAG constructor.
>>
>> Sergei
Tom,
The version of VLIWPacketizerList currently checked in was designed to work for postRA scheduling. I want it to be adapted for use in preRA (called by MachineScheduler), but Sergei and others are still engaged in that process. Since nothing is checked in yet, you'll be running into the same issues as Sergei.
To me, the definition of "isSchedulingBoundary" is "something the DAG builder cannot handle". If you look at MachineScheduler::runOnFunction it jumps through hoops to avoid exposing boundaries to buildSchedGraph(). VLIWPacketizer doesn't have that logic, so it only works if isSchedulingBoundary returns false. Alternatively, you can disable the buildSchedGraph assert and hope for the best--let's just say it's not supported.
The MachineScheduler driver allows for two approaches to bundling:
1) Bundler operates on a larger scope than the scheduler (preferred): isSchedulingBoundary is true for branches, calls, and other strange barriers. Instructions can be reordered between the boundaries during schedule(), and the bundler can operate across regions via exitRegion() and finishBlock(). You will still have DAG edges from the instruction that a branch depends on to the branch (captured by ExitSU), but you won't have any DAG edges crossing the boundary.
2) Scheduler and bundler operate on the same scope (an entire block): isSchedulingBoundary = false and all instructions must be correctly handled by the DAG builder. DAG edges will be built for all instructions within the block.
---
Now, regarding bundle creation. The FinalizeMachineBundles pass will "finalize" bundles for you after regalloc. You can enable that in your target config. e.g.
bool MyTargetPassConfig::addFinalizeRegAlloc() {
addPass(FinalizeMachineBundlesID);
}
This does not create bundles, rather it wraps bundles in an extra layer of abstraction by copying all of the MachineOperands into a bundle header. I think we have enough other bundle abstractions that this step could be eliminated, but it's a bootstrapping mechanism that we rely on for now during postRA passes only.
VLIWPacketizer calls finalizeBundle explicitly because it's running after regalloc.
We do not want to see any Bundle instructions prior to regalloc. So a preRA scheduler should not call finalizeBundles. It can create bundles simply by setting the isInsideBundle flag on MachineInstrs. If you allow internal register dependencies you would also need to set isInternalRead on those MachineOperands.
-Andy
More information about the llvm-dev
mailing list