|  | ; RUN: llc < %s -mtriple=s390x-linux-gnu | FileCheck %s | 
|  |  | 
|  | ; Test lowering of @llvm.frameaddress with packed-stack. | 
|  |  | 
|  | ; With back chain | 
|  | attributes #0 = { nounwind "packed-stack" "backchain" "use-soft-float"="true" } | 
|  | define ptr @fp0() #0 { | 
|  | entry: | 
|  | ; CHECK-LABEL: fp0: | 
|  | ; CHECK:      la   %r2, 152(%r15) | 
|  | ; CHECK-NEXT: br   %r14 | 
|  | %0 = tail call ptr @llvm.frameaddress(i32 0) | 
|  | ret ptr %0 | 
|  | } | 
|  |  | 
|  | define ptr @fp0f() #0 { | 
|  | entry: | 
|  | ; CHECK-LABEL: fp0f: | 
|  | ; CHECK:      lgr	%r1, %r15 | 
|  | ; CHECK-NEXT: aghi	%r15, -16 | 
|  | ; CHECK-NEXT: stg	%r1, 152(%r15) | 
|  | ; CHECK-NEXT: la	%r2, 168(%r15) | 
|  | ; CHECK-NEXT: aghi	%r15, 16 | 
|  | ; CHECK-NEXT: br	%r14 | 
|  | %0 = alloca i64, align 8 | 
|  | %1 = tail call ptr @llvm.frameaddress(i32 0) | 
|  | ret ptr %1 | 
|  | } | 
|  |  | 
|  | ; Check the caller's frame address. | 
|  | define ptr @fpcaller() #0 { | 
|  | entry: | 
|  | ; CHECK-LABEL: fpcaller: | 
|  | ; CHECK: lghi %r2, 152 | 
|  | ; CHECK: ag   %r2, 152(%r15) | 
|  | ; CHECK: br   %r14 | 
|  | %0 = tail call ptr @llvm.frameaddress(i32 1) | 
|  | ret ptr %0 | 
|  | } | 
|  |  | 
|  | ; Check the caller's caller's frame address. | 
|  | define ptr @fpcallercaller() #0 { | 
|  | entry: | 
|  | ; CHECK-LABEL: fpcallercaller: | 
|  | ; CHECK: lg   %r1, 152(%r15) | 
|  | ; CHECK: lghi %r2, 152 | 
|  | ; CHECK: ag   %r2, 152(%r1) | 
|  | ; CHECK: br   %r14 | 
|  | %0 = tail call ptr @llvm.frameaddress(i32 2) | 
|  | ret ptr %0 | 
|  | } | 
|  |  | 
|  | ; Without back chain | 
|  |  | 
|  | attributes #1 = { nounwind "packed-stack" } | 
|  | define ptr @fp1() #1 { | 
|  | entry: | 
|  | ; CHECK-LABEL: fp1: | 
|  | ; CHECK:      la   %r2, 152(%r15) | 
|  | ; CHECK-NEXT: br   %r14 | 
|  | %0 = tail call ptr @llvm.frameaddress(i32 0) | 
|  | ret ptr %0 | 
|  | } | 
|  |  | 
|  | ; No saved registers: returning address of unused slot where backcahin would | 
|  | ; have been located. | 
|  | define ptr @fp1f() #1 { | 
|  | entry: | 
|  | ; CHECK-LABEL: fp1f: | 
|  | ; CHECK:      aghi	%r15, -16 | 
|  | ; CHECK-NEXT: la	%r2, 168(%r15) | 
|  | ; CHECK-NEXT: aghi	%r15, 16 | 
|  | ; CHECK-NEXT: br	%r14 | 
|  | %0 = alloca i64, align 8 | 
|  | %1 = tail call ptr @llvm.frameaddress(i32 0) | 
|  | ret ptr %1 | 
|  | } | 
|  |  | 
|  | ; Saved registers: returning address for first saved GPR. | 
|  | declare void @foo(ptr %Arg) | 
|  | define ptr @fp2() #1 { | 
|  | entry: | 
|  | ; CHECK-LABEL: fp2: | 
|  | ; CHECK:      stmg      %r14, %r15, 144(%r15) | 
|  | ; CHECK-NEXT: aghi	%r15, -16 | 
|  | ; CHECK-NEXT: la	%r2, 168(%r15) | 
|  | ; CHECK-NEXT: brasl     %r14, foo@PLT | 
|  | ; CHECK-NEXT: la	%r2, 168(%r15) | 
|  | ; CHECK-NEXT: lmg       %r14, %r15, 160(%r15) | 
|  | ; CHECK-NEXT: br	%r14 | 
|  | %0 = tail call ptr @llvm.frameaddress(i32 0) | 
|  | call void @foo(ptr %0); | 
|  | ret ptr %0 | 
|  | } | 
|  |  | 
|  | declare ptr @llvm.frameaddress(i32) nounwind readnone |