| // NOTE: Assertions have been autogenerated by utils/update_cc_test_checks.py UTC_ARGS: --function-signature |
| // REQUIRES: webassembly-registered-target |
| |
| // Simple calls to known variadic functions that are completely elided when |
| // optimisations are on This is a functional check that the expand-variadic pass |
| // is consistent with clang's va_arg handling |
| |
| // When expand-variadics is added to the default pipeline, clang -O1 will |
| // suffice here -Wno-varargs avoids warning second argument to 'va_start' is not |
| // the last named parameter |
| |
| // RUN: %clang_cc1 %s -triple wasm32-unknown-unknown -Wno-varargs -O1 -emit-llvm -o - | opt - -S --passes='module(expand-variadics,default<O1>)' --expand-variadics-override=optimize -o - | FileCheck %s |
| |
| #include <stdarg.h> |
| #include <stdint.h> |
| |
| template <typename X, typename Y> static X first(...) { |
| va_list va; |
| __builtin_va_start(va, 0); |
| X r = va_arg(va, X); |
| va_end(va); |
| return r; |
| } |
| |
| template <typename X, typename Y> static Y second(...) { |
| va_list va; |
| __builtin_va_start(va, 0); |
| va_arg(va, X); |
| Y r = va_arg(va, Y); |
| va_end(va); |
| return r; |
| } |
| |
| extern "C" { |
| |
| // CHECK-LABEL: define {{[^@]+}}@first_pair_i32 |
| // CHECK-SAME: (i32 noundef returned [[X:%.*]], i32 noundef [[Y:%.*]]) local_unnamed_addr #[[ATTR0:[0-9]+]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: ret i32 [[X]] |
| // |
| int first_pair_i32(int x, int y) { return first<int, int>(x, y); } |
| |
| // CHECK-LABEL: define {{[^@]+}}@second_pair_i32 |
| // CHECK-SAME: (i32 noundef [[X:%.*]], i32 noundef returned [[Y:%.*]]) local_unnamed_addr #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: ret i32 [[Y]] |
| // |
| int second_pair_i32(int x, int y) { return second<int, int>(x, y); } |
| |
| // CHECK-LABEL: define {{[^@]+}}@first_pair_f64 |
| // CHECK-SAME: (double noundef returned [[X:%.*]], double noundef [[Y:%.*]]) local_unnamed_addr #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: ret double [[X]] |
| // |
| double first_pair_f64(double x, double y) { |
| return first<double, double>(x, y); |
| } |
| |
| // CHECK-LABEL: define {{[^@]+}}@second_pair_f64 |
| // CHECK-SAME: (double noundef [[X:%.*]], double noundef returned [[Y:%.*]]) local_unnamed_addr #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: ret double [[Y]] |
| // |
| double second_pair_f64(double x, double y) { |
| return second<double, double>(x, y); |
| } |
| } |
| |
| extern "C" { |
| |
| // CHECK-LABEL: define {{[^@]+}}@first_i32_f64 |
| // CHECK-SAME: (i32 noundef returned [[X:%.*]], double noundef [[Y:%.*]]) local_unnamed_addr #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: ret i32 [[X]] |
| // |
| int first_i32_f64(int x, double y) { return first<int, double>(x, y); } |
| |
| // CHECK-LABEL: define {{[^@]+}}@second_i32_f64 |
| // CHECK-SAME: (i32 noundef [[X:%.*]], double noundef returned [[Y:%.*]]) local_unnamed_addr #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: ret double [[Y]] |
| // |
| double second_i32_f64(int x, double y) { return second<int, double>(x, y); } |
| |
| // CHECK-LABEL: define {{[^@]+}}@first_f64_i32 |
| // CHECK-SAME: (double noundef returned [[X:%.*]], i32 noundef [[Y:%.*]]) local_unnamed_addr #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: ret double [[X]] |
| // |
| double first_f64_i32(double x, int y) { return first<double, int>(x, y); } |
| |
| // CHECK-LABEL: define {{[^@]+}}@second_f64_i32 |
| // CHECK-SAME: (double noundef [[X:%.*]], i32 noundef returned [[Y:%.*]]) local_unnamed_addr #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: ret i32 [[Y]] |
| // |
| int second_f64_i32(double x, int y) { return second<double, int>(x, y); } |
| } |
| |
| extern "C" { |
| typedef uint64_t ulong2 __attribute__((__vector_size__(16), __aligned__(16))); |
| |
| // CHECK-LABEL: define {{[^@]+}}@first_i32_ulong2 |
| // CHECK-SAME: (i32 noundef returned [[X:%.*]], ptr noundef readonly captures(none) [[Y:%.*]]) local_unnamed_addr #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: ret i32 [[X]] |
| // |
| int first_i32_ulong2(int x, ulong2 *y) { return first<int, ulong2>(x, *y); } |
| |
| // CHECK-LABEL: define {{[^@]+}}@second_i32_ulong2 |
| // CHECK-SAME: (i32 noundef [[X:%.*]], ptr noundef readonly captures(none) [[Y:%.*]], ptr noundef writeonly captures(none) initializes((0, 16)) [[R:%.*]]) local_unnamed_addr #[[ATTR1:[0-9]+]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[TMP0:%.*]] = load <2 x i64>, ptr [[Y]], align 16, !tbaa [[TBAA2:![0-9]+]] |
| // CHECK-NEXT: store <2 x i64> [[TMP0]], ptr [[R]], align 16, !tbaa [[TBAA2]] |
| // CHECK-NEXT: ret void |
| // |
| void second_i32_ulong2(int x, ulong2 *y, ulong2 *r) { |
| *r = second<int, ulong2>(x, *y); |
| } |
| |
| // CHECK-LABEL: define {{[^@]+}}@first_ulong2_i32 |
| // CHECK-SAME: (ptr noundef readonly captures(none) [[X:%.*]], i32 noundef [[Y:%.*]], ptr noundef writeonly captures(none) initializes((0, 16)) [[R:%.*]]) local_unnamed_addr #[[ATTR1]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: [[TMP0:%.*]] = load <2 x i64>, ptr [[X]], align 16, !tbaa [[TBAA2]] |
| // CHECK-NEXT: store <2 x i64> [[TMP0]], ptr [[R]], align 16, !tbaa [[TBAA2]] |
| // CHECK-NEXT: ret void |
| // |
| void first_ulong2_i32(ulong2 *x, int y, ulong2 *r) { |
| *r = first<ulong2, int>(*x, y); |
| } |
| |
| // CHECK-LABEL: define {{[^@]+}}@second_ulong2_i32 |
| // CHECK-SAME: (ptr noundef readonly captures(none) [[X:%.*]], i32 noundef returned [[Y:%.*]]) local_unnamed_addr #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: ret i32 [[Y]] |
| // |
| int second_ulong2_i32(ulong2 *x, int y) { return second<ulong2, int>(*x, y); } |
| } |
| |
| // ascending alignment |
| typedef struct { |
| char c; |
| short s; |
| int i; |
| long l; |
| float f; |
| double d; |
| } asc; |
| |
| extern "C" { |
| |
| // CHECK-LABEL: define {{[^@]+}}@first_i32_asc |
| // CHECK-SAME: (i32 noundef returned [[X:%.*]], ptr noundef readonly captures(none) [[Y:%.*]]) local_unnamed_addr #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: ret i32 [[X]] |
| // |
| int first_i32_asc(int x, asc *y) { return first<int, asc>(x, *y); } |
| |
| // CHECK-LABEL: define {{[^@]+}}@second_i32_asc |
| // CHECK-SAME: (i32 noundef [[X:%.*]], ptr noundef readonly captures(none) [[Y:%.*]], ptr noundef writeonly captures(none) initializes((0, 24)) [[R:%.*]]) local_unnamed_addr #[[ATTR1]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: tail call void @llvm.memmove.p0.p0.i32(ptr noundef nonnull align 8 dereferenceable(24) [[R]], ptr noundef nonnull align 1 dereferenceable(24) [[Y]], i32 24, i1 false) |
| // CHECK-NEXT: ret void |
| // |
| void second_i32_asc(int x, asc *y, asc *r) { *r = second<int, asc>(x, *y); } |
| |
| // CHECK-LABEL: define {{[^@]+}}@first_asc_i32 |
| // CHECK-SAME: (ptr noundef readonly captures(none) [[X:%.*]], i32 noundef [[Y:%.*]], ptr noundef writeonly captures(none) initializes((0, 24)) [[R:%.*]]) local_unnamed_addr #[[ATTR1]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: tail call void @llvm.memmove.p0.p0.i32(ptr noundef nonnull align 8 dereferenceable(24) [[R]], ptr noundef nonnull align 1 dereferenceable(24) [[X]], i32 24, i1 false) |
| // CHECK-NEXT: ret void |
| // |
| void first_asc_i32(asc *x, int y, asc *r) { *r = first<asc, int>(*x, y); } |
| |
| // CHECK-LABEL: define {{[^@]+}}@second_asc_i32 |
| // CHECK-SAME: (ptr noundef readonly captures(none) [[X:%.*]], i32 noundef returned [[Y:%.*]]) local_unnamed_addr #[[ATTR0]] { |
| // CHECK-NEXT: entry: |
| // CHECK-NEXT: ret i32 [[Y]] |
| // |
| int second_asc_i32(asc *x, int y) { return second<asc, int>(*x, y); } |
| } |