Add isRenamable() predicate to MachineOperand. This predicate can be
used by machine passes after register allocation to determine whether it
is safe to rename a given register operand. Register operands that
aren't marked as renamable may be required to be assigned their current
register to satisfy constraints that are not captured by the machine
IR (e.g. ABI or ISA constraints).
Details
Diff Detail
- Build Status
Buildable 12731 Build 12731: arc lint + arc unit
Event Timeline
Ping.
This change should be in pretty good shape now, but the MI interface changes could use a good review.
I had to update quite a few MIR tests to add the 'norename' flag. A lot of these could probably be fixed by getting the relevant passes to set the norename flag more precisely, but I figured it would be better to do that in separate changes.
include/llvm/CodeGen/MachineOperand.h | ||
---|---|---|
87–88 | Written like this it'll probably be part of the IsDef comment in doxygen. Maybe just append a "Only valid for MO_Register" to each of the four fields instead? | |
98–111 | One interesting discussion here is how "safe" things really are when this bit is set :) | |
99–101 | Should we state that this bit only affects physical registers? That way we don't need to set this bit everywhere a vreg is used. And I can't imagine any case where we have a non-renamable vreg or put another way: I do not want to add code dealing with non-renamable vregs. | |
101 | The last sentence is not true anymore. | |
311–312 | I assume changing this to assert(isReg() && IsDef()) breaks too much code? | |
366–370 | See my comment in setReg(), I think we shouldn't have this method. | |
413 | Maybe add assert(TargetRegisterInfo::isPhysicalRegister(getReg() && "Renamable flag may only be set for physregs"); | |
lib/CodeGen/MIRParser/MIParser.cpp | ||
1220 |
| |
lib/CodeGen/MIRParser/MIRParser.cpp | ||
420–428 ↗ | (On Diff #121965) | I'm not a fan of the parsing changing depending on a function property. I think the right thing to do is to encode "renamable" and not "norename" after all. If you are worried about mir output becoming noisy/hard to read after regalloc with "renamable" flags everywhere, maybe we can shorten "renamable" to some character like ~? |
lib/CodeGen/MachineInstr.cpp | ||
76–77 | I find it surprising that setReg() affects a flag as well. I'd rather just modify the regallocators to set the flag explicitely where possible. | |
lib/CodeGen/MachineVerifier.cpp | ||
366–367 | NoVRegs should mean exactly that: We are guaranteed that no vregs are used. It does not mean: We are in a pass after register allocation. | |
1218–1223 | As stated above I would go for Renamable only ever being set on physregs. That way the rules don't need to be conditional based on "regsAllocated". |
@MatzeB I have a new version that is much simpler that I believe addresses all of your points. I got rid of the context-dependent printing/parsing of the renamable flag (and changed it from 'norename') which ended up reducing the amount of lit test churn. I also restricted it to physical registers and tried to enforce the hasExtraRegAllocReq constraints, though that may not be complete.
include/llvm/CodeGen/MachineOperand.h | ||
---|---|---|
311–312 | Yeah, I tried adding it and gave up after fixing it in about 5 different passes. |
Thanks, conditional LGTM.
My only remaining issue here is addOperand() changing operands. If you can easily fix this just commit without further review, otherwise at least write a comment on why you need it.
include/llvm/CodeGen/MachineOperand.h | ||
---|---|---|
98–111 | I would still vote for a weaker language here by removing the word "safely". | |
lib/CodeGen/MachineInstr.cpp | ||
263–268 | Is this a good idea? Intuitively I would not expect an addOperand() function to change the operand. Admittedly someone already broke that rule with the setIsEarlyClobber() above but unless all other solutions are really bad, I'd still prefer not to have this here. |
It would also be nice to pick a suitable test in test/CodeGen/MIR/X86 and add one line to ensure .mir prints/parses the flag correctly.
I made the comment less strong and added a test case. Let me know what you think about the addOperand issue.
lib/CodeGen/MachineInstr.cpp | ||
---|---|---|
263–268 | Without this every bit of code that is builds an instruction whose opcode hasExtraRegAllocReq would need to remember to clear the appropriate renamable bit. It seems to me that the hasExtraRegAllocReq property is less useful that way. I suppose we could untie these two properties and ask that users check both the renamable bit and the hasExtraRegAllocReq (with a helper function), in which case the verification that the two properties are consistent goes away. |
lib/CodeGen/MachineInstr.cpp | ||
---|---|---|
263–268 | But unless I miss something, the default is not-renamable anyway so all the random code out there doesn't need to be changed. The only places that set the renamable bit are the regallocators and they already use MO.setIsRenamableIfNoExtraRegAllocReq() in your commit. So what other code is there that would set the renamable bit even though it shouldn't? |
lib/CodeGen/MachineInstr.cpp | ||
---|---|---|
263–268 | One example that occurs in the lit tests is code that is constructing a hasExtraReqAllocReq machine instruction after register allocation and copying over operands from an existing instruction. These particular operands are renamable since they were originally virtual regs. Once they are added to the new instruction they remain renamable, but are operands of a hasExtraRegAllocReq opcode so they fail verification. |
lib/CodeGen/MachineInstr.cpp | ||
---|---|---|
263–268 | I assume that will be mostly (or all) AMDGPU code, is adjusting this code so hard that it justifies extra logic in a core function like addOperand()? At least we have the asserts in place and won't silently miscompile... |
New version that doesn't force renamable flag to false for hasExtraRegAlloc opcodes in addOperand
lib/CodeGen/MachineInstr.cpp | ||
---|---|---|
263–268 | I added a new version that removes this change to addOperand and fixes the lit test failures (which only showed up for ARM). I'm a bit concerned that this approach will break other targets/tests, but I suppose we can always revisit this later if it is a problem. |
Written like this it'll probably be part of the IsDef comment in doxygen. Maybe just append a "Only valid for MO_Register" to each of the four fields instead?