View Source Membrane.Element.Action (Membrane Core v0.10.2)

This module contains type specifications of actions that can be returned from element callbacks.

Returning actions is a way of element interaction with other elements and parts of framework. Each action may be returned by any callback (except for Membrane.Element.Base.handle_init/1 and Membrane.Element.Base.handle_shutdown/2, as they do not support returning any actions) unless explicitly stated otherwise.

Link to this section Summary

Types

Sends buffers through a pad.

Sends caps through a pad.

Makes a demand on a pad.

Marks that processing via a pad (output) has been finished and the pad instance won't be used anymore.

Sends an event through a pad (input or output).

Sends buffers/caps/event to all output pads of element (or to input pads when event occurs on the output pad).

This action sets the latency for the element.

Sends a message to the parent.

Suspends/resumes change of playback state.

Executes Membrane.Element.WithOutputPads.handle_demand/5 callback for the given pad if its demand is greater than 0.

Allows to split callback execution into multiple applications of another callback (called from now sub-callback).

Starts a timer that will invoke Membrane.Element.Base.handle_tick/3 callback every interval according to the given clock.

Stops a timer started with start_timer_t/0 action.

t()

Type that defines a single action that may be returned from element callbacks.

Changes interval of a timer started with start_timer_t/0.

Link to this section Types

@type buffer_t() ::
  {:buffer, {Membrane.Pad.ref_t(), Membrane.Buffer.t() | [Membrane.Buffer.t()]}}

Sends buffers through a pad.

The pad must have output direction.

Allowed only when playback state is playing.

@type caps_t() :: {:caps, {Membrane.Pad.ref_t(), Membrane.Caps.t()}}

Sends caps through a pad.

The pad must have output direction. Sent caps must fit constraints on the pad.

Forbidden when playback state is stopped.

@type demand_size_t() :: pos_integer() | (pos_integer() -> non_neg_integer())
@type demand_t() :: {:demand, {Membrane.Pad.ref_t(), demand_size_t()}}

Makes a demand on a pad.

The pad must have input direction and work in pull mode. This action does NOT entail sending demand through the pad, but just requesting some amount of data from pad's internal queue, which sends demands automatically when it runs out of data. If there is any data available at the pad, the data is passed to Membrane.Filter.handle_process_list/4, Membrane.Endpoint.handle_write_list/4 or Membrane.Sink.handle_write_list/4 callback. Invoked callback is guaranteed not to receive more data than demanded.

Demand size can be either a non-negative integer, that overrides existing demand, or a function that is passed current demand, and is to return the new demand.

Allowed only when playback state is playing.

@type end_of_stream_t() :: {:end_of_stream, Membrane.Pad.ref_t()}

Marks that processing via a pad (output) has been finished and the pad instance won't be used anymore.

Triggers end_of_stream/3 callback at the receiver element. Allowed only when playback is in playing state.

@type event_t() :: {:event, {Membrane.Pad.ref_t(), Membrane.Event.t()}}

Sends an event through a pad (input or output).

Forbidden when playback state is stopped.

@type forward_t() ::
  {:forward,
   Membrane.Buffer.t()
   | [Membrane.Buffer.t()]
   | Membrane.Caps.t()
   | Membrane.Event.t()
   | :end_of_stream}

Sends buffers/caps/event to all output pads of element (or to input pads when event occurs on the output pad).

Used by default implementations of Membrane.Element.WithInputPads.handle_caps/4 and Membrane.Element.Base.handle_event/4 callbacks in filter.

Allowed only when all below conditions are met:

Keep in mind that Membrane.Filter.handle_process_list/4 can only forward buffers, Membrane.Element.WithInputPads.handle_caps/4 - caps and Membrane.Element.Base.handle_event/4 - events.

@type latency_t() :: {:latency, latency :: non_neg_integer()}

This action sets the latency for the element.

This action is not premitted in callback Membrane.Element.Base.handle_init/1.

@type notify_t() :: {:notify, Membrane.Notification.t()}

Sends a message to the parent.

