39 namespace ArgusSamples
44 : m_state(GST_STATE_NULL)
56 static const char *s_videoEncoderName =
"video encoder";
87 gst_object_unref(
m_p);
114 float frameRate,
const char *fileName,
VideoFormat videoFormat,
116 bool enableTwoPassCBR)
120 std::string videoFileName(fileName);
121 if (videoFileName !=
"/dev/null")
123 videoFileName +=
".";
129 gst_init(NULL, NULL);
132 m_pipeline = gst_pipeline_new(
"video_pipeline");
134 ORIGINATE_ERROR(
"Failed to create video pipeline");
137 GstElement *videoSource = gst_element_factory_make(
"nveglstreamsrc", NULL);
139 ORIGINATE_ERROR(
"Failed to create capture source element");
141 if (!gst_bin_add(GST_BIN(m_pipeline), videoSource))
142 ORIGINATE_ERROR(
"Failed to add video source to pipeline");
146 g_object_set(G_OBJECT(videoSource),
"eglstream", videoStream, NULL);
149 GstElement *queue = gst_element_factory_make(
"queue", NULL);
151 ORIGINATE_ERROR(
"Failed to create queue");
153 if (!gst_bin_add(GST_BIN(m_pipeline), queue))
154 ORIGINATE_ERROR(
"Failed to add queue to pipeline");
158 GstElement *videoEncoder = NULL;
162 videoEncoder = gst_element_factory_make(
"nvv4l2h264enc", s_videoEncoderName);
165 videoEncoder = gst_element_factory_make(
"nvv4l2h265enc", s_videoEncoderName);
168 printf(
"\n***vp8 encode is not supported for Jetson-Xavier & beyond\n");
169 videoEncoder = gst_element_factory_make(
"nvv4l2vp8enc", s_videoEncoderName);
172 videoEncoder = gst_element_factory_make(
"nvv4l2vp9enc", s_videoEncoderName);
175 ORIGINATE_ERROR(
"Unhandled video format");
178 ORIGINATE_ERROR(
"Failed to create video encoder");
179 unrefer.
set(videoEncoder);
180 if (!gst_bin_add(GST_BIN(m_pipeline), videoEncoder))
181 ORIGINATE_ERROR(
"Failed to add video encoder to pipeline");
189 else if (height < 1080)
191 else if (height <= 2160)
196 if (frameRate < 15.0)
198 bitRate = (uint32_t)((
float)bitRate/30.0*frameRate);
202 g_object_set(G_OBJECT(videoEncoder),
"bitrate", bitRate, NULL);
203 g_object_set(G_OBJECT(videoEncoder),
"control-rate", controlRate, NULL);
204 g_object_set(G_OBJECT(videoEncoder),
"EnableTwopassCBR", enableTwoPassCBR, NULL);
210 const uint32_t WIDTH_4K = 3840;
213 ORIGINATE_ERROR(
"\n Resolution > 4k requires videoformat H265 \n");
224 printf(
"\nThe VP9 video format is not supported on Jetson-tx1.\n");
230 printf(
"\nThe 3GP is only supported with H264 in current GST version. "
231 "Selecting other containers.\n");
238 printf(
"\nThe AVI is not supported with H265 in current GST version. "
239 "Selecting other containers.\n");
243 GstElement *videoParse = NULL;
247 videoParse = gst_element_factory_make(
"h264parse", NULL);
249 ORIGINATE_ERROR(
"Failed to create video parser");
252 videoParse = gst_element_factory_make(
"h265parse", NULL);
254 ORIGINATE_ERROR(
"Failed to create video parser");
260 ORIGINATE_ERROR(
"Unhandled video file type");
264 unrefer.
set(videoParse);
265 if (!gst_bin_add(GST_BIN(m_pipeline), videoParse))
266 ORIGINATE_ERROR(
"Failed to add video parser to pipeline");
270 GstElement *videoMuxer = NULL;
271 switch (videoFileType)
274 videoMuxer = gst_element_factory_make(
"qtmux", NULL);
277 videoMuxer = gst_element_factory_make(
"3gppmux", NULL);
280 videoMuxer = gst_element_factory_make(
"avimux", NULL);
283 videoMuxer = gst_element_factory_make(
"matroskamux", NULL);
286 ORIGINATE_ERROR(
"Unhandled video file type");
289 ORIGINATE_ERROR(
"Failed to create video muxer");
290 unrefer.
set(videoMuxer);
291 if (!gst_bin_add(GST_BIN(m_pipeline), videoMuxer))
292 ORIGINATE_ERROR(
"Failed to add video muxer to pipeline");
296 GstElement *videoSink = gst_element_factory_make(
"filesink", NULL);
298 ORIGINATE_ERROR(
"Failed to create video sink");
299 unrefer.
set(videoSink);
300 if (!gst_bin_add(GST_BIN(m_pipeline), videoSink))
301 ORIGINATE_ERROR(
"Failed to add video sink to pipeline");
304 g_object_set(G_OBJECT(videoSink),
"location", videoFileName.c_str(), NULL);
309 if (frameRate == 0.0f)
313 GstCaps *caps = gst_caps_new_simple(
"video/x-raw",
314 "format", G_TYPE_STRING,
"NV12",
315 "width", G_TYPE_INT, width,
316 "height", G_TYPE_INT, height,
317 "framerate", GST_TYPE_FRACTION, static_cast<gint>(frameRate * 100.f), 100,
320 ORIGINATE_ERROR(
"Failed to create caps");
322 GstCapsFeatures *feature = gst_caps_features_new(
"memory:NVMM", NULL);
325 gst_caps_unref(caps);
326 ORIGINATE_ERROR(
"Failed to create caps feature");
329 gst_caps_set_features(caps, 0, feature);
332 if (!gst_element_link_filtered(videoSource, queue, caps))
334 gst_caps_unref(caps);
335 ORIGINATE_ERROR(
"Failed to link source to queue");
337 gst_caps_unref(caps);
340 if (!gst_element_link(queue, videoEncoder))
341 ORIGINATE_ERROR(
"Failed to link queue to encoder");
346 if (!gst_element_link(videoEncoder, videoParse))
347 ORIGINATE_ERROR(
"Failed to link encoder to parser");
349 if (!gst_element_link(videoParse, videoMuxer))
350 ORIGINATE_ERROR(
"Failed to link parser to muxer");
354 if (!gst_element_link(videoEncoder, videoMuxer))
355 ORIGINATE_ERROR(
"Failed to link encoder to muxer");
359 if (!gst_element_link(videoMuxer, videoSink))
360 ORIGINATE_ERROR(
"Failed to link muxer to sink");
363 #else // GST_SUPPORTED
364 ORIGINATE_ERROR(
"Not supported");
365 #endif // GST_SUPPORTED
372 static bool objectModifyFlags(GObject *obj,
const char *flagName,
const char *valueName,
bool set)
375 GParamSpec **spec = g_object_class_list_properties(G_OBJECT_GET_CLASS(obj), &count);
377 for (guint index = 0; index < count; ++index)
379 GParamSpec *param = spec[index];
380 if (strcmp(param->name, flagName) == 0)
382 if (!G_IS_PARAM_SPEC_FLAGS(param))
383 ORIGINATE_ERROR(
"Param '%s' is not a flag", flagName);
385 GParamSpecFlags *pflags = G_PARAM_SPEC_FLAGS(param);
386 GFlagsValue *value = g_flags_get_value_by_nick(pflags->flags_class, valueName);
388 ORIGINATE_ERROR(
"Value '%s' of flag '%s' not found", valueName, flagName);
391 g_object_get(obj, flagName, &flags, NULL);
393 flags |= value->value;
395 flags &= ~value->value;
396 g_object_set(obj, flagName, flags, NULL);
402 ORIGINATE_ERROR(
"Param '%s' not found", flagName);
404 #endif // GST_SUPPORTED
410 gst_init(NULL, NULL);
413 m_pipeline = gst_element_factory_make(
"playbin",
"play");
415 ORIGINATE_ERROR(
"Failed to create playback pipeline");
418 char *uri = gst_filename_to_uri(fileName, NULL);
419 g_object_set(G_OBJECT(m_pipeline),
"uri", uri, NULL);
423 PROPAGATE_ERROR(objectModifyFlags(G_OBJECT(m_pipeline),
"flags",
"text",
false));
424 PROPAGATE_ERROR(objectModifyFlags(G_OBJECT(m_pipeline),
"flags",
"native-video",
true));
427 GstElement *audioSink = gst_element_factory_make(
"autoaudiosink",
"audio_sink");
429 ORIGINATE_ERROR(
"Failed to create audio sink");
432 g_object_set(G_OBJECT(m_pipeline),
"audio-sink", audioSink, NULL);
435 GstElement *videoSinkBin = gst_bin_new(
"video_sink_bin");
437 ORIGINATE_ERROR(
"Failed to create video sink bin");
440 g_object_set(G_OBJECT(m_pipeline),
"video-sink", videoSinkBin, NULL);
443 GstElement *videoConvert = gst_element_factory_make(
"nvvidconv",
"video converter");
445 ORIGINATE_ERROR(
"Failed to create video converter");
447 if (!gst_bin_add(GST_BIN(videoSinkBin), videoConvert))
448 ORIGINATE_ERROR(
"Failed to add video convert to video sink bin");
452 GstElement *videoSink = gst_element_factory_make(
"nvvideosink",
"video sink");
454 ORIGINATE_ERROR(
"Failed to create video sink");
455 unrefer.
set(videoSink);
456 if (!gst_bin_add(GST_BIN(videoSinkBin), videoSink))
457 ORIGINATE_ERROR(
"Failed to add video sink to video sink bin");
463 *videoStream = EGL_NO_STREAM_KHR;
464 g_object_get(G_OBJECT(videoSink),
"stream", videoStream, NULL);
465 if (*videoStream == EGL_NO_STREAM_KHR)
466 ORIGINATE_ERROR(
"Failed to get EGL stream from video sink");
468 if (!gst_element_link(videoConvert, videoSink))
469 ORIGINATE_ERROR(
"Failed to link video convert to video sink");
472 GstPad *pad = gst_element_get_static_pad(videoConvert,
"sink");
474 ORIGINATE_ERROR(
"Failed to get sink pad of video convert");
476 GstPad *ghostPad = gst_ghost_pad_new(
"sink", pad);
478 ORIGINATE_ERROR(
"Failed to create the ghost pad");
480 if (!gst_pad_set_active(ghostPad, TRUE))
481 ORIGINATE_ERROR(
"Failed to set pad active");
482 if (!gst_element_add_pad(videoSinkBin, ghostPad))
483 ORIGINATE_ERROR(
"Failed to add pad");
488 #else // GST_SUPPORTED
489 ORIGINATE_ERROR(
"Not supported");
490 #endif // GST_SUPPORTED
497 ORIGINATE_ERROR(
"Video pipeline is not set up");
499 if (m_state != GST_STATE_PLAYING)
502 if (gst_element_set_state(m_pipeline, GST_STATE_PLAYING) == GST_STATE_CHANGE_FAILURE)
503 ORIGINATE_ERROR(
"Failed to set playing state");
505 m_state = GST_STATE_PLAYING;
514 GST_DEBUG_BIN_TO_DOT_FILE_WITH_TS(GST_BIN(m_pipeline),
515 GST_DEBUG_GRAPH_SHOW_ALL,
"argus_camera");
519 #else // GST_SUPPORTED
520 ORIGINATE_ERROR(
"Not supported");
521 #endif // GST_SUPPORTED
528 ORIGINATE_ERROR(
"Video pipeline is not set up");
530 if (m_state != GST_STATE_PAUSED)
532 if (gst_element_set_state(m_pipeline, GST_STATE_PAUSED) == GST_STATE_CHANGE_FAILURE)
533 ORIGINATE_ERROR(
"Failed to set pause state");
534 m_state = GST_STATE_PAUSED;
538 #else // GST_SUPPORTED
539 ORIGINATE_ERROR(
"Not supported");
540 #endif // GST_SUPPORTED
548 ORIGINATE_ERROR(
"Video pipeline is not set up");
550 GstState newState = GST_STATE_NULL;
551 if (m_state == GST_STATE_PLAYING)
552 newState = GST_STATE_PAUSED;
553 else if (m_state == GST_STATE_PAUSED)
554 newState = GST_STATE_PLAYING;
556 ORIGINATE_ERROR(
"Invalid state");
558 if (gst_element_set_state(m_pipeline, newState) == GST_STATE_CHANGE_FAILURE)
559 ORIGINATE_ERROR(
"Failed to set pause state");
564 #else // GST_SUPPORTED
565 ORIGINATE_ERROR(
"Not supported");
566 #endif // GST_SUPPORTED
573 ORIGINATE_ERROR(
"Video pipeline is not set up");
575 if (!gst_element_seek_simple(m_pipeline, GST_FORMAT_TIME,
576 static_cast<GstSeekFlags>(GST_SEEK_FLAG_FLUSH | GST_SEEK_FLAG_KEY_UNIT), 0))
578 ORIGINATE_ERROR(
"Failed to rewind");
582 #else // GST_SUPPORTED
583 ORIGINATE_ERROR(
"Not supported");
584 #endif // GST_SUPPORTED
591 ORIGINATE_ERROR(
"Video pipeline is not set up");
593 if ((m_state == GST_STATE_PLAYING) || (m_state == GST_STATE_PAUSED))
596 GstElement *videoEncoder = gst_bin_get_by_name(GST_BIN(m_pipeline), s_videoEncoderName);
600 GstPad *pad = gst_element_get_static_pad(videoEncoder,
"sink");
602 ORIGINATE_ERROR(
"Failed to get 'sink' pad");
604 if (!gst_pad_send_event(pad, gst_event_new_eos()))
605 ORIGINATE_ERROR(
"Failed to send end of stream event encoder");
609 GstBus *bus = gst_pipeline_get_bus(GST_PIPELINE(m_pipeline));
611 ORIGINATE_ERROR(
"Failed to get bus");
613 if (!gst_bus_poll(bus, GST_MESSAGE_EOS, GST_CLOCK_TIME_NONE))
614 ORIGINATE_ERROR(
"Failed to wait for the eof event");
619 if (gst_element_set_state(m_pipeline, GST_STATE_NULL) != GST_STATE_CHANGE_SUCCESS)
620 ORIGINATE_ERROR(
"Failed to stop pipeline");
622 m_state = GST_STATE_NULL;
626 #else // GST_SUPPORTED
627 ORIGINATE_ERROR(
"Not supported");
628 #endif // GST_SUPPORTED
636 PROPAGATE_ERROR(
stop());
639 gst_object_unref(GST_OBJECT(m_pipeline));
645 #else // GST_SUPPORTED
646 ORIGINATE_ERROR(
"Not supported");
647 #endif // GST_SUPPORTED
668 return "Unhandled video file type";
673 if (aspectRatio == NULL)
674 ORIGINATE_ERROR(
"'aspectRatio' is NULL");
676 if ((m_state != GST_STATE_PLAYING) && (m_state != GST_STATE_PAUSED))
677 ORIGINATE_ERROR(
"Must be in paused or playing state.");
679 GstState state = GST_STATE_NULL;
680 while ((state != GST_STATE_PLAYING) && (state != GST_STATE_PAUSED))
682 if (gst_element_get_state(m_pipeline, &state, NULL, GST_CLOCK_TIME_NONE) ==
683 GST_STATE_CHANGE_FAILURE)
685 ORIGINATE_ERROR(
"gst_element_get_state failed");
690 GstElement *videoSink;
691 g_object_get(m_pipeline,
"video-sink", &videoSink, NULL);
693 ORIGINATE_ERROR(
"Failed to get video-sink");
696 GstPad *videoSinkPad = gst_element_get_static_pad(videoSink,
"sink");
698 ORIGINATE_ERROR(
"Failed to get video-sink pad");
700 GstCaps *caps = gst_pad_get_current_caps(videoSinkPad);
702 ORIGINATE_ERROR(
"Failed to get video-sink pad caps");
706 GstStructure *structure = gst_caps_get_structure(caps, 0);
709 gst_caps_unref(caps);
710 ORIGINATE_ERROR(
"Failed to get caps structure");
714 gint pixelAspectRatioNumerator, pixelAspectRatioDenominator;
716 if (!gst_structure_get_int(structure,
"width", &width) ||
717 !gst_structure_get_int(structure,
"height", &height) ||
718 !gst_structure_get_fraction(structure,
"pixel-aspect-ratio",
719 &pixelAspectRatioNumerator, &pixelAspectRatioDenominator))
721 gst_caps_unref(caps);
722 ORIGINATE_ERROR(
"Failed to get structure values");
725 *aspectRatio = (float)width / (
float)height;
726 *aspectRatio *= (float)pixelAspectRatioNumerator / (
float)pixelAspectRatioDenominator;
728 gst_caps_unref(caps);
731 #else // GST_SUPPORTED
732 ORIGINATE_ERROR(
"Not supported");
733 #endif // GST_SUPPORTED
740 #else // GST_SUPPORTED
742 #endif // GST_SUPPORTED