//===----------------------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
// <memory>
// Test arguments destruction order involving unique_ptr<T> with trivial_abi.
// ADDITIONAL_COMPILE_FLAGS: -Wno-macro-redefined -D_LIBCPP_ABI_ENABLE_UNIQUE_PTR_TRIVIAL_ABI
// XFAIL: gcc
#include <memory>
#include <cassert>
#include "test_macros.h"
__attribute__((noinline)) void call_something() { asm volatile(""); }
struct Base {
char* shared_buff;
int* cur_idx;
const char id;
explicit Base(char* buf, int* idx, char ch)
: shared_buff(buf), cur_idx(idx), id(ch) {}
Base(const Base& other) = default;
Base& operator=(const Base&) = delete;
~Base() { shared_buff[(*cur_idx)++] = id; }
};
struct A : Base {
explicit A(char* buf, int* idx) : Base(buf, idx, 'A') {}
};
struct B : Base {
explicit B(char* buf, int* idx) : Base(buf, idx, 'B') {}
};
struct C : Base {
explicit C(char* buf, int* idx) : Base(buf, idx, 'C') {}
};
__attribute__((noinline)) void func(A /*unused*/, std::unique_ptr<B> /*unused*/,
C /*unused*/) {
call_something();
}
int main(int, char**) {
char shared_buf[3] = {'0', '0', '0'};
int cur_idx = 0;
func(A(shared_buf, &cur_idx), std::unique_ptr<B>(new B(shared_buf, &cur_idx)),
C(shared_buf, &cur_idx));
#if defined(TEST_ABI_MICROSOFT)
// On Microsoft ABI, the dtor order is always A,B,C (because callee-destroyed)
assert(shared_buf[0] == 'A' && shared_buf[1] == 'B' && shared_buf[2] == 'C');
#else
// With trivial_abi, the std::unique_ptr<B> arg is always destructed first.
assert(shared_buf[0] == 'B');
#endif
return 0;
}