You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
82 lines
3.0 KiB
82 lines
3.0 KiB
#include <cassert>
|
|
#include "system/loggerd/encoder/encoder.h"
|
|
|
|
VideoEncoder::~VideoEncoder() {}
|
|
|
|
void VideoEncoder::publisher_init() {
|
|
// publish
|
|
service_name = this->type == DriverCam ? "driverEncodeData" :
|
|
(this->type == WideRoadCam ? "wideRoadEncodeData" :
|
|
(this->in_width == this->out_width ? "roadEncodeData" : "qRoadEncodeData"));
|
|
pm.reset(new PubMaster({service_name}));
|
|
}
|
|
|
|
void VideoEncoder::publisher_publish(VideoEncoder *e, int segment_num, uint32_t idx, VisionIpcBufExtra &extra,
|
|
unsigned int flags, kj::ArrayPtr<capnp::byte> header, kj::ArrayPtr<capnp::byte> dat) {
|
|
// broadcast packet
|
|
MessageBuilder msg;
|
|
auto event = msg.initEvent(true);
|
|
auto edat = (e->type == DriverCam) ? event.initDriverEncodeData() :
|
|
((e->type == WideRoadCam) ? event.initWideRoadEncodeData() :
|
|
(e->in_width == e->out_width ? event.initRoadEncodeData() : event.initQRoadEncodeData()));
|
|
auto edata = edat.initIdx();
|
|
struct timespec ts;
|
|
timespec_get(&ts, TIME_UTC);
|
|
edat.setUnixTimestampNanos((uint64_t)ts.tv_sec*1000000000 + ts.tv_nsec);
|
|
edata.setFrameId(extra.frame_id);
|
|
edata.setTimestampSof(extra.timestamp_sof);
|
|
edata.setTimestampEof(extra.timestamp_eof);
|
|
edata.setType(e->codec);
|
|
edata.setEncodeId(e->cnt++);
|
|
edata.setSegmentNum(segment_num);
|
|
edata.setSegmentId(idx);
|
|
edata.setFlags(flags);
|
|
edata.setLen(dat.size());
|
|
edat.setData(dat);
|
|
if (flags & V4L2_BUF_FLAG_KEYFRAME) edat.setHeader(header);
|
|
|
|
auto words = new kj::Array<capnp::word>(capnp::messageToFlatArray(msg));
|
|
auto bytes = words->asBytes();
|
|
e->pm->send(e->service_name, bytes.begin(), bytes.size());
|
|
if (e->write) {
|
|
e->to_write.push(words);
|
|
} else {
|
|
delete words;
|
|
}
|
|
}
|
|
|
|
// TODO: writing should be moved to loggerd
|
|
void VideoEncoder::write_handler(VideoEncoder *e, const char *path) {
|
|
VideoWriter writer(path, e->filename, e->codec != cereal::EncodeIndex::Type::FULL_H_E_V_C, e->out_width, e->out_height, e->fps, e->codec);
|
|
|
|
bool first = true;
|
|
kj::Array<capnp::word>* out_buf;
|
|
while ((out_buf = e->to_write.pop())) {
|
|
capnp::FlatArrayMessageReader cmsg(*out_buf);
|
|
cereal::Event::Reader event = cmsg.getRoot<cereal::Event>();
|
|
|
|
auto edata = (e->type == DriverCam) ? event.getDriverEncodeData() :
|
|
((e->type == WideRoadCam) ? event.getWideRoadEncodeData() :
|
|
(e->in_width == e->out_width ? event.getRoadEncodeData() : event.getQRoadEncodeData()));
|
|
auto idx = edata.getIdx();
|
|
auto flags = idx.getFlags();
|
|
|
|
if (first) {
|
|
assert(flags & V4L2_BUF_FLAG_KEYFRAME);
|
|
auto header = edata.getHeader();
|
|
writer.write((uint8_t *)header.begin(), header.size(), idx.getTimestampEof()/1000, true, false);
|
|
first = false;
|
|
}
|
|
|
|
// dangerous cast from const, but should be fine
|
|
auto data = edata.getData();
|
|
if (data.size() > 0) {
|
|
writer.write((uint8_t *)data.begin(), data.size(), idx.getTimestampEof()/1000, false, flags & V4L2_BUF_FLAG_KEYFRAME);
|
|
}
|
|
|
|
// free the data
|
|
delete out_buf;
|
|
}
|
|
|
|
// VideoWriter is freed on out of scope
|
|
}
|
|
|