File: builtin-std-move.cpp

package info (click to toggle)
llvm-toolchain-19 1%3A19.1.7-3
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 1,998,520 kB
  • sloc: cpp: 6,951,680; ansic: 1,486,157; asm: 913,598; python: 232,024; f90: 80,126; objc: 75,281; lisp: 37,276; pascal: 16,990; sh: 10,009; ml: 5,058; perl: 4,724; awk: 3,523; makefile: 3,167; javascript: 2,504; xml: 892; fortran: 664; cs: 573
file content (72 lines) | stat: -rw-r--r-- 2,527 bytes parent folder | download | duplicates (10)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
// RUN: %clang_cc1 -triple=x86_64-linux-gnu -emit-llvm -o - -std=c++17 %s | FileCheck %s --implicit-check-not=@_ZSt4move

namespace std {
  template<typename T> constexpr T &&move(T &val) { return static_cast<T&&>(val); }
  template<typename T> constexpr T &&move_if_noexcept(T &val);
  template<typename T> constexpr T &&forward(T &val);
  template<typename U, typename T> constexpr T &&forward_like(T &&val);
  template<typename T> constexpr const T &as_const(T &val);

  // Not the builtin.
  template<typename T, typename U> T move(U source, U source_end, T dest);
}

class T {};
extern "C" void take(T &&);
extern "C" void take_lval(const T &);

T a;

// Check emission of a constant-evaluated call.
// CHECK-DAG: @move_a = constant ptr @a
T &&move_a = std::move(a);
// CHECK-DAG: @move_if_noexcept_a = constant ptr @a
T &&move_if_noexcept_a = std::move_if_noexcept(a);
// CHECK-DAG: @forward_a = constant ptr @a
T &forward_a = std::forward<T&>(a);
// CHECK-DAG: @forward_like_a = constant ptr @a
T &forward_like_a = std::forward_like<int&>(a);

// Check emission of a non-constant call.
// CHECK-LABEL: define {{.*}} void @test
extern "C" void test(T &t) {
  // CHECK: store ptr %{{.*}}, ptr %[[T_REF:[^,]*]]
  // CHECK: %0 = load ptr, ptr %[[T_REF]]
  // CHECK: call void @take(ptr {{.*}} %0)
  take(std::move(t));
  // CHECK: %1 = load ptr, ptr %[[T_REF]]
  // CHECK: call void @take(ptr {{.*}} %1)
  take(std::move_if_noexcept(t));
  // CHECK: %2 = load ptr, ptr %[[T_REF]]
  // CHECK: call void @take(ptr {{.*}} %2)
  take(std::forward<T&&>(t));
  // CHECK: %3 = load ptr, ptr %[[T_REF]]
  // CHECK: call void @take_lval(ptr {{.*}} %3)
  take_lval(std::forward_like<int&>(t));
  // CHECK: %4 = load ptr, ptr %[[T_REF]]
  // CHECK: call void @take_lval(ptr {{.*}} %4)
  take_lval(std::as_const<T&&>(t));

  // CHECK: call {{.*}} @_ZSt4moveI1TS0_ET_T0_S2_S1_
  std::move(t, t, t);
}

// CHECK: declare {{.*}} @_ZSt4moveI1TS0_ET_T0_S2_S1_

// Check that we instantiate and emit if the address is taken.
// CHECK-LABEL: define {{.*}} @use_address
extern "C" void *use_address() {
  // CHECK: ret {{.*}} @_ZSt4moveIiEOT_RS0_
  return (void*)&std::move<int>;
}

// CHECK: define {{.*}} ptr @_ZSt4moveIiEOT_RS0_(ptr

extern "C" void take_const_int_rref(const int &&);
// CHECK-LABEL: define {{.*}} @move_const_int(
extern "C" void move_const_int() {
  // CHECK: store i32 5, ptr %[[N_ADDR:[^,]*]]
  const int n = 5;
  // CHECK: call {{.*}} @take_const_int_rref(ptr {{.*}} %[[N_ADDR]])
  take_const_int_rref(std::move(n));
}