You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			91 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			LLVM
		
	
	
	
	
	
		
		
			
		
	
	
			91 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			LLVM
		
	
	
	
	
	
|   | ; RUN: llc -mtriple=i686-- < %s | FileCheck %s
 | ||
|  | ; RUN: llc -mtriple=i686-- -O0 < %s | FileCheck %s
 | ||
|  | ; RUN: llc -mtriple=i686-- -disable-tail-calls < %s | FileCheck %s
 | ||
|  | 
 | ||
|  | declare void @t1_callee(i8*) | ||
|  | define void @t1(i32* %a) { | ||
|  | ; CHECK-LABEL: t1:
 | ||
|  | ; CHECK: jmp {{_?}}t1_callee
 | ||
|  |   %b = bitcast i32* %a to i8* | ||
|  |   musttail call void @t1_callee(i8* %b) | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | declare i8* @t2_callee() | ||
|  | define i32* @t2() { | ||
|  | ; CHECK-LABEL: t2:
 | ||
|  | ; CHECK: jmp {{_?}}t2_callee
 | ||
|  |   %v = musttail call i8* @t2_callee() | ||
|  |   %w = bitcast i8* %v to i32* | ||
|  |   ret i32* %w | ||
|  | } | ||
|  | 
 | ||
|  | ; Complex frame layout: stack realignment with dynamic alloca.
 | ||
|  | define void @t3(i32 %n) alignstack(32) nounwind { | ||
|  | entry: | ||
|  | ; CHECK: t3:
 | ||
|  | ; CHECK: pushl %ebp
 | ||
|  | ; CHECK: pushl %esi
 | ||
|  | ; CHECK: andl $-32, %esp
 | ||
|  | ; CHECK: movl %esp, %esi
 | ||
|  | ; CHECK: popl %esi
 | ||
|  | ; CHECK: popl %ebp
 | ||
|  | ; CHECK-NEXT: jmp {{_?}}t3_callee
 | ||
|  |   %a = alloca i8, i32 %n | ||
|  |   call void @capture(i8* %a) | ||
|  |   musttail call void @t3_callee(i32 %n) nounwind | ||
|  |   ret void | ||
|  | } | ||
|  | 
 | ||
|  | declare void @capture(i8*) | ||
|  | declare void @t3_callee(i32) | ||
|  | 
 | ||
|  | ; Test that we actually copy in and out stack arguments that aren't forwarded
 | ||
|  | ; without modification.
 | ||
|  | define i32 @t4({}* %fn, i32 %n, i32 %r) { | ||
|  | ; CHECK-LABEL: t4:
 | ||
|  | ; CHECK: incl %[[r:.*]]
 | ||
|  | ; CHECK: decl %[[n:.*]]
 | ||
|  | ; CHECK: movl %[[r]], {{[0-9]+}}(%esp)
 | ||
|  | ; CHECK: movl %[[n]], {{[0-9]+}}(%esp)
 | ||
|  | ; CHECK: jmpl *%{{.*}}
 | ||
|  | 
 | ||
|  | entry: | ||
|  |   %r1 = add i32 %r, 1 | ||
|  |   %n1 = sub i32 %n, 1 | ||
|  |   %fn_cast = bitcast {}* %fn to i32 ({}*, i32, i32)* | ||
|  |   %r2 = musttail call i32 %fn_cast({}* %fn, i32 %n1, i32 %r1) | ||
|  |   ret i32 %r2 | ||
|  | } | ||
|  | 
 | ||
|  | ; Combine the complex stack frame with the parameter modification.
 | ||
|  | define i32 @t5({}* %fn, i32 %n, i32 %r) alignstack(32) { | ||
|  | ; CHECK-LABEL: t5:
 | ||
|  | ; CHECK: pushl %ebp
 | ||
|  | ; CHECK: movl %esp, %ebp
 | ||
|  | ; CHECK: pushl %esi
 | ||
|  | ; 	Align the stack.
 | ||
|  | ; CHECK: andl $-32, %esp
 | ||
|  | ; CHECK: movl %esp, %esi
 | ||
|  | ; 	Modify the args.
 | ||
|  | ; CHECK: incl %[[r:.*]]
 | ||
|  | ; CHECK: decl %[[n:.*]]
 | ||
|  | ; 	Store them through ebp, since that's the only stable arg pointer.
 | ||
|  | ; CHECK: movl %[[r]], {{[0-9]+}}(%ebp)
 | ||
|  | ; CHECK: movl %[[n]], {{[0-9]+}}(%ebp)
 | ||
|  | ; 	Epilogue.
 | ||
|  | ; CHECK: leal {{[-0-9]+}}(%ebp), %esp
 | ||
|  | ; CHECK: popl %esi
 | ||
|  | ; CHECK: popl %ebp
 | ||
|  | ; CHECK: jmpl *%{{.*}}
 | ||
|  | 
 | ||
|  | entry: | ||
|  |   %a = alloca i8, i32 %n | ||
|  |   call void @capture(i8* %a) | ||
|  |   %r1 = add i32 %r, 1 | ||
|  |   %n1 = sub i32 %n, 1 | ||
|  |   %fn_cast = bitcast {}* %fn to i32 ({}*, i32, i32)* | ||
|  |   %r2 = musttail call i32 %fn_cast({}* %fn, i32 %n1, i32 %r1) | ||
|  |   ret i32 %r2 | ||
|  | } |