This is an archive of the discontinued LLVM Phabricator instance.

[IR] BasicBlock updatePHIs function for updating predecessors
AbandonedPublic

Authored by Marandil on Jun 1 2017, 10:10 AM.

Details

Summary

[IR] Add a function to update all PHI nodes within the block to change incoming values from IncomingOld to IncomingNew if IncomingOld is no longer a valid predecessor, or add new incoming edges in PHI nodes with IncomingNew otherwise.

The rationale behind this function is the amount of PHI node updates in my IR-based implementation of Jump Maps.

I've added a few unit tests for the function.

Diff Detail

Event Timeline

Marandil created this revision.Jun 1 2017, 10:10 AM
dberlin added inline comments.
lib/IR/BasicBlock.cpp
449

I'm not sure this loop is necessary.
If this funciton is how to update phi edges, you should guarantee an input state.
IE either pred edges removed or not removed.

457

This should be an assert

466

Can you explain why?
This seems like a bug?

dberlin added inline comments.Jun 1 2017, 10:30 AM
lib/IR/BasicBlock.cpp
464

It should be an assert that the old predecessor existed in the phi

468

This will put the phi arguments out of order with the predecessors, which is smoethign we would like to avoid.

470

I looked at the code for this function, and i don't see why it would fail.
Can you explain?

Marandil added inline comments.Jun 1 2017, 10:38 AM
lib/IR/BasicBlock.cpp
466

Consider this:

switch (a) {
  default: BB0;
  case 0: BB0;
  case 1: BB1;
  case 2: BB2;
  case 19: BB0;
  case 20: BB1;
  case 21: BB2;
}

changed into:

if (0 <= a <=2) {
  switch (a) {
    default: unreachable;
    case 0: BB0;
    case 1: BB1;
    case 2: BB2;
  }
} else {
  switch (a-19) {
    default: BB0;
    case 0: BB0;
    case 1: BB1;
    case 2: BB2;
  }
}

For BB0 a single predecessor changed into 2 different possible predecessors, that should share the same incoming value.

That just m

lib/IR/BasicBlock.cpp
466

That just makes me believe the API specification for this function is confusing.
Either your goal is to update all phis from one block to another, or it isn't.
In the above, you aren't.

Marandil added inline comments.Jun 1 2017, 11:01 AM
lib/IR/BasicBlock.cpp
466

The specification says:

/// \brief Replace (or add) occurrences of IncomingOld in PHIs with IncomingNew,
/// depending on whether or not IncomingOld is still a valid predecessor.

Consider a shorter example:

BB0:
  br i1 %0, label %BB2, label %BB2
BB1:
  br label %BB2
BB2:
  %1 = phi i32 [0, %BB0], [1, %BB1]
  ret i32 %1

Now, I modify the second edge of the br in BB0 (oblivious on the dest of the first one)

BB0:
  br i1 %0, label %BB2, label %BB3
BB1:
  br label %BB2
BB2:
  %1 = phi i32 [0, %BB0], [1, %BB1]
  ret i32 %1
BB3:
  br label %BB0

Obviously, the PHI node in BB2 is now invalid, as BB3->BB2 is another valid edge; so we need to update it:

BB0:
  br i1 %0, label %BB2, label %BB3
BB1:
  br label %BB2
BB2:
  %1 = phi i32 [0, %BB0], [1, %BB1], [0, %BB3]
  ret i32 %1
BB3:
  br label %BB0

The function is supposed to detect whether or not a new value in PHI should be added and it is inclined in the description.

dberlin added inline comments.Jun 1 2017, 11:16 AM
lib/IR/BasicBlock.cpp
466

I understand you have documented what it does.
I'm saying
A. "what it does" seems very specific to your use case
B. What it does in a specific case is not easy to understand just from looking at the call.
C. It is not what i would expect from a function named "updatephis" , I would expect such a thing to do *either* addition *or* replacement, not "add/replac depending on a bunch of variables and factors"
etc

IE This function seems incredibly specific to your case of updating.

I can say for certain in all the phi updating i've encountered, it would not be usable or useful.
A one to one replacement function would be useful (IE replace old with new)
An "add new edge to every phi" would be useful.
A function that tries to combine them and do something depending on other state that isn't passed - not so much.

But i'm going to leave it to others to see if they think this functionality is useful.
I can only request if they do, you don't call it "updatephis" because that is a very generic name for something that is quite specific.

Marandil added inline comments.Jun 1 2017, 11:23 AM
lib/IR/BasicBlock.cpp
466

How about splitting the current function into two, one for each case + one helper that detects the case and uses the appropriate version?
I have actually encountered this behavior several times with switch instructions (IR-level range splitting, etc) so I don't want to "keep it for myself".
Regarding the name: what name(s) would you propose? I have originally called this function fixPHIs, but later decided to rename it to updatePHIs. Maybe you have better suggestions.

Marandil updated this revision to Diff 101068.Jun 1 2017, 12:00 PM

I have changed checks to asserts as requested.

Marandil marked 2 inline comments as done.Jun 1 2017, 12:01 PM