blob: 8834581c892bea3bc7c346d69ffb1540f9353f1f [file] [log] [blame]
David Tolnay736cbca2020-03-11 16:49:18 -07001#include "../include/cxx.h"
David Tolnay7db73692019-10-20 14:51:12 -04002#include <cstring>
David Tolnay71918ec2020-04-11 21:52:09 -07003#include <exception>
David Tolnay001102a2020-03-01 20:05:04 -08004#include <iostream>
David Tolnay7db73692019-10-20 14:51:12 -04005#include <memory>
6#include <stdexcept>
7
David Tolnayce5af542020-04-10 18:08:30 -07008template <typename Exception>
9[[noreturn]] static void panic(const char *msg) {
10#if defined(RUST_CXX_NO_EXCEPTIONS)
11 std::cerr << "Error: " << msg << ". Aborting." << std::endl;
David Tolnay71918ec2020-04-11 21:52:09 -070012 std::terminate();
David Tolnayce5af542020-04-10 18:08:30 -070013#else
14 throw Exception(msg);
15#endif
16}
17
David Tolnay7db73692019-10-20 14:51:12 -040018extern "C" {
David Tolnay8c730492020-03-13 01:29:06 -070019const char *cxxbridge02$cxx_string$data(const std::string &s) noexcept {
David Tolnay7db73692019-10-20 14:51:12 -040020 return s.data();
21}
22
David Tolnay8c730492020-03-13 01:29:06 -070023size_t cxxbridge02$cxx_string$length(const std::string &s) noexcept {
David Tolnay7db73692019-10-20 14:51:12 -040024 return s.length();
25}
26
David Tolnay750755e2020-03-01 13:04:08 -080027// rust::String
David Tolnay8c730492020-03-13 01:29:06 -070028void cxxbridge02$string$new(rust::String *self) noexcept;
29void cxxbridge02$string$clone(rust::String *self,
David Tolnay6c089102020-03-02 00:21:13 -080030 const rust::String &other) noexcept;
David Tolnay8c730492020-03-13 01:29:06 -070031bool cxxbridge02$string$from(rust::String *self, const char *ptr,
David Tolnay6c089102020-03-02 00:21:13 -080032 size_t len) noexcept;
David Tolnay8c730492020-03-13 01:29:06 -070033void cxxbridge02$string$drop(rust::String *self) noexcept;
34const char *cxxbridge02$string$ptr(const rust::String *self) noexcept;
35size_t cxxbridge02$string$len(const rust::String *self) noexcept;
David Tolnay7db73692019-10-20 14:51:12 -040036
David Tolnay750755e2020-03-01 13:04:08 -080037// rust::Str
David Tolnay8c730492020-03-13 01:29:06 -070038bool cxxbridge02$str$valid(const char *ptr, size_t len) noexcept;
David Tolnay7db73692019-10-20 14:51:12 -040039} // extern "C"
40
David Tolnay750755e2020-03-01 13:04:08 -080041namespace rust {
David Tolnay8c730492020-03-13 01:29:06 -070042inline namespace cxxbridge02 {
David Tolnay7db73692019-10-20 14:51:12 -040043
David Tolnay8c730492020-03-13 01:29:06 -070044String::String() noexcept { cxxbridge02$string$new(this); }
David Tolnay7db73692019-10-20 14:51:12 -040045
David Tolnay56082162020-03-01 12:57:33 -080046String::String(const String &other) noexcept {
David Tolnay8c730492020-03-13 01:29:06 -070047 cxxbridge02$string$clone(this, other);
David Tolnay7db73692019-10-20 14:51:12 -040048}
49
David Tolnay56082162020-03-01 12:57:33 -080050String::String(String &&other) noexcept {
David Tolnay7db73692019-10-20 14:51:12 -040051 this->repr = other.repr;
David Tolnay8c730492020-03-13 01:29:06 -070052 cxxbridge02$string$new(&other);
David Tolnay7db73692019-10-20 14:51:12 -040053}
54
David Tolnay8c730492020-03-13 01:29:06 -070055String::~String() noexcept { cxxbridge02$string$drop(this); }
David Tolnay7db73692019-10-20 14:51:12 -040056
David Tolnay56082162020-03-01 12:57:33 -080057String::String(const std::string &s) {
David Tolnay7db73692019-10-20 14:51:12 -040058 auto ptr = s.data();
59 auto len = s.length();
David Tolnay8c730492020-03-13 01:29:06 -070060 if (!cxxbridge02$string$from(this, ptr, len)) {
David Tolnayce5af542020-04-10 18:08:30 -070061 panic<std::invalid_argument>("data for rust::String is not utf-8");
David Tolnay7db73692019-10-20 14:51:12 -040062 }
63}
64
David Tolnayd9c4ac92020-03-01 20:33:58 -080065String::String(const char *s) {
David Tolnaybb07a4f2020-03-16 23:04:00 -070066 auto len = std::strlen(s);
David Tolnay8c730492020-03-13 01:29:06 -070067 if (!cxxbridge02$string$from(this, s, len)) {
David Tolnayce5af542020-04-10 18:08:30 -070068 panic<std::invalid_argument>("data for rust::String is not utf-8");
David Tolnayd9c4ac92020-03-01 20:33:58 -080069 }
David Tolnay7db73692019-10-20 14:51:12 -040070}
71
David Tolnay56082162020-03-01 12:57:33 -080072String &String::operator=(const String &other) noexcept {
David Tolnay7db73692019-10-20 14:51:12 -040073 if (this != &other) {
David Tolnay8c730492020-03-13 01:29:06 -070074 cxxbridge02$string$drop(this);
75 cxxbridge02$string$clone(this, other);
David Tolnay7db73692019-10-20 14:51:12 -040076 }
77 return *this;
78}
79
David Tolnay56082162020-03-01 12:57:33 -080080String &String::operator=(String &&other) noexcept {
David Tolnay7db73692019-10-20 14:51:12 -040081 if (this != &other) {
David Tolnay8c730492020-03-13 01:29:06 -070082 cxxbridge02$string$drop(this);
David Tolnay7db73692019-10-20 14:51:12 -040083 this->repr = other.repr;
David Tolnay8c730492020-03-13 01:29:06 -070084 cxxbridge02$string$new(&other);
David Tolnay7db73692019-10-20 14:51:12 -040085 }
86 return *this;
87}
88
David Tolnayd9c4ac92020-03-01 20:33:58 -080089String::operator std::string() const {
90 return std::string(this->data(), this->size());
91}
92
David Tolnay56082162020-03-01 12:57:33 -080093const char *String::data() const noexcept {
David Tolnay8c730492020-03-13 01:29:06 -070094 return cxxbridge02$string$ptr(this);
David Tolnay7db73692019-10-20 14:51:12 -040095}
96
David Tolnay8c730492020-03-13 01:29:06 -070097size_t String::size() const noexcept { return cxxbridge02$string$len(this); }
David Tolnay7db73692019-10-20 14:51:12 -040098
David Tolnay8c730492020-03-13 01:29:06 -070099size_t String::length() const noexcept { return cxxbridge02$string$len(this); }
David Tolnay7db73692019-10-20 14:51:12 -0400100
David Tolnayd1e2efc2020-03-03 22:25:43 -0800101String::String(unsafe_bitcopy_t, const String &bits) noexcept
102 : repr(bits.repr) {}
103
David Tolnay56082162020-03-01 12:57:33 -0800104std::ostream &operator<<(std::ostream &os, const String &s) {
David Tolnay7db73692019-10-20 14:51:12 -0400105 os.write(s.data(), s.size());
106 return os;
107}
108
David Tolnay69fe4c22020-03-01 13:57:24 -0800109Str::Str() noexcept : repr(Repr{reinterpret_cast<const char *>(this), 0}) {}
David Tolnay7db73692019-10-20 14:51:12 -0400110
David Tolnayd9c4ac92020-03-01 20:33:58 -0800111Str::Str(const Str &) noexcept = default;
David Tolnay7db73692019-10-20 14:51:12 -0400112
David Tolnay09dbe752020-03-01 13:00:40 -0800113Str::Str(const std::string &s) : repr(Repr{s.data(), s.length()}) {
David Tolnay8c730492020-03-13 01:29:06 -0700114 if (!cxxbridge02$str$valid(this->repr.ptr, this->repr.len)) {
David Tolnayce5af542020-04-10 18:08:30 -0700115 panic<std::invalid_argument>("data for rust::Str is not utf-8");
David Tolnay7db73692019-10-20 14:51:12 -0400116 }
117}
118
David Tolnaybb07a4f2020-03-16 23:04:00 -0700119Str::Str(const char *s) : repr(Repr{s, std::strlen(s)}) {
David Tolnay8c730492020-03-13 01:29:06 -0700120 if (!cxxbridge02$str$valid(this->repr.ptr, this->repr.len)) {
David Tolnayce5af542020-04-10 18:08:30 -0700121 panic<std::invalid_argument>("data for rust::Str is not utf-8");
David Tolnayd9c4ac92020-03-01 20:33:58 -0800122 }
123}
David Tolnay7db73692019-10-20 14:51:12 -0400124
David Tolnay09dbe752020-03-01 13:00:40 -0800125Str &Str::operator=(Str other) noexcept {
David Tolnay7db73692019-10-20 14:51:12 -0400126 this->repr = other.repr;
127 return *this;
128}
129
David Tolnay09dbe752020-03-01 13:00:40 -0800130Str::operator std::string() const {
David Tolnay7db73692019-10-20 14:51:12 -0400131 return std::string(this->data(), this->size());
132}
133
David Tolnay09dbe752020-03-01 13:00:40 -0800134const char *Str::data() const noexcept { return this->repr.ptr; }
David Tolnay7db73692019-10-20 14:51:12 -0400135
David Tolnay09dbe752020-03-01 13:00:40 -0800136size_t Str::size() const noexcept { return this->repr.len; }
David Tolnay7db73692019-10-20 14:51:12 -0400137
David Tolnay09dbe752020-03-01 13:00:40 -0800138size_t Str::length() const noexcept { return this->repr.len; }
David Tolnay7db73692019-10-20 14:51:12 -0400139
David Tolnay09dbe752020-03-01 13:00:40 -0800140Str::Str(Repr repr_) noexcept : repr(repr_) {}
David Tolnay7db73692019-10-20 14:51:12 -0400141
David Tolnay09dbe752020-03-01 13:00:40 -0800142Str::operator Repr() noexcept { return this->repr; }
David Tolnay7db73692019-10-20 14:51:12 -0400143
David Tolnay09dbe752020-03-01 13:00:40 -0800144std::ostream &operator<<(std::ostream &os, const Str &s) {
David Tolnay7db73692019-10-20 14:51:12 -0400145 os.write(s.data(), s.size());
146 return os;
147}
148
David Tolnay1e548172020-03-16 13:37:09 -0700149extern "C" {
150const char *cxxbridge02$error(const char *ptr, size_t len) {
151 char *copy = new char[len];
152 strncpy(copy, ptr, len);
153 return copy;
154}
155} // extern "C"
156
157Error::Error(Str::Repr msg) noexcept : msg(msg) {}
158
159Error::Error(const Error &other) {
160 this->msg.ptr = cxxbridge02$error(other.msg.ptr, other.msg.len);
161 this->msg.len = other.msg.len;
162}
163
164Error::Error(Error &&other) noexcept {
165 delete[] this->msg.ptr;
166 this->msg = other.msg;
167 other.msg.ptr = nullptr;
168 other.msg.len = 0;
169}
170
171Error::~Error() noexcept { delete[] this->msg.ptr; }
172
173const char *Error::what() const noexcept { return this->msg.ptr; }
174
David Tolnay8c730492020-03-13 01:29:06 -0700175} // namespace cxxbridge02
David Tolnay750755e2020-03-01 13:04:08 -0800176} // namespace rust
David Tolnay7db73692019-10-20 14:51:12 -0400177
178extern "C" {
David Tolnay8c730492020-03-13 01:29:06 -0700179void cxxbridge02$unique_ptr$std$string$null(
David Tolnay7db73692019-10-20 14:51:12 -0400180 std::unique_ptr<std::string> *ptr) noexcept {
181 new (ptr) std::unique_ptr<std::string>();
182}
David Tolnay8c730492020-03-13 01:29:06 -0700183void cxxbridge02$unique_ptr$std$string$raw(std::unique_ptr<std::string> *ptr,
David Tolnay7db73692019-10-20 14:51:12 -0400184 std::string *raw) noexcept {
185 new (ptr) std::unique_ptr<std::string>(raw);
186}
David Tolnay8c730492020-03-13 01:29:06 -0700187const std::string *cxxbridge02$unique_ptr$std$string$get(
David Tolnay7db73692019-10-20 14:51:12 -0400188 const std::unique_ptr<std::string> &ptr) noexcept {
189 return ptr.get();
190}
David Tolnay8c730492020-03-13 01:29:06 -0700191std::string *cxxbridge02$unique_ptr$std$string$release(
David Tolnay7db73692019-10-20 14:51:12 -0400192 std::unique_ptr<std::string> &ptr) noexcept {
193 return ptr.release();
194}
David Tolnay8c730492020-03-13 01:29:06 -0700195void cxxbridge02$unique_ptr$std$string$drop(
David Tolnay7db73692019-10-20 14:51:12 -0400196 std::unique_ptr<std::string> *ptr) noexcept {
197 ptr->~unique_ptr();
198}
199} // extern "C"