FreeCalypso > hg > fc-tourmaline
diff src/cs/services/audio/audio_driver.c @ 0:4e78acac3d88
src/{condat,cs,gpf,nucleus}: import from Selenite
author | Mychaela Falconia <falcon@freecalypso.org> |
---|---|
date | Fri, 16 Oct 2020 06:23:26 +0000 |
parents | |
children |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/cs/services/audio/audio_driver.c Fri Oct 16 06:23:26 2020 +0000 @@ -0,0 +1,810 @@ +/****************************************************************************/ +/* */ +/* File Name: audio_driver.c */ +/* */ +/* Purpose: This file contains all the functions used to manage the */ +/* driver. */ +/* */ +/* Version 0.1 */ +/* */ +/* Date Modification */ +/* ------------------------------------ */ +/* 09 December 2002 Create */ +/* */ +/* Author Frederic Turgis */ +/* */ +/* (C) Copyright 2002 by Texas Instruments Incorporated, All Rights Reserved*/ +/****************************************************************************/ + +#include "rv/rv_defined_swe.h" +#ifdef RVM_AUDIO_MAIN_SWE +#ifndef _WINDOWS + #include "config/swconfig.cfg" + #include "config/sys.cfg" + #include "config/chipset.cfg" +#endif + + #include "l1_confg.h" + #include "rvf/rvf_api.h" + #include "rv/rv_general.h" + #include "rvm/rvm_gen.h" + #include "audio/audio_features_i.h" + #include "audio/audio_api.h" + #include "audio/audio_env_i.h" + #include "audio/audio_ffs_i.h" + #include "audio/audio_structs_i.h" + #include "audio/audio_macro_i.h" + #include "rvf/rvf_target.h" + #include "audio/audio_const_i.h" + #include "audio/audio_var_i.h" + #include "audio/audio_error_hdlr_i.h" + #include "audio/audio_messages_i.h" + + #ifndef _WINDOWS + // include the usefull L1 header + #define BOOL_FLAG + #define CHAR_FLAG + #include "l1_types.h" + #include "cust_os.h" + #include "l1audio_cust.h" + #include "l1audio_msgty.h" + #include "l1audio_signa.h" + #include "l1_signa.h" + #else + // include the usefull L1 header + #define BOOL_FLAG + #define CHAR_FLAG + #include "l1_types.h" + #include "l1audio_const.h" + #include "l1audio_cust.h" + #include "l1audio_defty.h" + #include "l1audio_msgty.h" + #include "l1audio_signa.h" + #include "l1_const.h" + #include "l1_defty.h" + #include "l1_msgty.h" + #include "l1_signa.h" + #include "l1_varex.h" + #include "audio/tests/audio_test.h" + #endif + +#if (L1_AUDIO_DRIVER) + #if (L1_VOICE_MEMO_AMR) + extern T_AUDIO_RET audio_driver_vm_amr_play_manager (T_RV_HDR *p_message, T_AUDIO_DRIVER_SESSION *p_session); + extern T_AUDIO_RET audio_driver_vm_amr_record_manager (T_RV_HDR *p_message, T_AUDIO_DRIVER_SESSION *p_session); + extern T_AUDIO_RET audio_driver_midi_manager (T_RV_HDR *p_message, T_AUDIO_DRIVER_SESSION *p_session); + #endif + + UINT8 audio_driver_message_switch(T_RV_HDR *p_message); + void audio_driver_send_status (T_AUDIO_RET status, + UINT8 status_type, + UINT8 channel_id, + T_RV_RETURN return_path); + void audio_driver_init_send_status (T_AUDIO_RET status, + UINT8 session_id, + UINT8 channel_id, + T_RV_RETURN return_path); + void audio_driver_manager(T_RV_HDR *p_message); + T_AUDIO_RET audio_driver_get_play_buffer(UINT8 channel_id, UINT8 **pp_buffer); + T_AUDIO_RET audio_driver_play_buffer(UINT8 channel_id, UINT8 *p_buffer); + + UINT8 audio_driver_message_switch(T_RV_HDR *p_message) + { + switch (p_message->msg_id) + { + case AUDIO_DRIVER_START_SESSION: + case AUDIO_DRIVER_STOP_SESSION: + case AUDIO_DRIVER_FREE_SESSION: + return(AUDIO_DRIVER_SWITCH); + break; + #if (L1_VOICE_MEMO_AMR) + case AUDIO_DRIVER_INIT_VM_AMR_RECORD_SESSION: + case MMI_VM_AMR_RECORD_START_CON: + case MMI_VM_AMR_RECORD_STOP_CON: + return(AUDIO_DRIVER_VM_AMR_RECORD_SESSION_SWITCH); + break; + case AUDIO_DRIVER_INIT_VM_AMR_PLAY_SESSION: + case MMI_VM_AMR_PLAY_START_CON: + case MMI_VM_AMR_PLAY_STOP_CON: + return(AUDIO_DRIVER_VM_AMR_PLAY_SESSION_SWITCH); + break; + #endif + #if (L1_MIDI == 1) + case AUDIO_DRIVER_INIT_MIDI_SESSION: + case MMI_MIDI_START_CON: + case MMI_MIDI_STOP_CON: + return(AUDIO_DRIVER_MIDI_SESSION_SWITCH); + break; + #endif + default: + return(AUDIO_DRIVER_NONE); + break; + } // switch + } + + void audio_driver_send_status (T_AUDIO_RET status, + UINT8 status_type, + UINT8 channel_id, + T_RV_RETURN return_path) + { + T_AUDIO_DRIVER_STATUS *p_send_message; + T_RVF_MB_STATUS mb_status = RVF_RED; + + while (mb_status == RVF_RED) + { + // allocate the message buffer + mb_status = rvf_get_buf (p_audio_gbl_var->mb_external, + sizeof (T_AUDIO_DRIVER_STATUS), + (T_RVF_BUFFER **) (&p_send_message)); + + // If insufficient resources, then report a memory error and abort. + // and wait until more ressource is given + if (mb_status == RVF_RED) + { + audio_driver_error_trace(AUDIO_ENTITY_NO_MEMORY); + rvf_delay(RVF_MS_TO_TICKS(1000)); + } + } + + // fill the header of the message + parameters + p_send_message->os_hdr.msg_id = AUDIO_DRIVER_STATUS_MSG; + p_send_message->status = status; + p_send_message->status_type = status_type; + p_send_message->channel_id = channel_id; + + // send message or call callback + if (return_path.callback_func == NULL) + rvf_send_msg (return_path.addr_id, p_send_message); + else + { + (*return_path.callback_func)((void *)(p_send_message)); + rvf_free_buf((T_RVF_BUFFER *)p_send_message); + } + } + + void audio_driver_init_send_status (T_AUDIO_RET status, + UINT8 session_id, + UINT8 channel_id, + T_RV_RETURN return_path) + { + T_AUDIO_DRIVER_INIT_STATUS *p_send_message; + T_RVF_MB_STATUS mb_status = RVF_RED; + + while (mb_status == RVF_RED) + { + // allocate the message buffer + mb_status = rvf_get_buf (p_audio_gbl_var->mb_external, + sizeof (T_AUDIO_DRIVER_INIT_STATUS), + (T_RVF_BUFFER **) (&p_send_message)); + + // If insufficient resources, then report a memory error and abort. + // and wait until more ressource is given + if (mb_status == RVF_RED) + { + audio_driver_error_trace(AUDIO_ENTITY_NO_MEMORY); + rvf_delay(RVF_MS_TO_TICKS(1000)); + } + } + + // fill the header of the message + p_send_message->os_hdr.msg_id = AUDIO_DRIVER_INIT_STATUS_MSG; + + // fill the status parameters + p_send_message->status = status; + p_send_message->session_id = session_id; + p_send_message->channel_id = channel_id; + + // send message or call callback + if (return_path.callback_func == NULL) + rvf_send_msg (return_path.addr_id, p_send_message); + else + { + (*return_path.callback_func)((void *)(p_send_message)); + rvf_free_buf((T_RVF_BUFFER *)p_send_message); + } + } + + void audio_driver_manager(T_RV_HDR *p_message) + { + T_AUDIO_DRIVER_SESSION *p_session; + T_AUDIO_DRIVER_PARAMETER *driver_parameter; + T_RV_RETURN return_path; + T_RVF_MB_STATUS mb_status; + UINT8 session_id, channel_id, state, j; + + // initialize return_path to default values + return_path.callback_func=NULL; + return_path.addr_id=0; + + // find state + extract information:channel_id or session_id + driver_parameter + switch (p_message->msg_id) + { + // start/stop session messages have channel_id so we know the driver_session + case AUDIO_DRIVER_START_SESSION: + case AUDIO_DRIVER_STOP_SESSION: + case AUDIO_DRIVER_FREE_SESSION: + channel_id = ((T_AUDIO_DRIVER_HANDLE_SESSION *)p_message)->channel_id; + p_session = &(p_audio_gbl_var->audio_driver_session[channel_id]); + state = p_session->session_info.state; + if (p_message->msg_id != AUDIO_DRIVER_STOP_SESSION) + return_path = ((T_AUDIO_DRIVER_HANDLE_SESSION *)p_message)->return_path; + else + return_path = p_session->session_req.return_path; + break; + // messages which contain only session_id, must look for it in ACTIVE driver_session + default: + { + switch (p_message->msg_id) + { + #if (L1_VOICE_MEMO_AMR) + case AUDIO_DRIVER_INIT_VM_AMR_RECORD_SESSION: + session_id = AUDIO_VM_AMR_RECORD_SESSION_ID; + driver_parameter = &(((T_AUDIO_DRIVER_INIT_VM_AMR_RECORD_SESSION *)p_message)->driver_parameter); + return_path = ((T_AUDIO_DRIVER_INIT_VM_AMR_RECORD_SESSION *)p_message)->return_path; + break; + case AUDIO_DRIVER_INIT_VM_AMR_PLAY_SESSION: + session_id = AUDIO_VM_AMR_PLAY_SESSION_ID; + driver_parameter = & (((T_AUDIO_DRIVER_INIT_VM_AMR_PLAY_SESSION *)p_message)->driver_parameter); + return_path = ((T_AUDIO_DRIVER_INIT_VM_AMR_PLAY_SESSION *)p_message)->return_path; + break; + case MMI_VM_AMR_RECORD_START_CON: + case MMI_VM_AMR_RECORD_STOP_CON: + session_id = AUDIO_VM_AMR_RECORD_SESSION_ID; + break; + case MMI_VM_AMR_PLAY_START_CON: + case MMI_VM_AMR_PLAY_STOP_CON: + session_id = AUDIO_VM_AMR_PLAY_SESSION_ID; + break; + #endif + #if (L1_MIDI == 1) + case AUDIO_DRIVER_INIT_MIDI_SESSION: + session_id = AUDIO_MIDI_SESSION_ID; + driver_parameter = &(((T_AUDIO_DRIVER_INIT_MIDI_SESSION *)p_message)->driver_parameter); + return_path = ((T_AUDIO_DRIVER_INIT_MIDI_SESSION *)p_message)->return_path; + break; + case MMI_MIDI_START_CON: + case MMI_MIDI_STOP_CON: + session_id = AUDIO_MIDI_SESSION_ID; + break; + #endif + } + + // initialize channel_id to browse all driver channels + channel_id = 0; + state = AUDIO_DRIVER_CHANNEL_WAIT_INIT; + + // look for an active session, which session_id matches the one from the message + while ( (channel_id < AUDIO_DRIVER_MAX_CHANNEL)&& + ((p_audio_gbl_var->audio_driver_session[channel_id].session_info.state == AUDIO_DRIVER_CHANNEL_WAIT_INIT)|| + (p_audio_gbl_var->audio_driver_session[channel_id].session_req.session_id != session_id)) ) + { + channel_id++; + } + // if channel_id < MAX_CHANNEL, we found an active channel so we can derive channel_id + state + if (channel_id < AUDIO_DRIVER_MAX_CHANNEL) + { + p_session = &(p_audio_gbl_var->audio_driver_session[channel_id]); + state = p_session->session_info.state; + return_path = p_session->session_req.return_path; + } + } + break; // default + } + + switch (state) + { + case AUDIO_DRIVER_CHANNEL_WAIT_INIT: + { + switch (p_message->msg_id) + { + #if (L1_VOICE_MEMO_AMR) + case AUDIO_DRIVER_INIT_VM_AMR_RECORD_SESSION: + case AUDIO_DRIVER_INIT_VM_AMR_PLAY_SESSION: + #endif + #if (L1_MIDI == 1) + case AUDIO_DRIVER_INIT_MIDI_SESSION: + #endif + { + T_AUDIO_RET result; + + // Find a free channel + channel_id = 0; + while ( (p_audio_gbl_var->audio_driver_session[channel_id].session_info.state != AUDIO_DRIVER_CHANNEL_WAIT_INIT) && + (channel_id < AUDIO_DRIVER_MAX_CHANNEL) ) + channel_id++; + + if (channel_id == AUDIO_DRIVER_MAX_CHANNEL) + { + AUDIO_SEND_TRACE("no driver channel available", RV_TRACE_LEVEL_DEBUG_LOW); + audio_driver_init_send_status(AUDIO_ERROR, session_id, channel_id, return_path); + return; + } + + AUDIO_SEND_TRACE_PARAM("AUDIO DRIVER: open channel", channel_id, RV_TRACE_LEVEL_DEBUG_LOW); + + // get session descriptor and fill REQ parameters + p_session = &(p_audio_gbl_var->audio_driver_session[channel_id]); + p_session->session_req.session_id = session_id; + p_session->session_req.size = driver_parameter->buffer_size << 1;// we request 16-bit words buffers + p_session->session_req.nb_buffer = driver_parameter->nb_buffer; + // fill return_path parameters, may be used if next message is STOP_SESSION + p_session->session_req.return_path.callback_func = return_path.callback_func; + p_session->session_req.return_path.addr_id = return_path.addr_id; + + /************************************************************/ + /* the driver must allocate the RAM buffers pointer */ + /************************************************************/ + mb_status = rvf_get_buf (p_audio_gbl_var->mb_internal, + sizeof(T_AUDIO_DRIVER_BUFFER_INFO)*p_session->session_req.nb_buffer, + (T_RVF_BUFFER **) (&p_session->session_info.buffer)); + + // If insufficient resources, then report a memory error and abort. + if (mb_status == RVF_RED) + { + audio_driver_error_trace(AUDIO_ENTITY_NO_MEMORY); + audio_driver_init_send_status(AUDIO_ERROR, session_id, channel_id, return_path); + return; + } + + /************************************************************/ + /* the driver must allocate the RAM buffers */ + /************************************************************/ + for (j = 0; j < p_session->session_req.nb_buffer; j++) + { + mb_status = rvf_get_buf (p_audio_gbl_var->mb_audio_ffs, + p_session->session_req.size, + (T_RVF_BUFFER **) (&p_session->session_info.buffer[j].p_start_pointer)); + + // If insufficient resources, then report a memory error and abort. + if (mb_status == RVF_RED) + { + UINT8 i; + // free already allocated buffers + buffer pointer + if (j > 0) + { + for (i = j - 1; i >= 0; i--) + rvf_free_buf((T_RVF_BUFFER *)p_session->session_info.buffer[i].p_start_pointer); + } + rvf_free_buf((T_RVF_BUFFER *)p_session->session_info.buffer); + audio_driver_error_trace(AUDIO_ENTITY_NO_MEMORY); + audio_driver_init_send_status(AUDIO_ERROR, session_id, channel_id, return_path); + return; + } + + // initialize parameters + p_session->session_info.buffer[j].size = p_session->session_req.size; + + AUDIO_SEND_TRACE_PARAM("AUDIO DRIVER: allocate buffer", + p_session->session_info.buffer[j].p_start_pointer, RV_TRACE_LEVEL_DEBUG_LOW); + } + + // info parameters (state must be changed before driver specific functions call + p_session->session_info.index_l1 = 0; + p_session->session_info.index_appli = 0; + p_session->session_info.stop_request = 0; + + // conversion of parameters + prepare l1 start message + switch(p_message->msg_id) + { + #if (L1_VOICE_MEMO_AMR) + case AUDIO_DRIVER_INIT_VM_AMR_RECORD_SESSION: + result=audio_driver_vm_amr_record_manager(p_message,p_session); + break; + case AUDIO_DRIVER_INIT_VM_AMR_PLAY_SESSION: + result=audio_driver_vm_amr_play_manager(p_message,p_session); + break; + #endif + #if (L1_MIDI == 1) + case AUDIO_DRIVER_INIT_MIDI_SESSION: + result=audio_driver_midi_manager(p_message,p_session); + break; + #endif + } + + // check L1 msg allocation was successfull + if (result != AUDIO_OK) + { + AUDIO_SEND_TRACE("AUDIO DRIVER MANAGER: L1 msg allocation failed", RV_TRACE_LEVEL_DEBUG_LOW); + audio_driver_init_send_status(AUDIO_ERROR, session_id, channel_id, return_path); + return; + } + + // state (must be changed after driver specific functions calls) + p_session->session_info.state = AUDIO_DRIVER_CHANNEL_WAIT_START; + + audio_driver_init_send_status(AUDIO_OK, session_id, channel_id, return_path); + } //case AUDIO_DRIVER_INIT_..._SESSION: + break; + case AUDIO_DRIVER_START_SESSION: + audio_driver_error_trace(AUDIO_ERROR_START_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_START_STATUS, channel_id, return_path); + break; + case AUDIO_DRIVER_STOP_SESSION: + audio_driver_error_trace(AUDIO_ERROR_STOP_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_STOP_STATUS, channel_id, return_path); + break; + case AUDIO_DRIVER_FREE_SESSION: + audio_driver_error_trace(AUDIO_ERROR_FREE_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_FREE_STATUS, channel_id, return_path); + break; + } // switch (p_message->msg_id) + } //case AUDIO_DRIVER_CHANNEL_WAIT_INIT: + break; + case AUDIO_DRIVER_CHANNEL_WAIT_START: + { + switch (p_message->msg_id) + { + case AUDIO_DRIVER_START_SESSION: + { + T_AUDIO_RET result; + + // fill notification parameters + p_session->session_req.return_path.callback_func = return_path.callback_func; + p_session->session_req.return_path.addr_id = return_path.addr_id; + + // send l1 message + switch(p_session->session_req.session_id) + { + #if (L1_VOICE_MEMO_AMR) + case AUDIO_VM_AMR_RECORD_SESSION_ID: + result=audio_driver_vm_amr_record_manager(p_message,p_session); + break; + case AUDIO_VM_AMR_PLAY_SESSION_ID: + result=audio_driver_vm_amr_play_manager(p_message,p_session); + break; + #endif + #if (L1_MIDI == 1) + case AUDIO_MIDI_SESSION_ID: + result=audio_driver_midi_manager(p_message,p_session); + break; + #endif + } + + if (result != AUDIO_OK) + { + audio_driver_error_trace(AUDIO_ERROR_L1_START_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_START_STATUS, channel_id, return_path); + return; + } + + // change state + p_session->session_info.state = AUDIO_DRIVER_CHANNEL_WAIT_START_CON; + } + break; + case AUDIO_DRIVER_STOP_SESSION: + { + // deallocate buffers + for(j=0; j<p_session->session_req.nb_buffer; j++) + { + mb_status=rvf_free_buf((T_RVF_BUFFER *)(p_session->session_info.buffer[j].p_start_pointer)); + + AUDIO_SEND_TRACE_PARAM("AUDIO DRIVER MANAGER: deallocate buffer", + j, RV_TRACE_LEVEL_DEBUG_LOW); + + if (mb_status != RVF_GREEN) + AUDIO_SEND_TRACE_PARAM("can't deallocate buffer", j, RV_TRACE_LEVEL_ERROR); + } + + // Deallocate buffers pointer + rvf_free_buf((T_RVF_BUFFER *)p_session->session_info.buffer); + + // deallocate l1 message + audio_deallocate_l1_message(p_session->session_req.p_l1_send_message); + + // send status OK + audio_driver_send_status(AUDIO_OK,AUDIO_STOP_STATUS,channel_id,return_path); + + // change state + p_session->session_info.state=AUDIO_DRIVER_CHANNEL_WAIT_INIT; + } + break; + #if (L1_VOICE_MEMO_AMR) + case AUDIO_DRIVER_INIT_VM_AMR_RECORD_SESSION: + case AUDIO_DRIVER_INIT_VM_AMR_PLAY_SESSION: + #endif + #if (L1_MIDI == 1) + case AUDIO_DRIVER_INIT_MIDI_SESSION: + #endif + audio_driver_error_trace(AUDIO_ERROR_INIT_EVENT); + audio_driver_init_send_status(AUDIO_ERROR, session_id, 0, return_path); + break; + case AUDIO_DRIVER_FREE_SESSION: + audio_driver_error_trace(AUDIO_ERROR_FREE_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_FREE_STATUS, channel_id, return_path); + break; + } // switch + } //case AUDIO_DRIVER_CHANNEL_WAIT_START: + break; + case AUDIO_DRIVER_CHANNEL_WAIT_START_CON: + { + switch (p_message->msg_id) + { + #if (L1_VOICE_MEMO_AMR) + case MMI_VM_AMR_RECORD_START_CON: + case MMI_VM_AMR_PLAY_START_CON: + #endif + #if (L1_MIDI == 1) + case MMI_MIDI_START_CON: + #endif + // change state + p_session->session_info.state = AUDIO_DRIVER_CHANNEL_WAIT_STOP; + break; + case AUDIO_DRIVER_STOP_SESSION: + // change state + p_session->session_info.state = AUDIO_DRIVER_CHANNEL_WAIT_START_CON_TO_STOP; + break; + #if (L1_VOICE_MEMO_AMR) + case AUDIO_DRIVER_INIT_VM_AMR_RECORD_SESSION: + case AUDIO_DRIVER_INIT_VM_AMR_PLAY_SESSION: + #endif + #if (L1_MIDI == 1) + case AUDIO_DRIVER_INIT_MIDI_SESSION: + #endif + audio_driver_error_trace(AUDIO_ERROR_INIT_EVENT); + audio_driver_init_send_status(AUDIO_ERROR, session_id, channel_id, return_path); + break; + case AUDIO_DRIVER_START_SESSION: + audio_driver_error_trace(AUDIO_ERROR_START_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_START_STATUS, channel_id, return_path); + break; + case AUDIO_DRIVER_FREE_SESSION: + audio_driver_error_trace(AUDIO_ERROR_FREE_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_FREE_STATUS, channel_id, return_path); + break; + } // switch + } //case AUDIO_DRIVER_CHANNEL_WAIT_START_CON: + break; + case AUDIO_DRIVER_CHANNEL_WAIT_STOP: + { + switch (p_message->msg_id) + { + case AUDIO_DRIVER_STOP_SESSION: + { + // 1st stop request + if (p_session->session_info.stop_request == 0) + { + T_AUDIO_RET result; + + // send l1 message STOP_REQ + switch(p_session->session_req.session_id) + { + #if (L1_VOICE_MEMO_AMR) + case AUDIO_VM_AMR_RECORD_SESSION_ID: + result=audio_driver_vm_amr_record_manager(p_message,p_session); + break; + case AUDIO_VM_AMR_PLAY_SESSION_ID: + result=audio_driver_vm_amr_play_manager(p_message,p_session); + break; + #endif + #if (L1_MIDI == 1) + case AUDIO_MIDI_SESSION_ID: + result=audio_driver_midi_manager(p_message,p_session); + break; + #endif + } + + // allocate or send L1 msg went wrong + if (result != AUDIO_OK) + { + audio_driver_error_trace(AUDIO_ERROR_L1_STOP_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_STOP_STATUS, channel_id, return_path); + return; + } + + // STOP has been requested, no longer accept it + p_session->session_info.stop_request = 1; + } + else + { + audio_driver_error_trace(AUDIO_ERROR_STOP_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_STOP_STATUS, channel_id, return_path); + } + } + break; + #if (L1_VOICE_MEMO_AMR) + case MMI_VM_AMR_RECORD_STOP_CON: + { + // send last notification + T_AUDIO_DRIVER_LAST_NOTIFICATION *p_status_message; + + // Allocate the Riviera buffer + mb_status = RVF_RED; + while (mb_status == RVF_RED) + { + mb_status = rvf_get_buf (p_audio_gbl_var->mb_internal, + sizeof (T_AUDIO_DRIVER_LAST_NOTIFICATION), + (T_RVF_BUFFER **) (&p_status_message)); + // If insufficient resources, then report a memory error and abort. + if (mb_status == RVF_RED) + { + // the memory is insufficient to continue the non regression test + audio_driver_error_trace(AUDIO_ENTITY_NO_MEMORY); + rvf_delay(RVF_MS_TO_TICKS(1000)); + } + } + + // Fill the message ID + p_status_message->header.msg_id = AUDIO_DRIVER_LAST_NOTIFICATION_MSG; + + // fill parameters + p_status_message->channel_id = channel_id; + p_status_message->recorded_size = ((T_MMI_VM_AMR_RECORD_CON *)p_message)->recorded_size - SC_VM_AMR_END_MASK_SIZE; + p_status_message->p_buffer = + (UINT16 *)p_session->session_info.buffer[p_session->session_info.index_l1].p_start_pointer; + + if (return_path.callback_func == NULL) + rvf_send_msg (return_path.addr_id, p_status_message); + else + { + // call the callback function + (*(return_path.callback_func))((void *)(p_status_message)); + rvf_free_buf((T_RVF_BUFFER *)p_status_message); + } + + // change state + p_session->session_info.state = AUDIO_DRIVER_CHANNEL_WAIT_FREE; + } + break; + #endif // #if (L1_VOICE_MEMO_AMR) + #if (L1_VOICE_MEMO_AMR) + case MMI_VM_AMR_PLAY_STOP_CON: + #endif + #if (L1_MIDI == 1) + case MMI_MIDI_STOP_CON: + #endif + { + // deallocate buffers + for (j = 0; j < p_session->session_req.nb_buffer; j++) + { + mb_status = rvf_free_buf ((T_RVF_BUFFER *) (p_session->session_info.buffer[j].p_start_pointer)); + + AUDIO_SEND_TRACE_PARAM("AUDIO DRIVER MANAGER: deallocate buffer", j, RV_TRACE_LEVEL_DEBUG_LOW); + + if (mb_status != RVF_GREEN) + AUDIO_SEND_TRACE_PARAM("can't deallocate buffer", j, RV_TRACE_LEVEL_ERROR); + } + // Deallocate buffers pointer + rvf_free_buf((T_RVF_BUFFER *)p_session->session_info.buffer); + + audio_driver_send_status (AUDIO_OK, AUDIO_STOP_STATUS, channel_id, + return_path); + // change state + p_session->session_info.state = AUDIO_DRIVER_CHANNEL_WAIT_INIT; + } + break; + #if (L1_VOICE_MEMO_AMR) + case AUDIO_DRIVER_INIT_VM_AMR_RECORD_SESSION: + case AUDIO_DRIVER_INIT_VM_AMR_PLAY_SESSION: + #endif + #if (L1_MIDI == 1) + case AUDIO_DRIVER_INIT_MIDI_SESSION: + #endif + audio_driver_error_trace(AUDIO_ERROR_INIT_EVENT); + audio_driver_init_send_status(AUDIO_ERROR, session_id, channel_id, return_path); + break; + case AUDIO_DRIVER_START_SESSION: + audio_driver_error_trace(AUDIO_ERROR_START_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_START_STATUS, channel_id, return_path); + break; + case AUDIO_DRIVER_FREE_SESSION: + audio_driver_error_trace(AUDIO_ERROR_FREE_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_FREE_STATUS, channel_id, return_path); + break; + } //switch + } //case AUDIO_DRIVER_CHANNEL_WAIT_STOP: + break; + case AUDIO_DRIVER_CHANNEL_WAIT_START_CON_TO_STOP: + { + T_AUDIO_RET result = AUDIO_OK; + switch (p_message->msg_id) + { + #if (L1_VOICE_MEMO_AMR) + case AUDIO_DRIVER_INIT_VM_AMR_RECORD_SESSION: + case AUDIO_DRIVER_INIT_VM_AMR_PLAY_SESSION: + #endif + #if (L1_MIDI == 1) + case AUDIO_DRIVER_INIT_MIDI_SESSION: + #endif + audio_driver_error_trace(AUDIO_ERROR_INIT_EVENT); + audio_driver_init_send_status(AUDIO_ERROR, session_id, channel_id, return_path); + return; + break; + case AUDIO_DRIVER_START_SESSION: + audio_driver_error_trace(AUDIO_ERROR_START_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_START_STATUS, channel_id, return_path); + return; + break; + case AUDIO_DRIVER_STOP_SESSION: + audio_driver_error_trace(AUDIO_ERROR_STOP_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_STOP_STATUS, channel_id, return_path); + return; + break; + case AUDIO_DRIVER_FREE_SESSION: + audio_driver_error_trace(AUDIO_ERROR_FREE_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_FREE_STATUS, channel_id, return_path); + return; + break; + #if (L1_VOICE_MEMO_AMR) + case MMI_VM_AMR_RECORD_START_CON: + // send L1 stop msg + result=audio_driver_vm_amr_record_manager(p_message, p_session); + break; + case MMI_VM_AMR_PLAY_START_CON: + // send L1 stop msg + result=audio_driver_vm_amr_play_manager(p_message, p_session); + break; + #endif + #if (L1_MIDI == 1) + case MMI_MIDI_START_CON: + // send L1 stop msg + result=audio_driver_midi_manager(p_message,p_session); + break; + #endif + } + + if (result != AUDIO_OK) + { + audio_driver_send_status(AUDIO_ERROR, AUDIO_STOP_STATUS, channel_id, return_path); + return; + } + else + { + // next state should be WAIT_STOP_CON, which is WAIT_STOP + stop_request == 1 + p_session->session_info.stop_request = 1; + // change state + p_session->session_info.state = AUDIO_DRIVER_CHANNEL_WAIT_STOP; + } + } + break; + case AUDIO_DRIVER_CHANNEL_WAIT_FREE: + { + switch (p_message->msg_id) + { + case AUDIO_DRIVER_FREE_SESSION: + { + // deallocate buffers + for (j = 0; j < p_session->session_req.nb_buffer; j++) + { + mb_status = rvf_free_buf ((T_RVF_BUFFER *) (p_session->session_info.buffer[j].p_start_pointer)); + + AUDIO_SEND_TRACE_PARAM("AUDIO DRIVER MANAGER: deallocate buffer", j, RV_TRACE_LEVEL_DEBUG_LOW); + + if (mb_status != RVF_GREEN) + AUDIO_SEND_TRACE_PARAM("can't deallocate buffer", j, RV_TRACE_LEVEL_ERROR); + } + // Deallocate buffers pointer + rvf_free_buf((T_RVF_BUFFER *)p_session->session_info.buffer); + + audio_driver_send_status (AUDIO_OK, AUDIO_FREE_STATUS, channel_id, + return_path); + + // change state + p_session->session_info.state = AUDIO_DRIVER_CHANNEL_WAIT_INIT; + } + break; + #if (L1_VOICE_MEMO_AMR) + case AUDIO_DRIVER_INIT_VM_AMR_RECORD_SESSION: + case AUDIO_DRIVER_INIT_VM_AMR_PLAY_SESSION: + #endif + #if (L1_MIDI == 1) + case AUDIO_DRIVER_INIT_MIDI_SESSION: + #endif + audio_driver_error_trace(AUDIO_ERROR_INIT_EVENT); + audio_driver_init_send_status(AUDIO_ERROR, session_id, channel_id, return_path); + break; + case AUDIO_DRIVER_START_SESSION: + audio_driver_error_trace(AUDIO_ERROR_START_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_START_STATUS, channel_id, return_path); + break; + case AUDIO_DRIVER_STOP_SESSION: + audio_driver_error_trace(AUDIO_ERROR_STOP_EVENT); + audio_driver_send_status(AUDIO_ERROR, AUDIO_STOP_STATUS, channel_id, return_path); + break; + } // switch + } + break; + } + } +#endif // (L1_AUDIO_DRIVER) + +#endif // #ifdef RVM_AUDIO_MAIN_SWE