问题
I have an audio stream that I'd like to save as single playable files split by time.
If I understand correctly, splitmuxsink does this for video files, but not for audio-only files.
There are some tips on how to accomplish this here: http://gstreamer-devel.966125.n4.nabble.com/Dynamically-updating-filesink-location-at-run-time-on-the-fly-td4660569.html I'm struggling to reproduce this and the mentioned functions were for gstreamer-0.1.
回答1:
From the description of splitmuxsink is seems that audio is included:
Pad Templates:
SINK template: 'video'
Availability: On request
Capabilities:
ANY
SINK template: 'audio_%u'
Availability: On request
Capabilities:
ANY
SINK template: 'subtitle_%u'
Availability: On request
Capabilities:
ANY
So I don't see why audio only files should not work with it. Especially since video files are usually considered video files with audio only.
Edit The command below should generate AC3/MP4 files with each 1 minute long:
gst-launch-1.0 -e audiotestsrc ! avenc_ac3 ! ac3parse ! mux.audio_0 splitmuxsink name=mux max-size-time=60000000000 location=out_%d.mp4
回答2:
I got ogg-encoded audio file splitting working with dynamic pipelining in C.
Florian Zwoch in a previous answer shows how splitmuxsink can split audio stream.
A workaround to file splitting using dynamic pipelines:
#include <stdio.h>
#define GLIB_DISABLE_DEPRECATION_WARNINGS
#include <gst/gst.h>
static GstElement *pipeline;
static GstPad *queue_src_pad;
static GstElement *bins[2];
static GstPad *bin_pads[2];
static GstElement *filesink[2];
static size_t current_bin = 0;
static size_t current_file = 0;
static GstPadProbeReturn
pad_probe_cb (GstPad * pad, GstPadProbeInfo * info, gpointer user_data) {
gst_pad_remove_probe (pad, GST_PAD_PROBE_INFO_ID (info));
gst_pad_unlink(queue_src_pad, bin_pads[current_bin]);
gst_pad_send_event(bin_pads[current_bin], gst_event_new_eos());
gst_bin_remove(GST_BIN(pipeline), bins[current_bin]);
gst_element_set_state(bins[current_bin], GST_STATE_NULL);
current_file++;
current_bin = (current_file % 2);
{
char file_location[32];
sprintf(file_location, "recording_%ld.ogg", current_file);
g_object_set(G_OBJECT(
filesink[current_bin]), "location", file_location, NULL);
printf("now writing to %s\n", file_location);
}
gst_bin_add(GST_BIN(pipeline), bins[current_bin]);
gst_pad_link(queue_src_pad, bin_pads[current_bin]);
gst_element_sync_state_with_parent(bins[current_bin]);
return GST_PAD_PROBE_OK;
}
static gboolean timeout_cb(gpointer user_data) {
gst_pad_add_probe (queue_src_pad, GST_PAD_PROBE_TYPE_BLOCK_DOWNSTREAM,
pad_probe_cb, NULL, NULL);
return TRUE;
}
int main(int argc, char *argv[]) {
GstElement *audiosrc, *queue;
GstElement *vorbisenc[2], *oggmux[2];
GstBus *bus;
GMainLoop *loop;
gst_init (&argc, &argv);
//audiosrc = gst_element_factory_make("audiotestsrc", "audiosrc");
//g_object_set (G_OBJECT (audiosrc), "is-live", TRUE, NULL);
audiosrc = gst_element_factory_make("pulsesrc", "audiosrc");
queue = gst_element_factory_make("queue", "queue");
bins[0] = gst_bin_new ("bin0");
bins[1] = gst_bin_new ("bin1");
vorbisenc[0] = gst_element_factory_make("vorbisenc", "vorbisenc0");
vorbisenc[1] = gst_element_factory_make("vorbisenc", "vorbisenc1");
oggmux[0] = gst_element_factory_make("oggmux", "oggmux0");
oggmux[1] = gst_element_factory_make("oggmux", "oggmux1");
filesink[0] = gst_element_factory_make("filesink", "filesink0");
filesink[1] = gst_element_factory_make("filesink", "filesink1");
pipeline = gst_pipeline_new ("test-pipeline");
if (!pipeline || !audiosrc || !queue
|| !vorbisenc[0] || !oggmux[0] || !filesink[0]
|| !vorbisenc[1] || !oggmux[1] || !filesink[1]
) {
g_printerr ("not all elements could be created\n");
return -1;
}
gst_bin_add_many(
GST_BIN(bins[0]), vorbisenc[0], oggmux[0], filesink[0], NULL);
gst_bin_add_many(
GST_BIN(bins[1]), vorbisenc[1], oggmux[1], filesink[1], NULL);
gst_bin_add_many(
GST_BIN(pipeline), audiosrc, queue, bins[0], NULL);
g_assert (gst_element_link(audiosrc, queue));
g_assert (gst_element_link_many(
vorbisenc[0], oggmux[0], filesink[0], NULL));
g_assert (gst_element_link_many(
vorbisenc[1], oggmux[1], filesink[1], NULL));
{
GstPad* pad = gst_element_get_static_pad(vorbisenc[0], "sink");
gst_element_add_pad(bins[0], gst_ghost_pad_new("sink", pad));
gst_object_unref(pad);
}
{
GstPad* pad = gst_element_get_static_pad(vorbisenc[1], "sink");
gst_element_add_pad(bins[1], gst_ghost_pad_new("sink", pad));
gst_object_unref(pad);
}
bin_pads[0] = gst_element_get_static_pad(bins[0], "sink");
bin_pads[1] = gst_element_get_static_pad(bins[1], "sink");
current_bin = 0;
gst_element_link(queue, bins[current_bin]);
g_object_set (filesink[current_bin], "location", "recording_0.ogg", NULL);
queue_src_pad = gst_element_get_static_pad (queue, "src");
bus = gst_element_get_bus (pipeline);
gst_element_set_state (pipeline, GST_STATE_PLAYING);
loop = g_main_loop_new (NULL, FALSE);
g_timeout_add_seconds (3, timeout_cb, NULL);
g_main_loop_run (loop);
gst_object_unref (bus);
gst_element_set_state (pipeline, GST_STATE_NULL);
gst_object_unref (pipeline);
return 0;
}
来源:https://stackoverflow.com/questions/47550863/gstreamer-split-audio-stream-into-files