blob: 759aaebb007eecc5487af52038e01f324cfff999 [file] [log] [blame]
#pragma once
#include <cassert>
#include <ast_video_types.hpp>
#include <iostream>
#include <mutex>
#include <vector>
#include <boost/asio.hpp>
namespace AstVideo {
//
// Cursor struct is used in User Mode
//
struct AST_CUR_ATTRIBUTION_TAG {
unsigned int posX;
unsigned int posY;
unsigned int cur_width;
unsigned int cur_height;
unsigned int cur_type; // 0:mono 1:color 2:disappear cursor
unsigned int cur_change_flag;
};
//
// For storing Cursor Information
//
struct AST_CURSOR_TAG {
AST_CUR_ATTRIBUTION_TAG attr;
// unsigned char icon[MAX_CUR_OFFSETX*MAX_CUR_OFFSETY*2];
unsigned char *icon; //[64*64*2];
};
//
// For select image format, i.e. 422 JPG420, 444 JPG444, lumin/chrom table, 0
// ~ 11, low to high
//
struct FEATURES_TAG {
short jpg_fmt; // 422:JPG420, 444:JPG444
short lumin_tbl;
short chrom_tbl;
short tolerance_noise;
int w;
int h;
unsigned char *buf;
};
//
// For configure video engine control registers
//
struct IMAGE_INFO {
short do_image_refresh; // Action 0:motion 1:fullframe 2:quick cursor
char qc_valid; // quick cursor enable/disable
unsigned int len;
int crypttype;
char cryptkey[16];
union {
FEATURES_TAG features;
AST_CURSOR_TAG cursor_info;
} parameter;
};
class SimpleVideoPuller {
public:
SimpleVideoPuller() : image_info(){};
void initialize() {
std::cout << "Opening /dev/video\n";
video_fd = open("/dev/video", O_RDWR);
if (video_fd == 0) {
std::cout << "Failed to open /dev/video\n";
throw std::runtime_error("Failed to open /dev/video");
}
std::cout << "Opened successfully\n";
}
RawVideoBuffer read_video() {
assert(video_fd != 0);
RawVideoBuffer raw;
image_info.do_image_refresh = 1; // full frame refresh
image_info.qc_valid = 0; // quick cursor disabled
image_info.parameter.features.buf =
reinterpret_cast<unsigned char *>(raw.buffer.data());
image_info.crypttype = -1;
std::cout << "Writing\n";
int status;
/*
status = write(video_fd, reinterpret_cast<char*>(&image_info),
sizeof(image_info));
if (status != sizeof(image_info)) {
std::cout << "Write failed. Return: " << status << "\n";
perror("perror output:");
}
std::cout << "Write done\n";
*/
std::cout << "Reading\n";
status = read(video_fd, reinterpret_cast<char *>(&image_info),
sizeof(image_info));
std::cout << "Done reading\n";
if (status != 0) {
std::cerr << "Read failed with status " << status << "\n";
}
raw.buffer.resize(image_info.len);
raw.height = image_info.parameter.features.h;
raw.width = image_info.parameter.features.w;
if (image_info.parameter.features.jpg_fmt == 422) {
raw.mode = YuvMode::YUV420;
} else {
raw.mode = YuvMode::YUV444;
}
return raw;
}
private:
int video_fd{};
IMAGE_INFO image_info;
};
#if defined(BOOST_ASIO_HAS_POSIX_STREAM_DESCRIPTOR)
class AsyncVideoPuller {
public:
using video_callback = std::function<void (RawVideoBuffer &)>;
explicit AsyncVideoPuller(boost::asio::io_service &io_service)
: image_info(), dev_video(io_service, open("/dev/video", O_RDWR)) {
videobuf = std::make_shared<RawVideoBuffer>();
image_info.do_image_refresh = 1; // full frame refresh
image_info.qc_valid = 0; // quick cursor disabled
image_info.parameter.features.buf =
reinterpret_cast<unsigned char *>(videobuf->buffer.data());
image_info.crypttype = -1;
};
void register_callback(video_callback &callback) {
std::lock_guard<std::mutex> lock(callback_mutex);
callbacks.push_back(callback);
start_read();
}
void start_read() {
auto mutable_buffer = boost::asio::buffer(&image_info, sizeof(image_info));
boost::asio::async_read(
dev_video, mutable_buffer, [this](const boost::system::error_code &ec,
std::size_t bytes_transferred) {
if (ec != nullptr) {
std::cerr << "Read failed with status " << ec << "\n";
} else {
this->read_done();
}
});
}
void read_done() {
std::cout << "Done reading\n";
videobuf->buffer.resize(image_info.len);
videobuf->height = image_info.parameter.features.h;
videobuf->width = image_info.parameter.features.w;
if (image_info.parameter.features.jpg_fmt == 422) {
videobuf->mode = YuvMode::YUV420;
} else {
videobuf->mode = YuvMode::YUV444;
}
std::lock_guard<std::mutex> lock(callback_mutex);
for (auto &callback : callbacks) {
// TODO(ed) call callbacks async and double buffer frames
callback(*videobuf);
}
}
private:
std::shared_ptr<RawVideoBuffer> videobuf;
boost::asio::posix::stream_descriptor dev_video;
IMAGE_INFO image_info;
std::mutex callback_mutex;
std::vector<video_callback> callbacks;
};
#endif // defined(BOOST_ASIO_HAS_POSIX_STREAM_DESCRIPTOR)
} // namespace AstVideo