@type playback_change_t() :: {:playback_change, :suspend | :resume}

Suspends/resumes change of playback state.

There is no straight limit how long playback change can take, but keep in mind that it may affect application quality if not done quick enough.

@type redemand_t() :: {:redemand, Membrane.Pad.ref_t()}

Executes Membrane.Element.WithOutputPads.handle_demand/5 callback for the given pad if its demand is greater than 0.

The pad must have output direction and work in pull mode.

redemand-in-sources-and-endpoints

Redemand in Sources and Endpoints

In case of Sources and Endpoints, :redemand is just a helper that simplifies element's code. The element doesn't need to generate the whole demand synchronously at handle_demand or store current demand size in its state, but it can just generate one buffer and return :redemand action. If there is still one or more buffers to produce, returning :redemand triggers the next invocation of handle_demand. In such case, the element is to produce next buffer and call :redemand again. If there are no more buffers demanded, handle_demand is not invoked and the loop ends. One more advantage of the approach with :redemand action is that produced buffers are sent one after another in separate messages and this can possibly improve the latency.

redemand-in-filters

Redemand in Filters

Redemand in Filters is useful in a situation where not the entire demand of output pad has been satisfied and there is a need to send a demand for additional buffers through the input pad. A typical example of this situation is a parser that has not demanded enough bytes to parse the whole frame.

usage-limitations

Usage limitations

Allowed only when playback state is playing.

@type split_t() :: {:split, {callback_name :: atom(), args_list :: [[any()]]}}

Allows to split callback execution into multiple applications of another callback (called from now sub-callback).

Executions are synchronous in the element process, and each of them passes subsequent arguments from the args_list, along with the element state (passed as the last argument each time).

Return value of each execution of sub-callback can be any valid return value of the original callback (this also means sub-callback can return any action valid for the original callback, unless expliciltly stated). Returned actions are executed immediately (they are NOT accumulated and executed after all sub-callback executions are finished).

Useful when a long action is to be undertaken, and partial results need to be returned before entire process finishes (e.g. default implementation of Membrane.Filter.handle_process_list/4 uses split action to invoke Membrane.Filter.handle_process/4 with each buffer)

@type start_timer_t() ::
  {:start_timer,
   {timer_id :: any(), interval :: Ratio.t() | non_neg_integer() | :no_interval}
   | {timer_id :: any(),
      interval :: Ratio.t() | non_neg_integer() | :no_interval,
      clock :: Membrane.Clock.t()}}

Starts a timer that will invoke Membrane.Element.Base.handle_tick/3 callback every interval according to the given clock.

The timer's id is passed to the Membrane.Element.Base.handle_tick/3 callback and can be used for changing its interval via timer_interval_t/0 or stopping it via stop_timer_t/0.

If interval is set to :no_interval, the timer won't issue any ticks until the interval is set with timer_interval_t/0 action.

If no clock is passed, parent's clock is chosen.

Timers use Process.send_after/3 under the hood.

@type stop_timer_t() :: {:stop_timer, timer_id :: any()}

Stops a timer started with start_timer_t/0 action.

This action is atomic: stopping timer guarantees that no ticks will arrive from it.

Type that defines a single action that may be returned from element callbacks.

Depending on element type, callback, current playback state and other circumstances there may be different actions available.

@type timer_interval_t() ::
  {:timer_interval,
   {timer_id :: any(), interval :: Ratio.t() | non_neg_integer() | :no_interval}}

Changes interval of a timer started with start_timer_t/0.

Permitted only from Membrane.Element.Base.handle_tick/3, unless the interval was previously set to :no_interval.

If the interval is :no_interval, the timer won't issue any ticks until another timer_interval_t/0 action. Otherwise, the timer will issue ticks every new interval. The next tick after interval change is scheduled at new_interval + previous_time, where previous_time is the time of the latest tick or the time of returning start_timer_t/0 action if no tick has been sent yet. Note that if current_time - previous_time > new_interval, a burst of div(current_time - previous_time, new_interval) ticks is issued immediately.