| 1 | /* |
| 2 | Copyright (c) 2012, Broadcom Europe Ltd |
| 3 | All rights reserved. |
| 4 | |
| 5 | Redistribution and use in source and binary forms, with or without |
| 6 | modification, are permitted provided that the following conditions are met: |
| 7 | * Redistributions of source code must retain the above copyright |
| 8 | notice, this list of conditions and the following disclaimer. |
| 9 | * Redistributions in binary form must reproduce the above copyright |
| 10 | notice, this list of conditions and the following disclaimer in the |
| 11 | documentation and/or other materials provided with the distribution. |
| 12 | * Neither the name of the copyright holder nor the |
| 13 | names of its contributors may be used to endorse or promote products |
| 14 | derived from this software without specific prior written permission. |
| 15 | |
| 16 | THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND |
| 17 | ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
| 18 | WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| 19 | DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY |
| 20 | DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
| 21 | (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
| 22 | LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
| 23 | ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| 24 | (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
| 25 | SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| 26 | */ |
| 27 | |
| 28 | #include "mmal.h" |
| 29 | #include "core/mmal_component_private.h" |
| 30 | #include "core/mmal_port_private.h" |
| 31 | #include "mmal_logging.h" |
| 32 | |
| 33 | #define PASSTHROUGH_PORTS_NUM 1 |
| 34 | |
| 35 | /*****************************************************************************/ |
| 36 | typedef struct MMAL_COMPONENT_MODULE_T |
| 37 | { |
| 38 | MMAL_BOOL_T error; /**< Error state */ |
| 39 | |
| 40 | } MMAL_COMPONENT_MODULE_T; |
| 41 | |
| 42 | typedef struct MMAL_PORT_MODULE_T |
| 43 | { |
| 44 | MMAL_QUEUE_T *queue; /**< queue for the buffers sent to the ports */ |
| 45 | |
| 46 | } MMAL_PORT_MODULE_T; |
| 47 | |
| 48 | /*****************************************************************************/ |
| 49 | |
| 50 | /** Destroy a previously created component */ |
| 51 | static MMAL_STATUS_T passthrough_component_destroy(MMAL_COMPONENT_T *component) |
| 52 | { |
| 53 | unsigned int i; |
| 54 | |
| 55 | for(i = 0; i < component->input_num; i++) |
| 56 | if(component->input[i]->priv->module->queue) |
| 57 | mmal_queue_destroy(component->input[i]->priv->module->queue); |
| 58 | if(component->input_num) |
| 59 | mmal_ports_free(component->input, component->input_num); |
| 60 | |
| 61 | for(i = 0; i < component->output_num; i++) |
| 62 | if(component->output[i]->priv->module->queue) |
| 63 | mmal_queue_destroy(component->output[i]->priv->module->queue); |
| 64 | if(component->output_num) |
| 65 | mmal_ports_free(component->output, component->output_num); |
| 66 | |
| 67 | vcos_free(component->priv->module); |
| 68 | return MMAL_SUCCESS; |
| 69 | } |
| 70 | |
| 71 | /** Enable processing on a port */ |
| 72 | static MMAL_STATUS_T passthrough_port_enable(MMAL_PORT_T *port, MMAL_PORT_BH_CB_T cb) |
| 73 | { |
| 74 | MMAL_PARAM_UNUSED(port); |
| 75 | MMAL_PARAM_UNUSED(cb); |
| 76 | return MMAL_SUCCESS; |
| 77 | } |
| 78 | |
| 79 | /** Flush a port */ |
| 80 | static MMAL_STATUS_T passthrough_port_flush(MMAL_PORT_T *port) |
| 81 | { |
| 82 | MMAL_PORT_MODULE_T *port_module = port->priv->module; |
| 83 | MMAL_BUFFER_HEADER_T *buffer; |
| 84 | |
| 85 | /* Flush buffers that our component is holding on to */ |
| 86 | buffer = mmal_queue_get(port_module->queue); |
| 87 | while(buffer) |
| 88 | { |
| 89 | mmal_port_buffer_header_callback(port, buffer); |
| 90 | buffer = mmal_queue_get(port_module->queue); |
| 91 | } |
| 92 | |
| 93 | return MMAL_SUCCESS; |
| 94 | } |
| 95 | |
| 96 | /** Disable processing on a port */ |
| 97 | static MMAL_STATUS_T passthrough_port_disable(MMAL_PORT_T *port) |
| 98 | { |
| 99 | /* We just need to flush our internal queue */ |
| 100 | return passthrough_port_flush(port); |
| 101 | } |
| 102 | |
| 103 | /** Send a buffer header to a port */ |
| 104 | static MMAL_STATUS_T passthrough_port_send(MMAL_PORT_T *port, MMAL_BUFFER_HEADER_T *buffer) |
| 105 | { |
| 106 | MMAL_COMPONENT_T *component = port->component; |
| 107 | MMAL_COMPONENT_MODULE_T *module = component->priv->module; |
| 108 | MMAL_PORT_T **other_port, *in_port, *out_port; |
| 109 | MMAL_BUFFER_HEADER_T **other_buffer, *in = 0, *out = 0; |
| 110 | MMAL_STATUS_T status; |
| 111 | |
| 112 | if (module->error) |
| 113 | { |
| 114 | mmal_queue_put(port->priv->module->queue, buffer); |
| 115 | return MMAL_SUCCESS; /* Just do nothing */ |
| 116 | } |
| 117 | |
| 118 | in_port = port->component->input[port->index]; |
| 119 | out_port = port->component->output[port->index]; |
| 120 | |
| 121 | if (port->type == MMAL_PORT_TYPE_INPUT) |
| 122 | { |
| 123 | other_port = &out_port; |
| 124 | other_buffer = &out; |
| 125 | in = buffer; |
| 126 | } |
| 127 | else |
| 128 | { |
| 129 | other_port = &in_port; |
| 130 | other_buffer = ∈ |
| 131 | out = buffer; |
| 132 | } |
| 133 | |
| 134 | /* Get a buffer header from the matching port */ |
| 135 | *other_buffer = mmal_queue_get((*other_port)->priv->module->queue); |
| 136 | if (!*other_buffer) |
| 137 | { |
| 138 | /* None available. Just queue the buffer header for now. */ |
| 139 | mmal_queue_put(port->priv->module->queue, buffer); |
| 140 | return MMAL_SUCCESS; |
| 141 | } |
| 142 | |
| 143 | /* Copy our input buffer header */ |
| 144 | status = mmal_buffer_header_replicate(out, in); |
| 145 | if (status != MMAL_SUCCESS) |
| 146 | goto error; |
| 147 | |
| 148 | /* Consume the input buffer */ |
| 149 | in->length = 0; |
| 150 | |
| 151 | /* Send buffers back */ |
| 152 | mmal_port_buffer_header_callback(in_port, in); |
| 153 | mmal_port_buffer_header_callback(out_port, out); |
| 154 | |
| 155 | return MMAL_SUCCESS; |
| 156 | |
| 157 | error: |
| 158 | mmal_queue_put(in_port->priv->module->queue, in); |
| 159 | mmal_queue_put(out_port->priv->module->queue, out); |
| 160 | status = mmal_event_error_send(port->component, status); |
| 161 | if (status != MMAL_SUCCESS) |
| 162 | { |
| 163 | LOG_ERROR("unable to send an error event buffer (%i)" , (int)status); |
| 164 | return MMAL_SUCCESS; |
| 165 | } |
| 166 | module->error = 1; |
| 167 | return MMAL_SUCCESS; |
| 168 | } |
| 169 | |
| 170 | /** Set format on a port */ |
| 171 | static MMAL_STATUS_T passthrough_port_format_commit(MMAL_PORT_T *port) |
| 172 | { |
| 173 | /* Sanity check */ |
| 174 | if (port->type == MMAL_PORT_TYPE_OUTPUT) |
| 175 | { |
| 176 | LOG_ERROR("output port is read-only" ); |
| 177 | return MMAL_EINVAL; |
| 178 | } |
| 179 | |
| 180 | return mmal_format_full_copy(port->component->output[port->index]->format, port->format); |
| 181 | } |
| 182 | |
| 183 | static MMAL_STATUS_T passthrough_port_parameter_set(MMAL_PORT_T *port, const MMAL_PARAMETER_HEADER_T *param) |
| 184 | { |
| 185 | MMAL_COMPONENT_T *component = port->component; |
| 186 | MMAL_PORT_T *in = component->input[port->index], *out = component->input[port->index]; |
| 187 | |
| 188 | switch (param->id) |
| 189 | { |
| 190 | case MMAL_PARAMETER_BUFFER_REQUIREMENTS: |
| 191 | { |
| 192 | /* Propagate the requirements to the matching input and output the ports */ |
| 193 | const MMAL_PARAMETER_BUFFER_REQUIREMENTS_T *req = (const MMAL_PARAMETER_BUFFER_REQUIREMENTS_T *)param; |
| 194 | uint32_t buffer_num_min = MMAL_MAX(port->buffer_num_min, req->buffer_num_min); |
| 195 | uint32_t buffer_num_recommended = MMAL_MAX(port->buffer_num_recommended, req->buffer_num_recommended); |
| 196 | uint32_t buffer_size_min = MMAL_MAX(port->buffer_size_min, req->buffer_size_min); |
| 197 | uint32_t buffer_size_recommended = MMAL_MAX(port->buffer_size_recommended, req->buffer_size_recommended); |
| 198 | |
| 199 | in->buffer_num_min = buffer_num_min; |
| 200 | in->buffer_num_recommended = buffer_num_recommended; |
| 201 | in->buffer_size_min = buffer_size_min; |
| 202 | in->buffer_size_recommended = buffer_size_recommended; |
| 203 | out->buffer_num_min = buffer_num_min; |
| 204 | out->buffer_num_recommended = buffer_num_recommended; |
| 205 | out->buffer_size_min = buffer_size_min; |
| 206 | out->buffer_size_recommended = buffer_size_recommended; |
| 207 | } |
| 208 | return MMAL_SUCCESS; |
| 209 | |
| 210 | default: |
| 211 | return MMAL_ENOSYS; |
| 212 | } |
| 213 | } |
| 214 | |
| 215 | /** Create an instance of a component */ |
| 216 | static MMAL_STATUS_T mmal_component_create_passthrough(const char *name, MMAL_COMPONENT_T *component) |
| 217 | { |
| 218 | MMAL_COMPONENT_MODULE_T *module; |
| 219 | MMAL_STATUS_T status = MMAL_ENOMEM; |
| 220 | unsigned int i; |
| 221 | MMAL_PARAM_UNUSED(name); |
| 222 | |
| 223 | /* Allocate the context for our module */ |
| 224 | component->priv->module = module = vcos_malloc(sizeof(*module), "mmal module" ); |
| 225 | if (!module) |
| 226 | return MMAL_ENOMEM; |
| 227 | memset(module, 0, sizeof(*module)); |
| 228 | |
| 229 | component->priv->pf_destroy = passthrough_component_destroy; |
| 230 | |
| 231 | /* Allocate and initialise all the ports for this component */ |
| 232 | component->input = mmal_ports_alloc(component, PASSTHROUGH_PORTS_NUM, |
| 233 | MMAL_PORT_TYPE_INPUT, sizeof(MMAL_PORT_MODULE_T)); |
| 234 | if(!component->input) |
| 235 | goto error; |
| 236 | component->input_num = PASSTHROUGH_PORTS_NUM; |
| 237 | for(i = 0; i < component->input_num; i++) |
| 238 | { |
| 239 | component->input[i]->priv->pf_enable = passthrough_port_enable; |
| 240 | component->input[i]->priv->pf_disable = passthrough_port_disable; |
| 241 | component->input[i]->priv->pf_flush = passthrough_port_flush; |
| 242 | component->input[i]->priv->pf_send = passthrough_port_send; |
| 243 | component->input[i]->priv->pf_set_format = passthrough_port_format_commit; |
| 244 | component->input[i]->priv->pf_parameter_set = passthrough_port_parameter_set; |
| 245 | component->input[i]->buffer_num_min = 1; |
| 246 | component->input[i]->buffer_num_recommended = 0; |
| 247 | component->input[i]->priv->module->queue = mmal_queue_create(); |
| 248 | if(!component->input[i]->priv->module->queue) |
| 249 | goto error; |
| 250 | } |
| 251 | |
| 252 | component->output = mmal_ports_alloc(component, PASSTHROUGH_PORTS_NUM, |
| 253 | MMAL_PORT_TYPE_OUTPUT, sizeof(MMAL_PORT_MODULE_T)); |
| 254 | if(!component->output) |
| 255 | goto error; |
| 256 | component->output_num = PASSTHROUGH_PORTS_NUM; |
| 257 | for(i = 0; i < component->output_num; i++) |
| 258 | { |
| 259 | component->output[i]->priv->pf_enable = passthrough_port_enable; |
| 260 | component->output[i]->priv->pf_disable = passthrough_port_disable; |
| 261 | component->output[i]->priv->pf_flush = passthrough_port_flush; |
| 262 | component->output[i]->priv->pf_send = passthrough_port_send; |
| 263 | component->output[i]->priv->pf_set_format = passthrough_port_format_commit; |
| 264 | component->output[i]->priv->pf_parameter_set = passthrough_port_parameter_set; |
| 265 | component->output[i]->buffer_num_min = 1; |
| 266 | component->output[i]->buffer_num_recommended = 0; |
| 267 | component->output[i]->capabilities = MMAL_PORT_CAPABILITY_PASSTHROUGH; |
| 268 | component->output[i]->priv->module->queue = mmal_queue_create(); |
| 269 | if(!component->output[i]->priv->module->queue) |
| 270 | goto error; |
| 271 | } |
| 272 | |
| 273 | return MMAL_SUCCESS; |
| 274 | |
| 275 | error: |
| 276 | passthrough_component_destroy(component); |
| 277 | return status; |
| 278 | } |
| 279 | |
| 280 | MMAL_CONSTRUCTOR(mmal_register_component_passthrough); |
| 281 | void mmal_register_component_passthrough(void) |
| 282 | { |
| 283 | mmal_component_supplier_register("passthrough" , mmal_component_create_passthrough); |
| 284 | } |
| 285 | |