Use packet source trait in demuxer
This commit is contained in:
parent
c39054a63d
commit
f3197e178d
3 changed files with 19 additions and 88 deletions
|
@ -112,65 +112,6 @@ sc_demuxer_recv_packet(struct sc_demuxer *demuxer, AVPacket *packet) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool
|
|
||||||
push_packet_to_sinks(struct sc_demuxer *demuxer, const AVPacket *packet) {
|
|
||||||
for (unsigned i = 0; i < demuxer->sink_count; ++i) {
|
|
||||||
struct sc_packet_sink *sink = demuxer->sinks[i];
|
|
||||||
if (!sink->ops->push(sink, packet)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool
|
|
||||||
sc_demuxer_push_packet(struct sc_demuxer *demuxer, AVPacket *packet) {
|
|
||||||
bool ok = push_packet_to_sinks(demuxer, packet);
|
|
||||||
if (!ok) {
|
|
||||||
LOGE("Demuxer '%s': could not process packet", demuxer->name);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
sc_demuxer_close_first_sinks(struct sc_demuxer *demuxer, unsigned count) {
|
|
||||||
while (count) {
|
|
||||||
struct sc_packet_sink *sink = demuxer->sinks[--count];
|
|
||||||
sink->ops->close(sink);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void
|
|
||||||
sc_demuxer_close_sinks(struct sc_demuxer *demuxer) {
|
|
||||||
sc_demuxer_close_first_sinks(demuxer, demuxer->sink_count);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool
|
|
||||||
sc_demuxer_open_sinks(struct sc_demuxer *demuxer, const AVCodec *codec) {
|
|
||||||
for (unsigned i = 0; i < demuxer->sink_count; ++i) {
|
|
||||||
struct sc_packet_sink *sink = demuxer->sinks[i];
|
|
||||||
if (!sink->ops->open(sink, codec)) {
|
|
||||||
sc_demuxer_close_first_sinks(demuxer, i);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
sc_demuxer_disable_sinks(struct sc_demuxer *demuxer) {
|
|
||||||
for (unsigned i = 0; i < demuxer->sink_count; ++i) {
|
|
||||||
struct sc_packet_sink *sink = demuxer->sinks[i];
|
|
||||||
if (sink->ops->disable) {
|
|
||||||
sink->ops->disable(sink);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
run_demuxer(void *data) {
|
run_demuxer(void *data) {
|
||||||
struct sc_demuxer *demuxer = data;
|
struct sc_demuxer *demuxer = data;
|
||||||
|
@ -189,7 +130,7 @@ run_demuxer(void *data) {
|
||||||
if (raw_codec_id == 0) {
|
if (raw_codec_id == 0) {
|
||||||
LOGW("Demuxer '%s': stream explicitly disabled by the device",
|
LOGW("Demuxer '%s': stream explicitly disabled by the device",
|
||||||
demuxer->name);
|
demuxer->name);
|
||||||
sc_demuxer_disable_sinks(demuxer);
|
sc_packet_source_sinks_disable(&demuxer->packet_source);
|
||||||
status = SC_DEMUXER_STATUS_DISABLED;
|
status = SC_DEMUXER_STATUS_DISABLED;
|
||||||
goto end;
|
goto end;
|
||||||
}
|
}
|
||||||
|
@ -204,7 +145,7 @@ run_demuxer(void *data) {
|
||||||
if (codec_id == AV_CODEC_ID_NONE) {
|
if (codec_id == AV_CODEC_ID_NONE) {
|
||||||
LOGE("Demuxer '%s': stream disabled due to unsupported codec",
|
LOGE("Demuxer '%s': stream disabled due to unsupported codec",
|
||||||
demuxer->name);
|
demuxer->name);
|
||||||
sc_demuxer_disable_sinks(demuxer);
|
sc_packet_source_sinks_disable(&demuxer->packet_source);
|
||||||
goto end;
|
goto end;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -212,11 +153,11 @@ run_demuxer(void *data) {
|
||||||
if (!codec) {
|
if (!codec) {
|
||||||
LOGE("Demuxer '%s': stream disabled due to missing decoder",
|
LOGE("Demuxer '%s': stream disabled due to missing decoder",
|
||||||
demuxer->name);
|
demuxer->name);
|
||||||
sc_demuxer_disable_sinks(demuxer);
|
sc_packet_source_sinks_disable(&demuxer->packet_source);
|
||||||
goto end;
|
goto end;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!sc_demuxer_open_sinks(demuxer, codec)) {
|
if (!sc_packet_source_sinks_open(&demuxer->packet_source, codec)) {
|
||||||
goto end;
|
goto end;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -253,10 +194,10 @@ run_demuxer(void *data) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ok = sc_demuxer_push_packet(demuxer, packet);
|
ok = sc_packet_source_sinks_push(&demuxer->packet_source, packet);
|
||||||
av_packet_unref(packet);
|
av_packet_unref(packet);
|
||||||
if (!ok) {
|
if (!ok) {
|
||||||
// cannot process packet (error already logged)
|
// The sink already logged its concrete error
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -269,7 +210,7 @@ run_demuxer(void *data) {
|
||||||
|
|
||||||
av_packet_free(&packet);
|
av_packet_free(&packet);
|
||||||
finally_close_sinks:
|
finally_close_sinks:
|
||||||
sc_demuxer_close_sinks(demuxer);
|
sc_packet_source_sinks_close(&demuxer->packet_source);
|
||||||
end:
|
end:
|
||||||
demuxer->cbs->on_ended(demuxer, status, demuxer->cbs_userdata);
|
demuxer->cbs->on_ended(demuxer, status, demuxer->cbs_userdata);
|
||||||
|
|
||||||
|
@ -283,7 +224,7 @@ sc_demuxer_init(struct sc_demuxer *demuxer, const char *name, sc_socket socket,
|
||||||
|
|
||||||
demuxer->name = name; // statically allocated
|
demuxer->name = name; // statically allocated
|
||||||
demuxer->socket = socket;
|
demuxer->socket = socket;
|
||||||
demuxer->sink_count = 0;
|
sc_packet_source_init(&demuxer->packet_source);
|
||||||
|
|
||||||
assert(cbs && cbs->on_ended);
|
assert(cbs && cbs->on_ended);
|
||||||
|
|
||||||
|
@ -291,14 +232,6 @@ sc_demuxer_init(struct sc_demuxer *demuxer, const char *name, sc_socket socket,
|
||||||
demuxer->cbs_userdata = cbs_userdata;
|
demuxer->cbs_userdata = cbs_userdata;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
|
||||||
sc_demuxer_add_sink(struct sc_demuxer *demuxer, struct sc_packet_sink *sink) {
|
|
||||||
assert(demuxer->sink_count < SC_DEMUXER_MAX_SINKS);
|
|
||||||
assert(sink);
|
|
||||||
assert(sink->ops);
|
|
||||||
demuxer->sinks[demuxer->sink_count++] = sink;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
bool
|
||||||
sc_demuxer_start(struct sc_demuxer *demuxer) {
|
sc_demuxer_start(struct sc_demuxer *demuxer) {
|
||||||
LOGD("Demuxer '%s': starting thread", demuxer->name);
|
LOGD("Demuxer '%s': starting thread", demuxer->name);
|
||||||
|
|
|
@ -8,21 +8,19 @@
|
||||||
#include <libavcodec/avcodec.h>
|
#include <libavcodec/avcodec.h>
|
||||||
#include <libavformat/avformat.h>
|
#include <libavformat/avformat.h>
|
||||||
|
|
||||||
|
#include "trait/packet_source.h"
|
||||||
#include "trait/packet_sink.h"
|
#include "trait/packet_sink.h"
|
||||||
#include "util/net.h"
|
#include "util/net.h"
|
||||||
#include "util/thread.h"
|
#include "util/thread.h"
|
||||||
|
|
||||||
#define SC_DEMUXER_MAX_SINKS 2
|
|
||||||
|
|
||||||
struct sc_demuxer {
|
struct sc_demuxer {
|
||||||
|
struct sc_packet_source packet_source; // packet source trait
|
||||||
|
|
||||||
const char *name; // must be statically allocated (e.g. a string literal)
|
const char *name; // must be statically allocated (e.g. a string literal)
|
||||||
|
|
||||||
sc_socket socket;
|
sc_socket socket;
|
||||||
sc_thread thread;
|
sc_thread thread;
|
||||||
|
|
||||||
struct sc_packet_sink *sinks[SC_DEMUXER_MAX_SINKS];
|
|
||||||
unsigned sink_count;
|
|
||||||
|
|
||||||
const struct sc_demuxer_callbacks *cbs;
|
const struct sc_demuxer_callbacks *cbs;
|
||||||
void *cbs_userdata;
|
void *cbs_userdata;
|
||||||
};
|
};
|
||||||
|
@ -43,9 +41,6 @@ void
|
||||||
sc_demuxer_init(struct sc_demuxer *demuxer, const char *name, sc_socket socket,
|
sc_demuxer_init(struct sc_demuxer *demuxer, const char *name, sc_socket socket,
|
||||||
const struct sc_demuxer_callbacks *cbs, void *cbs_userdata);
|
const struct sc_demuxer_callbacks *cbs, void *cbs_userdata);
|
||||||
|
|
||||||
void
|
|
||||||
sc_demuxer_add_sink(struct sc_demuxer *demuxer, struct sc_packet_sink *sink);
|
|
||||||
|
|
||||||
bool
|
bool
|
||||||
sc_demuxer_start(struct sc_demuxer *demuxer);
|
sc_demuxer_start(struct sc_demuxer *demuxer);
|
||||||
|
|
||||||
|
|
|
@ -444,11 +444,13 @@ scrcpy(struct scrcpy_options *options) {
|
||||||
#endif
|
#endif
|
||||||
if (needs_video_decoder) {
|
if (needs_video_decoder) {
|
||||||
sc_decoder_init(&s->video_decoder, "video");
|
sc_decoder_init(&s->video_decoder, "video");
|
||||||
sc_demuxer_add_sink(&s->video_demuxer, &s->video_decoder.packet_sink);
|
sc_packet_source_add_sink(&s->video_demuxer.packet_source,
|
||||||
|
&s->video_decoder.packet_sink);
|
||||||
}
|
}
|
||||||
if (needs_audio_decoder) {
|
if (needs_audio_decoder) {
|
||||||
sc_decoder_init(&s->audio_decoder, "audio");
|
sc_decoder_init(&s->audio_decoder, "audio");
|
||||||
sc_demuxer_add_sink(&s->audio_demuxer, &s->audio_decoder.packet_sink);
|
sc_packet_source_add_sink(&s->audio_demuxer.packet_source,
|
||||||
|
&s->audio_decoder.packet_sink);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (options->record_filename) {
|
if (options->record_filename) {
|
||||||
|
@ -467,9 +469,10 @@ scrcpy(struct scrcpy_options *options) {
|
||||||
}
|
}
|
||||||
recorder_started = true;
|
recorder_started = true;
|
||||||
|
|
||||||
sc_demuxer_add_sink(&s->video_demuxer, &s->recorder.video_packet_sink);
|
sc_packet_source_add_sink(&s->video_demuxer.packet_source,
|
||||||
|
&s->recorder.video_packet_sink);
|
||||||
if (options->audio) {
|
if (options->audio) {
|
||||||
sc_demuxer_add_sink(&s->audio_demuxer,
|
sc_packet_source_add_sink(&s->audio_demuxer.packet_source,
|
||||||
&s->recorder.audio_packet_sink);
|
&s->recorder.audio_packet_sink);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue