Skip to content

Commit 96e16b1

Browse files
authored
Set up basic diplomat workflow (#163)
Progress on #104 This adds a simple diplomat integration, but does not use it yet. When we start using it we should add CI that runs `cargo run -p diplomat-gen` and ensures there is no diff.
1 parent 3a8f8c1 commit 96e16b1

File tree

11 files changed

+1098
-0
lines changed

11 files changed

+1098
-0
lines changed

Cargo.lock

+415
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Cargo.toml

+1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
[workspace]
22
resolver = "2"
3+
members = ["diplomat-gen", "temporal_capi"]
34

45
[workspace.package]
56
edition = "2021"

diplomat-gen/Cargo.toml

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
[package]
2+
name = "diplomat-gen"
3+
edition.workspace = true
4+
version.workspace = true
5+
rust-version.workspace = true
6+
authors.workspace = true
7+
license.workspace = true
8+
repository.workspace = true
9+
readme.workspace = true
10+
exclude.workspace = true
11+
12+
[dependencies]
13+
diplomat-tool = "0.9.0"

diplomat-gen/src/main.rs

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
use std::path::Path;
2+
3+
fn main() -> std::io::Result<()> {
4+
let manifest = Path::new(env!("CARGO_MANIFEST_DIR"));
5+
6+
let capi = manifest.parent().unwrap().join("temporal_capi");
7+
8+
let library_config = Default::default();
9+
10+
diplomat_tool::gen(
11+
&capi.join("src/lib.rs"),
12+
"cpp",
13+
&{
14+
let include = capi.join("bindings").join("cpp");
15+
std::fs::remove_dir_all(&include)?;
16+
std::fs::create_dir(&include)?;
17+
include
18+
},
19+
&Default::default(),
20+
library_config,
21+
false,
22+
)
23+
}

temporal_capi/Cargo.toml

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
[package]
2+
name = "temporal_capi"
3+
edition.workspace = true
4+
version.workspace = true
5+
rust-version.workspace = true
6+
authors.workspace = true
7+
license.workspace = true
8+
repository.workspace = true
9+
readme.workspace = true
10+
exclude.workspace = true
11+
12+
[dependencies]
13+
diplomat = "0.9.0"
14+
diplomat-runtime = "0.9.0"
15+
temporal_rs = { version = "0.0.4", path = ".." }
16+
icu_calendar = { version = "2.0.0-beta1", default-features = false}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,215 @@
1+
#ifndef DIPLOMAT_RUNTIME_CPP_H
2+
#define DIPLOMAT_RUNTIME_CPP_H
3+
4+
#include <optional>
5+
#include <string>
6+
#include <type_traits>
7+
#include <variant>
8+
9+
#if __cplusplus >= 202002L
10+
#include <span>
11+
#else
12+
#include <array>
13+
#endif
14+
15+
namespace diplomat {
16+
17+
namespace capi {
18+
extern "C" {
19+
20+
static_assert(sizeof(char) == sizeof(uint8_t), "your architecture's `char` is not 8 bits");
21+
static_assert(sizeof(char16_t) == sizeof(uint16_t), "your architecture's `char16_t` is not 16 bits");
22+
static_assert(sizeof(char32_t) == sizeof(uint32_t), "your architecture's `char32_t` is not 32 bits");
23+
24+
typedef struct DiplomatWrite {
25+
void* context;
26+
char* buf;
27+
size_t len;
28+
size_t cap;
29+
bool grow_failed;
30+
void (*flush)(struct DiplomatWrite*);
31+
bool (*grow)(struct DiplomatWrite*, size_t);
32+
} DiplomatWrite;
33+
34+
bool diplomat_is_str(const char* buf, size_t len);
35+
36+
#define MAKE_SLICES(name, c_ty) \
37+
typedef struct Diplomat##name##View { \
38+
const c_ty* data; \
39+
size_t len; \
40+
} Diplomat##name##View; \
41+
typedef struct Diplomat##name##ViewMut { \
42+
c_ty* data; \
43+
size_t len; \
44+
} Diplomat##name##ViewMut; \
45+
typedef struct Diplomat##name##Array { \
46+
const c_ty* data; \
47+
size_t len; \
48+
} Diplomat##name##Array;
49+
50+
#define MAKE_SLICES_AND_OPTIONS(name, c_ty) \
51+
MAKE_SLICES(name, c_ty) \
52+
typedef struct Option##name {union { c_ty ok; }; bool is_ok; } Option##name;
53+
54+
MAKE_SLICES_AND_OPTIONS(I8, int8_t)
55+
MAKE_SLICES_AND_OPTIONS(U8, uint8_t)
56+
MAKE_SLICES_AND_OPTIONS(I16, int16_t)
57+
MAKE_SLICES_AND_OPTIONS(U16, uint16_t)
58+
MAKE_SLICES_AND_OPTIONS(I32, int32_t)
59+
MAKE_SLICES_AND_OPTIONS(U32, uint32_t)
60+
MAKE_SLICES_AND_OPTIONS(I64, int64_t)
61+
MAKE_SLICES_AND_OPTIONS(U64, uint64_t)
62+
MAKE_SLICES_AND_OPTIONS(Isize, intptr_t)
63+
MAKE_SLICES_AND_OPTIONS(Usize, size_t)
64+
MAKE_SLICES_AND_OPTIONS(F32, float)
65+
MAKE_SLICES_AND_OPTIONS(F64, double)
66+
MAKE_SLICES_AND_OPTIONS(Bool, bool)
67+
MAKE_SLICES_AND_OPTIONS(Char, char32_t)
68+
MAKE_SLICES(String, char)
69+
MAKE_SLICES(String16, char16_t)
70+
MAKE_SLICES(Strings, DiplomatStringView)
71+
MAKE_SLICES(Strings16, DiplomatString16View)
72+
73+
} // extern "C"
74+
} // namespace capi
75+
76+
extern "C" inline void _flush(capi::DiplomatWrite* w) {
77+
std::string* string = reinterpret_cast<std::string*>(w->context);
78+
string->resize(w->len);
79+
};
80+
81+
extern "C" inline bool _grow(capi::DiplomatWrite* w, uintptr_t requested) {
82+
std::string* string = reinterpret_cast<std::string*>(w->context);
83+
string->resize(requested);
84+
w->cap = string->length();
85+
w->buf = &(*string)[0];
86+
return true;
87+
};
88+
89+
inline capi::DiplomatWrite WriteFromString(std::string& string) {
90+
capi::DiplomatWrite w;
91+
w.context = &string;
92+
w.buf = &string[0];
93+
w.len = string.length();
94+
w.cap = string.length();
95+
// Will never become true, as _grow is infallible.
96+
w.grow_failed = false;
97+
w.flush = _flush;
98+
w.grow = _grow;
99+
return w;
100+
};
101+
102+
template<class T> struct Ok {
103+
T inner;
104+
Ok(T&& i): inner(std::move(i)) {}
105+
// We don't want to expose an lvalue-capable constructor in general
106+
// however there is no problem doing this for trivially copyable types
107+
template<typename X = T, typename = typename std::enable_if<std::is_trivially_copyable<X>::value>::type>
108+
Ok(T i): inner(i) {}
109+
Ok() = default;
110+
Ok(Ok&&) noexcept = default;
111+
Ok(const Ok &) = default;
112+
Ok& operator=(const Ok&) = default;
113+
Ok& operator=(Ok&&) noexcept = default;
114+
};
115+
116+
template<class T> struct Err {
117+
T inner;
118+
Err(T&& i): inner(std::move(i)) {}
119+
// We don't want to expose an lvalue-capable constructor in general
120+
// however there is no problem doing this for trivially copyable types
121+
template<typename X = T, typename = typename std::enable_if<std::is_trivially_copyable<X>::value>::type>
122+
Err(T i): inner(i) {}
123+
Err() = default;
124+
Err(Err&&) noexcept = default;
125+
Err(const Err &) = default;
126+
Err& operator=(const Err&) = default;
127+
Err& operator=(Err&&) noexcept = default;
128+
};
129+
130+
template<class T, class E>
131+
class result {
132+
private:
133+
std::variant<Ok<T>, Err<E>> val;
134+
public:
135+
result(Ok<T>&& v): val(std::move(v)) {}
136+
result(Err<E>&& v): val(std::move(v)) {}
137+
result() = default;
138+
result(const result &) = default;
139+
result& operator=(const result&) = default;
140+
result& operator=(result&&) noexcept = default;
141+
result(result &&) noexcept = default;
142+
~result() = default;
143+
bool is_ok() const {
144+
return std::holds_alternative<Ok<T>>(this->val);
145+
};
146+
bool is_err() const {
147+
return std::holds_alternative<Err<E>>(this->val);
148+
};
149+
150+
std::optional<T> ok() && {
151+
if (!this->is_ok()) {
152+
return std::nullopt;
153+
}
154+
return std::make_optional(std::move(std::get<Ok<T>>(std::move(this->val)).inner));
155+
};
156+
std::optional<E> err() && {
157+
if (!this->is_err()) {
158+
return std::nullopt;
159+
}
160+
return std::make_optional(std::move(std::get<Err<E>>(std::move(this->val)).inner));
161+
}
162+
163+
void set_ok(T&& t) {
164+
this->val = Ok<T>(std::move(t));
165+
}
166+
167+
void set_err(E&& e) {
168+
this->val = Err<E>(std::move(e));
169+
}
170+
171+
template<typename T2>
172+
result<T2, E> replace_ok(T2&& t) {
173+
if (this->is_err()) {
174+
return result<T2, E>(Err<E>(std::get<Err<E>>(std::move(this->val))));
175+
} else {
176+
return result<T2, E>(Ok<T2>(std::move(t)));
177+
}
178+
}
179+
};
180+
181+
class Utf8Error {};
182+
183+
// Use custom std::span on C++17, otherwise use std::span
184+
#if __cplusplus >= 202002L
185+
186+
template<class T> using span = std::span<T>;
187+
188+
#else // __cplusplus < 202002L
189+
190+
// C++-17-compatible std::span
191+
template<class T>
192+
class span {
193+
194+
public:
195+
constexpr span(T* data, size_t size)
196+
: data_(data), size_(size) {}
197+
template<size_t N>
198+
constexpr span(std::array<typename std::remove_const<T>::type, N>& arr)
199+
: data_(const_cast<T*>(arr.data())), size_(N) {}
200+
constexpr T* data() const noexcept {
201+
return this->data_;
202+
}
203+
constexpr size_t size() const noexcept {
204+
return this->size_;
205+
}
206+
private:
207+
T* data_;
208+
size_t size_;
209+
};
210+
211+
#endif // __cplusplus >= 202002L
212+
213+
} // namespace diplomat
214+
215+
#endif

