mirror of
				https://github.com/c64scene-ar/llvm-6502.git
				synced 2025-10-25 10:27:04 +00:00 
			
		
		
		
	Summary:
Currently, we give an error if %z is used with non-immediates, instead of continuing as if the %z isn't there.
For example, you use the %z operand modifier along with the "Jr" constraints ("r" makes the operand a register, and "J" makes it an immediate, but only if its value is 0). 
In this case, you want the compiler to print "$0" if the inline asm input operand turns out to be an immediate zero and you want it to print the register containing the operand, if it's not.
We give an error in the latter case, and we shouldn't (GCC also doesn't).
Reviewers: dsanders
Reviewed By: dsanders
Subscribers: llvm-commits
Differential Revision: http://reviews.llvm.org/D6023
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@221453 91177308-0d34-0410-b5e6-96231b3b80d8
		
	
		
			
				
	
	
		
			181 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			LLVM
		
	
	
	
	
	
			
		
		
	
	
			181 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			LLVM
		
	
	
	
	
	
| ; Positive test for inline register constraints
 | |
| ;
 | |
| ; RUN: llc -march=mipsel < %s  | FileCheck -check-prefix=CHECK_LITTLE_32 %s
 | |
| ; RUN: llc -march=mips < %s  | FileCheck -check-prefix=CHECK_BIG_32 %s
 | |
| 
 | |
| %union.u_tag = type { i64 }
 | |
| %struct.anon = type { i32, i32 }
 | |
| @uval = common global %union.u_tag zeroinitializer, align 8
 | |
| 
 | |
| ; X with -3
 | |
| define i32 @constraint_X() nounwind {
 | |
| entry:
 | |
| ;CHECK_LITTLE_32-LABEL:   constraint_X:
 | |
| ;CHECK_LITTLE_32: #APP
 | |
| ;CHECK_LITTLE_32: addiu ${{[0-9]+}},${{[0-9]+}},0xfffffffffffffffd
 | |
| ;CHECK_LITTLE_32: #NO_APP
 | |
|   tail call i32 asm sideeffect "addiu $0,$1,${2:X}", "=r,r,I"(i32 7, i32 -3) ;
 | |
|   ret i32 0
 | |
| }
 | |
| 
 | |
| ; x with -3
 | |
| define i32 @constraint_x() nounwind {
 | |
| entry:
 | |
| ;CHECK_LITTLE_32-LABEL:   constraint_x:
 | |
| ;CHECK_LITTLE_32: #APP
 | |
| ;CHECK_LITTLE_32: addiu ${{[0-9]+}},${{[0-9]+}},0xfffd
 | |
| ;CHECK_LITTLE_32: #NO_APP
 | |
|   tail call i32 asm sideeffect "addiu $0,$1,${2:x}", "=r,r,I"(i32 7, i32 -3) ;
 | |
|   ret i32 0
 | |
| }
 | |
| 
 | |
| ; d with -3
 | |
| define i32 @constraint_d() nounwind {
 | |
| entry:
 | |
| ;CHECK_LITTLE_32-LABEL:   constraint_d:
 | |
| ;CHECK_LITTLE_32:   #APP
 | |
| ;CHECK_LITTLE_32:   addiu ${{[0-9]+}},${{[0-9]+}},-3
 | |
| ;CHECK_LITTLE_32:   #NO_APP
 | |
|   tail call i32 asm sideeffect "addiu $0,$1,${2:d}", "=r,r,I"(i32 7, i32 -3) ;
 | |
|   ret i32 0
 | |
| }
 | |
| 
 | |
| ; m with -3
 | |
| define i32 @constraint_m() nounwind {
 | |
| entry:
 | |
| ;CHECK_LITTLE_32-LABEL:   constraint_m:
 | |
| ;CHECK_LITTLE_32:   #APP
 | |
| ;CHECK_LITTLE_32:   addiu ${{[0-9]+}},${{[0-9]+}},-4
 | |
| ;CHECK_LITTLE_32:   #NO_APP
 | |
|   tail call i32 asm sideeffect "addiu $0,$1,${2:m}", "=r,r,I"(i32 7, i32 -3) ;
 | |
|   ret i32 0
 | |
| }
 | |
| 
 | |
| ; z with -3
 | |
