Move header include path to rust/cxx.h
diff --git a/src/cxx.cc b/src/cxx.cc
new file mode 100644
index 0000000..f7b02ec
--- /dev/null
+++ b/src/cxx.cc
@@ -0,0 +1,166 @@
+#include "../include/cxx.h"
+#include <cstring>
+#include <iostream>
+#include <memory>
+#include <stdexcept>
+
+extern "C" {
+const char *cxxbridge01$cxx_string$data(const std::string &s) noexcept {
+  return s.data();
+}
+
+size_t cxxbridge01$cxx_string$length(const std::string &s) noexcept {
+  return s.length();
+}
+
+// rust::String
+void cxxbridge01$string$new(rust::String *self) noexcept;
+void cxxbridge01$string$clone(rust::String *self,
+                              const rust::String &other) noexcept;
+bool cxxbridge01$string$from(rust::String *self, const char *ptr,
+                             size_t len) noexcept;
+void cxxbridge01$string$drop(rust::String *self) noexcept;
+const char *cxxbridge01$string$ptr(const rust::String *self) noexcept;
+size_t cxxbridge01$string$len(const rust::String *self) noexcept;
+
+// rust::Str
+bool cxxbridge01$str$valid(const char *ptr, size_t len) noexcept;
+} // extern "C"
+
+namespace rust {
+inline namespace cxxbridge01 {
+
+String::String() noexcept { cxxbridge01$string$new(this); }
+
+String::String(const String &other) noexcept {
+  cxxbridge01$string$clone(this, other);
+}
+
+String::String(String &&other) noexcept {
+  this->repr = other.repr;
+  cxxbridge01$string$new(&other);
+}
+
+String::~String() noexcept { cxxbridge01$string$drop(this); }
+
+String::String(const std::string &s) {
+  auto ptr = s.data();
+  auto len = s.length();
+  if (!cxxbridge01$string$from(this, ptr, len)) {
+    throw std::invalid_argument("data for rust::String is not utf-8");
+  }
+}
+
+String::String(const char *s) {
+  auto len = strlen(s);
+  if (!cxxbridge01$string$from(this, s, len)) {
+    throw std::invalid_argument("data for rust::String is not utf-8");
+  }
+}
+
+String &String::operator=(const String &other) noexcept {
+  if (this != &other) {
+    cxxbridge01$string$drop(this);
+    cxxbridge01$string$clone(this, other);
+  }
+  return *this;
+}
+
+String &String::operator=(String &&other) noexcept {
+  if (this != &other) {
+    cxxbridge01$string$drop(this);
+    this->repr = other.repr;
+    cxxbridge01$string$new(&other);
+  }
+  return *this;
+}
+
+String::operator std::string() const {
+  return std::string(this->data(), this->size());
+}
+
+const char *String::data() const noexcept {
+  return cxxbridge01$string$ptr(this);
+}
+
+size_t String::size() const noexcept { return cxxbridge01$string$len(this); }
+
+size_t String::length() const noexcept { return cxxbridge01$string$len(this); }
+
+String::String(unsafe_bitcopy_t, const String &bits) noexcept
+    : repr(bits.repr) {}
+
+std::ostream &operator<<(std::ostream &os, const String &s) {
+  os.write(s.data(), s.size());
+  return os;
+}
+
+Str::Str() noexcept : repr(Repr{reinterpret_cast<const char *>(this), 0}) {}
+
+Str::Str(const Str &) noexcept = default;
+
+Str::Str(const std::string &s) : repr(Repr{s.data(), s.length()}) {
+  if (!cxxbridge01$str$valid(this->repr.ptr, this->repr.len)) {
+    throw std::invalid_argument("data for rust::Str is not utf-8");
+  }
+}
+
+Str::Str(const char *s) : repr(Repr{s, strlen(s)}) {
+  if (!cxxbridge01$str$valid(this->repr.ptr, this->repr.len)) {
+    throw std::invalid_argument("data for rust::Str is not utf-8");
+  }
+}
+
+Str &Str::operator=(Str other) noexcept {
+  this->repr = other.repr;
+  return *this;
+}
+
+Str::operator std::string() const {
+  return std::string(this->data(), this->size());
+}
+
+const char *Str::data() const noexcept { return this->repr.ptr; }
+
+size_t Str::size() const noexcept { return this->repr.len; }
+
+size_t Str::length() const noexcept { return this->repr.len; }
+
+Str::Str(Repr repr_) noexcept : repr(repr_) {}
+
+Str::operator Repr() noexcept { return this->repr; }
+
+std::ostream &operator<<(std::ostream &os, const Str &s) {
+  os.write(s.data(), s.size());
+  return os;
+}
+
+} // namespace cxxbridge01
+} // namespace rust
+
+extern "C" {
+void cxxbridge01$unique_ptr$std$string$null(
+    std::unique_ptr<std::string> *ptr) noexcept {
+  new (ptr) std::unique_ptr<std::string>();
+}
+void cxxbridge01$unique_ptr$std$string$new(std::unique_ptr<std::string> *ptr,
+                                           std::string *value) noexcept {
+  new (ptr) std::unique_ptr<std::string>(new std::string(std::move(*value)));
+}
+void cxxbridge01$unique_ptr$std$string$raw(std::unique_ptr<std::string> *ptr,
+                                           std::string *raw) noexcept {
+  new (ptr) std::unique_ptr<std::string>(raw);
+}
+const std::string *cxxbridge01$unique_ptr$std$string$get(
+    const std::unique_ptr<std::string> &ptr) noexcept {
+  return ptr.get();
+}
+std::string *cxxbridge01$unique_ptr$std$string$release(
+    std::unique_ptr<std::string> &ptr) noexcept {
+  return ptr.release();
+}
+void cxxbridge01$unique_ptr$std$string$drop(
+    std::unique_ptr<std::string> *ptr) noexcept {
+  ptr->~unique_ptr();
+}
+} // extern "C"