// RUN: %clang_cc1 -triple wasm32-unknown-unknown %s -emit-llvm -o - \ // RUN: | FileCheck %s -check-prefix=WEBASSEMBLY32 // RUN: %clang_cc1 -triple wasm64-unknown-unknown %s -emit-llvm -o - \ // RUN: | FileCheck %s -check-prefix=WEBASSEMBLY64 // RUN: %clang_cc1 -triple wasm32-unknown-unknown %s -target-abi experimental-mv -emit-llvm -o - \ // RUN: | FileCheck %s -check-prefix=EXPERIMENTAL-MV // Basic argument/attribute and return tests for WebAssembly // WEBASSEMBLY32: define void @misc_args(i32 %i, i32 %j, i64 %k, double %l, fp128 %m) // WEBASSEMBLY64: define void @misc_args(i32 %i, i64 %j, i64 %k, double %l, fp128 %m) void misc_args(int i, long j, long long k, double l, long double m) {} typedef struct { int aa; int bb; } s1; // Structs should be passed byval and not split up. // WEBASSEMBLY32: define void @struct_arg(%struct.s1* byval(%struct.s1) align 4 %i) // WEBASSEMBLY64: define void @struct_arg(%struct.s1* byval(%struct.s1) align 4 %i) // Except in the experimental multivalue ABI, where structs are passed in args // EXPERIMENTAL-MV: define void @struct_arg(i32 %i.0, i32 %i.1) void struct_arg(s1 i) {} // Structs should be returned sret and not simplified by the frontend. // WEBASSEMBLY32: define void @struct_ret(%struct.s1* noalias sret(%struct.s1) align 4 %agg.result) // WEBASSEMBLY32: ret void // WEBASSEMBLY64: define void @struct_ret(%struct.s1* noalias sret(%struct.s1) align 4 %agg.result) // WEBASSEMBLY64: ret void // Except with the experimental multivalue ABI, which returns structs by value // EXPERIMENTAL-MV: define %struct.s1 @struct_ret() // EXPERIMENTAL-MV: ret %struct.s1 %0 s1 struct_ret() { s1 foo; return foo; } typedef struct { int cc; } s2; // Single-element structs should be passed as the one element. // WEBASSEMBLY32: define void @single_elem_arg(i32 %i.coerce) // WEBASSEMBLY64: define void @single_elem_arg(i32 %i.coerce) // EXPERIMENTAL-MV: define void @single_elem_arg(i32 %i.coerce) void single_elem_arg(s2 i) {} // Single-element structs should be passed as the one element. // WEBASSEMBLY32: define i32 @single_elem_ret() // WEBASSEMBLY32: ret i32 // WEBASSEMBLY64: define i32 @single_elem_ret() // EXPERIMENTAL-MV: define i32 @single_elem_ret() s2 single_elem_ret() { s2 foo; return foo; } // WEBASSEMBLY32: define void @long_long_arg(i64 %i) // WEBASSEMBLY64: define void @long_long_arg(i64 %i) void long_long_arg(long long i) {} // i8/i16 should be signext, i32 and higher should not. // WEBASSEMBLY32: define void @char_short_arg(i8 signext %a, i16 signext %b) // WEBASSEMBLY64: define void @char_short_arg(i8 signext %a, i16 signext %b) void char_short_arg(char a, short b) {} // WEBASSEMBLY32: define void @uchar_ushort_arg(i8 zeroext %a, i16 zeroext %b) // WEBASSEMBLY64: define void @uchar_ushort_arg(i8 zeroext %a, i16 zeroext %b) void uchar_ushort_arg(unsigned char a, unsigned short b) {} enum my_enum { ENUM1, ENUM2, ENUM3, }; // Enums should be treated as the underlying i32. // WEBASSEMBLY32: define void @enum_arg(i32 %a) // WEBASSEMBLY64: define void @enum_arg(i32 %a) void enum_arg(enum my_enum a) {} enum my_big_enum { ENUM4 = 0xFFFFFFFFFFFFFFFF, }; // Big enums should be treated as the underlying i64. // WEBASSEMBLY32: define void @big_enum_arg(i64 %a) // WEBASSEMBLY64: define void @big_enum_arg(i64 %a) void big_enum_arg(enum my_big_enum a) {} union simple_union { int a; char b; }; // Unions should be passed as byval structs. // WEBASSEMBLY32: define void @union_arg(%union.simple_union* byval(%union.simple_union) align 4 %s) // WEBASSEMBLY64: define void @union_arg(%union.simple_union* byval(%union.simple_union) align 4 %s) // EXPERIMENTAL-MV: define void @union_arg(i32 %s.0) void union_arg(union simple_union s) {} // Unions should be returned sret and not simplified by the frontend. // WEBASSEMBLY32: define void @union_ret(%union.simple_union* noalias sret(%union.simple_union) align 4 %agg.result) // WEBASSEMBLY32: ret void // WEBASSEMBLY64: define void @union_ret(%union.simple_union* noalias sret(%union.simple_union) align 4 %agg.result) // WEBASSEMBLY64: ret void // The experimental multivalue ABI returns them by value, though. // EXPERIMENTAL-MV: define %union.simple_union @union_ret() // EXPERIMENTAL-MV: ret %union.simple_union %0 union simple_union union_ret() { union simple_union bar; return bar; } typedef struct { int b4 : 4; int b3 : 3; int b8 : 8; } bitfield1; // Bitfields should be passed as byval structs. // WEBASSEMBLY32: define void @bitfield_arg(%struct.bitfield1* byval(%struct.bitfield1) align 4 %bf1) // WEBASSEMBLY64: define void @bitfield_arg(%struct.bitfield1* byval(%struct.bitfield1) align 4 %bf1) // EXPERIMENTAL-MV: define void @bitfield_arg(%struct.bitfield1* byval(%struct.bitfield1) align 4 %bf1) void bitfield_arg(bitfield1 bf1) {} // And returned via sret pointers. // WEBASSEMBLY32: define void @bitfield_ret(%struct.bitfield1* noalias sret(%struct.bitfield1) align 4 %agg.result) // WEBASSEMBLY64: define void @bitfield_ret(%struct.bitfield1* noalias sret(%struct.bitfield1) align 4 %agg.result) // Except, of course, in the experimental multivalue ABI // EXPERIMENTAL-MV: define %struct.bitfield1 @bitfield_ret() bitfield1 bitfield_ret() { bitfield1 baz; return baz; }