123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345 |
- /*
- Copyright (c) 2012, Broadcom Europe Ltd
- All rights reserved.
-
- Redistribution and use in source and binary forms, with or without
- modification, are permitted provided that the following conditions are met:
- * Redistributions of source code must retain the above copyright
- notice, this list of conditions and the following disclaimer.
- * Redistributions in binary form must reproduce the above copyright
- notice, this list of conditions and the following disclaimer in the
- documentation and/or other materials provided with the distribution.
- * Neither the name of the copyright holder nor the
- names of its contributors may be used to endorse or promote products
- derived from this software without specific prior written permission.
-
- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
- ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
- DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY
- DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
- (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
- LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
- ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
- SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- */
-
- #include "mmal.h"
- #include "core/mmal_component_private.h"
- #include "core/mmal_port_private.h"
- #include "mmal_logging.h"
-
- #define SPLITTER_OUTPUT_PORTS_NUM 4 /* 4 should do for now */
-
- /*****************************************************************************/
- typedef struct MMAL_COMPONENT_MODULE_T
- {
- uint32_t enabled_flags; /**< Flags indicating which output port is enabled */
- uint32_t sent_flags; /**< Flags indicating which output port we've already sent data to */
- MMAL_BOOL_T error; /**< Error state */
-
- } MMAL_COMPONENT_MODULE_T;
-
- typedef struct MMAL_PORT_MODULE_T
- {
- MMAL_QUEUE_T *queue; /**< queue for the buffers sent to the ports */
-
- } MMAL_PORT_MODULE_T;
-
- /*****************************************************************************/
-
- /** Destroy a previously created component */
- static MMAL_STATUS_T splitter_component_destroy(MMAL_COMPONENT_T *component)
- {
- unsigned int i;
-
- for(i = 0; i < component->input_num; i++)
- if(component->input[i]->priv->module->queue)
- mmal_queue_destroy(component->input[i]->priv->module->queue);
- if(component->input_num)
- mmal_ports_free(component->input, component->input_num);
-
- for(i = 0; i < component->output_num; i++)
- if(component->output[i]->priv->module->queue)
- mmal_queue_destroy(component->output[i]->priv->module->queue);
- if(component->output_num)
- mmal_ports_free(component->output, component->output_num);
-
- vcos_free(component->priv->module);
- return MMAL_SUCCESS;
- }
-
- /** Enable processing on a port */
- static MMAL_STATUS_T splitter_port_enable(MMAL_PORT_T *port, MMAL_PORT_BH_CB_T cb)
- {
- #if 0
- MMAL_COMPONENT_T *component = port->component;
- uint32_t buffer_num, buffer_size;
- unsigned int i;
-
- /* Find the max and apply that to all ports */
- buffer_num = component->input[0]->buffer_num;
- buffer_size = component->input[0]->buffer_size;
- for (i = 0; i < component->output_num; i++)
- {
- buffer_num = MMAL_MAX(buffer_num, component->output[i]->buffer_num);
- buffer_size = MMAL_MAX(buffer_num, component->output[i]->buffer_size);
- }
- component->input[0]->buffer_num = buffer_num;
- component->input[0]->buffer_size = buffer_size;
- for (i = 0; i < component->output_num; i++)
- {
- component->output[i]->buffer_num = buffer_num;
- component->output[i]->buffer_size = buffer_num;
- }
- #endif
-
- MMAL_PARAM_UNUSED(cb);
- if (port->buffer_size)
- if (port->type == MMAL_PORT_TYPE_OUTPUT)
- port->component->priv->module->enabled_flags |= (1<<port->index);
- return MMAL_SUCCESS;
- }
-
- /** Flush a port */
- static MMAL_STATUS_T splitter_port_flush(MMAL_PORT_T *port)
- {
- MMAL_PORT_MODULE_T *port_module = port->priv->module;
- MMAL_BUFFER_HEADER_T *buffer;
-
- /* Flush buffers that our component is holding on to */
- buffer = mmal_queue_get(port_module->queue);
- while(buffer)
- {
- mmal_port_buffer_header_callback(port, buffer);
- buffer = mmal_queue_get(port_module->queue);
- }
-
- if (port->type == MMAL_PORT_TYPE_INPUT)
- port->component->priv->module->sent_flags = 0;
-
- return MMAL_SUCCESS;
- }
-
- /** Disable processing on a port */
- static MMAL_STATUS_T splitter_port_disable(MMAL_PORT_T *port)
- {
- if (port->type == MMAL_PORT_TYPE_OUTPUT)
- port->component->priv->module->enabled_flags &= ~(1<<port->index);
-
- /* We just need to flush our internal queue */
- return splitter_port_flush(port);
- }
-
- /** Send a buffer header to a port */
- static MMAL_STATUS_T splitter_send_output(MMAL_BUFFER_HEADER_T *buffer, MMAL_PORT_T *out_port)
- {
- MMAL_BUFFER_HEADER_T *out;
- MMAL_STATUS_T status;
-
- /* Get a buffer header from output port */
- out = mmal_queue_get(out_port->priv->module->queue);
- if (!out)
- return MMAL_EAGAIN;
-
- /* Copy our input buffer header */
- status = mmal_buffer_header_replicate(out, buffer);
- if (status != MMAL_SUCCESS)
- goto error;
-
- /* Send buffer back */
- mmal_port_buffer_header_callback(out_port, out);
- return MMAL_SUCCESS;
-
- error:
- mmal_queue_put_back(out_port->priv->module->queue, out);
- return status;
- }
-
- /** Send a buffer header to a port */
- static MMAL_STATUS_T splitter_port_send(MMAL_PORT_T *port, MMAL_BUFFER_HEADER_T *buffer)
- {
- MMAL_COMPONENT_T *component = port->component;
- MMAL_COMPONENT_MODULE_T *module = component->priv->module;
- MMAL_PORT_T *in_port, *out_port;
- MMAL_BUFFER_HEADER_T *in;
- MMAL_STATUS_T status;
- unsigned int i;
-
- mmal_queue_put(port->priv->module->queue, buffer);
-
- if (module->error)
- return MMAL_SUCCESS; /* Just do nothing */
-
- /* Get input buffer header */
- in_port = component->input[0];
- in = mmal_queue_get(in_port->priv->module->queue);
- if (!in)
- return MMAL_SUCCESS; /* Nothing to do */
-
- for (i = 0; i < component->output_num; i++)
- {
- out_port = component->output[i];
- status = splitter_send_output(in, out_port);
-
- if (status != MMAL_SUCCESS && status != MMAL_EAGAIN)
- goto error;
-
- if (status == MMAL_SUCCESS)
- module->sent_flags |= (1<<i);
- }
-
- /* Check if we're done with the input buffer */
- if ((module->sent_flags & module->enabled_flags) == module->enabled_flags)
- {
- in->length = 0; /* Consume the input buffer */
- mmal_port_buffer_header_callback(in_port, in);
- module->sent_flags = 0;
- return MMAL_SUCCESS;
- }
-
- /* We're not done yet so put the buffer back in the queue */
- mmal_queue_put(in_port->priv->module->queue, in);
- return MMAL_SUCCESS;
-
- error:
- mmal_queue_put(in_port->priv->module->queue, in);
-
- status = mmal_event_error_send(port->component, status);
- if (status != MMAL_SUCCESS)
- {
- LOG_ERROR("unable to send an error event buffer (%i)", (int)status);
- return MMAL_SUCCESS;
- }
- module->error = 1;
- return MMAL_SUCCESS;
- }
-
- /** Set format on a port */
- static MMAL_STATUS_T splitter_port_format_commit(MMAL_PORT_T *port)
- {
- MMAL_COMPONENT_T *component = port->component;
- MMAL_STATUS_T status;
- unsigned int i;
-
- /* Sanity check */
- if (port->type == MMAL_PORT_TYPE_OUTPUT)
- {
- LOG_ERROR("output port is read-only");
- return MMAL_EINVAL;
- }
-
- /* Commit the format on all output ports */
- for (i = 0; i < component->output_num; i++)
- {
- status = mmal_format_full_copy(component->output[i]->format, port->format);
- if (status != MMAL_SUCCESS)
- return status;
- }
-
- return MMAL_SUCCESS;
- }
-
- static MMAL_STATUS_T splitter_port_parameter_set(MMAL_PORT_T *port, const MMAL_PARAMETER_HEADER_T *param)
- {
- MMAL_COMPONENT_T *component = port->component;
- unsigned int i;
-
- switch (param->id)
- {
- case MMAL_PARAMETER_BUFFER_REQUIREMENTS:
- {
- /* Propagate the requirements to all the ports */
- const MMAL_PARAMETER_BUFFER_REQUIREMENTS_T *req = (const MMAL_PARAMETER_BUFFER_REQUIREMENTS_T *)param;
- uint32_t buffer_num_min = MMAL_MAX(port->buffer_num_min, req->buffer_num_min);
- uint32_t buffer_num_recommended = MMAL_MAX(port->buffer_num_recommended, req->buffer_num_recommended);
- uint32_t buffer_size_min = MMAL_MAX(port->buffer_size_min, req->buffer_size_min);
- uint32_t buffer_size_recommended = MMAL_MAX(port->buffer_size_recommended, req->buffer_size_recommended);
-
- component->input[0]->buffer_num_min = buffer_num_min;
- component->input[0]->buffer_num_recommended = buffer_num_recommended;
- component->input[0]->buffer_size_min = buffer_size_min;
- component->input[0]->buffer_size_recommended = buffer_size_recommended;
- for (i = 0; i < component->output_num; i++)
- {
- component->output[i]->buffer_num_min = buffer_num_min;
- component->output[i]->buffer_num_recommended = buffer_num_recommended;
- component->output[i]->buffer_size_min = buffer_size_min;
- component->output[i]->buffer_size_recommended = buffer_size_recommended;
- }
-
- }
- return MMAL_SUCCESS;
-
- default:
- return MMAL_ENOSYS;
- }
- }
-
- /** Create an instance of a component */
- static MMAL_STATUS_T mmal_component_create_splitter(const char *name, MMAL_COMPONENT_T *component)
- {
- MMAL_COMPONENT_MODULE_T *module;
- MMAL_STATUS_T status = MMAL_ENOMEM;
- unsigned int i;
- MMAL_PARAM_UNUSED(name);
-
- /* Allocate the context for our module */
- component->priv->module = module = vcos_malloc(sizeof(*module), "mmal module");
- if (!module)
- return MMAL_ENOMEM;
- memset(module, 0, sizeof(*module));
-
- component->priv->pf_destroy = splitter_component_destroy;
-
- /* Allocate and initialise all the ports for this component */
- component->input = mmal_ports_alloc(component, 1, MMAL_PORT_TYPE_INPUT, sizeof(MMAL_PORT_MODULE_T));
- if(!component->input)
- goto error;
- component->input_num = 1;
- component->input[0]->priv->pf_enable = splitter_port_enable;
- component->input[0]->priv->pf_disable = splitter_port_disable;
- component->input[0]->priv->pf_flush = splitter_port_flush;
- component->input[0]->priv->pf_send = splitter_port_send;
- component->input[0]->priv->pf_set_format = splitter_port_format_commit;
- component->input[0]->priv->pf_parameter_set = splitter_port_parameter_set;
- component->input[0]->buffer_num_min = 1;
- component->input[0]->buffer_num_recommended = 0;
- component->input[0]->priv->module->queue = mmal_queue_create();
- if(!component->input[0]->priv->module->queue)
- goto error;
-
- component->output = mmal_ports_alloc(component, SPLITTER_OUTPUT_PORTS_NUM,
- MMAL_PORT_TYPE_OUTPUT, sizeof(MMAL_PORT_MODULE_T));
- if(!component->output)
- goto error;
- component->output_num = SPLITTER_OUTPUT_PORTS_NUM;
- for(i = 0; i < component->output_num; i++)
- {
- component->output[i]->priv->pf_enable = splitter_port_enable;
- component->output[i]->priv->pf_disable = splitter_port_disable;
- component->output[i]->priv->pf_flush = splitter_port_flush;
- component->output[i]->priv->pf_send = splitter_port_send;
- component->output[i]->priv->pf_set_format = splitter_port_format_commit;
- component->output[i]->priv->pf_parameter_set = splitter_port_parameter_set;
- component->output[i]->buffer_num_min = 1;
- component->output[i]->buffer_num_recommended = 0;
- component->output[i]->capabilities = MMAL_PORT_CAPABILITY_PASSTHROUGH;
- component->output[i]->priv->module->queue = mmal_queue_create();
- if(!component->output[i]->priv->module->queue)
- goto error;
- }
-
- return MMAL_SUCCESS;
-
- error:
- splitter_component_destroy(component);
- return status;
- }
-
- MMAL_CONSTRUCTOR(mmal_register_component_splitter);
- void mmal_register_component_splitter(void)
- {
- mmal_component_supplier_register("splitter", mmal_component_create_splitter);
- }
|