temporal_capi/src/calendar.rs

+61
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
#[diplomat::bridge]
2+
#[diplomat::abi_rename = "temporal_rs_{0}"]
3+
#[diplomat::attr(auto, namespace = "temporal_rs")]
4+
pub mod ffi {
5+
use crate::error::ffi::TemporalError;
6+
use diplomat_runtime::DiplomatStr;
7+
8+
#[diplomat::enum_convert(icu_calendar::any_calendar::AnyCalendarKind, needs_wildcard)]
9+
pub enum AnyCalendarKind {
10+
Buddhist,
11+
Chinese,
12+
Coptic,
13+
Dangi,
14+
Ethiopian,
15+
EthiopianAmeteAlem,
16+
Gregorian,
17+
Hebrew,
18+
Indian,
19+
IslamicCivil,
20+
IslamicObservational,
21+
IslamicTabular,
22+
IslamicUmmAlQura,
23+
Iso,
24+
Japanese,
25+
JapaneseExtended,
26+
Persian,
27+
Roc,
28+
}
29+
30+
impl AnyCalendarKind {
31+
pub fn get_for_bcp47_string(s: &DiplomatStr) -> Option<Self> {
32+
icu_calendar::any_calendar::AnyCalendarKind::get_for_bcp47_bytes(s).map(Into::into)
33+
}
34+
}
35+
36+
#[diplomat::opaque]
37+
#[diplomat::transparent_convert]
38+
pub struct Calendar(pub temporal_rs::Calendar);
39+
40+
impl Calendar {
41+
pub fn create(kind: AnyCalendarKind) -> Box<Self> {
42+
Box::new(Calendar(temporal_rs::Calendar::new(kind.into())))
43+
}
44+
45+
pub fn from_utf8(s: &DiplomatStr) -> Result<Box<Self>, TemporalError> {
46+
temporal_rs::Calendar::from_utf8(s)
47+
.map(|c| Box::new(Calendar(c)))
48+
.map_err(Into::into)
49+
}
50+
51+
pub fn is_iso(&self) -> bool {
52+
self.0.is_iso()
53+
}
54+
55+
pub fn identifier(&self) -> &'static str {
56+
self.0.identifier()
57+
}
58+
59+
// TODO the rest of calendar (needs all the date/time types)
60+
}
61+
}

