blob: 21637d8d69bc4990fa3514e62080d60ab6296f01 [file] [log] [blame]
David Tolnay97c72102020-01-25 16:49:00 -08001#include "tests/ffi/tests.h"
David Tolnay1a2683a2020-03-17 19:09:29 -07002#include "tests/ffi/lib.rs.h"
David Tolnayb6c5ea72020-03-16 13:36:28 -07003#include <cstring>
David Tolnay37dd7e12020-04-25 12:51:59 -07004#include <numeric>
David Tolnayebef4a22020-03-17 15:33:47 -07005#include <stdexcept>
David Tolnay97c72102020-01-25 16:49:00 -08006
David Tolnay2fe58c62020-03-06 16:23:09 -08007extern "C" void cxx_test_suite_set_correct() noexcept;
8extern "C" tests::R *cxx_test_suite_get_box() noexcept;
9extern "C" bool cxx_test_suite_r_is_correct(const tests::R *) noexcept;
David Tolnay3fd7f562020-01-26 17:47:11 -080010
David Tolnay97c72102020-01-25 16:49:00 -080011namespace tests {
12
David Tolnay4037de22020-04-30 19:51:04 -070013static constexpr char SLICE_DATA[] = "2020";
Adrian Taylorf5dd5522020-04-13 16:50:14 -070014
David Tolnayad5b8af2020-01-26 16:59:13 -080015C::C(size_t n) : n(n) {}
16
David Tolnay3fd7f562020-01-26 17:47:11 -080017size_t C::get() const { return this->n; }
18
Joel Galensone1e969d2020-04-21 12:50:20 -070019size_t C::get2() const { return this->n; }
20
Joel Galenson3d4f6122020-04-07 15:54:05 -070021size_t C::set(size_t n) {
22 this->n = n;
23 return this->n;
24}
25
Joel Galensone1e969d2020-04-21 12:50:20 -070026size_t C::set2(size_t n) {
27 this->n = n;
28 return this->n;
29}
30
David Tolnayde5340e2020-04-25 14:03:21 -070031const std::vector<uint8_t> &C::get_v() const { return this->v; }
32
David Tolnayb8715772020-01-28 00:54:05 -080033size_t c_return_primitive() { return 2020; }
David Tolnayad5b8af2020-01-26 16:59:13 -080034
David Tolnayb8715772020-01-28 00:54:05 -080035Shared c_return_shared() { return Shared{2020}; }
David Tolnayad5b8af2020-01-26 16:59:13 -080036
David Tolnaybe13d8a2020-03-06 15:45:39 -080037rust::Box<R> c_return_box() {
38 return rust::Box<R>::from_raw(cxx_test_suite_get_box());
39}
40
David Tolnayad5b8af2020-01-26 16:59:13 -080041std::unique_ptr<C> c_return_unique_ptr() {
42 return std::unique_ptr<C>(new C{2020});
43}
44
David Tolnayb8715772020-01-28 00:54:05 -080045const size_t &c_return_ref(const Shared &shared) { return shared.z; }
David Tolnayad5b8af2020-01-26 16:59:13 -080046
David Tolnay750755e2020-03-01 13:04:08 -080047rust::Str c_return_str(const Shared &shared) {
David Tolnayad5b8af2020-01-26 16:59:13 -080048 (void)shared;
49 return "2020";
50}
51
David Tolnayeb952ba2020-04-14 15:02:24 -070052rust::Slice<uint8_t> c_return_sliceu8(const Shared &shared) {
Adrian Taylorf5dd5522020-04-13 16:50:14 -070053 (void)shared;
David Tolnay4037de22020-04-30 19:51:04 -070054 return rust::Slice<uint8_t>(reinterpret_cast<const uint8_t *>(SLICE_DATA),
55 sizeof(SLICE_DATA));
Adrian Taylorf5dd5522020-04-13 16:50:14 -070056}
57
David Tolnay750755e2020-03-01 13:04:08 -080058rust::String c_return_rust_string() { return "2020"; }
David Tolnayad5b8af2020-01-26 16:59:13 -080059
60std::unique_ptr<std::string> c_return_unique_ptr_string() {
61 return std::unique_ptr<std::string>(new std::string("2020"));
62}
63
Myron Ahneba35cf2020-02-05 19:41:51 +070064std::unique_ptr<std::vector<uint8_t>> c_return_unique_ptr_vector_u8() {
David Tolnay85db5a02020-04-25 13:17:27 -070065 auto vec = std::unique_ptr<std::vector<uint8_t>>(new std::vector<uint8_t>());
66 vec->push_back(86);
67 vec->push_back(75);
68 vec->push_back(30);
69 vec->push_back(9);
70 return vec;
Myron Ahneba35cf2020-02-05 19:41:51 +070071}
72
73std::unique_ptr<std::vector<double>> c_return_unique_ptr_vector_f64() {
David Tolnay85db5a02020-04-25 13:17:27 -070074 auto vec = std::unique_ptr<std::vector<double>>(new std::vector<double>());
75 vec->push_back(86.0);
76 vec->push_back(75.0);
77 vec->push_back(30.0);
78 vec->push_back(9.5);
79 return vec;
Myron Ahneba35cf2020-02-05 19:41:51 +070080}
81
82std::unique_ptr<std::vector<Shared>> c_return_unique_ptr_vector_shared() {
David Tolnay85db5a02020-04-25 13:17:27 -070083 auto vec = std::unique_ptr<std::vector<Shared>>(new std::vector<Shared>());
84 vec->push_back(Shared{1010});
85 vec->push_back(Shared{1011});
86 return vec;
Myron Ahneba35cf2020-02-05 19:41:51 +070087}
88
David Tolnay1bcc9fe2020-04-25 13:51:07 -070089std::unique_ptr<std::vector<C>> c_return_unique_ptr_vector_opaque() {
90 return std::unique_ptr<std::vector<C>>(new std::vector<C>());
91}
92
David Tolnayde5340e2020-04-25 14:03:21 -070093const std::vector<uint8_t> &c_return_ref_vector(const C &c) {
94 return c.get_v();
95}
96
David Tolnayb41e74c2020-04-25 15:06:18 -070097rust::Vec<uint8_t> c_return_rust_vec() {
98 throw std::runtime_error("unimplemented");
99}
100
David Tolnay77989692020-04-25 15:57:47 -0700101const rust::Vec<uint8_t> &c_return_ref_rust_vec(const C &c) {
102 (void)c;
103 throw std::runtime_error("unimplemented");
104}
105
David Tolnay378ca502020-04-27 16:47:55 -0700106size_t c_return_identity(size_t n) { return n; }
Joel Galensonba676072020-04-27 15:55:45 -0700107
David Tolnay378ca502020-04-27 16:47:55 -0700108size_t c_return_sum(size_t n1, size_t n2) { return n1 + n2; }
Joel Galensonba676072020-04-27 15:55:45 -0700109
Joel Galensonc03402a2020-04-23 17:31:09 -0700110Enum c_return_enum(uint32_t n) {
111 if (n <= static_cast<uint32_t>(Enum::AVal)) {
112 return Enum::AVal;
113 } else if (n <= static_cast<uint32_t>(Enum::BVal)) {
114 return Enum::BVal;
115 } else {
116 return Enum::CVal;
117 }
118}
119
David Tolnay3fd7f562020-01-26 17:47:11 -0800120void c_take_primitive(size_t n) {
121 if (n == 2020) {
122 cxx_test_suite_set_correct();
123 }
124}
David Tolnayad5b8af2020-01-26 16:59:13 -0800125
David Tolnay3fd7f562020-01-26 17:47:11 -0800126void c_take_shared(Shared shared) {
127 if (shared.z == 2020) {
128 cxx_test_suite_set_correct();
129 }
130}
David Tolnayad5b8af2020-01-26 16:59:13 -0800131
David Tolnay750755e2020-03-01 13:04:08 -0800132void c_take_box(rust::Box<R> r) {
David Tolnaya7d00e82020-03-06 15:50:14 -0800133 if (cxx_test_suite_r_is_correct(&*r)) {
134 cxx_test_suite_set_correct();
135 }
David Tolnay3fd7f562020-01-26 17:47:11 -0800136}
David Tolnayad5b8af2020-01-26 16:59:13 -0800137
David Tolnay3fd7f562020-01-26 17:47:11 -0800138void c_take_unique_ptr(std::unique_ptr<C> c) {
139 if (c->get() == 2020) {
140 cxx_test_suite_set_correct();
141 }
142}
David Tolnayad5b8af2020-01-26 16:59:13 -0800143
David Tolnaya7d00e82020-03-06 15:50:14 -0800144void c_take_ref_r(const R &r) {
145 if (cxx_test_suite_r_is_correct(&r)) {
146 cxx_test_suite_set_correct();
147 }
148}
David Tolnayad5b8af2020-01-26 16:59:13 -0800149
David Tolnay3fd7f562020-01-26 17:47:11 -0800150void c_take_ref_c(const C &c) {
151 if (c.get() == 2020) {
152 cxx_test_suite_set_correct();
153 }
154}
David Tolnayad5b8af2020-01-26 16:59:13 -0800155
David Tolnay750755e2020-03-01 13:04:08 -0800156void c_take_str(rust::Str s) {
David Tolnay3fd7f562020-01-26 17:47:11 -0800157 if (std::string(s) == "2020") {
158 cxx_test_suite_set_correct();
159 }
160}
David Tolnayad5b8af2020-01-26 16:59:13 -0800161
Adrian Taylorf5dd5522020-04-13 16:50:14 -0700162void c_take_sliceu8(rust::Slice<uint8_t> s) {
David Tolnay633b1f52020-04-14 16:33:14 -0700163 if (std::string(reinterpret_cast<const char *>(s.data()), s.size()) ==
164 "2020") {
Adrian Taylorf5dd5522020-04-13 16:50:14 -0700165 cxx_test_suite_set_correct();
166 }
167}
168
David Tolnay750755e2020-03-01 13:04:08 -0800169void c_take_rust_string(rust::String s) {
David Tolnay3fd7f562020-01-26 17:47:11 -0800170 if (std::string(s) == "2020") {
171 cxx_test_suite_set_correct();
172 }
173}
David Tolnayad5b8af2020-01-26 16:59:13 -0800174
David Tolnay3fd7f562020-01-26 17:47:11 -0800175void c_take_unique_ptr_string(std::unique_ptr<std::string> s) {
176 if (*s == "2020") {
177 cxx_test_suite_set_correct();
178 }
179}
David Tolnayad5b8af2020-01-26 16:59:13 -0800180
Myron Ahneba35cf2020-02-05 19:41:51 +0700181void c_take_unique_ptr_vector_u8(std::unique_ptr<std::vector<uint8_t>> v) {
182 if (v->size() == 4) {
183 cxx_test_suite_set_correct();
184 }
185}
186
187void c_take_unique_ptr_vector_f64(std::unique_ptr<std::vector<double>> v) {
188 if (v->size() == 4) {
189 cxx_test_suite_set_correct();
190 }
191}
192
193void c_take_unique_ptr_vector_shared(std::unique_ptr<std::vector<Shared>> v) {
194 if (v->size() == 2) {
195 cxx_test_suite_set_correct();
196 }
197}
198
David Tolnay2244d1f2020-04-25 13:58:18 -0700199void c_take_ref_vector(const std::vector<uint8_t> &v) {
200 if (v.size() == 4) {
201 cxx_test_suite_set_correct();
202 }
203}
204
David Tolnayd2ce8a92020-04-25 16:16:45 -0700205void c_take_rust_vec(rust::Vec<uint8_t> v) { c_take_ref_rust_vec(v); }
Myron Ahneba35cf2020-02-05 19:41:51 +0700206
David Tolnayd2ce8a92020-04-25 16:16:45 -0700207void c_take_rust_vec_shared(rust::Vec<Shared> v) {
Myron Ahneba35cf2020-02-05 19:41:51 +0700208 uint32_t sum = 0;
David Tolnayc87c2152020-04-24 17:07:41 -0700209 for (auto i : v) {
Myron Ahneba35cf2020-02-05 19:41:51 +0700210 sum += i.z;
211 }
212 if (sum == 2021) {
213 cxx_test_suite_set_correct();
214 }
215}
216
myronahnda9be502020-04-29 05:47:23 +0700217void c_take_rust_vec_shared_forward_iterator(rust::Vec<Shared> v) {
218 // Exercise requirements of ForwardIterator
219 // https://en.cppreference.com/w/cpp/named_req/ForwardIterator
220 uint32_t sum = 0;
221 for (auto it = v.begin(), it_end = v.end(); it != it_end; it++) {
222 sum += it->z;
223 }
224 if (sum == 2021) {
225 cxx_test_suite_set_correct();
226 }
227}
228
David Tolnayd2ce8a92020-04-25 16:16:45 -0700229void c_take_ref_rust_vec(const rust::Vec<uint8_t> &v) {
230 uint8_t sum = std::accumulate(v.begin(), v.end(), 0);
231 if (sum == 200) {
232 cxx_test_suite_set_correct();
233 }
234}
235
myronahnda9be502020-04-29 05:47:23 +0700236void c_take_ref_rust_vec_copy(const rust::Vec<uint8_t> &v) {
237 // The std::copy() will make sure rust::Vec<>::const_iterator satisfies the
238 // requirements for std::iterator_traits.
239 // https://en.cppreference.com/w/cpp/iterator/iterator_traits
David Tolnayf5aeea22020-04-30 19:34:51 -0700240 std::vector<uint8_t> stdv;
241 std::copy(v.begin(), v.end(), std::back_inserter(stdv));
242 uint8_t sum = std::accumulate(stdv.begin(), stdv.end(), 0);
myronahnda9be502020-04-29 05:47:23 +0700243 if (sum == 200) {
244 cxx_test_suite_set_correct();
245 }
246}
247
David Tolnay75dca2e2020-03-25 20:17:52 -0700248void c_take_callback(rust::Fn<size_t(rust::String)> callback) {
249 callback("2020");
250}
251
Joel Galensonc03402a2020-04-23 17:31:09 -0700252void c_take_enum(Enum e) {
253 if (e == Enum::AVal) {
254 cxx_test_suite_set_correct();
255 }
256}
257
David Tolnayebef4a22020-03-17 15:33:47 -0700258void c_try_return_void() {}
259
260size_t c_try_return_primitive() { return 2020; }
261
262size_t c_fail_return_primitive() { throw std::logic_error("logic error"); }
263
David Tolnay99642622020-03-25 13:07:35 -0700264rust::Box<R> c_try_return_box() { return c_return_box(); }
Myron Ahn84849302020-03-25 22:00:58 +0700265
David Tolnay99642622020-03-25 13:07:35 -0700266const rust::String &c_try_return_ref(const rust::String &s) { return s; }
267
268rust::Str c_try_return_str(rust::Str s) { return s; }
269
Adrian Taylorec9430e2020-04-14 16:09:58 -0700270rust::Slice<uint8_t> c_try_return_sliceu8(rust::Slice<uint8_t> s) { return s; }
Adrian Taylorf5dd5522020-04-13 16:50:14 -0700271
David Tolnay99642622020-03-25 13:07:35 -0700272rust::String c_try_return_rust_string() { return c_return_rust_string(); }
273
274std::unique_ptr<std::string> c_try_return_unique_ptr_string() {
275 return c_return_unique_ptr_string();
Myron Ahn84849302020-03-25 22:00:58 +0700276}
277
David Tolnay8b9d1762020-04-25 16:05:46 -0700278rust::Vec<uint8_t> c_try_return_rust_vec() {
279 throw std::runtime_error("unimplemented");
280}
281
David Tolnay77989692020-04-25 15:57:47 -0700282const rust::Vec<uint8_t> &c_try_return_ref_rust_vec(const C &c) {
283 (void)c;
284 throw std::runtime_error("unimplemented");
285}
286
David Tolnay2fe58c62020-03-06 16:23:09 -0800287extern "C" C *cxx_test_suite_get_unique_ptr() noexcept {
David Tolnay4b3a66e2020-03-06 16:14:00 -0800288 return std::unique_ptr<C>(new C{2020}).release();
289}
290
David Tolnay85db24862020-03-06 16:24:41 -0800291extern "C" std::string *cxx_test_suite_get_unique_ptr_string() noexcept {
292 return std::unique_ptr<std::string>(new std::string("2020")).release();
293}
294
David Tolnayf306da42020-02-22 19:55:43 -0800295extern "C" const char *cxx_run_test() noexcept {
296#define STRINGIFY(x) #x
297#define TOSTRING(x) STRINGIFY(x)
298#define ASSERT(x) \
299 do { \
300 if (!(x)) { \
301 return "Assertion failed: `" #x "`, " __FILE__ ":" TOSTRING(__LINE__); \
302 } \
303 } while (false)
304
305 ASSERT(r_return_primitive() == 2020);
306 ASSERT(r_return_shared().z == 2020);
David Tolnay5cd8d612020-03-06 15:56:30 -0800307 ASSERT(cxx_test_suite_r_is_correct(&*r_return_box()));
David Tolnay4b3a66e2020-03-06 16:14:00 -0800308 ASSERT(r_return_unique_ptr()->get() == 2020);
David Tolnayf306da42020-02-22 19:55:43 -0800309 ASSERT(r_return_ref(Shared{2020}) == 2020);
310 ASSERT(std::string(r_return_str(Shared{2020})) == "2020");
311 ASSERT(std::string(r_return_rust_string()) == "2020");
David Tolnay85db24862020-03-06 16:24:41 -0800312 ASSERT(*r_return_unique_ptr_string() == "2020");
Joel Galensonba676072020-04-27 15:55:45 -0700313 ASSERT(r_return_identity(2020) == 2020);
314 ASSERT(r_return_sum(2020, 1) == 2021);
Joel Galensonc03402a2020-04-23 17:31:09 -0700315 ASSERT(r_return_enum(0) == Enum::AVal);
316 ASSERT(r_return_enum(1) == Enum::BVal);
317 ASSERT(r_return_enum(2021) == Enum::CVal);
David Tolnayf306da42020-02-22 19:55:43 -0800318
319 r_take_primitive(2020);
320 r_take_shared(Shared{2020});
321 r_take_unique_ptr(std::unique_ptr<C>(new C{2020}));
322 r_take_ref_c(C{2020});
David Tolnay750755e2020-03-01 13:04:08 -0800323 r_take_str(rust::Str("2020"));
David Tolnay4037de22020-04-30 19:51:04 -0700324 r_take_sliceu8(rust::Slice<uint8_t>(
325 reinterpret_cast<const uint8_t *>(SLICE_DATA), sizeof(SLICE_DATA)));
David Tolnay40226ab2020-03-03 00:05:35 -0800326 r_take_rust_string(rust::String("2020"));
David Tolnayf306da42020-02-22 19:55:43 -0800327 r_take_unique_ptr_string(
328 std::unique_ptr<std::string>(new std::string("2020")));
Joel Galensonc03402a2020-04-23 17:31:09 -0700329 r_take_enum(Enum::AVal);
David Tolnayf306da42020-02-22 19:55:43 -0800330
David Tolnayb6c5ea72020-03-16 13:36:28 -0700331 ASSERT(r_try_return_primitive() == 2020);
332 try {
333 r_fail_return_primitive();
334 ASSERT(false);
335 } catch (const rust::Error &e) {
336 ASSERT(std::strcmp(e.what(), "rust error") == 0);
337 }
338
Joel Galensonc1c4e7a2020-04-15 10:21:00 -0700339 auto r2 = r_return_r2(2020);
340 ASSERT(r2->get() == 2020);
341 ASSERT(r2->set(2021) == 2021);
342 ASSERT(r2->get() == 2021);
343 ASSERT(r2->set(2020) == 2020);
344 ASSERT(r2->get() == 2020);
345
David Tolnayf306da42020-02-22 19:55:43 -0800346 cxx_test_suite_set_correct();
347 return nullptr;
348}
349
David Tolnay97c72102020-01-25 16:49:00 -0800350} // namespace tests