Use getLoopLatch in place of isLoopSimplifyForm. we do not need
to know whether the loop has a preheader nor dedicated exits.
Details
Diff Detail
- Build Status
- Buildable 2920 - Build 2920: arc lint + arc unit 
Event Timeline
This looks good to me (aside from the fact that, as Sanjoy pointed out, it needs a test case).
Add an assertion to check that in case we have a single latch, we actually use that
latch to get the loop id. This is the fast case !.
Add a test case to make sure fast case is exercised and works correctly.
Some comments on the test case.
| unittests/Analysis/LoopInfoTest.cpp | ||
|---|---|---|
| 20 ↗ | (On Diff #84499) | I'd rather put this inside the LoopInfoTest struct as a public instance variable. | 
| 30 ↗ | (On Diff #84499) | Can you ASSERT or assert (up to you which one) that the name of Header is for.cond, just to make sure we're doing what we think we're doing? | 
| 52 ↗ | (On Diff #84499) | Make this a static function. | 
| 66 ↗ | (On Diff #84499) | This is totally up to you, but I tend to prefer not using explicit "\n" s but instead end lines with a space. That makes the string more readable and it works fine since LLVM is (mostly?) whitespace insensitive. | 
| 81 ↗ | (On Diff #84499) | Creating a pass for this seems unnecessary -- why can't you directly call LoopInfo::analyze on the relevant function? | 
| unittests/Analysis/LoopInfoTest.cpp | ||
|---|---|---|
| 85 ↗ | (On Diff #84499) | Also, I'd add a check that the loop is not in loop simplify form, to make sure we're testing the right thing. | 
Why was this implemented as a unittest? Wouldn't this work just as good by inserting some debug prints and then using lit+opt+FileCheck to test?
Admittedly the same is true for many of the other tests in unittests/Analysis. Most of the times I just don't see the benefits of using unittests over lit+FileCheck, they are usually a lot harder to understand and debug.
One problem I can see with LIT test here is that getLoopID() is not a transformation by itself, it depends on other transformations to call/use it. Would not using a LIT test for that tranformation making the testing a little bit more fragile/unpredictable, i.e. what if the transformation later is changed not to call this API or call it in a different way. Testing this way is a bit more specific I feel.
You can just request a single analysis from opt without running any transformation. It should work similar to this opt -mypass -debug-only=mypass input.ll if you use DEBUG(dbgs() << ) statements, alternatively there is the -analyze flag to opt, which ends up calling MyAnalysisPass::print().
There are number of tests in test/Analysis that work this way.
I think that makes sense when you're testing an analysis (e.g. does AA think these two pointers may alias or not?). Here the test is for an API that analyses and transformations uses, so to me it is more understandable to test it this way (since you're directly testing what you changed).
However, we could be better about encoding IR in a way that they're easily auto-upgraded, since IMO that's one of the drawbacks of writing unittests/.
I think that would work for LoopInfo, with some additional debug printing. One problem/question I have is whether its a good idea to call getLoopID()/setLoopID() in ::print(), should not print() be primarily used to dump existing information, than testing some APIs in the analysis.
If course print() should not change any information.
At a first glance this looked to me like you just test the result of an anlysis. But indeed if you need to test actual interaction with an API the unittest is the right tool.
Admittedly, I did not think how this should be tested through completely. (I know what I need to test, but as pointed out there are more than 1 way to test this). I sort of followed what @sanjoy suggested. Thanks for raising the concern though.