temporal_capi/src/error.rs

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
#[diplomat::bridge]
2+
#[diplomat::abi_rename = "temporal_rs_{0}"]
3+
#[diplomat::attr(auto, namespace = "temporal_rs")]
4+
pub mod ffi {
5+
6+
#[diplomat::enum_convert(temporal_rs::error::ErrorKind)]
7+
pub enum ErrorKind {
8+
Generic,
9+
Type,
10+
Range,
11+
Syntax,
12+
Assert,
13+
}
14+
15+
// In the future we might turn this into an opaque type with a msg() field
16+
pub struct TemporalError {
17+
pub kind: ErrorKind,
18+
}
19+
20+
impl TemporalError {
21+
// internal
22+
pub(crate) fn syntax() -> Self {
23+
TemporalError {
24+
kind: ErrorKind::Syntax,
25+
}
26+
}
27+
}
28+
}
29+
30+
impl From<temporal_rs::TemporalError> for ffi::TemporalError {
31+
fn from(other: temporal_rs::TemporalError) -> Self {
32+
Self {
33+
kind: other.kind().into(),
34+
}
35+
}
36+
}

temporal_capi/src/lib.rs

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
#![allow(unused)] // Until we add all the APIs
2+
#![allow(clippy::needless_lifetimes)] // Diplomat requires explicit lifetimes at times
3+
4+
mod calendar;
5+
mod error;
6+
mod options;
7+
8+
mod plain_date;

0 commit comments

Comments
 (0)