/* 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 "mmalomx.h" #include "mmalomx_commands.h" #include "mmalomx_buffer.h" #include "mmalomx_logging.h" typedef struct { OMX_STATETYPE state; OMX_STATETYPE request; uint32_t actions; } MMALOMX_STATE_TRANSITION_T; MMALOMX_STATE_TRANSITION_T state_transition_table[] = { {OMX_StateInvalid, OMX_StateInvalid, 0}, {OMX_StateLoaded, OMX_StateIdle, MMALOMX_ACTION_CHECK_ALLOCATED|MMALOMX_ACTION_ENABLE}, {OMX_StateLoaded, OMX_StateWaitForResources, 0}, {OMX_StateWaitForResources, OMX_StateLoaded, 0}, {OMX_StateWaitForResources, OMX_StateIdle, MMALOMX_ACTION_CHECK_ALLOCATED|MMALOMX_ACTION_ENABLE}, {OMX_StateIdle, OMX_StateLoaded, MMALOMX_ACTION_CHECK_DEALLOCATED|MMALOMX_ACTION_DISABLE}, {OMX_StateIdle, OMX_StateExecuting, 0}, {OMX_StateIdle, OMX_StatePause, 0}, {OMX_StateExecuting, OMX_StateIdle, MMALOMX_ACTION_FLUSH|MMALOMX_ACTION_CHECK_FLUSHED}, {OMX_StateExecuting, OMX_StatePause, 0}, {OMX_StatePause, OMX_StateIdle, 0}, {OMX_StatePause, OMX_StateExecuting, 0}, {OMX_StateMax, OMX_StateMax, 0} }; /*****************************************************************************/ static unsigned int mmalomx_state_transition_get(OMX_STATETYPE state, OMX_STATETYPE request) { unsigned int i; for (i = 0; state_transition_table[i].state != OMX_StateMax; i++) if (state_transition_table[i].state == state && state_transition_table[i].request == request) break; return state_transition_table[i].state != OMX_StateMax ? i : 0; } /*****************************************************************************/ static void mmalomx_buffer_cb_io(MMAL_PORT_T *port, MMAL_BUFFER_HEADER_T *buffer) { mmalomx_buffer_return((MMALOMX_PORT_T *)port->userdata, buffer); } /*****************************************************************************/ static void mmalomx_commands_check_port_actions(MMALOMX_COMPONENT_T *component, MMALOMX_PORT_T *port) { uint32_t exec_actions = 0; MMAL_STATUS_T status; MMALOMX_LOCK_PORT(component, port); if (!port->actions) { MMALOMX_UNLOCK_PORT(component, port); return; } if (port->actions & MMALOMX_ACTION_FLUSH) { port->actions &= ~MMALOMX_ACTION_FLUSH; port->actions |= MMALOMX_ACTION_PENDING_FLUSH; exec_actions |= MMALOMX_ACTION_PENDING_FLUSH; } if ((port->actions & MMALOMX_ACTION_DISABLE) && (!port->buffers_in_transit || !(port->actions & MMALOMX_ACTION_CHECK_FLUSHED))) { port->actions &= ~MMALOMX_ACTION_DISABLE; port->actions |= MMALOMX_ACTION_PENDING_DISABLE; exec_actions |= MMALOMX_ACTION_PENDING_DISABLE; } if ((port->actions & MMALOMX_ACTION_ENABLE) && port->buffers) { /* We defer enabling the mmal port until the first buffer allocation * has been done. Only at that point do we know for sure whether we * are going to use shared memory or not. * We might want to delay it to just before sending the event to the client ??? */ port->actions &= ~MMALOMX_ACTION_ENABLE; port->actions |= MMALOMX_ACTION_PENDING_ENABLE; exec_actions |= MMALOMX_ACTION_PENDING_ENABLE; } MMALOMX_UNLOCK_PORT(component, port); if (exec_actions & MMALOMX_ACTION_PENDING_FLUSH) mmal_port_flush(port->mmal); if (exec_actions & MMALOMX_ACTION_PENDING_DISABLE) { mmal_port_disable(port->mmal); /* If there was a port format changed event, we need to make sure * the new format has been committed */ if (port->format_changed) { status = mmal_port_format_commit(port->mmal); if (status != MMAL_SUCCESS) LOG_WARN("could not commit new format (%i)", status); port->format_changed = MMAL_FALSE; } } if (exec_actions & MMALOMX_ACTION_PENDING_ENABLE) { status = mmal_port_enable(port->mmal, mmalomx_buffer_cb_io); if (status == MMAL_SUCCESS) status = mmal_pool_resize(port->pool, port->mmal->buffer_num, 0); if (status != MMAL_SUCCESS) mmalomx_callback_event_handler(component, OMX_EventError, mmalil_error_to_omx(status), 0, NULL); /* FIXME: we're still going to generate a cmd complete. Not sure if that's an issue. */ } MMALOMX_LOCK_PORT(component, port); port->actions &= ~exec_actions; if ((port->actions & MMALOMX_ACTION_CHECK_ALLOCATED) && port->populated) port->actions &= ~MMALOMX_ACTION_CHECK_ALLOCATED; if ((port->actions & MMALOMX_ACTION_CHECK_DEALLOCATED) && !port->buffers) port->actions &= ~MMALOMX_ACTION_CHECK_DEALLOCATED; if ((port->actions & MMALOMX_ACTION_CHECK_FLUSHED) && !port->buffers_in_transit) port->actions &= ~MMALOMX_ACTION_CHECK_FLUSHED; exec_actions = port->actions; if (port->actions == MMALOMX_ACTION_NOTIFY_FLUSH || port->actions == MMALOMX_ACTION_NOTIFY_ENABLE || port->actions == MMALOMX_ACTION_NOTIFY_DISABLE) port->actions = 0; /* We're done */ MMALOMX_UNLOCK_PORT(component, port); if (exec_actions == MMALOMX_ACTION_NOTIFY_FLUSH) mmalomx_callback_event_handler(component, OMX_EventCmdComplete, OMX_CommandFlush, port->index, NULL); else if (exec_actions == MMALOMX_ACTION_NOTIFY_ENABLE) mmalomx_callback_event_handler(component, OMX_EventCmdComplete, OMX_CommandPortEnable, port->index, NULL); else if (exec_actions == MMALOMX_ACTION_NOTIFY_DISABLE) mmalomx_callback_event_handler(component, OMX_EventCmdComplete, OMX_CommandPortDisable, port->index, NULL); } /*****************************************************************************/ void mmalomx_commands_actions_check(MMALOMX_COMPONENT_T *component) { uint32_t actions_left = 0; unsigned int i; for (i = 0; i < component->ports_num; i++) mmalomx_commands_check_port_actions(component, &component->ports[i]); MMALOMX_LOCK(component); for (i = 0; i < component->ports_num; i++) actions_left |= component->ports[i].actions; if (!actions_left && component->state_transition) { component->state = state_transition_table[component->state_transition].request; component->state_transition = 0; actions_left = MMALOMX_ACTION_NOTIFY_STATE; } MMALOMX_UNLOCK(component); if (actions_left == MMALOMX_ACTION_NOTIFY_STATE) { mmalomx_callback_event_handler(component, OMX_EventCmdComplete, OMX_CommandStateSet, component->state, NULL); actions_left = 0; } /* If we're not currently processing a command, we can start processing * the next one. */ if (!actions_left) mmalomx_commands_actions_next(component); } /*****************************************************************************/ void mmalomx_commands_actions_signal(MMALOMX_COMPONENT_T *component) { if (component->cmd_thread_used) vcos_semaphore_post(&component->cmd_sema); else mmalomx_commands_actions_check(component); } /*****************************************************************************/ OMX_ERRORTYPE mmalomx_command_state_set( OMX_HANDLETYPE hComponent, OMX_STATETYPE state) { MMALOMX_COMPONENT_T *component = (MMALOMX_COMPONENT_T *)hComponent; unsigned int i, transition; if (component->state == state) { mmalomx_callback_event_handler(component, OMX_EventError, OMX_ErrorSameState, 0, NULL); return OMX_ErrorNone; } /* We're asked to transition to StateInvalid */ if (state == OMX_StateInvalid) { component->state = state; mmalomx_callback_event_handler(component, OMX_EventError, OMX_ErrorInvalidState, 0, NULL); return OMX_ErrorNone; } /* Commands are being queued so we should never get into that state */ vcos_assert(!component->state_transition); /* Check the transition is valid */ transition = mmalomx_state_transition_get(component->state, state); if (!transition) { mmalomx_callback_event_handler(component, OMX_EventError, OMX_ErrorIncorrectStateTransition, 0, NULL); return OMX_ErrorNone; } /* Special case for transition in and out of Executing */ if (state == OMX_StateExecuting || component->state == OMX_StateExecuting) { MMAL_STATUS_T status; if (state == OMX_StateExecuting) status = mmal_component_enable(component->mmal); else status = mmal_component_disable(component->mmal); if (status != MMAL_SUCCESS) { LOG_ERROR("could not %s %s", state == OMX_StateExecuting ? "enable" : "disable", component->name); mmalomx_callback_event_handler(component, OMX_EventError, mmalil_error_to_omx(status), 0, NULL); return OMX_ErrorNone; } } MMALOMX_LOCK(component); component->state_transition = transition; for (i = 0; i < component->ports_num; i++) { if (!component->ports[i].enabled) continue; MMALOMX_LOCK_PORT(component, component->ports + i); component->ports[i].actions = state_transition_table[transition].actions; /* If we're transitionning from Idle to Loaded we'd rather do a flush first * to avoid the cmd thread to block for too long (mmal_disable is a * blocking call). */ if (state_transition_table[transition].state == OMX_StateIdle && state_transition_table[transition].request == OMX_StateLoaded && component->cmd_thread_used) component->ports[i].actions |= MMALOMX_ACTION_FLUSH|MMALOMX_ACTION_CHECK_FLUSHED; MMALOMX_UNLOCK_PORT(component, component->ports + i); } MMALOMX_UNLOCK(component); mmalomx_commands_actions_check(component); return OMX_ErrorNone; } /*****************************************************************************/ OMX_ERRORTYPE mmalomx_command_port_mark( OMX_HANDLETYPE hComponent, OMX_U32 nPortIndex, OMX_PTR *pCmdData) { MMALOMX_COMPONENT_T *component = (MMALOMX_COMPONENT_T *)hComponent; OMX_MARKTYPE *mark = (OMX_MARKTYPE *)pCmdData; MMALOMX_PORT_T *port; if (nPortIndex >= component->ports_num) return OMX_ErrorBadPortIndex; port = &component->ports[nPortIndex]; if (port->marks_num == MAX_MARKS_NUM) return OMX_ErrorInsufficientResources; port->marks[(port->marks_first + port->marks_num) % MAX_MARKS_NUM] = *mark; port->marks_num++; return OMX_ErrorNone; } /*****************************************************************************/ OMX_ERRORTYPE mmalomx_command_port_flush( OMX_HANDLETYPE hComponent, OMX_U32 nPortIndex) { MMALOMX_COMPONENT_T *component = (MMALOMX_COMPONENT_T *)hComponent; MMALOMX_LOCK_PORT(component, &component->ports[nPortIndex]); component->ports[nPortIndex].actions = MMALOMX_ACTION_FLUSH|MMALOMX_ACTION_CHECK_FLUSHED|MMALOMX_ACTION_NOTIFY_FLUSH; MMALOMX_UNLOCK_PORT(component, &component->ports[nPortIndex]); mmalomx_commands_actions_check(component); return OMX_ErrorNone; } /*****************************************************************************/ OMX_ERRORTYPE mmalomx_command_port_enable( OMX_HANDLETYPE hComponent, OMX_U32 nPortIndex) { MMALOMX_COMPONENT_T *component = (MMALOMX_COMPONENT_T *)hComponent; component->ports[nPortIndex].enabled = MMAL_TRUE; if (component->state == OMX_StateLoaded || component->state == OMX_StateWaitForResources) { mmalomx_callback_event_handler(component, OMX_EventCmdComplete, OMX_CommandPortEnable, nPortIndex, NULL); return OMX_ErrorNone; } MMALOMX_LOCK_PORT(component, &component->ports[nPortIndex]); component->ports[nPortIndex].actions = MMALOMX_ACTION_CHECK_ALLOCATED|MMALOMX_ACTION_ENABLE|MMALOMX_ACTION_NOTIFY_ENABLE; MMALOMX_UNLOCK_PORT(component, &component->ports[nPortIndex]); mmalomx_commands_actions_check(component); return OMX_ErrorNone; } /*****************************************************************************/ OMX_ERRORTYPE mmalomx_command_port_disable( OMX_HANDLETYPE hComponent, OMX_U32 nPortIndex) { MMALOMX_COMPONENT_T *component = (MMALOMX_COMPONENT_T *)hComponent; component->ports[nPortIndex].enabled = MMAL_FALSE; if (component->state == OMX_StateLoaded || component->state == OMX_StateWaitForResources) { mmalomx_callback_event_handler(component, OMX_EventCmdComplete, OMX_CommandPortDisable, nPortIndex, NULL); return OMX_ErrorNone; } MMALOMX_LOCK_PORT(component, &component->ports[nPortIndex]); component->ports[nPortIndex].actions = MMALOMX_ACTION_DISABLE|MMALOMX_ACTION_CHECK_DEALLOCATED|MMALOMX_ACTION_NOTIFY_DISABLE; if (component->cmd_thread_used) component->ports[nPortIndex].actions |= MMALOMX_ACTION_FLUSH|MMALOMX_ACTION_CHECK_FLUSHED; MMALOMX_UNLOCK_PORT(component, &component->ports[nPortIndex]); mmalomx_commands_actions_check(component); return OMX_ErrorNone; } /*****************************************************************************/ OMX_ERRORTYPE mmalomx_command_queue( MMALOMX_COMPONENT_T *component, OMX_U32 arg1, OMX_U32 arg2) { MMAL_BUFFER_HEADER_T *cmd = mmal_queue_get(component->cmd_pool->queue); if (!vcos_verify(cmd)) { LOG_ERROR("command queue too small"); return OMX_ErrorInsufficientResources; } cmd->cmd = arg1; cmd->offset = arg2; mmal_queue_put(component->cmd_queue, cmd); mmalomx_commands_actions_signal(component); return OMX_ErrorNone; } /*****************************************************************************/ OMX_ERRORTYPE mmalomx_command_dequeue( MMALOMX_COMPONENT_T *component, OMX_U32 *arg1, OMX_U32 *arg2) { MMAL_BUFFER_HEADER_T *cmd = mmal_queue_get(component->cmd_queue); if (!cmd) return OMX_ErrorNoMore; *arg1 = cmd->cmd; *arg2 = cmd->offset; mmal_buffer_header_release(cmd); return OMX_ErrorNone; } /*****************************************************************************/ void mmalomx_commands_actions_next(MMALOMX_COMPONENT_T *component) { OMX_ERRORTYPE status = OMX_ErrorNone; OMX_COMMANDTYPE cmd; OMX_U32 arg1, arg2, nParam1; unsigned int i; status = mmalomx_command_dequeue(component, &arg1, &arg2); if (status != OMX_ErrorNone) return; cmd = (OMX_COMMANDTYPE)arg1; nParam1 = arg2; if (cmd == OMX_CommandStateSet) { mmalomx_command_state_set((OMX_HANDLETYPE)&component->omx, nParam1); } else if (cmd == OMX_CommandFlush) { for (i = 0; i < component->ports_num; i++) if (i == nParam1 || nParam1 == OMX_ALL) mmalomx_command_port_flush((OMX_HANDLETYPE)&component->omx, i); } else if (cmd == OMX_CommandPortEnable) { for (i = 0; i < component->ports_num; i++) if (i == nParam1 || nParam1 == OMX_ALL) mmalomx_command_port_enable((OMX_HANDLETYPE)&component->omx, i); } else if (cmd == OMX_CommandPortDisable) { for (i = 0; i < component->ports_num; i++) if (i == nParam1 || nParam1 == OMX_ALL) mmalomx_command_port_disable((OMX_HANDLETYPE)&component->omx, i); } }