CPP_SRCS = $(shell find src -name \*.cpp)
CPP_DEPS = $(shell find src -name \*.h)
-LIBS = cairo libavformat libavcodec libavutil libswresample libswscale
+LIBS = cairo freetype2 libavformat libavcodec libavutil libswresample libswscale
main: $(CPP_SRCS) $(CPP_DEPS)
clang++ -g $(shell pkg-config --cflags --libs $(LIBS)) $(CPP_SRCS) -o $@
--- /dev/null
+#ifndef TEST_APP_RENDERER_H_
+#define TEST_APP_RENDERER_H_
+
+#include <cstdint>
+extern "C" {
+#include "cairo.h"
+}
+
+#include "../freetype/Face.h"
+#include "../freetype/Library.h"
+#include "../cairo/Context.h"
+#include "../cairo/Face.h"
+#include "../cairo/Surface.h"
+
+namespace app {
+
+class Renderer {
+
+public:
+ Renderer(uint8_t *plane, int linesize, int width, int height)
+ : ft()
+ , face(ft.NewFace("/usr/share/fonts/truetype/dejavu/DejaVuSans.ttf", 0))
+ , font(face)
+ , surface(plane, linesize, CAIRO_FORMAT_ARGB32, width, height)
+ , ctx(surface.CreateContext())
+ , width(width)
+ , height(height) {
+ }
+ ~Renderer() {
+ }
+
+ Renderer(const Renderer &) = delete;
+ Renderer &operator =(const Renderer &) = delete;
+
+public:
+ void RenderVideoFrame(int64_t num, int64_t time_ms) {
+ ctx.SetSourceRGB(0, 0, 0);
+ ctx.Paint();
+
+ ctx.MoveTo(50, 50);
+ ctx.SetFontFace(font);
+ ctx.SetFontSize(16);
+ ctx.SetSourceRGB(1, 1, 1);
+ ctx.ShowText("Hello");
+
+ surface.Flush();
+ }
+
+
+private:
+ freetype::Library ft;
+ freetype::Face face;
+ cairo::Face font;
+ cairo::Surface surface;
+ cairo::Context ctx;
+
+ int width;
+ int height;
+
+};
+
+}
+
+#endif
}
#include "../ffmpeg/Encoder.h"
+#include "../ffmpeg/Network.h"
#include "../ffmpeg/OutputContext.h"
#include "../ffmpeg/Resampler.h"
#include "../ffmpeg/Scaler.h"
}
private:
+ ffmpeg::Network net;
ffmpeg::OutputContext output;
ffmpeg::Encoder audio_encoder;
ffmpeg::Encoder video_encoder;
#include <ostream>
#include "Error.h"
+#include "Face.h"
namespace cairo {
cairo_select_font_face(ctx, family, slant, weight);
}
+ void SetFontFace(Face &face) {
+ cairo_set_font_face(ctx, face.GetFace());
+ }
+
void SetFontSize(double size) {
cairo_set_font_size(ctx, size);
}
#include <cairo.h>
namespace {
-std::string make_message(const std::string &msg, cairo_status_t status) {
+std::string cairo_make_message(const std::string &msg, cairo_status_t status) {
return msg + ": " + std::string(cairo_status_to_string(status));
}
}
class Error: public std::runtime_error {
public:
- Error(const std::string &msg, cairo_status_t status): std::runtime_error(make_message(msg, status)) {
+ Error(const std::string &msg, cairo_status_t status): std::runtime_error(cairo_make_message(msg, status)) {
}
~Error() {
}
--- /dev/null
+#ifndef TEST_CAIRO_FACE_H_
+#define TEST_CAIRO_FACE_H_
+
+extern "C" {
+#include <cairo.h>
+#include <cairo-ft.h>
+}
+
+#include "../freetype/Face.h"
+
+namespace cairo {
+
+class Face {
+
+public:
+ explicit Face(const freetype::Face &face)
+ : font(cairo_ft_font_face_create_for_ft_face(face.GetFace(), 0))
+ , face(face) {
+ }
+ ~Face() {
+ cairo_font_face_destroy(font);
+ }
+
+ Face(const Face &) = delete;
+ Face &operator =(const Face &) = delete;
+
+public:
+ cairo_font_face_t *GetFace() {
+ return font;
+ }
+
+ const cairo_font_face_t *GetFace() const {
+ return font;
+ }
+
+private:
+ cairo_font_face_t *font;
+ freetype::Face face;
+
+};
+
+}
+
+#endif
}
namespace {
-std::string make_message(const std::string &msg, int code) {
+std::string ffmpeg_make_message(const std::string &msg, int code) {
char buf[128] = {0};
av_strerror(code, buf, sizeof(buf));
return msg + ": " + std::string(buf);
class Error: public std::runtime_error {
public:
- Error(const std::string &msg, int code): std::runtime_error(make_message(msg, code)) {
+ Error(const std::string &msg, int code): std::runtime_error(ffmpeg_make_message(msg, code)) {
}
~Error() {
}
--- /dev/null
+#ifndef TEST_FFMPEG_NETWORK_H_
+#define TEST_FFMPEG_NETWORK_H_
+
+extern "C" {
+#include <libavformat/avformat.h>
+}
+
+#include "Error.h"
+
+namespace ffmpeg {
+
+class Network {
+
+public:
+ Network() {
+ int res = avformat_network_init();
+ if (res != 0) {
+ throw Error("failed to initialize network", res);
+ }
+ }
+ ~Network() {
+ avformat_network_deinit();
+ }
+
+ Network(const Network &) = delete;
+ Network &operator =(const Network &) = delete;
+
+};
+
+}
+
+#endif
--- /dev/null
+#ifndef TEST_FREETYPE_ERROR_H_
+#define TEST_FREETYPE_ERROR_H_
+
+#include <stdexcept>
+#include <string>
+
+extern "C" {
+#include <ft2build.h>
+#include FT_FREETYPE_H
+}
+
+
+namespace {
+std::string freetype_make_message(const std::string &msg, FT_Error status) {
+ return msg + ": " + std::string(FT_Error_String(status));
+}
+}
+
+namespace freetype {
+
+class Error: public std::runtime_error {
+
+public:
+ Error(const std::string &msg, FT_Error status): std::runtime_error(freetype_make_message(msg, status)) {
+ }
+ ~Error() {
+ }
+
+};
+
+}
+
+#endif
--- /dev/null
+#ifndef TEST_FREETYPE_FACE_H_
+#define TEST_FREETYPE_FACE_H_
+
+extern "C" {
+#include <ft2build.h>
+#include FT_FREETYPE_H
+}
+
+#include "Error.h"
+
+namespace freetype {
+
+class Face {
+
+public:
+ Face(FT_Library lib, const char *path, FT_Long index) {
+ FT_Error err = FT_New_Face(lib, path, index, &face);
+ if (err) {
+ throw Error("failed to open face", err);
+ }
+ }
+ ~Face() {
+ FT_Done_Face(face);
+ }
+ Face(const Face &other): face(other.face) {
+ FT_Reference_Face(face);
+ }
+ Face &operator =(const Face &other) {
+ FT_Reference_Face(other.face);
+ FT_Done_Face(face);
+ face = other.face;
+ return *this;
+ }
+
+public:
+ const FT_Face &GetFace() const {
+ return face;
+ }
+
+private:
+ FT_Face face;
+
+};
+
+}
+
+#endif
--- /dev/null
+#ifndef TEST_FREETYPE_LIBRARY_H_
+#define TEST_FREETYPE_LIBRARY_H_
+
+extern "C" {
+#include <ft2build.h>
+#include FT_FREETYPE_H
+}
+
+#include "Error.h"
+#include "Face.h"
+
+namespace freetype {
+
+class Library {
+
+public:
+ Library() {
+ FT_Error err = FT_Init_FreeType(&lib);
+ if (err) {
+ throw Error("failed to initialize library", err);
+ }
+ }
+ ~Library() {
+ FT_Done_FreeType(lib);
+ }
+
+ Library(const Library &) = delete;
+ Library &operator =(const Library &) = delete;
+
+public:
+ Face NewFace(const char *path, FT_Long index) {
+ return Face(lib, path, index);
+ }
+
+private:
+ FT_Library lib;
+
+};
+
+}
+
+#endif
-#include "cairo.h"
-#include "cairo/Context.h"
#include <chrono>
#include <csignal>
#include <cstdint>
#include <iostream>
-#include <string>
#include <thread>
extern "C" {
#include <libavutil/timestamp.h>
}
+#include "app/Renderer.h"
#include "app/Stream.h"
-#include "cairo/Surface.h"
-
-#include "ffmpeg/Error.h"
-
namespace {
running = false;
}
-void RenderFrame(int64_t num, int64_t time_ms, cairo::Context &c) {
- c.SetSourceRGB(0, 0, 0);
- c.Paint();
-}
-
}
const int FPS = 60;
const char *URL = "rtmp://localhost/localhorsttv";
- int res = avformat_network_init();
- if (res != 0) {
- throw ffmpeg::Error("network init failed", res);
- }
-
app::Stream stream(URL, WIDTH, HEIGHT, FPS);
running = true;
int16_t *audio_plane = stream.GetAudioPlane();
const int audio_channels = stream.GetAudioChannels();
- cairo::Surface surface(plane, linesize, CAIRO_FORMAT_ARGB32, WIDTH, HEIGHT);
- cairo::Context context(surface.CreateContext());
+ app::Renderer renderer(plane, linesize, WIDTH, HEIGHT);
stream.Start();
if (stream.GetVideoFrameCounter() > 0 && difference < 0) {
std::cout << (difference / 1000.0) << "s behind schedule, dropping frame" << std::endl;
} else {
- RenderFrame(stream.GetVideoFrameCounter(), target, context);
- surface.Flush();
+ renderer.RenderVideoFrame(stream.GetVideoFrameCounter(), target);
}
stream.PushVideoFrame();