Files
idevice/cpp/src/tcp_callback_feeder.cpp
2025-08-29 14:19:28 -06:00

119 lines
3.3 KiB
C++

// Jackson Coxson
#include <idevice++/bindings.hpp>
#include <idevice++/tcp_object_stack.hpp>
namespace IdeviceFFI {
// ---------- TcpFeeder ----------
bool TcpObjectStackFeeder::write(const uint8_t* data, std::size_t len, FfiError& err) const {
if (IdeviceFfiError* e = ::idevice_tcp_feed_object_write(h_, data, len)) {
err = FfiError(e);
return false;
}
return true;
}
// ---------- TcpEater ----------
bool TcpObjectStackEater::read(OwnedBuffer& out, FfiError& err) const {
uint8_t* ptr = nullptr;
std::size_t len = 0;
if (IdeviceFfiError* e = ::idevice_tcp_eat_object_read(h_, &ptr, &len)) {
err = FfiError(e);
return false;
}
// Success: adopt the buffer (freed via idevice_data_free in OwnedBuffer dtor)
out.adopt(ptr, len);
return true;
}
// ---------- TcpStackFromCallback ----------
Result<TcpObjectStack, FfiError> TcpObjectStack::create(const std::string& our_ip,
const std::string& their_ip) {
::TcpFeedObject* feeder_h = nullptr;
::TcpEatObject* eater_h = nullptr;
::AdapterHandle* adapter_h = nullptr;
FfiError e(::idevice_tcp_stack_into_sync_objects(
our_ip.c_str(), their_ip.c_str(), &feeder_h, &eater_h, &adapter_h));
if (e) {
return Err(e);
}
auto impl = std::make_unique<Impl>();
impl->feeder = TcpObjectStackFeeder(feeder_h);
impl->eater = TcpObjectStackEater(eater_h);
impl->adapter = Adapter::adopt(adapter_h);
TcpObjectStack out;
out.impl_ = std::move(impl);
return Ok(std::move(out));
}
TcpObjectStackFeeder& TcpObjectStack::feeder() {
return impl_->feeder;
}
const TcpObjectStackFeeder& TcpObjectStack::feeder() const {
return impl_->feeder;
}
TcpObjectStackEater& TcpObjectStack::eater() {
return impl_->eater;
}
const TcpObjectStackEater& TcpObjectStack::eater() const {
return impl_->eater;
}
Adapter& TcpObjectStack::adapter() {
if (!impl_ || impl_->adapter.is_some()) {
static Adapter* never = nullptr;
return *never;
}
return (impl_->adapter.unwrap());
}
const Adapter& TcpObjectStack::adapter() const {
if (!impl_ || impl_->adapter.is_none()) {
static Adapter* never = nullptr;
return *never;
}
return (impl_->adapter.unwrap());
}
// ---------- Release APIs ----------
Option<TcpObjectStackFeeder> TcpObjectStack::release_feeder() {
if (!impl_) {
return None;
}
auto has = impl_->feeder.raw() != nullptr;
if (!has) {
return None;
}
TcpObjectStackFeeder out = std::move(impl_->feeder);
// impl_->feeder is now empty (h_ == nullptr) thanks to move
return Some(std::move(out));
}
Option<TcpObjectStackEater> TcpObjectStack::release_eater() {
if (!impl_) {
return None;
}
auto has = impl_->eater.raw() != nullptr;
if (!has) {
return None;
}
TcpObjectStackEater out = std::move(impl_->eater);
return Some(std::move(out));
}
Option<Adapter> TcpObjectStack::release_adapter() {
if (!impl_ || impl_->adapter.is_none()) {
return None;
}
// Move out and clear our optional
auto out = std::move((impl_->adapter.unwrap()));
impl_->adapter.reset();
return Some(std::move(out));
}
} // namespace IdeviceFFI