mirror of
https://github.com/m1k1o/neko.git
synced 2024-07-24 14:40:50 +12:00
gst refactor to use ctx.
This commit is contained in:
parent
7c3a24a613
commit
b932e94f77
@ -113,7 +113,7 @@ func (manager *BroacastManagerCtx) destroyPipeline() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
manager.pipeline.Stop()
|
manager.pipeline.Destroy()
|
||||||
manager.logger.Info().Msgf("destroying pipeline")
|
manager.logger.Info().Msgf("destroying pipeline")
|
||||||
manager.pipeline = nil
|
manager.pipeline = nil
|
||||||
}
|
}
|
||||||
|
@ -1,9 +1,5 @@
|
|||||||
#include "gst.h"
|
#include "gst.h"
|
||||||
|
|
||||||
typedef struct SampleHandlerUserData {
|
|
||||||
int pipelineId;
|
|
||||||
} SampleHandlerUserData;
|
|
||||||
|
|
||||||
void gstreamer_init(void) {
|
void gstreamer_init(void) {
|
||||||
gst_init(NULL, NULL);
|
gst_init(NULL, NULL);
|
||||||
}
|
}
|
||||||
@ -14,27 +10,29 @@ void gstreamer_loop(void) {
|
|||||||
g_main_loop_run(gstreamer_main_loop);
|
g_main_loop_run(gstreamer_main_loop);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void gstreamer_pipeline_log(char* level, const char* format, ...) {
|
static void gstreamer_pipeline_log(GstPipelineCtx *ctx, char* level, const char* format, ...) {
|
||||||
va_list argptr;
|
va_list argptr;
|
||||||
va_start(argptr, format);
|
va_start(argptr, format);
|
||||||
char buffer[100];
|
char buffer[100];
|
||||||
vsprintf(buffer, format, argptr);
|
vsprintf(buffer, format, argptr);
|
||||||
va_end(argptr);
|
va_end(argptr);
|
||||||
goPipelineLog(level, buffer);
|
goPipelineLog(level, buffer, ctx->pipelineId);
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean gstreamer_bus_call(GstBus *bus, GstMessage *msg, gpointer data) {
|
static gboolean gstreamer_bus_call(GstBus *bus, GstMessage *msg, gpointer user_data) {
|
||||||
|
GstPipelineCtx *ctx = (GstPipelineCtx *)user_data;
|
||||||
|
|
||||||
switch (GST_MESSAGE_TYPE(msg)) {
|
switch (GST_MESSAGE_TYPE(msg)) {
|
||||||
case GST_MESSAGE_EOS: {
|
case GST_MESSAGE_EOS: {
|
||||||
gstreamer_pipeline_log("panic", "end of stream");
|
gstreamer_pipeline_log(ctx, "fatal", "end of stream");
|
||||||
exit(1);
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case GST_MESSAGE_STATE_CHANGED: {
|
case GST_MESSAGE_STATE_CHANGED: {
|
||||||
GstState old_state, new_state;
|
GstState old_state, new_state;
|
||||||
gst_message_parse_state_changed(msg, &old_state, &new_state, NULL);
|
gst_message_parse_state_changed(msg, &old_state, &new_state, NULL);
|
||||||
|
|
||||||
gstreamer_pipeline_log("debug",
|
gstreamer_pipeline_log(ctx, "debug",
|
||||||
"element %s changed state from %s to %s",
|
"element %s changed state from %s to %s",
|
||||||
GST_OBJECT_NAME(msg->src),
|
GST_OBJECT_NAME(msg->src),
|
||||||
gst_element_state_get_name(old_state),
|
gst_element_state_get_name(old_state),
|
||||||
@ -46,7 +44,7 @@ static gboolean gstreamer_bus_call(GstBus *bus, GstMessage *msg, gpointer data)
|
|||||||
GstTagList *tags = NULL;
|
GstTagList *tags = NULL;
|
||||||
gst_message_parse_tag(msg, &tags);
|
gst_message_parse_tag(msg, &tags);
|
||||||
|
|
||||||
gstreamer_pipeline_log("debug",
|
gstreamer_pipeline_log(ctx, "debug",
|
||||||
"got tags from element %s",
|
"got tags from element %s",
|
||||||
GST_OBJECT_NAME(msg->src));
|
GST_OBJECT_NAME(msg->src));
|
||||||
|
|
||||||
@ -59,10 +57,10 @@ static gboolean gstreamer_bus_call(GstBus *bus, GstMessage *msg, gpointer data)
|
|||||||
gchar *dbg_info = NULL;
|
gchar *dbg_info = NULL;
|
||||||
gst_message_parse_error(msg, &err, &dbg_info);
|
gst_message_parse_error(msg, &err, &dbg_info);
|
||||||
|
|
||||||
gstreamer_pipeline_log("error",
|
gstreamer_pipeline_log(ctx, "error",
|
||||||
"error from element %s: %s",
|
"error from element %s: %s",
|
||||||
GST_OBJECT_NAME(msg->src), err->message);
|
GST_OBJECT_NAME(msg->src), err->message);
|
||||||
gstreamer_pipeline_log("warn",
|
gstreamer_pipeline_log(ctx, "warn",
|
||||||
"debugging info: %s",
|
"debugging info: %s",
|
||||||
(dbg_info) ? dbg_info : "none");
|
(dbg_info) ? dbg_info : "none");
|
||||||
|
|
||||||
@ -72,26 +70,42 @@ static gboolean gstreamer_bus_call(GstBus *bus, GstMessage *msg, gpointer data)
|
|||||||
}
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
gstreamer_pipeline_log("trace", "unknown message");
|
gstreamer_pipeline_log(ctx, "trace", "unknown message");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
GstPipelineCtx *gstreamer_pipeline_create(char *pipelineStr, int pipelineId, GError **error) {
|
||||||
|
GstElement *pipeline = gst_parse_launch(pipelineStr, error);
|
||||||
|
if (pipeline == NULL) return NULL;
|
||||||
|
|
||||||
|
// create gstreamer pipeline context
|
||||||
|
GstPipelineCtx *ctx = calloc(1, sizeof(GstPipelineCtx));
|
||||||
|
ctx->pipelineId = pipelineId;
|
||||||
|
ctx->pipeline = pipeline;
|
||||||
|
|
||||||
|
GstBus *bus = gst_pipeline_get_bus(GST_PIPELINE(pipeline));
|
||||||
|
gst_bus_add_watch(bus, gstreamer_bus_call, ctx);
|
||||||
|
gst_object_unref(bus);
|
||||||
|
|
||||||
|
return ctx;
|
||||||
|
}
|
||||||
|
|
||||||
static GstFlowReturn gstreamer_send_new_sample_handler(GstElement *object, gpointer user_data) {
|
static GstFlowReturn gstreamer_send_new_sample_handler(GstElement *object, gpointer user_data) {
|
||||||
|
GstPipelineCtx *ctx = (GstPipelineCtx *)user_data;
|
||||||
GstSample *sample = NULL;
|
GstSample *sample = NULL;
|
||||||
GstBuffer *buffer = NULL;
|
GstBuffer *buffer = NULL;
|
||||||
gpointer copy = NULL;
|
gpointer copy = NULL;
|
||||||
gsize copy_size = 0;
|
gsize copy_size = 0;
|
||||||
SampleHandlerUserData *s = (SampleHandlerUserData *)user_data;
|
|
||||||
|
|
||||||
g_signal_emit_by_name(object, "pull-sample", &sample);
|
g_signal_emit_by_name(object, "pull-sample", &sample);
|
||||||
if (sample) {
|
if (sample) {
|
||||||
buffer = gst_sample_get_buffer(sample);
|
buffer = gst_sample_get_buffer(sample);
|
||||||
if (buffer) {
|
if (buffer) {
|
||||||
gst_buffer_extract_dup(buffer, 0, gst_buffer_get_size(buffer), ©, ©_size);
|
gst_buffer_extract_dup(buffer, 0, gst_buffer_get_size(buffer), ©, ©_size);
|
||||||
goHandlePipelineBuffer(copy, copy_size, GST_BUFFER_DURATION(buffer), s->pipelineId);
|
goHandlePipelineBuffer(copy, copy_size, GST_BUFFER_DURATION(buffer), ctx->pipelineId);
|
||||||
}
|
}
|
||||||
gst_sample_unref(sample);
|
gst_sample_unref(sample);
|
||||||
}
|
}
|
||||||
@ -99,39 +113,29 @@ static GstFlowReturn gstreamer_send_new_sample_handler(GstElement *object, gpoin
|
|||||||
return GST_FLOW_OK;
|
return GST_FLOW_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
void gstreamer_pipeline_attach_appsink(GstElement *pipeline, char *sinkName, int pipelineId) {
|
void gstreamer_pipeline_attach_appsink(GstPipelineCtx *ctx, char *sinkName) {
|
||||||
SampleHandlerUserData *s = calloc(1, sizeof(SampleHandlerUserData));
|
GstElement *appsink = gst_bin_get_by_name(GST_BIN(ctx->pipeline), sinkName);
|
||||||
s->pipelineId = pipelineId;
|
|
||||||
|
|
||||||
GstElement *appsink = gst_bin_get_by_name(GST_BIN(pipeline), sinkName);
|
|
||||||
g_object_set(appsink, "emit-signals", TRUE, NULL);
|
g_object_set(appsink, "emit-signals", TRUE, NULL);
|
||||||
g_signal_connect(appsink, "new-sample", G_CALLBACK(gstreamer_send_new_sample_handler), s);
|
g_signal_connect(appsink, "new-sample", G_CALLBACK(gstreamer_send_new_sample_handler), ctx);
|
||||||
gst_object_unref(appsink);
|
gst_object_unref(appsink);
|
||||||
}
|
}
|
||||||
|
|
||||||
GstElement *gstreamer_pipeline_create(char *pipelineStr, GError **error) {
|
void gstreamer_pipeline_play(GstPipelineCtx *ctx) {
|
||||||
GstElement *pipeline = gst_parse_launch(pipelineStr, error);
|
gst_element_set_state(GST_ELEMENT(ctx->pipeline), GST_STATE_PLAYING);
|
||||||
|
|
||||||
if (pipeline != NULL) {
|
|
||||||
GstBus *bus = gst_pipeline_get_bus(GST_PIPELINE(pipeline));
|
|
||||||
gst_bus_add_watch(bus, gstreamer_bus_call, NULL);
|
|
||||||
gst_object_unref(bus);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return pipeline;
|
void gstreamer_pipeline_pause(GstPipelineCtx *ctx) {
|
||||||
|
gst_element_set_state(GST_ELEMENT(ctx->pipeline), GST_STATE_PAUSED);
|
||||||
}
|
}
|
||||||
|
|
||||||
void gstreamer_pipeline_play(GstElement *pipeline) {
|
void gstreamer_pipeline_destory(GstPipelineCtx *ctx) {
|
||||||
gst_element_set_state(pipeline, GST_STATE_PLAYING);
|
gst_element_set_state(GST_ELEMENT(ctx->pipeline), GST_STATE_NULL);
|
||||||
|
gst_object_unref(ctx->pipeline);
|
||||||
|
free(ctx);
|
||||||
}
|
}
|
||||||
|
|
||||||
void gstreamer_pipeline_stop(GstElement *pipeline) {
|
void gstreamer_pipeline_push(GstPipelineCtx *ctx, char *srcName, void *buffer, int bufferLen) {
|
||||||
gst_element_set_state(pipeline, GST_STATE_NULL);
|
GstElement *src = gst_bin_get_by_name(GST_BIN(ctx->pipeline), srcName);
|
||||||
gst_object_unref(pipeline);
|
|
||||||
}
|
|
||||||
|
|
||||||
void gstreamer_pipeline_push(GstElement *pipeline, char *srcName, void *buffer, int bufferLen) {
|
|
||||||
GstElement *src = gst_bin_get_by_name(GST_BIN(pipeline), srcName);
|
|
||||||
|
|
||||||
if (src != NULL) {
|
if (src != NULL) {
|
||||||
gpointer p = g_memdup(buffer, bufferLen);
|
gpointer p = g_memdup(buffer, bufferLen);
|
||||||
|
@ -19,10 +19,10 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type Pipeline struct {
|
type Pipeline struct {
|
||||||
Pipeline *C.GstElement
|
|
||||||
Sample chan types.Sample
|
|
||||||
Src string
|
|
||||||
id int
|
id int
|
||||||
|
Src string
|
||||||
|
Ctx *C.GstPipelineCtx
|
||||||
|
Sample chan types.Sample
|
||||||
}
|
}
|
||||||
|
|
||||||
var pipelines = make(map[int]*Pipeline)
|
var pipelines = make(map[int]*Pipeline)
|
||||||
@ -43,10 +43,10 @@ func CreatePipeline(pipelineStr string) (*Pipeline, error) {
|
|||||||
pipelinesLock.Lock()
|
pipelinesLock.Lock()
|
||||||
defer pipelinesLock.Unlock()
|
defer pipelinesLock.Unlock()
|
||||||
|
|
||||||
var gstPipeline *C.GstElement
|
id := len(pipelines)
|
||||||
var gstError *C.GError
|
|
||||||
|
|
||||||
gstPipeline = C.gstreamer_pipeline_create(pipelineStrUnsafe, &gstError)
|
var gstError *C.GError
|
||||||
|
ctx := C.gstreamer_pipeline_create(pipelineStrUnsafe, C.int(id), &gstError)
|
||||||
|
|
||||||
if gstError != nil {
|
if gstError != nil {
|
||||||
defer C.g_error_free(gstError)
|
defer C.g_error_free(gstError)
|
||||||
@ -54,10 +54,10 @@ func CreatePipeline(pipelineStr string) (*Pipeline, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
p := &Pipeline{
|
p := &Pipeline{
|
||||||
Pipeline: gstPipeline,
|
id: id,
|
||||||
Sample: make(chan types.Sample),
|
|
||||||
Src: pipelineStr,
|
Src: pipelineStr,
|
||||||
id: len(pipelines),
|
Ctx: ctx,
|
||||||
|
Sample: make(chan types.Sample),
|
||||||
}
|
}
|
||||||
|
|
||||||
pipelines[p.id] = p
|
pipelines[p.id] = p
|
||||||
@ -68,15 +68,21 @@ func (p *Pipeline) AttachAppsink(sinkName string) {
|
|||||||
sinkNameUnsafe := C.CString(sinkName)
|
sinkNameUnsafe := C.CString(sinkName)
|
||||||
defer C.free(unsafe.Pointer(sinkNameUnsafe))
|
defer C.free(unsafe.Pointer(sinkNameUnsafe))
|
||||||
|
|
||||||
C.gstreamer_pipeline_attach_appsink(p.Pipeline, sinkNameUnsafe, C.int(p.id))
|
C.gstreamer_pipeline_attach_appsink(p.Ctx, sinkNameUnsafe)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Pipeline) Play() {
|
func (p *Pipeline) Play() {
|
||||||
C.gstreamer_pipeline_play(p.Pipeline)
|
C.gstreamer_pipeline_play(p.Ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Pipeline) Stop() {
|
func (p *Pipeline) Pause() {
|
||||||
C.gstreamer_pipeline_stop(p.Pipeline)
|
C.gstreamer_pipeline_pause(p.Ctx)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *Pipeline) Destroy() {
|
||||||
|
C.gstreamer_pipeline_destory(p.Ctx)
|
||||||
|
p.Ctx = nil
|
||||||
|
close(p.Sample)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Pipeline) Push(srcName string, buffer []byte) {
|
func (p *Pipeline) Push(srcName string, buffer []byte) {
|
||||||
@ -86,7 +92,7 @@ func (p *Pipeline) Push(srcName string, buffer []byte) {
|
|||||||
bytes := C.CBytes(buffer)
|
bytes := C.CBytes(buffer)
|
||||||
defer C.free(bytes)
|
defer C.free(bytes)
|
||||||
|
|
||||||
C.gstreamer_pipeline_push(p.Pipeline, srcNameUnsafe, bytes, C.int(len(buffer)))
|
C.gstreamer_pipeline_push(p.Ctx, srcNameUnsafe, bytes, C.int(len(buffer)))
|
||||||
}
|
}
|
||||||
|
|
||||||
// gst-inspect-1.0
|
// gst-inspect-1.0
|
||||||
@ -121,18 +127,30 @@ func goHandlePipelineBuffer(buffer unsafe.Pointer, bufferLen C.int, duration C.i
|
|||||||
log.Warn().
|
log.Warn().
|
||||||
Str("module", "capture").
|
Str("module", "capture").
|
||||||
Str("submodule", "gstreamer").
|
Str("submodule", "gstreamer").
|
||||||
Msgf("discarding buffer, no pipeline with id %d", int(pipelineID))
|
Msgf("discarding sample, no pipeline with id %d", int(pipelineID))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//export goPipelineLog
|
//export goPipelineLog
|
||||||
func goPipelineLog(levelUnsafe *C.char, msgUnsafe *C.char) {
|
func goPipelineLog(levelUnsafe *C.char, msgUnsafe *C.char, pipelineID C.int) {
|
||||||
levelStr := C.GoString(levelUnsafe)
|
levelStr := C.GoString(levelUnsafe)
|
||||||
msg := C.GoString(msgUnsafe)
|
msg := C.GoString(msgUnsafe)
|
||||||
|
|
||||||
level, _ := zerolog.ParseLevel(levelStr)
|
logger := log.With().
|
||||||
log.WithLevel(level).
|
|
||||||
Str("module", "capture").
|
Str("module", "capture").
|
||||||
Str("submodule", "gstreamer").
|
Str("submodule", "gstreamer").
|
||||||
Msg(msg)
|
Logger()
|
||||||
|
|
||||||
|
pipelinesLock.Lock()
|
||||||
|
pipeline, ok := pipelines[int(pipelineID)]
|
||||||
|
pipelinesLock.Unlock()
|
||||||
|
|
||||||
|
if ok {
|
||||||
|
logger = logger.With().
|
||||||
|
Int("id", pipeline.id).
|
||||||
|
Logger()
|
||||||
|
}
|
||||||
|
|
||||||
|
level, _ := zerolog.ParseLevel(levelStr)
|
||||||
|
logger.WithLevel(level).Msg(msg)
|
||||||
}
|
}
|
||||||
|
@ -4,14 +4,20 @@
|
|||||||
#include <gst/gst.h>
|
#include <gst/gst.h>
|
||||||
#include <gst/app/gstappsrc.h>
|
#include <gst/app/gstappsrc.h>
|
||||||
|
|
||||||
extern void goHandlePipelineBuffer(void *buffer, int bufferLen, int samples, int pipelineId);
|
typedef struct GstPipelineCtx {
|
||||||
extern void goPipelineLog(char *level, char *msg);
|
int pipelineId;
|
||||||
|
GstElement *pipeline;
|
||||||
|
} GstPipelineCtx;
|
||||||
|
|
||||||
void gstreamer_pipeline_attach_appsink(GstElement *pipeline, char *sinkName, int pipelineId);
|
extern void goHandlePipelineBuffer(void *buffer, int bufferLen, int samples, int pipelineId);
|
||||||
GstElement *gstreamer_pipeline_create(char *pipelineStr, GError **error);
|
extern void goPipelineLog(char *level, char *msg, int pipelineId);
|
||||||
void gstreamer_pipeline_play(GstElement *pipeline);
|
|
||||||
void gstreamer_pipeline_stop(GstElement *pipeline);
|
GstPipelineCtx *gstreamer_pipeline_create(char *pipelineStr, int pipelineId, GError **error);
|
||||||
void gstreamer_pipeline_push(GstElement *pipeline, char *srcName, void *buffer, int bufferLen);
|
void gstreamer_pipeline_attach_appsink(GstPipelineCtx *ctx, char *sinkName);
|
||||||
|
void gstreamer_pipeline_play(GstPipelineCtx *ctx);
|
||||||
|
void gstreamer_pipeline_pause(GstPipelineCtx *ctx);
|
||||||
|
void gstreamer_pipeline_destory(GstPipelineCtx *ctx);
|
||||||
|
void gstreamer_pipeline_push(GstPipelineCtx *ctx, char *srcName, void *buffer, int bufferLen);
|
||||||
|
|
||||||
void gstreamer_init(void);
|
void gstreamer_init(void);
|
||||||
void gstreamer_loop(void);
|
void gstreamer_loop(void);
|
||||||
|
@ -186,7 +186,7 @@ func (manager *ScreencastManagerCtx) destroyPipeline() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
manager.pipeline.Stop()
|
manager.pipeline.Destroy()
|
||||||
manager.logger.Info().Msgf("destroying pipeline")
|
manager.logger.Info().Msgf("destroying pipeline")
|
||||||
manager.pipeline = nil
|
manager.pipeline = nil
|
||||||
}
|
}
|
||||||
|
@ -262,7 +262,7 @@ func (manager *StreamSinkManagerCtx) destroyPipeline() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
manager.pipeline.Stop()
|
manager.pipeline.Destroy()
|
||||||
manager.logger.Info().Msgf("destroying pipeline")
|
manager.logger.Info().Msgf("destroying pipeline")
|
||||||
manager.pipeline = nil
|
manager.pipeline = nil
|
||||||
}
|
}
|
||||||
|
@ -90,7 +90,7 @@ func (manager *StreamSrcManagerCtx) Stop() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
manager.pipeline.Stop()
|
manager.pipeline.Destroy()
|
||||||
manager.logger.Info().Msgf("destroying pipeline")
|
manager.logger.Info().Msgf("destroying pipeline")
|
||||||
manager.pipeline = nil
|
manager.pipeline = nil
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user