| define i32 @constraint_z() nounwind {
 | |
| entry:
 | |
| ;CHECK_LITTLE_32-LABEL: constraint_z:
 | |
| ;CHECK_LITTLE_32:    #APP
 | |
| ;CHECK_LITTLE_32:    addiu ${{[0-9]+}},${{[0-9]+}},-3
 | |
| ;CHECK_LITTLE_32:    #NO_APP
 | |
|   tail call i32 asm sideeffect "addiu $0,$1,${2:z}", "=r,r,I"(i32 7, i32 -3) ;
 | |
| 
 | |
| ; z with 0
 | |
| ;CHECK_LITTLE_32:    #APP
 | |
| ;CHECK_LITTLE_32:    addiu ${{[0-9]+}},${{[0-9]+}},$0
 | |
| ;CHECK_LITTLE_32:    #NO_APP
 | |
|   tail call i32 asm sideeffect "addiu $0,$1,${2:z}", "=r,r,I"(i32 7, i32 0) nounwind
 | |
| 
 | |
| ; z with non-zero and the "r"(register) and "J"(integer zero) constraints
 | |
| ;CHECK_LITTLE_32:    #APP
 | |
| ;CHECK_LITTLE_32:    mtc0 ${{[1-9][0-9]?}}, ${{[0-9]+}}
 | |
| ;CHECK_LITTLE_32:    #NO_APP
 | |
|   call void asm sideeffect "mtc0 ${0:z}, $$12", "Jr"(i32 7) nounwind
 | |
| 
 | |
| ; z with zero and the "r"(register) and "J"(integer zero) constraints
 | |
| ;CHECK_LITTLE_32:    #APP
 | |
| ;CHECK_LITTLE_32:    mtc0 $0, ${{[0-9]+}}
 | |
| ;CHECK_LITTLE_32:    #NO_APP
 | |
|   call void asm sideeffect "mtc0 ${0:z}, $$12", "Jr"(i32 0) nounwind
 | |
| 
 | |
| ; z with non-zero and just the "r"(register) constraint
 | |
| ;CHECK_LITTLE_32:    #APP
 | |
| ;CHECK_LITTLE_32:    mtc0 ${{[1-9][0-9]?}}, ${{[0-9]+}}
 | |
| ;CHECK_LITTLE_32:    #NO_APP
 | |
|   call void asm sideeffect "mtc0 ${0:z}, $$12", "r"(i32 7) nounwind
 | |
| 
 | |
| ; z with zero and just the "r"(register) constraint
 | |
| ; FIXME: Check for $0, instead of other registers.
 | |
| ;        We should be using $0 directly in this case, not real registers.
 | |
| ;        When the materialization of 0 gets fixed, this test will fail.
 | |
| ;CHECK_LITTLE_32:    #APP
 | |
| ;CHECK_LITTLE_32:    mtc0 ${{[1-9][0-9]?}}, ${{[0-9]+}}
 | |
| ;CHECK_LITTLE_32:    #NO_APP
 | |
|   call void asm sideeffect "mtc0 ${0:z}, $$12", "r"(i32 0) nounwind
 | |
|   ret i32 0
 | |
| }
 | |
| 
 | |
| ; a long long in 32 bit mode (use to assert)
 | |
| define i32 @constraint_longlong() nounwind {
 | |
| entry:
 | |
| ;CHECK_LITTLE_32-LABEL: constraint_longlong:
 | |
| ;CHECK_LITTLE_32:    #APP
 | |
| ;CHECK_LITTLE_32:    addiu ${{[0-9]+}},${{[0-9]+}},3
 | |
| ;CHECK_LITTLE_32:    #NO_APP
 | |
|   tail call i64 asm sideeffect "addiu $0,$1,$2 \0A\09", "=r,r,X"(i64 1229801703532086340, i64 3) nounwind
 | |
|   ret i32 0
 | |
| }
 | |
| 
 | |
| ; D, in little endian the source reg will be 4 bytes into the long long
 | |
| define i32 @constraint_D() nounwind {
 | |
| entry:
 | |
| ;CHECK_LITTLE_32-LABEL: constraint_D:
 | |
| ;CHECK_LITTLE_32:    lw ${{[0-9]+}}, %got(uval)(${{[0-9,a-z]+}})
 | |
| ;CHECK_LITTLE_32:    lw $[[SECOND:[0-9]+]], 4(${{[0-9]+}})
 | |
| ;CHECK_LITTLE_32:    lw $[[FIRST:[0-9]+]], 0(${{[0-9]+}})
 | |
| ;CHECK_LITTLE_32:    #APP
 | |
| ;CHECK_LITTLE_32:    or ${{[0-9]+}},$[[SECOND]],${{[0-9]+}}
 | |
| ;CHECK_LITTLE_32:    #NO_APP
 | |
| 
 | |
| ; D, in big endian the source reg will also be 4 bytes into the long long
 | |
| ;CHECK_BIG_32-LABEL:    constraint_D:
 | |
| ;CHECK_BIG_32:       lw ${{[0-9]+}}, %got(uval)(${{[0-9,a-z]+}})
 | |
| ;CHECK_BIG_32:       lw $[[SECOND:[0-9]+]], 4(${{[0-9]+}})
 | |
| ;CHECK_BIG_32:       lw $[[FIRST:[0-9]+]], 0(${{[0-9]+}})
 | |
| ;CHECK_BIG_32:       #APP
 | |
| ;CHECK_BIG_32:       or ${{[0-9]+}},$[[SECOND]],${{[0-9]+}}
 | |
| ;CHECK_BIG_32:       #NO_APP
 | |
|   %bosco = load i64* getelementptr inbounds (%union.u_tag* @uval, i32 0, i32 0), align 8
 | |
|   %trunc1 = trunc i64 %bosco to i32
 | |
|   tail call i32 asm sideeffect "or $0,${1:D},$2", "=r,r,r"(i64 %bosco, i32 %trunc1) nounwind
 | |
|   ret i32 0
 | |
| }
 | |
