Membrane FFmpeg SWScale plugin

Hex.pmAPI DocsCircleCI

Plugin providing an element scaling raw video frames and performing pixel format conversions, using SWScale module of FFmpeg library.

It is a part of Membrane Multimedia Framework.

Installation

Add the following line to your deps in mix.exs. Run mix deps.get.

    {:membrane_ffmpeg_swscale_plugin, "~> 0.11.1"}

You also need to have FFmpeg library with development headers installed on your system.

Description

PixelFormatConverter

PixelFormatConverter accepts raw video in any of the pixel formats specified in type Membrane.RawVideo.pixel_format_t() as input and is capable of producing output in any of these pixel formats.

When creating the element you need to specify a single option format defining the desired pixel format of the output. The element requires Membrane.RawVideo stream format on the input with aligned: true constraint, meaning that each buffer must contain exactly one raw video frame.

Scaler

Scaler needs input in the YUV420p format, processes one frame at a time and requires getting stream format with input video width and height. To meet all requirements either Membrane.RawVideo.Parser or some decoder (e.g. Membrane.H264.FFmpeg.Decoder) have to precede Scaler in the pipeline.

There are two options that have to be specified when creating the element:

Both need to be even numbers.

Scaling consists of two operations:

The output of the element is also in the YUV420p format. It has the size as specified in the options.

Usage

Membrane.FFmpeg.SWScale.Scaler

Scaling an encoded (using H.264 standard) video requires parser and decoder because Scaler scales only raw, decoded video. The pipeline scales the video and reencodes it.

defmodule Scaling.Pipeline do
  use Membrane.Pipeline

  @impl true
  def handle_init(_ctx, _options) do
    structure = [
      child(:file_src, %Membrane.File.Source{location: "/tmp/input.h264"})
      |> child(:parser, Membrane.H264.FFmpeg.Parser)
      |> child(:decoder, Membrane.H264.FFmpeg.Decoder)
      |> child(:scaler, %Membrane.FFmpeg.SWScale.Scaler{output_width: 640, output_height: 640})
      |> child(:encoder, Membrane.H264.FFmpeg.Encoder)
      |> child(:file_sink, %Membrane.File.Sink{location: "/tmp/output.h264"})
    ]

    {[spec: structure}, %{}}
  end
end

Membrane.FFmpeg.SWScale.PixelFormatConverter

Converting the pixel format of an encoded (using H.264 standard) video requires a parser and a decoder because the Converter performs conversion only on the raw , decoded video. The pipeline changes the pixel format of the video to the I422 format and reencodes it.

defmodule Converting.Pipeline do
  use Membrane.Pipeline

  @impl true
  def handle_init(_ctx, _options) do
    structure = [
      child(:file_src, %Membrane.File.Source{location: "/tmp/input.h264"})
      |> child(:parser, Membrane.H264.FFmpeg.Parser)
      |> child(:decoder, Membrane.H264.FFmpeg.Decoder)
      |> child(:converter, %Membrane.FFmpeg.SWScale.PixelFormatConverter{format: :I422})
      |> child(:encoder, Membrane.H264.FFmpeg.Encoder)
      |> child(:file_sink, %Membrane.File.Sink{location: "/tmp/output.h264"})
    ]
    
    {[spec: structure], %{}}
  end
end

Copyright and License

Copyright 2021, Software Mansion

Software Mansion

Licensed under the Apache License, Version 2.0