blob: e8ec979d62eb013bcdde3a74e6e321be46f9b218 [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 Tolnayc2279002020-04-11 22:06:00 -07008template <typename Exception> static void panic [[noreturn]] (const char *msg) {
David Tolnayce5af542020-04-10 18:08:30 -07009#if defined(RUST_CXX_NO_EXCEPTIONS)
10 std::cerr << "Error: " << msg << ". Aborting." << std::endl;
David Tolnay71918ec2020-04-11 21:52:09 -070011 std::terminate();
David Tolnayce5af542020-04-10 18:08:30 -070012#else
13 throw Exception(msg);
14#endif
15}
16
David Tolnay7db73692019-10-20 14:51:12 -040017extern "C" {
David Tolnay8c730492020-03-13 01:29:06 -070018const char *cxxbridge02$cxx_string$data(const std::string &s) noexcept {
David Tolnay7db73692019-10-20 14:51:12 -040019 return s.data();
20}
21
David Tolnay8c730492020-03-13 01:29:06 -070022size_t cxxbridge02$cxx_string$length(const std::string &s) noexcept {
David Tolnay7db73692019-10-20 14:51:12 -040023 return s.length();
24}
25
David Tolnay750755e2020-03-01 13:04:08 -080026// rust::String
David Tolnay8c730492020-03-13 01:29:06 -070027void cxxbridge02$string$new(rust::String *self) noexcept;
28void cxxbridge02$string$clone(rust::String *self,
David Tolnay6c089102020-03-02 00:21:13 -080029 const rust::String &other) noexcept;
David Tolnay8c730492020-03-13 01:29:06 -070030bool cxxbridge02$string$from(rust::String *self, const char *ptr,
David Tolnay6c089102020-03-02 00:21:13 -080031 size_t len) noexcept;
David Tolnay8c730492020-03-13 01:29:06 -070032void cxxbridge02$string$drop(rust::String *self) noexcept;
33const char *cxxbridge02$string$ptr(const rust::String *self) noexcept;
34size_t cxxbridge02$string$len(const rust::String *self) noexcept;
David Tolnay7db73692019-10-20 14:51:12 -040035
David Tolnay750755e2020-03-01 13:04:08 -080036// rust::Str
David Tolnay8c730492020-03-13 01:29:06 -070037bool cxxbridge02$str$valid(const char *ptr, size_t len) noexcept;
David Tolnay7db73692019-10-20 14:51:12 -040038} // extern "C"
39
David Tolnay750755e2020-03-01 13:04:08 -080040namespace rust {
David Tolnay8c730492020-03-13 01:29:06 -070041inline namespace cxxbridge02 {
David Tolnay7db73692019-10-20 14:51:12 -040042
David Tolnay8c730492020-03-13 01:29:06 -070043String::String() noexcept { cxxbridge02$string$new(this); }
David Tolnay7db73692019-10-20 14:51:12 -040044
David Tolnay56082162020-03-01 12:57:33 -080045String::String(const String &other) noexcept {
David Tolnay8c730492020-03-13 01:29:06 -070046 cxxbridge02$string$clone(this, other);
David Tolnay7db73692019-10-20 14:51:12 -040047}
48
David Tolnay56082162020-03-01 12:57:33 -080049String::String(String &&other) noexcept {
David Tolnay7db73692019-10-20 14:51:12 -040050 this->repr = other.repr;
David Tolnay8c730492020-03-13 01:29:06 -070051 cxxbridge02$string$new(&other);
David Tolnay7db73692019-10-20 14:51:12 -040052}
53
David Tolnay8c730492020-03-13 01:29:06 -070054String::~String() noexcept { cxxbridge02$string$drop(this); }
David Tolnay7db73692019-10-20 14:51:12 -040055
David Tolnay56082162020-03-01 12:57:33 -080056String::String(const std::string &s) {
David Tolnay7db73692019-10-20 14:51:12 -040057 auto ptr = s.data();
58 auto len = s.length();
David Tolnay8c730492020-03-13 01:29:06 -070059 if (!cxxbridge02$string$from(this, ptr, len)) {
David Tolnayce5af542020-04-10 18:08:30 -070060 panic<std::invalid_argument>("data for rust::String is not utf-8");
David Tolnay7db73692019-10-20 14:51:12 -040061 }
62}
63
David Tolnayd9c4ac92020-03-01 20:33:58 -080064String::String(const char *s) {
David Tolnaybb07a4f2020-03-16 23:04:00 -070065 auto len = std::strlen(s);
David Tolnay8c730492020-03-13 01:29:06 -070066 if (!cxxbridge02$string$from(this, s, len)) {
David Tolnayce5af542020-04-10 18:08:30 -070067 panic<std::invalid_argument>("data for rust::String is not utf-8");
David Tolnayd9c4ac92020-03-01 20:33:58 -080068 }
David Tolnay7db73692019-10-20 14:51:12 -040069}
70
David Tolnay56082162020-03-01 12:57:33 -080071String &String::operator=(const String &other) noexcept {
David Tolnay7db73692019-10-20 14:51:12 -040072 if (this != &other) {
David Tolnay8c730492020-03-13 01:29:06 -070073 cxxbridge02$string$drop(this);
74 cxxbridge02$string$clone(this, other);
David Tolnay7db73692019-10-20 14:51:12 -040075 }
76 return *this;
77}
78
David Tolnay56082162020-03-01 12:57:33 -080079String &String::operator=(String &&other) noexcept {
David Tolnay7db73692019-10-20 14:51:12 -040080 if (this != &other) {
David Tolnay8c730492020-03-13 01:29:06 -070081 cxxbridge02$string$drop(this);
David Tolnay7db73692019-10-20 14:51:12 -040082 this->repr = other.repr;
David Tolnay8c730492020-03-13 01:29:06 -070083 cxxbridge02$string$new(&other);
David Tolnay7db73692019-10-20 14:51:12 -040084 }
85 return *this;
86}
87
David Tolnayd9c4ac92020-03-01 20:33:58 -080088String::operator std::string() const {
89 return std::string(this->data(), this->size());
90}
91
David Tolnay56082162020-03-01 12:57:33 -080092const char *String::data() const noexcept {
David Tolnay8c730492020-03-13 01:29:06 -070093 return cxxbridge02$string$ptr(this);
David Tolnay7db73692019-10-20 14:51:12 -040094}
95
David Tolnay8c730492020-03-13 01:29:06 -070096size_t String::size() const noexcept { return cxxbridge02$string$len(this); }
David Tolnay7db73692019-10-20 14:51:12 -040097
David Tolnay8c730492020-03-13 01:29:06 -070098size_t String::length() const noexcept { return cxxbridge02$string$len(this); }
David Tolnay7db73692019-10-20 14:51:12 -040099
David Tolnayd1e2efc2020-03-03 22:25:43 -0800100String::String(unsafe_bitcopy_t, const String &bits) noexcept
101 : repr(bits.repr) {}
102
David Tolnay56082162020-03-01 12:57:33 -0800103std::ostream &operator<<(std::ostream &os, const String &s) {
David Tolnay7db73692019-10-20 14:51:12 -0400104 os.write(s.data(), s.size());
105 return os;
106}
107
David Tolnay69fe4c22020-03-01 13:57:24 -0800108Str::Str() noexcept : repr(Repr{reinterpret_cast<const char *>(this), 0}) {}
David Tolnay7db73692019-10-20 14:51:12 -0400109
David Tolnayd9c4ac92020-03-01 20:33:58 -0800110Str::Str(const Str &) noexcept = default;
David Tolnay7db73692019-10-20 14:51:12 -0400111
David Tolnay09dbe752020-03-01 13:00:40 -0800112Str::Str(const std::string &s) : repr(Repr{s.data(), s.length()}) {
David Tolnay8c730492020-03-13 01:29:06 -0700113 if (!cxxbridge02$str$valid(this->repr.ptr, this->repr.len)) {
David Tolnayce5af542020-04-10 18:08:30 -0700114 panic<std::invalid_argument>("data for rust::Str is not utf-8");
David Tolnay7db73692019-10-20 14:51:12 -0400115 }
116}
117
David Tolnaybb07a4f2020-03-16 23:04:00 -0700118Str::Str(const char *s) : repr(Repr{s, std::strlen(s)}) {
David Tolnay8c730492020-03-13 01:29:06 -0700119 if (!cxxbridge02$str$valid(this->repr.ptr, this->repr.len)) {
David Tolnayce5af542020-04-10 18:08:30 -0700120 panic<std::invalid_argument>("data for rust::Str is not utf-8");
David Tolnayd9c4ac92020-03-01 20:33:58 -0800121 }
122}
David Tolnay7db73692019-10-20 14:51:12 -0400123
David Tolnay09dbe752020-03-01 13:00:40 -0800124Str &Str::operator=(Str other) noexcept {
David Tolnay7db73692019-10-20 14:51:12 -0400125 this->repr = other.repr;
126 return *this;
127}
128
David Tolnay09dbe752020-03-01 13:00:40 -0800129Str::operator std::string() const {
David Tolnay7db73692019-10-20 14:51:12 -0400130 return std::string(this->data(), this->size());
131}
132
David Tolnay09dbe752020-03-01 13:00:40 -0800133const char *Str::data() const noexcept { return this->repr.ptr; }
David Tolnay7db73692019-10-20 14:51:12 -0400134
David Tolnay09dbe752020-03-01 13:00:40 -0800135size_t Str::size() const noexcept { return this->repr.len; }
David Tolnay7db73692019-10-20 14:51:12 -0400136
David Tolnay09dbe752020-03-01 13:00:40 -0800137size_t Str::length() const noexcept { return this->repr.len; }
David Tolnay7db73692019-10-20 14:51:12 -0400138
David Tolnay09dbe752020-03-01 13:00:40 -0800139Str::Str(Repr repr_) noexcept : repr(repr_) {}
David Tolnay7db73692019-10-20 14:51:12 -0400140
David Tolnay09dbe752020-03-01 13:00:40 -0800141Str::operator Repr() noexcept { return this->repr; }
David Tolnay7db73692019-10-20 14:51:12 -0400142
David Tolnay09dbe752020-03-01 13:00:40 -0800143std::ostream &operator<<(std::ostream &os, const Str &s) {
David Tolnay7db73692019-10-20 14:51:12 -0400144 os.write(s.data(), s.size());
145 return os;
146}
147
David Tolnay1e548172020-03-16 13:37:09 -0700148extern "C" {
149const char *cxxbridge02$error(const char *ptr, size_t len) {
150 char *copy = new char[len];
151 strncpy(copy, ptr, len);
152 return copy;
153}
154} // extern "C"
155
156Error::Error(Str::Repr msg) noexcept : msg(msg) {}
157
158Error::Error(const Error &other) {
159 this->msg.ptr = cxxbridge02$error(other.msg.ptr, other.msg.len);
160 this->msg.len = other.msg.len;
161}
162
163Error::Error(Error &&other) noexcept {
164 delete[] this->msg.ptr;
165 this->msg = other.msg;
166 other.msg.ptr = nullptr;
167 other.msg.len = 0;
168}
169
170Error::~Error() noexcept { delete[] this->msg.ptr; }
171
172const char *Error::what() const noexcept { return this->msg.ptr; }
173
David Tolnay8c730492020-03-13 01:29:06 -0700174} // namespace cxxbridge02
David Tolnay750755e2020-03-01 13:04:08 -0800175} // namespace rust
David Tolnay7db73692019-10-20 14:51:12 -0400176
177extern "C" {
David Tolnay8c730492020-03-13 01:29:06 -0700178void cxxbridge02$unique_ptr$std$string$null(
David Tolnay7db73692019-10-20 14:51:12 -0400179 std::unique_ptr<std::string> *ptr) noexcept {
180 new (ptr) std::unique_ptr<std::string>();
181}
David Tolnay8c730492020-03-13 01:29:06 -0700182void cxxbridge02$unique_ptr$std$string$raw(std::unique_ptr<std::string> *ptr,
David Tolnay7db73692019-10-20 14:51:12 -0400183 std::string *raw) noexcept {
184 new (ptr) std::unique_ptr<std::string>(raw);
185}
David Tolnay8c730492020-03-13 01:29:06 -0700186const std::string *cxxbridge02$unique_ptr$std$string$get(
David Tolnay7db73692019-10-20 14:51:12 -0400187 const std::unique_ptr<std::string> &ptr) noexcept {
188 return ptr.get();
189}
David Tolnay8c730492020-03-13 01:29:06 -0700190std::string *cxxbridge02$unique_ptr$std$string$release(
David Tolnay7db73692019-10-20 14:51:12 -0400191 std::unique_ptr<std::string> &ptr) noexcept {
192 return ptr.release();
193}
David Tolnay8c730492020-03-13 01:29:06 -0700194void cxxbridge02$unique_ptr$std$string$drop(
David Tolnay7db73692019-10-20 14:51:12 -0400195 std::unique_ptr<std::string> *ptr) noexcept {
196 ptr->~unique_ptr();
197}
198} // extern "C"