| 
 | |
| ; L, in little endian the source reg will be 0 bytes into the long long
 | |
| define i32 @constraint_L() nounwind {
 | |
| entry:
 | |
| ;CHECK_LITTLE_32-LABEL: constraint_L:
 | |
| ;CHECK_LITTLE_32:    lw ${{[0-9]+}}, %got(uval)(${{[0-9,a-z]+}})
 | |
| ;CHECK_LITTLE_32:    lw $[[SECOND:[0-9]+]], 4(${{[0-9]+}})
 | |
| ;CHECK_LITTLE_32:    lw $[[FIRST:[0-9]+]], 0(${{[0-9]+}})
 | |
| ;CHECK_LITTLE_32:    #APP
 | |
| ;CHECK_LITTLE_32:    or ${{[0-9]+}},$[[FIRST]],${{[0-9]+}}
 | |
| ;CHECK_LITTLE_32:    #NO_APP
 | |
| ; L, in big endian the source reg will be 4 bytes into the long long
 | |
| ;CHECK_BIG_32-LABEL: constraint_L:
 | |
| ;CHECK_BIG_32:       lw ${{[0-9]+}}, %got(uval)(${{[0-9,a-z]+}})
 | |
| ;CHECK_BIG_32:       lw $[[SECOND:[0-9]+]], 4(${{[0-9]+}})
 | |
| ;CHECK_BIG_32:       lw $[[FIRST:[0-9]+]], 0(${{[0-9]+}})
 | |
| ;CHECK_BIG_32:       #APP
 | |
| ;CHECK_BIG_32:       or ${{[0-9]+}},$[[SECOND]],${{[0-9]+}}
 | |
| ;CHECK_BIG_32:       #NO_APP
 | |
|   %bosco = load i64* getelementptr inbounds (%union.u_tag* @uval, i32 0, i32 0), align 8
 | |
|   %trunc1 = trunc i64 %bosco to i32
 | |
|   tail call i32 asm sideeffect "or $0,${1:L},$2", "=r,r,r"(i64 %bosco, i32 %trunc1) nounwind
 | |
|   ret i32 0
 | |
| }
 | |
| 
 | |
| ; M, in little endian the source reg will be 4 bytes into the long long
 | |
| define i32 @constraint_M() nounwind {
 | |
| entry:
 | |
| ;CHECK_LITTLE_32-LABEL: constraint_M:
 | |
| ;CHECK_LITTLE_32:    lw ${{[0-9]+}}, %got(uval)(${{[0-9,a-z]+}})
 | |
| ;CHECK_LITTLE_32:    lw $[[SECOND:[0-9]+]], 4(${{[0-9]+}})
 | |
| ;CHECK_LITTLE_32:    lw $[[FIRST:[0-9]+]], 0(${{[0-9]+}})
 | |
| ;CHECK_LITTLE_32:    #APP
 | |
| ;CHECK_LITTLE_32:    or ${{[0-9]+}},$[[SECOND]],${{[0-9]+}}
 | |
| ;CHECK_LITTLE_32:    #NO_APP
 | |
| ; M, in big endian the source reg will be 0 bytes into the long long
 | |
| ;CHECK_BIG_32-LABEL:    constraint_M:
 | |
| ;CHECK_BIG_32:       lw ${{[0-9]+}}, %got(uval)(${{[0-9,a-z]+}})
 | |
| ;CHECK_BIG_32:       lw $[[SECOND:[0-9]+]], 4(${{[0-9]+}})
 | |
| ;CHECK_BIG_32:       lw $[[FIRST:[0-9]+]], 0(${{[0-9]+}})
 | |
| ;CHECK_BIG_32:       #APP
 | |
| ;CHECK_BIG_32:       or ${{[0-9]+}},$[[FIRST]],${{[0-9]+}}
 | |
| ;CHECK_BIG_32:       #NO_APP
 | |
|   %bosco = load i64* getelementptr inbounds (%union.u_tag* @uval, i32 0, i32 0), align 8
 | |
|   %trunc1 = trunc i64 %bosco to i32
 | |
|   tail call i32 asm sideeffect "or $0,${1:M},$2", "=r,r,r"(i64 %bosco, i32 %trunc1) nounwind
 | |
|   ret i32 0
 | |
| }
 |