310 lines
8.3 KiB
C++
310 lines
8.3 KiB
C++
#pragma once
|
|
|
|
#include <solanaceae/object_store/fwd.hpp>
|
|
#include <solanaceae/object_store/object_store.hpp>
|
|
#include <solanaceae/util/time.hpp>
|
|
|
|
#include <entt/core/type_info.hpp>
|
|
|
|
#include "./frame_stream2.hpp"
|
|
|
|
#include <unordered_map>
|
|
#include <vector>
|
|
#include <memory>
|
|
#include <algorithm>
|
|
#include <thread>
|
|
#include <chrono>
|
|
#include <atomic>
|
|
|
|
// fwd
|
|
class StreamManager;
|
|
|
|
namespace Components {
|
|
|
|
// mark a source or sink as the(a) default
|
|
struct TagDefaultTarget {};
|
|
|
|
// mark a source/sink as to be connected to a default sink/source
|
|
// of the same type
|
|
struct TagConnectToDefault {};
|
|
|
|
struct StreamSource {
|
|
std::string name;
|
|
std::string frame_type_name;
|
|
|
|
std::function<bool(StreamManager&, Object, Object, bool)> connect_fn;
|
|
|
|
template<typename FrameType>
|
|
static StreamSource create(const std::string& name);
|
|
};
|
|
|
|
struct StreamSink {
|
|
std::string name;
|
|
std::string frame_type_name;
|
|
|
|
template<typename FrameType>
|
|
static StreamSink create(const std::string& name);
|
|
};
|
|
|
|
template<typename FrameType>
|
|
using FrameStream2Source = std::unique_ptr<FrameStream2SourceI<FrameType>>;
|
|
|
|
template<typename FrameType>
|
|
using FrameStream2Sink = std::unique_ptr<FrameStream2SinkI<FrameType>>;
|
|
|
|
// supported sources/sinks have this component
|
|
// to signal the backend sender/encoder a generic bitrate
|
|
// TODO: split into advanced encoder bitrate / target / range
|
|
struct Bitrate {
|
|
// in kilo bits per second
|
|
int64_t rate{0};
|
|
};
|
|
|
|
} // Components
|
|
|
|
|
|
class StreamManager : protected ObjectStoreEventI {
|
|
friend class StreamManagerUI; // TODO: make this go away
|
|
ObjectStore2& _os;
|
|
ObjectStore2::SubscriptionReference _os_sr;
|
|
|
|
struct Connection {
|
|
ObjectHandle src;
|
|
ObjectHandle sink;
|
|
|
|
struct Data {
|
|
virtual ~Data(void) {}
|
|
virtual const void* getReaderStream(void) = 0;
|
|
virtual const void* getWriterStream(void) = 0;
|
|
};
|
|
std::unique_ptr<Data> data; // stores reader writer type erased
|
|
std::function<void(Connection&)> pump_fn; // TODO: make it return next interval?
|
|
std::function<void(Connection&)> unsubscribe_fn;
|
|
|
|
bool on_main_thread {true};
|
|
std::atomic_bool stop {false}; // disconnect
|
|
std::atomic_bool finished {false}; // disconnect
|
|
|
|
// pump thread
|
|
std::thread pump_thread;
|
|
|
|
// frame interval counters and estimates
|
|
std::atomic<float> interval_avg {0.f}; // s
|
|
std::atomic<uint64_t> frames_total{0};
|
|
std::atomic<uint64_t> bytes_total{0}; // if it can be mesured
|
|
|
|
// moving avg
|
|
std::atomic<float> bytes_per_sec{0};
|
|
|
|
// temps for mesuring
|
|
uint64_t _last_ts {0}; // frame format OR ms if frame has no ts
|
|
|
|
|
|
Connection(void) = default;
|
|
Connection(
|
|
ObjectHandle src_,
|
|
ObjectHandle sink_,
|
|
std::unique_ptr<Data>&& data_,
|
|
std::function<void(Connection&)>&& pump_fn_,
|
|
std::function<void(Connection&)>&& unsubscribe_fn_,
|
|
bool on_main_thread_ = true
|
|
);
|
|
};
|
|
std::vector<std::unique_ptr<Connection>> _connections;
|
|
|
|
std::unordered_map<std::string, Object> _default_sources;
|
|
std::unordered_map<std::string, Object> _default_sinks;
|
|
|
|
public:
|
|
StreamManager(ObjectStore2& os);
|
|
virtual ~StreamManager(void);
|
|
|
|
template<typename FrameType>
|
|
bool connect(Object src, Object sink, bool threaded = true);
|
|
|
|
bool connect(Object src, Object sink, bool threaded = true);
|
|
bool disconnect(Object src, Object sink);
|
|
bool disconnectAll(Object o);
|
|
|
|
template<typename FN>
|
|
void forEachConnected(ObjectHandle o, FN&& fn) {
|
|
for (const auto& con : _connections) {
|
|
if (con->src == o) {
|
|
fn(con->sink);
|
|
} else if (con->sink == o) {
|
|
fn(con->src);
|
|
}
|
|
}
|
|
}
|
|
|
|
template<typename FN>
|
|
void forEachConnectedExt(ObjectHandle o, FN&& fn) {
|
|
for (const auto& con : _connections) {
|
|
if (con->src == o) {
|
|
// TODO: very ugly
|
|
fn(con->sink, con->data.get()->getReaderStream(), con->data.get()->getWriterStream());
|
|
} else if (con->sink == o) {
|
|
fn(con->src, con->data.get()->getReaderStream(), con->data.get()->getWriterStream());
|
|
}
|
|
}
|
|
}
|
|
|
|
// do we need the time delta?
|
|
float tick(float);
|
|
|
|
protected:
|
|
bool onEvent(const ObjectStore::Events::ObjectConstruct&) override;
|
|
bool onEvent(const ObjectStore::Events::ObjectUpdate&) override;
|
|
bool onEvent(const ObjectStore::Events::ObjectDestory&) override;
|
|
};
|
|
|
|
// template impls
|
|
|
|
namespace Components {
|
|
|
|
// we require the complete sm type here
|
|
template<typename FrameType>
|
|
StreamSource StreamSource::create(const std::string& name) {
|
|
return StreamSource{
|
|
name,
|
|
std::string{entt::type_name<FrameType>::value()},
|
|
+[](StreamManager& sm, Object src, Object sink, bool threaded) {
|
|
return sm.connect<FrameType>(src, sink, threaded);
|
|
},
|
|
};
|
|
}
|
|
|
|
template<typename FrameType>
|
|
StreamSink StreamSink::create(const std::string& name) {
|
|
return StreamSink{
|
|
name,
|
|
std::string{entt::type_name<FrameType>::value()},
|
|
};
|
|
}
|
|
|
|
} // Components
|
|
|
|
template<typename FrameType>
|
|
bool StreamManager::connect(Object src, Object sink, bool threaded) {
|
|
auto res = std::find_if(
|
|
_connections.cbegin(), _connections.cend(),
|
|
[&](const auto& a) { return a->src == src && a->sink == sink; }
|
|
);
|
|
if (res != _connections.cend()) {
|
|
// already exists
|
|
return false;
|
|
}
|
|
|
|
auto h_src = _os.objectHandle(src);
|
|
auto h_sink = _os.objectHandle(sink);
|
|
if (!static_cast<bool>(h_src) || !static_cast<bool>(h_sink)) {
|
|
// an object does not exist
|
|
return false;
|
|
}
|
|
|
|
if (!h_src.all_of<Components::FrameStream2Source<FrameType>>()) {
|
|
// src not stream source
|
|
return false;
|
|
}
|
|
|
|
if (!h_sink.all_of<Components::FrameStream2Sink<FrameType>>()) {
|
|
// sink not stream sink
|
|
return false;
|
|
}
|
|
|
|
auto& src_stream = h_src.get<Components::FrameStream2Source<FrameType>>();
|
|
auto& sink_stream = h_sink.get<Components::FrameStream2Sink<FrameType>>();
|
|
|
|
struct inlineData : public Connection::Data {
|
|
~inlineData(void) {}
|
|
std::shared_ptr<FrameStream2I<FrameType>> reader;
|
|
std::shared_ptr<FrameStream2I<FrameType>> writer;
|
|
const void* getReaderStream(void) override { return reader.get(); }
|
|
const void* getWriterStream(void) override { return writer.get(); }
|
|
};
|
|
|
|
auto our_data = std::make_unique<inlineData>();
|
|
|
|
our_data->reader = src_stream->subscribe();
|
|
if (!our_data->reader) {
|
|
return false;
|
|
}
|
|
our_data->writer = sink_stream->subscribe();
|
|
if (!our_data->writer) {
|
|
return false;
|
|
}
|
|
|
|
_connections.push_back(std::make_unique<Connection>(
|
|
h_src,
|
|
h_sink,
|
|
std::move(our_data),
|
|
[](Connection& con) -> void { // pump
|
|
// there might be more stored
|
|
for (size_t i = 0; i < 64; i++) {
|
|
auto new_frame_opt = static_cast<inlineData*>(con.data.get())->reader->pop();
|
|
// TODO: frame interval estimates
|
|
if (new_frame_opt.has_value()) {
|
|
con.frames_total++;
|
|
|
|
// TODO: opt-in ?
|
|
float delta{0.f}; // s
|
|
uint64_t ts{0};
|
|
if constexpr (frameHasTimestamp<FrameType>()) {
|
|
ts = frameGetTimestamp(new_frame_opt.value());
|
|
} else {
|
|
ts = getTimeMS(); // fallback
|
|
}
|
|
|
|
if (con._last_ts != 0 && ts > con._last_ts) {
|
|
// normalize to seconds
|
|
if constexpr (frameHasTimestamp<FrameType>()) {
|
|
delta = float(ts - con._last_ts) / frameGetTimestampDivision<FrameType>();
|
|
} else {
|
|
delta = float(ts - con._last_ts) / 1000.f; // fallback
|
|
}
|
|
|
|
if (con.interval_avg == 0.f) {
|
|
con.interval_avg = delta;
|
|
} else {
|
|
con.interval_avg = con.interval_avg*0.95f + delta*0.05f;
|
|
}
|
|
}
|
|
con._last_ts = ts;
|
|
|
|
if constexpr (frameHasBytes<FrameType>()) {
|
|
// we need to always run this, timing stuff below might not
|
|
const auto bytes = frameGetBytes(new_frame_opt.value());
|
|
con.bytes_total += bytes;
|
|
|
|
if (delta > 0.f) {
|
|
if (con.bytes_per_sec == 0.f) {
|
|
con.bytes_per_sec = bytes/delta;
|
|
} else {
|
|
con.bytes_per_sec = con.bytes_per_sec*0.95f + (bytes/delta)*0.05f;
|
|
}
|
|
}
|
|
}
|
|
|
|
static_cast<inlineData*>(con.data.get())->writer->push(new_frame_opt.value());
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
},
|
|
[](Connection& con) -> void { // disco
|
|
auto* src_stream_ptr = con.src.try_get<Components::FrameStream2Source<FrameType>>();
|
|
if (src_stream_ptr != nullptr) {
|
|
(*src_stream_ptr)->unsubscribe(static_cast<inlineData*>(con.data.get())->reader);
|
|
}
|
|
auto* sink_stream_ptr = con.sink.try_get<Components::FrameStream2Sink<FrameType>>();
|
|
if (sink_stream_ptr != nullptr) {
|
|
(*sink_stream_ptr)->unsubscribe(static_cast<inlineData*>(con.data.get())->writer);
|
|
}
|
|
},
|
|
!threaded
|
|
));
|
|
|
|
return true;
|
|
}
|
|
|