1; RUN: llc -march=mipsel -pre-RA-sched=source < %s | FileCheck %s 2 3; All test functions do the same thing - they return the first variable 4; argument. 5 6; All CHECK's do the same thing - they check whether variable arguments from 7; registers are placed on correct stack locations, and whether the first 8; variable argument is returned from the correct stack location. 9 10 11declare void @llvm.va_start(i8*) nounwind 12declare void @llvm.va_end(i8*) nounwind 13 14; return int 15define i32 @va1(i32 %a, ...) nounwind { 16entry: 17 %a.addr = alloca i32, align 4 18 %ap = alloca i8*, align 4 19 %b = alloca i32, align 4 20 store i32 %a, i32* %a.addr, align 4 21 %ap1 = bitcast i8** %ap to i8* 22 call void @llvm.va_start(i8* %ap1) 23 %0 = va_arg i8** %ap, i32 24 store i32 %0, i32* %b, align 4 25 %ap2 = bitcast i8** %ap to i8* 26 call void @llvm.va_end(i8* %ap2) 27 %tmp = load i32, i32* %b, align 4 28 ret i32 %tmp 29 30; CHECK-LABEL: va1: 31; CHECK: addiu $sp, $sp, -16 32; CHECK: sw $7, 28($sp) 33; CHECK: sw $6, 24($sp) 34; CHECK: sw $5, 20($sp) 35; CHECK: lw $2, 20($sp) 36} 37 38; check whether the variable double argument will be accessed from the 8-byte 39; aligned location (i.e. whether the address is computed by adding 7 and 40; clearing lower 3 bits) 41define double @va2(i32 %a, ...) nounwind { 42entry: 43 %a.addr = alloca i32, align 4 44 %ap = alloca i8*, align 4 45 %b = alloca double, align 8 46 store i32 %a, i32* %a.addr, align 4 47 %ap1 = bitcast i8** %ap to i8* 48 call void @llvm.va_start(i8* %ap1) 49 %0 = va_arg i8** %ap, double 50 store double %0, double* %b, align 8 51 %ap2 = bitcast i8** %ap to i8* 52 call void @llvm.va_end(i8* %ap2) 53 %tmp = load double, double* %b, align 8 54 ret double %tmp 55 56; CHECK-LABEL: va2: 57; CHECK: addiu $sp, $sp, -16 58; CHECK: sw $7, 28($sp) 59; CHECK: sw $6, 24($sp) 60; CHECK: sw $5, 20($sp) 61; CHECK: addiu $[[R0:[0-9]+]], $sp, 20 62; CHECK: addiu $[[R1:[0-9]+]], $[[R0]], 7 63; CHECK: addiu $[[R2:[0-9]+]], $zero, -8 64; CHECK: and $[[R3:[0-9]+]], $[[R1]], $[[R2]] 65; CHECK: ldc1 $f0, 0($[[R3]]) 66} 67 68; int 69define i32 @va3(double %a, ...) nounwind { 70entry: 71 %a.addr = alloca double, align 8 72 %ap = alloca i8*, align 4 73 %b = alloca i32, align 4 74 store double %a, double* %a.addr, align 8 75 %ap1 = bitcast i8** %ap to i8* 76 call void @llvm.va_start(i8* %ap1) 77 %0 = va_arg i8** %ap, i32 78 store i32 %0, i32* %b, align 4 79 %ap2 = bitcast i8** %ap to i8* 80 call void @llvm.va_end(i8* %ap2) 81 %tmp = load i32, i32* %b, align 4 82 ret i32 %tmp 83 84; CHECK-LABEL: va3: 85; CHECK: addiu $sp, $sp, -16 86; CHECK: sw $7, 28($sp) 87; CHECK: sw $6, 24($sp) 88; CHECK: lw $2, 24($sp) 89} 90 91; double 92define double @va4(double %a, ...) nounwind { 93entry: 94 %a.addr = alloca double, align 8 95 %ap = alloca i8*, align 4 96 %b = alloca double, align 8 97 store double %a, double* %a.addr, align 8 98 %ap1 = bitcast i8** %ap to i8* 99 call void @llvm.va_start(i8* %ap1) 100 %0 = va_arg i8** %ap, double 101 store double %0, double* %b, align 8 102 %ap2 = bitcast i8** %ap to i8* 103 call void @llvm.va_end(i8* %ap2) 104 %tmp = load double, double* %b, align 8 105 ret double %tmp 106 107; CHECK-LABEL: va4: 108; CHECK: addiu $sp, $sp, -24 109; CHECK: sw $7, 36($sp) 110; CHECK: sw $6, 32($sp) 111; CHECK: addiu ${{[0-9]+}}, $sp, 32 112; CHECK: ldc1 $f0, 32($sp) 113} 114 115; int 116define i32 @va5(i32 %a, i32 %b, i32 %c, ...) nounwind { 117entry: 118 %a.addr = alloca i32, align 4 119 %b.addr = alloca i32, align 4 120 %c.addr = alloca i32, align 4 121 %ap = alloca i8*, align 4 122 %d = alloca i32, align 4 123 store i32 %a, i32* %a.addr, align 4 124 store i32 %b, i32* %b.addr, align 4 125 store i32 %c, i32* %c.addr, align 4 126 %ap1 = bitcast i8** %ap to i8* 127 call void @llvm.va_start(i8* %ap1) 128 %0 = va_arg i8** %ap, i32 129 store i32 %0, i32* %d, align 4 130 %ap2 = bitcast i8** %ap to i8* 131 call void @llvm.va_end(i8* %ap2) 132 %tmp = load i32, i32* %d, align 4 133 ret i32 %tmp 134 135; CHECK-LABEL: va5: 136; CHECK: addiu $sp, $sp, -24 137; CHECK: sw $7, 36($sp) 138; CHECK: lw $2, 36($sp) 139} 140 141; double 142define double @va6(i32 %a, i32 %b, i32 %c, ...) nounwind { 143entry: 144 %a.addr = alloca i32, align 4 145 %b.addr = alloca i32, align 4 146 %c.addr = alloca i32, align 4 147 %ap = alloca i8*, align 4 148 %d = alloca double, align 8 149 store i32 %a, i32* %a.addr, align 4 150 store i32 %b, i32* %b.addr, align 4 151 store i32 %c, i32* %c.addr, align 4 152 %ap1 = bitcast i8** %ap to i8* 153 call void @llvm.va_start(i8* %ap1) 154 %0 = va_arg i8** %ap, double 155 store double %0, double* %d, align 8 156 %ap2 = bitcast i8** %ap to i8* 157 call void @llvm.va_end(i8* %ap2) 158 %tmp = load double, double* %d, align 8 159 ret double %tmp 160 161; CHECK-LABEL: va6: 162; CHECK: addiu $sp, $sp, -24 163; CHECK: sw $7, 36($sp) 164; CHECK: addiu $[[R0:[0-9]+]], $sp, 36 165; CHECK: addiu $[[R1:[0-9]+]], $[[R0]], 7 166; CHECK: addiu $[[R2:[0-9]+]], $zero, -8 167; CHECK: and $[[R3:[0-9]+]], $[[R1]], $[[R2]] 168; CHECK: ldc1 $f0, 0($[[R3]]) 169} 170 171; int 172define i32 @va7(i32 %a, double %b, ...) nounwind { 173entry: 174 %a.addr = alloca i32, align 4 175 %b.addr = alloca double, align 8 176 %ap = alloca i8*, align 4 177 %c = alloca i32, align 4 178 store i32 %a, i32* %a.addr, align 4 179 store double %b, double* %b.addr, align 8 180 %ap1 = bitcast i8** %ap to i8* 181 call void @llvm.va_start(i8* %ap1) 182 %0 = va_arg i8** %ap, i32 183 store i32 %0, i32* %c, align 4 184 %ap2 = bitcast i8** %ap to i8* 185 call void @llvm.va_end(i8* %ap2) 186 %tmp = load i32, i32* %c, align 4 187 ret i32 %tmp 188 189; CHECK-LABEL: va7: 190; CHECK: addiu $sp, $sp, -24 191; CHECK: lw $2, 40($sp) 192} 193 194; double 195define double @va8(i32 %a, double %b, ...) nounwind { 196entry: 197 %a.addr = alloca i32, align 4 198 %b.addr = alloca double, align 8 199 %ap = alloca i8*, align 4 200 %c = alloca double, align 8 201 store i32 %a, i32* %a.addr, align 4 202 store double %b, double* %b.addr, align 8 203 %ap1 = bitcast i8** %ap to i8* 204 call void @llvm.va_start(i8* %ap1) 205 %0 = va_arg i8** %ap, double 206 store double %0, double* %c, align 8 207 %ap2 = bitcast i8** %ap to i8* 208 call void @llvm.va_end(i8* %ap2) 209 %tmp = load double, double* %c, align 8 210 ret double %tmp 211 212; CHECK-LABEL: va8: 213; CHECK: addiu $sp, $sp, -32 214; CHECK: addiu ${{[0-9]+}}, $sp, 48 215; CHECK: ldc1 $f0, 48($sp) 216} 217 218; int 219define i32 @va9(double %a, double %b, i32 %c, ...) nounwind { 220entry: 221 %a.addr = alloca double, align 8 222 %b.addr = alloca double, align 8 223 %c.addr = alloca i32, align 4 224 %ap = alloca i8*, align 4 225 %d = alloca i32, align 4 226 store double %a, double* %a.addr, align 8 227 store double %b, double* %b.addr, align 8 228 store i32 %c, i32* %c.addr, align 4 229 %ap1 = bitcast i8** %ap to i8* 230 call void @llvm.va_start(i8* %ap1) 231 %0 = va_arg i8** %ap, i32 232 store i32 %0, i32* %d, align 4 233 %ap2 = bitcast i8** %ap to i8* 234 call void @llvm.va_end(i8* %ap2) 235 %tmp = load i32, i32* %d, align 4 236 ret i32 %tmp 237 238; CHECK-LABEL: va9: 239; CHECK: addiu $sp, $sp, -32 240; CHECK: lw $2, 52($sp) 241} 242 243; double 244define double @va10(double %a, double %b, i32 %c, ...) nounwind { 245entry: 246 %a.addr = alloca double, align 8 247 %b.addr = alloca double, align 8 248 %c.addr = alloca i32, align 4 249 %ap = alloca i8*, align 4 250 %d = alloca double, align 8 251 store double %a, double* %a.addr, align 8 252 store double %b, double* %b.addr, align 8 253 store i32 %c, i32* %c.addr, align 4 254 %ap1 = bitcast i8** %ap to i8* 255 call void @llvm.va_start(i8* %ap1) 256 %0 = va_arg i8** %ap, double 257 store double %0, double* %d, align 8 258 %ap2 = bitcast i8** %ap to i8* 259 call void @llvm.va_end(i8* %ap2) 260 %tmp = load double, double* %d, align 8 261 ret double %tmp 262 263; CHECK-LABEL: va10: 264; CHECK: addiu $sp, $sp, -32 265; CHECK: addiu $[[R0:[0-9]+]], $sp, 52 266; CHECK: addiu $[[R1:[0-9]+]], $[[R0]], 7 267; CHECK: addiu $[[R2:[0-9]+]], $zero, -8 268; CHECK: and $[[R3:[0-9]+]], $[[R1]], $[[R2]] 269; CHECK: ldc1 $f0, 0($[[R3]]) 270} 271