xref: /llvm-project/clang/test/CodeGenCXX/builtin-std-move.cpp (revision 0fd9c37d8cf20d1ee42263872ab2135265c3ac3d)
172315d02SRichard Smith // RUN: %clang_cc1 -triple=x86_64-linux-gnu -emit-llvm -o - -std=c++17 %s | FileCheck %s --implicit-check-not=@_ZSt4move
272315d02SRichard Smith 
372315d02SRichard Smith namespace std {
move(T & val)472315d02SRichard Smith   template<typename T> constexpr T &&move(T &val) { return static_cast<T&&>(val); }
572315d02SRichard Smith   template<typename T> constexpr T &&move_if_noexcept(T &val);
672315d02SRichard Smith   template<typename T> constexpr T &&forward(T &val);
7*0fd9c37dSAlexander Shaposhnikov   template<typename U, typename T> constexpr T &&forward_like(T &&val);
872315d02SRichard Smith   template<typename T> constexpr const T &as_const(T &val);
972315d02SRichard Smith 
1072315d02SRichard Smith   // Not the builtin.
1172315d02SRichard Smith   template<typename T, typename U> T move(U source, U source_end, T dest);
1272315d02SRichard Smith }
1372315d02SRichard Smith 
1472315d02SRichard Smith class T {};
1572315d02SRichard Smith extern "C" void take(T &&);
1672315d02SRichard Smith extern "C" void take_lval(const T &);
1772315d02SRichard Smith 
1872315d02SRichard Smith T a;
1972315d02SRichard Smith 
2072315d02SRichard Smith // Check emission of a constant-evaluated call.
2172315d02SRichard Smith // CHECK-DAG: @move_a = constant ptr @a
2272315d02SRichard Smith T &&move_a = std::move(a);
2372315d02SRichard Smith // CHECK-DAG: @move_if_noexcept_a = constant ptr @a
2472315d02SRichard Smith T &&move_if_noexcept_a = std::move_if_noexcept(a);
2572315d02SRichard Smith // CHECK-DAG: @forward_a = constant ptr @a
2672315d02SRichard Smith T &forward_a = std::forward<T&>(a);
27*0fd9c37dSAlexander Shaposhnikov // CHECK-DAG: @forward_like_a = constant ptr @a
28*0fd9c37dSAlexander Shaposhnikov T &forward_like_a = std::forward_like<int&>(a);
2972315d02SRichard Smith 
3072315d02SRichard Smith // Check emission of a non-constant call.
3172315d02SRichard Smith // CHECK-LABEL: define {{.*}} void @test
test(T & t)3272315d02SRichard Smith extern "C" void test(T &t) {
3372315d02SRichard Smith   // CHECK: store ptr %{{.*}}, ptr %[[T_REF:[^,]*]]
3472315d02SRichard Smith   // CHECK: %0 = load ptr, ptr %[[T_REF]]
3572315d02SRichard Smith   // CHECK: call void @take(ptr {{.*}} %0)
3672315d02SRichard Smith   take(std::move(t));
3772315d02SRichard Smith   // CHECK: %1 = load ptr, ptr %[[T_REF]]
3872315d02SRichard Smith   // CHECK: call void @take(ptr {{.*}} %1)
3972315d02SRichard Smith   take(std::move_if_noexcept(t));
4072315d02SRichard Smith   // CHECK: %2 = load ptr, ptr %[[T_REF]]
4172315d02SRichard Smith   // CHECK: call void @take(ptr {{.*}} %2)
4272315d02SRichard Smith   take(std::forward<T&&>(t));
4372315d02SRichard Smith   // CHECK: %3 = load ptr, ptr %[[T_REF]]
4472315d02SRichard Smith   // CHECK: call void @take_lval(ptr {{.*}} %3)
45*0fd9c37dSAlexander Shaposhnikov   take_lval(std::forward_like<int&>(t));
46*0fd9c37dSAlexander Shaposhnikov   // CHECK: %4 = load ptr, ptr %[[T_REF]]
47*0fd9c37dSAlexander Shaposhnikov   // CHECK: call void @take_lval(ptr {{.*}} %4)
4872315d02SRichard Smith   take_lval(std::as_const<T&&>(t));
4972315d02SRichard Smith 
5072315d02SRichard Smith   // CHECK: call {{.*}} @_ZSt4moveI1TS0_ET_T0_S2_S1_
5172315d02SRichard Smith   std::move(t, t, t);
5272315d02SRichard Smith }
5372315d02SRichard Smith 
5472315d02SRichard Smith // CHECK: declare {{.*}} @_ZSt4moveI1TS0_ET_T0_S2_S1_
5572315d02SRichard Smith 
5672315d02SRichard Smith // Check that we instantiate and emit if the address is taken.
5772315d02SRichard Smith // CHECK-LABEL: define {{.*}} @use_address
use_address()5872315d02SRichard Smith extern "C" void *use_address() {
5972315d02SRichard Smith   // CHECK: ret {{.*}} @_ZSt4moveIiEOT_RS0_
6072315d02SRichard Smith   return (void*)&std::move<int>;
6172315d02SRichard Smith }
6272315d02SRichard Smith 
6372315d02SRichard Smith // CHECK: define {{.*}} ptr @_ZSt4moveIiEOT_RS0_(ptr
6472315d02SRichard Smith 
6572315d02SRichard Smith extern "C" void take_const_int_rref(const int &&);
6672315d02SRichard Smith // CHECK-LABEL: define {{.*}} @move_const_int(
move_const_int()6772315d02SRichard Smith extern "C" void move_const_int() {
6872315d02SRichard Smith   // CHECK: store i32 5, ptr %[[N_ADDR:[^,]*]]
6972315d02SRichard Smith   const int n = 5;
7072315d02SRichard Smith   // CHECK: call {{.*}} @take_const_int_rref(ptr {{.*}} %[[N_ADDR]])
7172315d02SRichard Smith   take_const_int_rref(std::move(n));
7272315d02SRichard Smith }
73