FreeCalypso > hg > fc-magnetite
view src/g23m-gsm/sim/sim_stk.c @ 516:1ed9de6c90bd
src/g23m-gsm/sms/sms_for.c: bogus malloc removed
The new error handling code that was not present in TCS211 blob version
contains a malloc call that is bogus for 3 reasons:
1) The memory allocation in question is not needed in the first place;
2) libc malloc is used instead of one of the firmware's proper ways;
3) The memory allocation is made inside a function and then never freed,
i.e., a memory leak.
This bug was caught in gcc-built FreeCalypso fw projects (Citrine
and Selenite) because our gcc environment does not allow any use of
libc malloc (any reference to malloc produces a link failure),
but this code from TCS3.2 is wrong even for Magnetite: if this code
path is executed repeatedly over a long time, the many small allocations
made by this malloc call without a subsequent free will eventually
exhaust the malloc heap provided by the TMS470 environment, malloc will
start returning NULL, and the bogus code will treat it as an error.
Because the memory allocation in question is not needed at all,
the fix entails simply removing it.
author | Mychaela Falconia <falcon@freecalypso.org> |
---|---|
date | Sun, 22 Jul 2018 06:04:49 +0000 |
parents | 27a4235405c6 |
children |
line wrap: on
line source
/* +----------------------------------------------------------------------------- | Project : GSM-F&D (8411) | Modul : SIM_STK +----------------------------------------------------------------------------- | Copyright 2002 Texas Instruments Berlin, AG | All rights reserved. | | This file is confidential and a trade secret of Texas | Instruments Berlin, AG | The receipt of or possession of this file does not convey | any rights to reproduce or disclose its contents or to | manufacture, use, or sell anything it may describe, in | whole, or in part, without the specific written consent of | Texas Instruments Berlin, AG. +----------------------------------------------------------------------------- | Purpose : This modul defines the SIM Toolkit Upgrade. +----------------------------------------------------------------------------- */ #ifndef SIM_STK_C #define SIM_STK_C #define ENTITY_SIM /*==== INCLUDES ===================================================*/ #include <string.h> #include "typedefs.h" #include "pcm.h" #include "pconst.cdg" #include "message.h" #include "ccdapi.h" #include "vsi.h" #include "custom.h" #include "gsm.h" #include "cnf_sim.h" #include "mon_sim.h" #include "prim.h" #include "pei.h" #include "tok.h" #include "sim.h" #include "sim_em.h" #include "cl_imei.h" /* IMEI common library */ #include "cl_shrd.h" #ifdef TI_PS_UICC_CHIPSET_15 #include "8010_136_SIMDRV_SAP_inline.h" #endif EXTERN USHORT stk_l_cmd; /*==== EXPORT =====================================================*/ /* * These Functions are only temporary valid and should be replaced as soon as possible */ EXTERN UBYTE get_network_meas (UBYTE * chan_list); EXTERN UBYTE get_bcch_chan_list (stk_data_type * out_stk_data); /*==== PRIVAT =====================================================*/ #ifdef FF_SAT_E LOCAL void stk_handle_ccd_error(UBYTE notOK, UBYTE* status, UBYTE* general_result, UBYTE* add_info_result); LOCAL void stk_bip_decode_stk_command(T_sdu* message, T_cmd_details* cmd_details, UBYTE* status, UBYTE* general_result, UBYTE* add_info_result); LOCAL void stk_close_dti_connection(UBYTE open); LOCAL void stk_close_bip_channel(UBYTE general_result, UBYTE add_info_result); /* LOCAL void stk_dti_inform_mmi_old (UBYTE dti_conn_req); */ LOCAL void stk_dti_inform_mmi(UBYTE dti_conn_req, UBYTE bip_conn_req ); LOCAL void stk_dti_send_data(); LOCAL void stk_bip_send_data_terminal_response(UBYTE general_result, UBYTE add_info_result); #endif /* FF_SAT_E */ /*==== TYPES ======================================================*/ typedef struct { UBYTE tag; UBYTE min_len; UBYTE mand; UBYTE min_req; } T_SAT_TAG_DESC; typedef struct { UBYTE cmd_type; UBYTE tp_i; UBYTE tp_flag; UBYTE max_tag; const T_SAT_TAG_DESC *tag_desc; } T_SAT_CMD_DESC; /*==== CONSTANTS ==================================================*/ const T_SAT_TAG_DESC sat_def_cmd[2] = { { STK_COMMAND_DETAILS_TAG, STK_COMMAND_DETAILS_LEN, TRUE, TRUE }, { STK_DEVICE_IDENTITY_TAG, STK_DEVICE_IDENTITY_LEN, TRUE, TRUE } }; // valid tags for any command const T_SAT_TAG_DESC sat_cmd_poll_itv[3] = { { STK_COMMAND_DETAILS_TAG, STK_COMMAND_DETAILS_LEN, TRUE, TRUE }, { STK_DEVICE_IDENTITY_TAG, STK_DEVICE_IDENTITY_LEN, TRUE, TRUE }, { STK_DURATION_TAG, STK_DURATION_LEN, TRUE, TRUE } }; // valid tags for POLL INTERVALL const T_SAT_TAG_DESC sat_cmd_refresh[3] = { { STK_COMMAND_DETAILS_TAG, STK_COMMAND_DETAILS_LEN, TRUE, TRUE }, { STK_DEVICE_IDENTITY_TAG, STK_DEVICE_IDENTITY_LEN, TRUE, TRUE }, { STK_FILE_LIST_TAG, STK_FILE_LIST_LEN, FALSE, TRUE } }; // valid tags for REFRESH const T_SAT_TAG_DESC sat_cmd_event_list[3] = { { STK_COMMAND_DETAILS_TAG, STK_COMMAND_DETAILS_LEN, TRUE, TRUE }, { STK_DEVICE_IDENTITY_TAG, STK_DEVICE_IDENTITY_LEN, TRUE, TRUE }, { STK_EVENT_LIST_TAG, STK_EVENT_LIST_LEN, TRUE, TRUE } }; // valid tags for SETUP EVENT LIST const T_SAT_TAG_DESC sat_cmd_timer[4] = { { STK_COMMAND_DETAILS_TAG, STK_COMMAND_DETAILS_LEN, TRUE, TRUE }, { STK_DEVICE_IDENTITY_TAG, STK_DEVICE_IDENTITY_LEN, TRUE, TRUE }, { STK_TIMER_ID_TAG, STK_TIMER_ID_LEN, TRUE, TRUE }, { STK_TIMER_VALUE_TAG, STK_TIMER_VALUE_LEN, FALSE, FALSE } }; // valid tags for TIMER MANAGEMENT const T_SAT_TAG_DESC sat_cmd_receive_data[5] = { { STK_COMMAND_DETAILS_TAG, STK_COMMAND_DETAILS_LEN, TRUE, TRUE }, { STK_DEVICE_IDENTITY_TAG, STK_DEVICE_IDENTITY_LEN, TRUE, TRUE }, { STK_ALPHA_IDENTITY_TAG, STK_ALPHA_IDENTITY_LEN, FALSE, FALSE }, { STK_ICON_IDENTITY_TAG, STK_ICON_IDENTITY_LEN, FALSE, FALSE }, { STK_CHANNEL_DATA_LGTH_TAG, STK_CHANNEL_DATA_LGTH_LEN, TRUE, TRUE } }; // valid tags for RECEIVE DATA const T_SAT_TAG_DESC sat_cmd_send_data[5] = { { STK_COMMAND_DETAILS_TAG, STK_COMMAND_DETAILS_LEN, TRUE, TRUE }, { STK_DEVICE_IDENTITY_TAG, STK_DEVICE_IDENTITY_LEN, TRUE, TRUE }, { STK_ALPHA_IDENTITY_TAG, STK_ALPHA_IDENTITY_LEN, FALSE, FALSE }, { STK_ICON_IDENTITY_TAG, STK_ICON_IDENTITY_LEN, FALSE, FALSE }, { STK_CHANNEL_DATA_TAG, STK_CHANNEL_DATA_LEN, TRUE, TRUE } }; // valid tags for SEND DATA const T_SAT_CMD_DESC sat_cmd_list[] = { { 0, 0, 0xFF, item_of(sat_def_cmd), sat_def_cmd }, // command not yet known { STK_REFRESH, 2, SAT_TP3_REFRESH, item_of(sat_cmd_refresh), sat_cmd_refresh }, { STK_MORE_TIME, 2, SAT_TP3_MORE_TIME, item_of(sat_def_cmd), sat_def_cmd }, { STK_POLL_INTERVALL, 2, SAT_TP3_POLL_ITV, item_of(sat_cmd_poll_itv), sat_cmd_poll_itv }, { STK_POLLING_OFF, 2, SAT_TP3_POLL_OFF, item_of(sat_def_cmd), sat_def_cmd }, { STK_SET_UP_CALL, 3, SAT_TP4_SETUP_CALL, item_of(sat_def_cmd), sat_def_cmd }, { STK_SEND_SS, 3, SAT_TP4_SEND_SS, item_of(sat_def_cmd), sat_def_cmd }, { STK_SEND_SMS, 3, SAT_TP4_SEND_SMS, item_of(sat_def_cmd), sat_def_cmd }, { STK_PLAY_TONE, 2, SAT_TP3_PLAY_TONE, item_of(sat_def_cmd), sat_def_cmd }, { STK_DISPLAY_TEXT, 2, SAT_TP3_DSPL_TXT, item_of(sat_def_cmd), sat_def_cmd }, { STK_GET_INKEY, 2, SAT_TP3_GET_INKEY, item_of(sat_def_cmd), sat_def_cmd }, { STK_GET_INPUT, 2, SAT_TP3_GET_INPUT, item_of(sat_def_cmd), sat_def_cmd }, { STK_SELECT_ITEM, 3, SAT_TP4_SEL_ITEM, item_of(sat_def_cmd), sat_def_cmd }, { STK_SET_UP_MENU, 3, SAT_TP4_SETUP_MENU, item_of(sat_def_cmd), sat_def_cmd }, { STK_PROVIDE_LOCAL_INFO, 3, SAT_TP4_PLI_PLMN_IMEI, item_of(sat_def_cmd), sat_def_cmd }, { STK_TIMER_MANAGEMENT, 7, SAT_TP8_TMNG_ST, item_of(sat_cmd_timer), sat_cmd_timer }, { STK_SETUP_EVENT_LIST, 4, SAT_TP5_EVENT_LIST, item_of(sat_def_cmd), sat_def_cmd }, { STK_SEND_USSD, 3, SAT_TP4_SEND_USSD, item_of(sat_def_cmd), sat_def_cmd }, { STK_SEND_DTMF, 8, SAT_TP9_DTMF_CMD, item_of(sat_def_cmd), sat_def_cmd }, { STK_SETUP_IDLE_TEXT, 7, SAT_TP8_IDLE_TXT, item_of(sat_def_cmd), sat_def_cmd }, { STK_RUN_AT_CMD, 7, SAT_TP8_AT_CMD, item_of(sat_def_cmd), sat_def_cmd }, { STK_LANGUAGE_NOTIFICATION, 8, SAT_TP9_LANG_NOTIFY, item_of(sat_def_cmd), sat_def_cmd } #ifdef FF_SAT_C , { STK_LAUNCH_BROWSER, 8, SAT_TP9_LAUNCH_BROWSER, item_of(sat_def_cmd), sat_def_cmd } #endif /* FF_SAT_C */ #ifdef FF_SAT_E , { STK_OPEN_CHANNEL, 11, SAT_TP12_OPEN_CHANNEL, item_of(sat_def_cmd), sat_def_cmd }, { STK_CLOSE_CHANNEL, 11, SAT_TP12_CLOSE_CHANNEL, item_of(sat_def_cmd), sat_def_cmd }, { STK_RECEIVE_DATA, 11, SAT_TP12_RECEIVE_DATA, item_of(sat_cmd_receive_data), sat_cmd_receive_data }, { STK_SEND_DATA, 11, SAT_TP12_SEND_DATA, item_of(sat_cmd_send_data), sat_cmd_send_data }, { STK_GET_CHANNEL_STAT, 11, SAT_TP12_GET_CHANNEL_STAT, item_of(sat_def_cmd), sat_def_cmd } #endif /* FF_SAT_E */ }; /* Terminal Profile bits for which SIM is exclusively responsible. They are discarded from the profile provided by MMI */ static const UBYTE sat_tp_sim_exclusive[MAX_STK_PRF] = { /* SAT_TP1_PRF_DNL |*/ SAT_TP1_TIMER_EXP | SAT_TP1_CC_ON_REDIAL, 0x0, SAT_TP3_MORE_TIME | SAT_TP3_POLL_ITV | SAT_TP3_POLL_OFF, SAT_TP4_PLI_PLMN_IMEI | SAT_TP4_PLI_NMR, 0x0 /*SAT_TP5_LOC_STATUS*/, 0x0, 0xFF, SAT_TP8_TMNG_ST | SAT_TP8_TMNG_VAL, SAT_TP9_BCCH_COD | SAT_TP9_PLI_TIMING_ADV, /* Timing Advance is to be supported */ 0x0, 0x0, SAT_TP12_RECEIVE_DATA | SAT_TP12_SEND_DATA, 0x0, 0x0, 0x0, 0x0, 0x0, 0xFF, 0x0, 0xFF}; /* Terminal Profile bits which have to be supported by SIM to be valid with the profile provided by MMI */ static const UBYTE sat_tp_sim_ability[MAX_STK_PRF] = { SAT_TP1_PRF_DNL, SAT_TP2_CMD_RES, SAT_TP3_DSPL_TXT | SAT_TP3_GET_INKEY | SAT_TP3_GET_INPUT | SAT_TP3_PLAY_TONE | SAT_TP3_REFRESH, SAT_TP4_SEL_ITEM | SAT_TP4_SEND_SMS | SAT_TP4_SEND_SS | SAT_TP4_SEND_USSD | SAT_TP4_SETUP_CALL | SAT_TP4_SETUP_MENU, SAT_TP5_EVENT_LIST, 0x0, 0x0, SAT_TP8_IDLE_TXT | SAT_TP8_AT_CMD | SAT_TP8_PLI_DTT, SAT_TP9_DTMF_CMD | SAT_TP9_LANG_NOTIFY | SAT_TP9_LAUNCH_BROWSER, 0x0, 0x0, SAT_TP12_OPEN_CHANNEL | SAT_TP12_CLOSE_CHANNEL | SAT_TP12_GET_CHANNEL_STAT, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}; /* Currently supported Terminal Profile bits by SIM */ static const UBYTE sat_tp_sim_enabled[MAX_STK_PRF] = { SAT_TP1_PRF_DNL | SAT_TP1_SMS_DNL | SAT_TP1_9E_XX | SAT_TP1_TIMER_EXP, SAT_TP2_CMD_RES, SAT_TP3_MORE_TIME | SAT_TP3_POLL_ITV | SAT_TP3_POLL_OFF | SAT_TP3_REFRESH, SAT_TP4_PLI_PLMN_IMEI | SAT_TP4_PLI_NMR, 0x0, 0x0, 0x0, SAT_TP8_TMNG_ST | SAT_TP8_TMNG_VAL | SAT_TP8_PLI_DTT | SAT_TP8_IDLE_TXT | SAT_TP8_AT_CMD, SAT_TP9_BCCH_COD | SAT_TP9_PLI_TIMING_ADV, /* Timing Advance is to be supported */ 0x0, 0x0, #ifdef FF_SAT_E SAT_TP12_OPEN_CHANNEL | SAT_TP12_CLOSE_CHANNEL | SAT_TP12_GET_CHANNEL_STAT | SAT_TP12_RECEIVE_DATA | SAT_TP12_SEND_DATA, #else 0x0, #endif 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}; static const UBYTE tbl_device_src_id[] = { DEV_SRC_KBD, DEV_SRC_DSP, DEV_SRC_EAR, DEV_SRC_CDR0, DEV_SRC_CDR1, DEV_SRC_CDR2, DEV_SRC_CDR3, DEV_SRC_CDR4, DEV_SRC_CDR5, DEV_SRC_CDR6, DEV_SRC_CDR7, DEV_SRC_CH1, DEV_SRC_CH2, DEV_SRC_CH3, DEV_SRC_CH4, DEV_SRC_CH5, DEV_SRC_CH6, DEV_SRC_CH7, DEV_SRC_SIM, DEV_SRC_ME, DEV_SRC_NTW, 0 }; /*==== VARIABLES ==================================================*/ BOOL startTimerPollOff = FALSE; USHORT cusSatMinPollItv = 50; // 5 seconds /*==== FUNCTIONS ==================================================*/ LOCAL UBYTE stk_dti_bip_receive_data (T_sdu* message, UBYTE result_code); /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_init_sim_data | +--------------------------------------------------------------------+ PURPOSE : Initialize the SIM data for the module SIM TOOLKIT. */ GLOBAL void stk_init_sim_data (void) { TRACE_FUNCTION ("stk_init_sim_data()"); /* * initialize SAT variables */ sim_data.sat_session = FALSE; sim_data.ext_sat_cmd = FALSE; sim_data.term_resp_sent = FALSE; sim_data.chk_sat_avail = FALSE; sim_data.context_switch_ptr = NULL; sim_data.cust_mode = FALSE; sim_data.user_confirmation_expected = FALSE; #ifdef FF_SAT_E /* * initialize DTI variables */ sim_data.dti_connection_state = SIM_DTI_CONNECTION_CLOSED; sim_data.dti_rx_state = SIM_DTI_RX_IDLE; sim_data.dti_tx_state = SIM_DTI_TX_IDLE; sim_data.event_data_avail = SIM_EVENT_DISABLE; sim_data.bip_state = SIM_BIP_CLOSED; sim_data.bip_suspend = FALSE; #ifdef _SIMULATION_ TRACE_EVENT("bip_rx_state = IDLE"); #endif sim_data.bip_rx_state = SIM_BIP_RX_IDLE; #ifdef _SIMULATION_ TRACE_EVENT("bip_tx_state = IDLE"); #endif sim_data.bip_tx_state = SIM_BIP_TX_IDLE; sim_data.bip_timer_state = SIM_BIP_TIMER_DISCONNECTED; sim_data.bip_release_time = SIM_NO_AUTO_RELEASE; sim_data.bip_general_result = RSLT_PERF_SUCCESS; sim_data.bip_add_info_result = ADD_NO_CAUSE; sim_data.con_type = SIM_CON_TYPE_UDP; sim_data.data_to_send.first = (ULONG)NULL; sim_data.data_to_send.list_len = 0; sim_data.prev_data_to_send.first = (ULONG)NULL; sim_data.prev_data_to_send.list_len = 0; sim_data.received_data.first = (ULONG)NULL; sim_data.received_data.list_len = 0; sim_data.received_data_pos = 0; sim_data.sim_dti_req = NULL; sim_data.sim_bip_req = NULL; sim_data.sim_bip_config_req = NULL; memset(&sim_data.udp_parameters, 0, sizeof(T_SRC_DES)); #endif /* FF_SAT_E */ } /* stk_init_sim_data() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_check_tp | +--------------------------------------------------------------------+ PURPOSE : Evaluates the received Terminal Profile against the capability of the SIM Entity. */ GLOBAL void stk_check_tp (UBYTE *out_prf, UBYTE *in_prf, USHORT len) { int i; UBYTE tp_tmp; /* TRACE_FUNCTION ("stk_check_tp()"); */ len = MINIMUM(len, MAX_STK_PRF); for (i = 0; i < len; i++) // check TP from MMI { #ifndef __OLD tp_tmp = sat_tp_sim_enabled[i] & ~sat_tp_sim_ability[i]; out_prf[i] = ((in_prf[i] & ~sat_tp_sim_exclusive[i]) & ~tp_tmp) | tp_tmp; #else out_prf[i] = ((in_prf[i] & ~sat_tp_sim_exclusive[i]) & (sat_tp_sim_enabled[i] | ~sat_tp_sim_ability[i])) | sat_tp_sim_enabled[i]; #endif } if (!(out_prf[0] & SAT_TP1_SMS_DNL)) // check SMS download out_prf[0] &= ~SAT_TP1_9E_XX; // only valid with SMS download if (len >= 6 AND !(out_prf[4] & SAT_TP5_EVENT_LIST)) // check event list consistency out_prf[5] = out_prf[4] = 0; // discard event list if (len >= 12 AND (out_prf[11] & STK_TP12_CLASS_E) NEQ STK_TP12_CLASS_E) { out_prf[11] &= ~STK_TP12_CLASS_E; if (len >= 13) out_prf[12] &= ~(SAT_TP13_CSD_SUPP_BY_ME | SAT_TP13_GPRS_SUPP_BY_ME); if (len >= 17) out_prf[16] &= ~(SAT_TP17_BEARER_IND_SUPP_TCP | SAT_TP17_BEARER_IND_SUPP_UDP); } TRACE_EVENT_P5("TP: %02X %02X %02X %02X ... (%d bytes)", out_prf[0], out_prf[1], out_prf[2], out_prf[3], len); } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_perform_profile_download | +--------------------------------------------------------------------+ PURPOSE : Initialize the SIM data for the module application. */ GLOBAL void stk_perform_profile_download (void) { USHORT result; USHORT used_tp = (USHORT)sizeof (sim_data.stk_profile); TRACE_FUNCTION ("stk_perform_profile_download()"); do { if (sim_data.stk_profile[--used_tp] NEQ 0) break; } while (used_tp >= 3); used_tp++; if ((result = FKT_TerminalProfile (sim_data.stk_profile, used_tp)) NEQ SIM_NO_ERROR) { TRACE_EVENT_P1("TP dnl error: %04X", result); } } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_proactive_polling | +--------------------------------------------------------------------+ PURPOSE : Polling of an proactive SIM card. */ static const UBYTE tag_dur [] = { STK_DURATION_TAG | STK_COMPREHENSION_REQUIRED, /* duration tag */ STK_DURATION_LEN, /* duration length */ 0, /* unit */ 0 /* number of units */ }; static const UBYTE terminal_response_loci [21] = { STK_COMMAND_DETAILS_TAG | STK_COMPREHENSION_REQUIRED, /* command details tag */ STK_COMMAND_DETAILS_LEN, /* command details length */ 0, /* command number */ STK_PROVIDE_LOCAL_INFO, /* command PROVIDE LOCATION INFO*/ 0, /* location information */ STK_DEVICE_IDENTITY_TAG | STK_COMPREHENSION_REQUIRED, /* device details tag */ STK_DEVICE_IDENTITY_LEN, /* device details length */ 0x82, /* source ME */ 0x81, /* destination SIM */ STK_RESULT_TAG | STK_COMPREHENSION_REQUIRED, /* result tag */ 1, /* result length */ 0, /* result OK */ 0x13, /* location information tag */ 7, /* location information length */ 0,0,0, /* MCC & MNC */ 0,0, /* Location area code */ 0,0 /* cell identity */ }; static const UBYTE terminal_response_imei [22] = { STK_COMMAND_DETAILS_TAG | STK_COMPREHENSION_REQUIRED, /* command details tag */ STK_COMMAND_DETAILS_LEN, /* command details length */ 0, /* command number */ STK_PROVIDE_LOCAL_INFO, /* command PROVIDE LOCATION INFO*/ 1, /* imei */ STK_DEVICE_IDENTITY_TAG | STK_COMPREHENSION_REQUIRED, /* device details tag */ STK_DEVICE_IDENTITY_LEN, /* device details length */ 0x82, /* source ME */ 0x81, /* destination SIM */ STK_RESULT_TAG | STK_COMPREHENSION_REQUIRED, /* result tag */ 1, /* result length */ 0, /* result OK */ 0x14, /* imei tag */ 8, /* imei length */ 0,0,0,0,0,0,0,0 /* imei digits */ }; static const UBYTE terminal_response_nmr [14] = { STK_COMMAND_DETAILS_TAG | STK_COMPREHENSION_REQUIRED, /* command details tag */ STK_COMMAND_DETAILS_LEN, /* command details length */ 0, /* command number */ STK_PROVIDE_LOCAL_INFO, /* command PROVIDE LOCATION INFO*/ 2, /* network measurement results */ STK_DEVICE_IDENTITY_TAG | STK_COMPREHENSION_REQUIRED, /* device details tag */ STK_DEVICE_IDENTITY_LEN, /* device details length */ 0x82, /* source ME */ 0x81, /* destination SIM */ STK_RESULT_TAG | STK_COMPREHENSION_REQUIRED, /* result tag */ 1, /* result length */ 0, /* result OK */ 0x16, /* nmr tag */ 16 /* nmr length */ }; /* * Terminal-Response for timing advance */ static const UBYTE terminal_response_timingadv [16] = { /* ---- cmd details ----- */ STK_COMMAND_DETAILS_TAG | STK_COMPREHENSION_REQUIRED, /* command details tag */ STK_COMMAND_DETAILS_LEN, /* command details length */ 0, /* command number */ STK_PROVIDE_LOCAL_INFO, /* command PROVIDE LOCAL INFO */ 5, /* timing advance */ /* ---- device ids ------ */ STK_DEVICE_IDENTITY_TAG | STK_COMPREHENSION_REQUIRED, /* device details tag */ STK_DEVICE_IDENTITY_LEN, /* device details length */ 0x82, /* source ME */ 0x81, /* destination SIM */ /* ------result --------- */ STK_RESULT_TAG | STK_COMPREHENSION_REQUIRED, /* result tag */ 1, /* result length */ 0, /* result OK */ /* --- timing advance --- */ 0x2E, /* timing advance tag */ 2, /* timing advance len */ 0, /* ME Status */ 0 /* Timing Advance */ }; static const T_SIM_TRSP_SIMPLE terminal_response_def = { STK_COMMAND_DETAILS_TAG | STK_COMPREHENSION_REQUIRED, /* command details tag */ STK_COMMAND_DETAILS_LEN, /* command details length */ 0, /* command number */ 0, /* command type place holder */ 0, /* command qualifier */ STK_DEVICE_IDENTITY_TAG | STK_COMPREHENSION_REQUIRED, /* device details tag */ STK_DEVICE_IDENTITY_LEN, /* device details length */ 0x82, /* source ME */ 0x81, /* destination SIM */ STK_RESULT_TAG | STK_COMPREHENSION_REQUIRED, /* result tag */ 1, /* result length place holder */ 0xFF, /* result place holder */ 0, /* result additional info */ }; #ifdef __INVALID static const UBYTE terminal_response_ok [12] = { STK_COMMAND_DETAILS_TAG | STK_COMPREHENSION_REQUIRED, /* command details tag */ STK_COMMAND_DETAILS_LEN, /* command details length */ 0, /* command number */ 2, /* command MORE TIME */ 0, /* not used */ STK_DEVICE_IDENTITY_TAG | STK_COMPREHENSION_REQUIRED, /* device details tag */ STK_DEVICE_IDENTITY_LEN, /* device details length */ 0x82, /* source ME */ 0x81, /* destination SIM */ STK_RESULT_TAG | STK_COMPREHENSION_REQUIRED, /* result tag */ 1, /* result length */ 0 /* result OK */ }; static const UBYTE terminal_response_loci_neg [13] = { STK_COMMAND_DETAILS_TAG | STK_COMPREHENSION_REQUIRED, /* command details tag */ STK_COMMAND_DETAILS_LEN, /* command details length */ 0, /* command number */ STK_PROVIDE_LOCAL_INFO, /* command PROVIDE LOCATION INFO*/ 0, /* location information */ STK_DEVICE_IDENTITY_TAG | STK_COMPREHENSION_REQUIRED, /* device details tag */ STK_DEVICE_IDENTITY_LEN, /* device details length */ 0x82, /* source ME */ 0x81, /* destination SIM */ STK_RESULT_TAG | STK_COMPREHENSION_REQUIRED, /* result tag */ 2, /* result length */ 0x20, /* result ME unable to process */ 4 /* result add. info no service */ }; static const UBYTE terminal_response_no_cap [12] = { STK_COMMAND_DETAILS_TAG | STK_COMPREHENSION_REQUIRED, /* command details tag */ STK_COMMAND_DETAILS_LEN, /* command details length */ 0, /* command number */ 0x26, /* command PROVIDE LOCATION INFO*/ 0, /* location information */ 2, /* device details tag */ 2, /* device details length */ 0x82, /* source ME */ 0x81, /* destination SIM */ 3, /* result tag */ 1, /* result length */ 0x30 /* result beyond ME capability */ }; #endif LOCAL UBYTE stk_process_tl (UBYTE **pp_stk, SHORT *p_ber_len, SHORT *p_tlv_len) { /* get tag and length and adjust pointer to get parameter values */ UBYTE tag = STK_COMPREHENSION_REQUIRED; SHORT tlv_len; if (*p_ber_len >= 2) { tag = *((*pp_stk)++); // get tag and adjust pointer if ((tlv_len = (SHORT)*((*pp_stk)++)) <= 0x7F) { // one byte length *p_tlv_len = tlv_len; // get length *p_ber_len -= 2; } else if (tlv_len EQ 0x81 AND *p_ber_len >= 3) { // two bytes length *p_tlv_len = (SHORT)*((*pp_stk)++); *p_ber_len -= 3; // get length and adjust pointer } else { // erroneous length coding *p_tlv_len = 0; *p_ber_len = 0; return STK_COMPREHENSION_REQUIRED; } } return tag; } LOCAL SHORT stk_build_response (UBYTE *p_response, UBYTE *p_cmd, int cmd_len, UBYTE *p_res, int res_len, UBYTE *p_prm, int prm_len) { /* * builds a TERMINAL RESPONSE */ static const UBYTE dev_resp[5] = { STK_DEVICE_IDENTITY_TAG | STK_COMPREHENSION_REQUIRED, /* device details tag */ STK_DEVICE_IDENTITY_LEN, /* device details length */ 0x82, /* source ME */ 0x81, /* destination SIM */ STK_RESULT_TAG | STK_COMPREHENSION_REQUIRED /* result tag */ }; int tr_len = 6; #ifdef _SIMULATION_ TRACE_FUNCTION ("stk_build_response()"); #endif TRACE_FUNCTION_P1 ("TERMINAL_RESPONSE: result=0x%02X", (int)*p_res); memcpy (p_response, p_cmd, cmd_len); memcpy (p_response + cmd_len, dev_resp, 5); p_response[cmd_len + 5] = (UBYTE)res_len; memcpy (p_response + cmd_len + 6, p_res, res_len); tr_len += cmd_len + res_len; if (p_prm NEQ NULL AND prm_len > 0) { memcpy (p_response + tr_len, p_prm, prm_len); tr_len += prm_len; } return (SHORT)tr_len; } GLOBAL int process_sim_refresh( T_CONTEXT_SWITCH *cmd_ptr ) { int i; SHORT resp_len = 0; #ifdef TI_PS_UICC_CHIPSET_15 U8 readerId = SIMDRV_VAL_READER_ID__RANGE_MIN; U8 voltageSelect = SIMDRV_REQ_VOLTAGE_SEL; #endif T_STK_POLL_DATA *p; MALLOC (p, sizeof (T_STK_POLL_DATA)); memset (p, 0, sizeof (T_STK_POLL_DATA)); TRACE_FUNCTION ("process_sim_refresh()"); /* process as before */ switch (cmd_ptr->p_cmd[4]) { case 1: /*check file list is empty */ if( ( 0 == cmd_ptr->fl_len ) OR ( 1 == cmd_ptr->fl_len ) ) { cmd_ptr->res_code[0] = STK_RES_ERR_MISS_VALUE; resp_len = stk_build_response (p->response, cmd_ptr->p_cmd, (int)cmd_ptr->cmd_len, cmd_ptr->res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("FCN: File List is empty"); #endif FKT_TerminalResponse (p->response, (USHORT)resp_len); break; } if (cmd_ptr->tag2 NEQ NULL AND cmd_ptr->fl_len >= 5) { BOOL sim_init = FALSE; PALLOC (file_update_sms, SIM_FILE_UPDATE_IND); memset (file_update_sms, 0, sizeof(T_SIM_FILE_UPDATE_IND)); file_update_sms->val_nr = *(cmd_ptr->tag2++); cmd_ptr->fl_len -= 3; /* align file id count */ for (i = 0; cmd_ptr->fl_len > 0 AND i < (int)file_update_sms->val_nr; i++) { do { cmd_ptr->tag2 += 2; /* Check if entry is a first level directory */ if(cmd_ptr->tag2[0] EQ 0x7F) { if(file_update_sms->file_info[i].v_path_info EQ FALSE) { file_update_sms->file_info[i].v_path_info = TRUE; file_update_sms->file_info[i].path_info.df_level1 = ((USHORT)cmd_ptr->tag2[0] << 8) | (USHORT)cmd_ptr->tag2[1]; } else { /* This has already been filled. There cannot be another first level directory. Since i is reduced, i will not equal to file_update_sms->val_nr and error will be sent to SIM */ i--; break; } } /* Check if entry is a second level directory and first level * directory is also filled*/ else if(cmd_ptr->tag2[0] EQ 0x5F) { if(file_update_sms->file_info[i].v_path_info EQ TRUE AND file_update_sms->file_info[i].path_info.v_df_level2 EQ FALSE) { file_update_sms->file_info[i].path_info.v_df_level2 = TRUE; file_update_sms->file_info[i].path_info.df_level2 = ((USHORT)cmd_ptr->tag2[0] << 8) | (USHORT)cmd_ptr->tag2[1]; } else { /* This has already been filled. There cannot be another second level directory */ i--; break; } } cmd_ptr->fl_len -= 2; } while (cmd_ptr->fl_len > 0 AND cmd_ptr->tag2[2] NEQ 0x3F); if (cmd_ptr->tag2[0] NEQ 0x2F AND cmd_ptr->tag2[0] NEQ 0x6F AND cmd_ptr->tag2[0] NEQ 0x4F) break; /* no EF code -> loop exit leads to error response */ file_update_sms->file_info[i].datafield = ((USHORT)cmd_ptr->tag2[0] << 8) | (USHORT)cmd_ptr->tag2[1]; if ((file_update_sms->file_info[i].datafield EQ SIM_PHASE) OR (file_update_sms->file_info[i].datafield EQ SIM_SST)) { /* SIM initialisation is needed! */ sim_init = TRUE; TRACE_FUNCTION ("FCN converted to INIT"); /* break; exit for loop */ } } if (cmd_ptr->fl_len > 0 OR i NEQ (int)file_update_sms->val_nr) { /* inconsistent TLV content */ PFREE(file_update_sms); cmd_ptr->res_code[0] = STK_RES_ERR_CMD_DATA; resp_len = stk_build_response (p->response, cmd_ptr->p_cmd, (int)cmd_ptr->cmd_len, cmd_ptr->res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("FCN: inconsistent TLV content"); #endif FKT_TerminalResponse (p->response, (USHORT)resp_len); break; /* exit switch 'cmd_qual' */ } if (!sim_init) { { PALLOC (file_update_mmi, SIM_FILE_UPDATE_IND); memcpy (file_update_mmi, file_update_sms, sizeof(T_SIM_FILE_UPDATE_IND)); PSENDX (MMI, file_update_mmi); } { PALLOC (file_update_mm, SIM_FILE_UPDATE_IND); memcpy (file_update_mm, file_update_sms, sizeof(T_SIM_FILE_UPDATE_IND)); PSENDX (MM, file_update_mm); } PSENDX (SMS, file_update_sms); sim_data.file_change_resp = 0x7; /* check response */ /* Force file selection */ sim_data.act_directory = NOT_PRESENT_16BIT; sim_data.act_field = NOT_PRESENT_16BIT; if ((USHORT)(cmd_ptr->cmd_len + 7) <= sizeof (sim_data.stk_response)) { sim_data.stk_resp_len = stk_build_response (sim_data.stk_response, cmd_ptr->p_cmd, (int)cmd_ptr->cmd_len, cmd_ptr->res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("FCN: prepare TR"); #endif } else sim_data.stk_resp_len = 0; break; } else { PFREE (file_update_sms); } } else { cmd_ptr->res_code[0] = STK_RES_ERR_MISS_VALUE; resp_len = stk_build_response (p->response, cmd_ptr->p_cmd, (int)cmd_ptr->cmd_len, cmd_ptr->res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("FCN: cmd incomplete"); #endif FKT_TerminalResponse (p->response, (USHORT)resp_len); break; } // no break for sim_init EQ TRUE ! case 3: case 2: if (cmd_ptr->res_code[0] EQ STK_RES_SUCCESS) // do not overwrite special result codes // these commands read more EFs than indicated cmd_ptr->res_code[0] = STK_RES_SUCC_ADD_EF_READ; case 0: if ( (SIM_IS_FLAG_SET (CALL_ACTIVE)) AND (sim_data.cust_mode EQ 0) ) { cmd_ptr->res_code[0] = STK_RES_BUSY_ME; cmd_ptr->res_code[1] = STK_RES_EXT_BUSY_CALL; resp_len = stk_build_response (p->response, cmd_ptr->p_cmd, (int)cmd_ptr->cmd_len, cmd_ptr->res_code, 2, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("INIT: call active"); #endif FKT_TerminalResponse (p->response, (USHORT)resp_len); } else { /* Force file selection */ sim_data.act_directory = NOT_PRESENT_16BIT; sim_data.act_field = NOT_PRESENT_16BIT; sim_data.status_time = THIRTY_SECONDS; app_sim_read_parameters (); /* The terminal response to be sent is created and stored in context. Once ACI and MM indicates the completion of the reading of the EFs through SIM_SYNC_REQ, the response will be sent. */ sim_data.stk_resp_len = stk_build_response (sim_data.stk_response, cmd_ptr->p_cmd, (int)cmd_ptr->cmd_len, cmd_ptr->res_code, 1, NULL, 0); sim_data.sync_awaited = SIM_SYNC_AWAIT_MM_READ | SIM_SYNC_AWAIT_MMI_READ; #ifdef _SIMULATION_ TRACE_EVENT("INIT: success"); #endif } break; case 4: if ( (SIM_IS_FLAG_SET (CALL_ACTIVE)) AND (sim_data.cust_mode EQ 0) ) { cmd_ptr->res_code[0] = STK_RES_BUSY_ME; cmd_ptr->res_code[1] = STK_RES_EXT_BUSY_CALL; resp_len = stk_build_response (p->response, cmd_ptr->p_cmd, (int)cmd_ptr->cmd_len, cmd_ptr->res_code, 2, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("RESET: call active"); #endif FKT_TerminalResponse (p->response, (USHORT)resp_len); } else { #ifndef TI_PS_UICC_CHIPSET_15 T_SIM_CARD sim_info; #endif /* TI_PS_UICC_CHIPSET_15 */ USHORT retcode; TRACE_ASSERT (cmd_ptr->sig_ptr); PFREE (cmd_ptr->sig_ptr); TRACE_ASSERT (p); MFREE (p); sim_data.remove_error = SIM_NO_ERROR; app_sim_remove (); sim_data.remove_error = SIM_CAUSE_CARD_REMOVED; #ifndef TI_PS_UICC_CHIPSET_15 SIM_PowerOff (); #else simdrv_poweroff(readerId); #endif /* Force file selection */ sim_data.act_directory = NOT_PRESENT_16BIT; sim_data.act_field = NOT_PRESENT_16BIT; /*retcode = SIM_Restart (&sim_info); Driver does not call 'insert()'! */ #ifndef TI_PS_UICC_CHIPSET_15 retcode = SIM_Reset (&sim_info); #else retcode = simdrv_reset (readerId,voltageSelect); #endif TRACE_EVENT_P1 ("Result SIM Restart = %d", (int)retcode); TRACE_FUNCTION ("process_sim_refresh() exited(1)"); return(1); //return and exit stk_proactive_polling(), if that is where called from } break; default: /* * not supported information request */ TRACE_FUNCTION ("process_sim_refresh() default:"); cmd_ptr->res_code[0] = STK_RES_ERR_CMD_TYPE; resp_len = stk_build_response (p->response, cmd_ptr->p_cmd, cmd_ptr->cmd_len, cmd_ptr->res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("RFR: unknown qualifier"); #endif FKT_TerminalResponse (p->response, (USHORT)resp_len); break; } TRACE_ASSERT (cmd_ptr->sig_ptr); PFREE (cmd_ptr->sig_ptr); TRACE_ASSERT (p); MFREE (p); TRACE_FUNCTION ("process_sim_refresh() exited(0)"); return(0); } GLOBAL void stk_proactive_polling (void) { UBYTE *stk, *p_tag, *p_cmd = NULL; UBYTE *found_tag[STK_MAX_EXP_TAG]; UBYTE compreh = 0, cmd_nr = 0, cmd_type, cmd_qual = 0; SHORT ber_len, tag_len, cmd_len, fl_len = 0, resp_len = 0; SHORT i, tag_i, cmd_i, tlv_len; const T_SAT_TAG_DESC *p_tag_desc; int offset; T_LOC_INFO loc_info; T_TIM_ADV tim_adv; TRACE_FUNCTION ("stk_proactive_polling()"); if (SIM_IS_FLAG_SET (PRO_ACTIVE_SIM)) { while (sim_data.proactive_sim_data_len > 0) { UINT in_queue, out_queue; UBYTE result = FALSE; UBYTE res_code[2] = {STK_RES_SUCCESS, 0}; T_TIME tm_val; T_STK_POLL_DATA *p; PALLOC (sim_toolkit_ind, SIM_TOOLKIT_IND); MALLOC (p, sizeof (T_STK_POLL_DATA)); /* * clear indication primitive */ memset (sim_toolkit_ind, 0, sizeof (T_SIM_TOOLKIT_IND)); sim_toolkit_ind->stk_cmd.l_cmd = ((USHORT)sim_data.proactive_sim_data_len) << 3; ber_len = (SHORT)sim_data.proactive_sim_data_len; sim_data.proactive_sim_data_len = 0; sim_data.term_resp_sent = FALSE; if (FKT_Fetch(sim_toolkit_ind->stk_cmd.cmd, ber_len) EQ SIM_NO_ERROR) { UBYTE tag; BOOL tag_not_found = TRUE; #ifdef _SIMULATION_ TRACE_EVENT("FKT_Fetch_EQ_SIM_NO_ERROR"); /* for debug purpose only */ #endif sim_data.ext_sat_cmd = FALSE; /* internal command by default */ stk = sim_toolkit_ind->stk_cmd.cmd; memset (found_tag, 0, sizeof(found_tag)); tag_i = cmd_i = 0; cmd_len = 0; cmd_type = 0; tag = stk_process_tl(&stk, &ber_len, &tlv_len); /* BER tag */ if (tag NEQ STK_PROACTIVE_SIM_COMMAND_TAG) { res_code[0] = STK_RES_ERR_MISS_VALUE; } else if (ber_len >= tlv_len) { ber_len = tlv_len; // omit data after BER-TLV result = TRUE; } else /* ber_len < tlv_len */ { if (tlv_len > 2) { res_code[0] = STK_RES_ERR_MISS_VALUE; result = TRUE; } else { res_code[0] = STK_RES_ERR_CMD_DATA; } } while (result AND ber_len > 0) // tag and length needed { UBYTE tag_simple; p_tag = stk; // begin of TLV tag_len = ber_len; tag_simple = stk_process_tl(&stk, &ber_len, &tlv_len); if (ber_len < tlv_len) // inconsistent length { if (res_code[0] NEQ STK_RES_ERR_MISS_VALUE) { TRACE_EVENT("ber_len < tlv_len -> STK_RES_ERR_CMD_DATA"); res_code[0] = STK_RES_ERR_CMD_DATA; } result = FALSE; break; } else ber_len -= tlv_len; // remaining BER-TLV tag_len -= ber_len; // length of TLV if (tag EQ (tag_simple & ~STK_COMPREHENSION_REQUIRED)) { #ifdef _SIMULATION_ TRACE_EVENT("TC_505A, TC_505B, TC_505C"); #endif stk += tlv_len; continue; } else { tag = tag_simple; tag_not_found = FALSE; } compreh = tag & STK_COMPREHENSION_REQUIRED; tag &= ~STK_COMPREHENSION_REQUIRED; do { p_tag_desc = &sat_cmd_list[cmd_i].tag_desc[tag_i]; if (tag EQ p_tag_desc->tag) { break; /* tag found !!! */ } else if (!p_tag_desc->mand) { tag_i++; /* skip non-mandatory tags */ } else /* mandatory tag missed */ { i = tag_i; res_code[0] = STK_RES_ERR_MISS_VALUE; while (i < (int)sat_cmd_list[cmd_i].max_tag) { p_tag_desc = &sat_cmd_list[cmd_i].tag_desc[i++]; if (tag EQ p_tag_desc->tag AND !p_tag_desc->mand) { #ifdef _SIMULATION_ TRACE_EVENT("TC_506"); #endif res_code[0] = STK_RES_ERR_CMD_DATA; break; } } if (!compreh AND i EQ (int)sat_cmd_list[cmd_i].max_tag) { #ifdef _SIMULATION_ TRACE_EVENT("TC_507"); #endif res_code[0] = STK_RES_SUCC_PART_COMPR; tag_not_found = TRUE; } break; } } while (tag_i < (int)sat_cmd_list[cmd_i].max_tag OR res_code[0]); if (tag_i >= (int)sat_cmd_list[cmd_i].max_tag) { i = (int)sat_cmd_list[cmd_i].tp_i; if ((~sat_tp_sim_ability[i] & sat_cmd_list[cmd_i].tp_flag) EQ 0) { break; /* the commmand is being handled outside the SIM */ } if (compreh) { #ifdef _SIMULATION_ TRACE_EVENT("TC_502C, TC_504"); #endif res_code[0] = STK_RES_ERR_CMD_DATA; result = FALSE; break; } } else { p_tag_desc = &sat_cmd_list[cmd_i].tag_desc[tag_i]; if (tag NEQ p_tag_desc->tag OR tlv_len < p_tag_desc->min_len) { if (compreh) { #ifdef _SIMULATION_ TRACE_EVENT("TC_503, TC_506"); #endif result = FALSE; break; } } else found_tag[tag_i] = stk; // points to parameters of tag } if (tag_not_found) { stk += tlv_len; continue; } switch (tag) { case STK_COMMAND_DETAILS_TAG: p_cmd = p_tag; // for TERMINAL RESPONSE cmd_len = tag_len; cmd_nr = stk[0]; cmd_type = stk[1]; cmd_qual = stk[2]; TRACE_FUNCTION_P2("COMMAND_DETAILS: type=0x%02X, qlf=0x%02X", (int)cmd_type, (int) cmd_qual); SIM_EM_PROACTIVE_COMMAND; for (i = 1; i < (SHORT)item_of(sat_cmd_list); i++) /* find out if known command */ { if (cmd_type EQ sat_cmd_list[i].cmd_type) { cmd_i = i; break; } } if (i EQ item_of(sat_cmd_list)) { TRACE_EVENT_P1("UNKNOWN COMMAND TYPE:0x%02X", cmd_type); } break; case STK_DEVICE_IDENTITY_TAG: #ifdef _SIMULATION_ TRACE_EVENT("found SIMPLE_TAG: STK_DEVICE_IDENTITY_TAG"); #endif break; case STK_CHANNEL_DATA_TAG: #ifdef _SIMULATION_ TRACE_EVENT("found SIMPLE_TAG: STK_CHANNEL_DATA_TAG"); #endif break; case STK_CHANNEL_DATA_LGTH_TAG: #ifdef _SIMULATION_ TRACE_EVENT("found SIMPLE_TAG: STK_CHANNEL_DATA_LGTH_TAG"); #endif break; case STK_FILE_LIST_TAG: fl_len = tlv_len; break; default: #ifdef _SIMULATION_ TRACE_EVENT_P1("other tag=0x%02X", tag); #endif break; } if (++tag_i >= STK_MAX_EXP_TAG) /* incremented index points to next tag_desc entry */ break; stk += tlv_len; } if (!result) { if (found_tag[0] EQ 0) { switch (res_code[0]) { case STK_RES_ERR_MISS_VALUE: #ifdef _SIMULATION_ TRACE_EVENT("TC_501, TC_508"); #endif memcpy (p->response, &terminal_response_def, SAT_TRSP_MIN_RES); ((T_SIM_TRSP_SIMPLE *)p->response)->cmd_nr = cmd_nr; ((T_SIM_TRSP_SIMPLE *)p->response)->cmd_type = cmd_type; ((T_SIM_TRSP_SIMPLE *)p->response)->res_gnrl = res_code[0]; FKT_TerminalResponse (p->response, SAT_TRSP_MIN_RES); break; default: break; } } else if (found_tag[0] AND cmd_i NEQ 0) { switch (res_code[0]) { case STK_RES_ERR_CMD_DATA: #ifdef _SIMULATION_ TRACE_EVENT("TC_502A, TC_502B, TC_502C, TC_506"); #endif memcpy (p->response, &terminal_response_def, SAT_TRSP_MIN_RES); ((T_SIM_TRSP_SIMPLE *)p->response)->cmd_nr = cmd_nr; ((T_SIM_TRSP_SIMPLE *)p->response)->cmd_type = cmd_type; ((T_SIM_TRSP_SIMPLE *)p->response)->res_gnrl = res_code[0]; FKT_TerminalResponse (p->response, SAT_TRSP_MIN_RES); break; case STK_RES_ERR_MISS_VALUE: #ifdef _SIMULATION_ TRACE_EVENT("TC_503, TC_509A"); #endif memcpy (p->response, &terminal_response_def, SAT_TRSP_MIN_RES); ((T_SIM_TRSP_SIMPLE *)p->response)->cmd_nr = cmd_nr; ((T_SIM_TRSP_SIMPLE *)p->response)->cmd_type = cmd_type; ((T_SIM_TRSP_SIMPLE *)p->response)->res_gnrl = res_code[0]; FKT_TerminalResponse (p->response, SAT_TRSP_MIN_RES); break; default: break; } } else if (cmd_i EQ 0) { #ifdef _SIMULATION_ TRACE_EVENT("TC_504"); #endif res_code[0] = STK_RES_ERR_CMD_TYPE; memcpy (p->response, &terminal_response_def, SAT_TRSP_MIN_RES); ((T_SIM_TRSP_SIMPLE *)p->response)->cmd_nr = cmd_nr; ((T_SIM_TRSP_SIMPLE *)p->response)->cmd_type = cmd_type; ((T_SIM_TRSP_SIMPLE *)p->response)->res_gnrl = res_code[0]; FKT_TerminalResponse (p->response, SAT_TRSP_MIN_RES); } else { resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("SAT: other error"); #endif FKT_TerminalResponse (p->response, (USHORT)resp_len); } PFREE (sim_toolkit_ind); } else switch (cmd_type) { case STK_DISPLAY_TEXT: case STK_GET_INKEY: case STK_GET_INPUT: case STK_PLAY_TONE: case STK_SET_UP_MENU: case STK_SELECT_ITEM: case STK_SEND_SMS: case STK_SEND_SS: case STK_SET_UP_CALL: case STK_SETUP_EVENT_LIST: case STK_SEND_USSD: case STK_SEND_DTMF: case STK_SETUP_IDLE_TEXT: case STK_LAUNCH_BROWSER: case STK_RUN_AT_CMD: case STK_LANGUAGE_NOTIFICATION: #ifdef FF_SAT_E case STK_OPEN_CHANNEL: case STK_CLOSE_CHANNEL: case STK_GET_CHANNEL_STAT: #endif /* FF_SAT_E */ /* * Operations processed by MMI / MFW */ sim_data.sat_session = TRUE; sim_data.ext_sat_cmd = TRUE; PSENDX (MMI, sim_toolkit_ind); break; case STK_POLL_INTERVALL: /* * SIM toolkit defines an idle poll interval. The timer in * SIM application is stopped and started with the * new value. A terminal response with the used time * is sent to SIM toolkit. */ if (found_tag[2] NEQ NULL) { BOOL org_val; TRACE_EVENT("Idle Polling switched on"); if (found_tag[2][1] NEQ 0) /* value not "reserved" */ { switch (found_tag[2][0]) { case 0: /* minutes */ if ((ULONG)found_tag[2][1] * 600L >= (ULONG)cusSatMinPollItv) { sim_data.status_time = ((T_TIME)found_tag[2][1]) * 60L * 1000 - TIMER_LATENCY; org_val = TRUE; } else org_val = FALSE; break; case 1: /* seconds */ if ((USHORT)found_tag[2][1] * 10 >= cusSatMinPollItv) { sim_data.status_time = ((T_TIME)found_tag[2][1]) * 1000L - TIMER_LATENCY; org_val = TRUE; } else org_val = FALSE; break; case 2: /* tenth seconds */ if ((USHORT)found_tag[2][1] >= cusSatMinPollItv) { sim_data.status_time = ((T_TIME)found_tag[2][1]) * 100L - TIMER_LATENCY; org_val = TRUE; } else org_val = FALSE; break; default: /* command data not understood */ res_code[0] = STK_RES_ERR_CMD_DATA; org_val = TRUE; break; /* return original values */ } } else /* return original values */ { res_code[0] = STK_RES_ERR_CMD_DATA; org_val = TRUE; } resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, (UBYTE *)tag_dur, sizeof(tag_dur)); #ifdef _SIMULATION_ TRACE_EVENT("POLL_ITV data not understood"); #endif if (org_val) { p->response[cmd_len+ 9] = found_tag[2][0]; /* copy unit */ /* copy number of units (1 to 255) or 0 !!! */ p->response[cmd_len+10] = found_tag[2][1]; } else { sim_data.status_time = ((T_TIME)cusSatMinPollItv) * 100L - TIMER_LATENCY; if (cusSatMinPollItv < 0x100) { p->response[cmd_len+ 9] = 2; // fits to tenth of seconds p->response[cmd_len+10] = (UBYTE)cusSatMinPollItv; } else if (cusSatMinPollItv < (0x100*10)) { p->response[cmd_len+ 9] = 1; // fits to seconds p->response[cmd_len+10] = (UBYTE)((cusSatMinPollItv / 5 + 1) / 2); } else { p->response[cmd_len+ 9] = 0; // fits to minutes p->response[cmd_len+10] = (UBYTE)((cusSatMinPollItv / 300 + 1) / 2); } } sim_data.idle_polling = TRUE; } else { res_code[0] = STK_RES_ERR_MISS_VALUE; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("POLL_ITV missing value"); #endif } FKT_TerminalResponse (p->response, (USHORT)resp_len); PFREE (sim_toolkit_ind); if (SIM_IS_FLAG_CLEARED (CALL_ACTIVE) OR sim_data.status_time < THIRTY_SECONDS) TIMER_PSTART( sim_handle, SIM_TIMER, sim_data.status_time, sim_data.status_time); break; case STK_PROVIDE_LOCAL_INFO: /* * SIM toolkit request local information from the * SIM application. These informations are: * MCC, MNC, LAC, Cell Identity or IMEI. * SIM application sends this information with * a terminal response to the SIM toolkit. */ if(p_cmd EQ NULL) { TRACE_EVENT("p_cmd is NULL"); return; } switch (cmd_qual) { case QLF_PLOI_LOC_INFO: /* * request location information */ #ifdef _SIMULATION_ if (sim_data.location_info.c_loc NEQ 0) { memcpy (p->response, p_cmd, cmd_len); /* location information available */ memcpy (&p->response[cmd_len], &terminal_response_loci[5], 9); /* copy MCC, MNC & LAC */ memcpy (&p->response[cmd_len+ 9], &sim_data.location_info.loc[4], 5); /* copy cell identity */ p->response[cmd_len+14] = sim_data.cell_identity >> 8; p->response[cmd_len+15] = sim_data.cell_identity & 0xFF; p->response[cmd_len+ 6] = res_code[0]; resp_len = cmd_len + 16; #else if((cl_shrd_get_loc (&loc_info) EQ TRUE) AND (loc_info.service_mode NEQ NO_SERVICE)) { memcpy (p->response, p_cmd, cmd_len); /* location information available */ memcpy (&p->response[cmd_len], &terminal_response_loci[5], 9); /* copy from loc_info to response */ p->response[cmd_len+9] = loc_info.mcc[1] << 4; p->response[cmd_len+9] += loc_info.mcc[0]; p->response[cmd_len+10] = loc_info.mnc[2] << 4; p->response[cmd_len+10] += loc_info.mcc[2]; p->response[cmd_len+11] = loc_info.mnc[1] << 4; p->response[cmd_len+11] += loc_info.mnc[0]; p->response[cmd_len+12] = loc_info.lac >> 8; p->response[cmd_len+13] = loc_info.lac & 0xff; p->response[cmd_len+14] = loc_info.cell_id >> 8; p->response[cmd_len+15] = loc_info.cell_id & 0xFF; if ( loc_info.service_mode EQ LIMITED_SERVICE ) { p->response[cmd_len+ 6] = STK_RES_SUCC_LIMITED_SERVICE; } else { p->response[cmd_len+ 6] = res_code[0]; } resp_len = cmd_len + 16; #endif /* _SIMULATION_ */ } else { /* location information not available */ res_code[0] = STK_RES_BUSY_ME; res_code[1] = STK_RES_EXT_NO_SERVICE; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 2, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("PLI: local info not available"); #endif } sim_data.ext_sat_cmd = FALSE; break; case QLF_PLOI_IMEI: /* * request IMEI */ memcpy (p->response, p_cmd, cmd_len); memcpy (&p->response[cmd_len], &terminal_response_imei[5], 17); cl_get_imeisv(CL_IMEI_SIZE, &p->response[cmd_len + 9], CL_IMEI_GET_STORED_IMEI); for (i = (int)cmd_len + 8 + CL_IMEI_SIZE; i > (int)cmd_len + 9; i--) { /* place nibbles according to 4.08 */ p->response[i] = (p->response[i] & 0xf0) | (p->response[i - 1] & 0x0f); } /* i points to first IMEI octet, set 'type of identity' */ p->response[i] = (p->response[i] & 0xf0) | 0xA; p->response[cmd_len+8+CL_IMEI_SIZE] &= 0xF; p->response[cmd_len+ 6] = res_code[0]; resp_len = cmd_len + 17; sim_data.ext_sat_cmd = FALSE; break; case QLF_PLOI_NTW_MSR: /* * request network measurement results and BCCH list */ if ((sat_tp_sim_enabled[3] & SAT_TP4_PLI_NMR) EQ 0) { /* if not effective */ res_code[0] = STK_RES_ERR_NO_SUPPORT; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("PLI: NMR not supported"); #endif break; } memcpy (p->response, p_cmd, cmd_len); /* location information available */ memcpy (&p->response[cmd_len], &terminal_response_nmr[5], 9); if (get_network_meas (&p->response[cmd_len+9])) { /* * set result code */ #ifdef _SIMULATION_ p->response[cmd_len+ 6] = res_code[0]; resp_len = cmd_len + 25; #else if ((cl_shrd_get_loc (&loc_info) EQ TRUE) AND (loc_info.service_mode NEQ NO_SERVICE)) { if ( loc_info.service_mode EQ LIMITED_SERVICE ) { p->response[cmd_len+ 6] = STK_RES_SUCC_LIMITED_SERVICE; } else { p->response[cmd_len+ 6] = res_code[0]; } resp_len = cmd_len + 25; } else { res_code[0] = STK_RES_BUSY_ME; res_code[1] = STK_RES_EXT_NO_SERVICE; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 2, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("PLI: service mode not available"); #endif /* _SIMULATION_ */ } #endif /* _SIMULATION_ */ /* * Add BCCH list */ if (get_bcch_chan_list (&p->stk_data)) { /* * copy BCCH data list for answer */ p->response[resp_len] = 0x1D; /* BCCH channel flag */ p->response[resp_len + 1] = p->stk_data.stk_length; memcpy (&p->response[resp_len + 2], p->stk_data.stk_parameter, p->stk_data.stk_length); resp_len += (2+p->stk_data.stk_length); } else { /* location information not available */ res_code[0] = STK_RES_BUSY_ME; res_code[1] = STK_RES_EXT_NO_SERVICE; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 2, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("PLI: BCCH list not available"); #endif } } else { /* location information not available */ res_code[0] = STK_RES_BUSY_ME; res_code[1] = STK_RES_EXT_NO_SERVICE; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 2, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("PLI: measurement not available"); #endif } sim_data.ext_sat_cmd = FALSE; break; case QLF_PLOI_DTT: /* * Date, Time and Timezone */ if ((sim_data.stk_profile[7] & SAT_TP8_PLI_DTT) EQ 0) { /* if not effective */ res_code[0] = STK_RES_ERR_NO_SUPPORT; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("PLI: not supported in TP"); #endif break; } /* * Operations processed by MMI / MFW */ sim_data.ext_sat_cmd = TRUE; PSENDX (MMI, sim_toolkit_ind); break; case QLF_PLOI_LANG_SET: /* * Language Setting to be supported (to be processed by ACI) */ if ((sim_data.stk_profile[8] & SAT_TP9_PLI_LANG) EQ 0) { /* if not effective */ res_code[0] = STK_RES_ERR_NO_SUPPORT; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("language setting not supported"); #endif break; } /* * To be processed by ACI */ sim_data.sat_session = TRUE; sim_data.ext_sat_cmd = TRUE; PSENDX (MMI, sim_toolkit_ind); break; case QLF_PLOI_TIM_ADV: if ((sat_tp_sim_enabled[8] & SAT_TP9_PLI_TIMING_ADV) EQ 0) { res_code[0] = STK_RES_ERR_NO_SUPPORT; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("PLI: timing advance not supported"); #endif } else { memset (&tim_adv, 0, sizeof(T_TIM_ADV)); memcpy (p->response, p_cmd, cmd_len); memcpy (&p->response[cmd_len], &terminal_response_timingadv[5], 11); #ifdef _SIMULATION_ p->response[cmd_len+ 6] = res_code[0]; resp_len = cmd_len + 11; #else if(cl_shrd_get_tim_adv (&tim_adv) EQ TRUE) { p->response[cmd_len+9] = tim_adv.me_status; p->response[cmd_len+10] = tim_adv.tm_adv; /* * Set result code. */ p->response[cmd_len+ 6] = res_code[0]; resp_len = cmd_len + 11; } else { res_code[0] = STK_RES_BUSY_ME; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("PLI: Timing Advance not available"); #endif } #endif /* _SIMULATION_ */ } sim_data.ext_sat_cmd = FALSE; break; default: /* * no valid command qualifier */ res_code[0] = STK_RES_ERR_CMD_TYPE; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("PLI: type not understood"); #endif sim_data.ext_sat_cmd = FALSE; break; } if (!sim_data.ext_sat_cmd) /* may depend on Command Qualifier */ { FKT_TerminalResponse (p->response, (USHORT)resp_len); PFREE (sim_toolkit_ind); } break; case STK_REFRESH: /* * process SIM REFRESH according to command qualifier */ TRACE_EVENT("process SIM REFRESH according to command qualifier"); /* MALLOC area for CONTEXT SWITCH */ if ( sim_data.context_switch_ptr != NULL ) { TRACE_EVENT("ERROR!! context_switch_ptr should be NULL - freeing"); MFREE ( sim_data.context_switch_ptr ); sim_data.context_switch_ptr = NULL; } MALLOC (sim_data.context_switch_ptr, sizeof (T_CONTEXT_SWITCH)); memset ( sim_data.context_switch_ptr,0, sizeof (T_CONTEXT_SWITCH)); /* Set sig_ptr to handle required signal */ #ifdef TI_PS_FF_AT_P_CMD_CUST if (sim_data.cust_mode EQ 1) { /* Cust_Mode 1 operation setup*/ PALLOC(copy_sim_toolkit_ind,SIM_TOOLKIT_IND); memcpy(copy_sim_toolkit_ind,sim_toolkit_ind,sizeof(T_SIM_TOOLKIT_IND)); sim_data.context_switch_ptr->sig_ptr = copy_sim_toolkit_ind; } else { /*default operation setup*/ sim_data.context_switch_ptr->sig_ptr = sim_toolkit_ind; } #else /*default operation setup*/ sim_data.context_switch_ptr->sig_ptr = sim_toolkit_ind; #endif /* TI_PS_FF_AT_P_CMD_CUST */ /* Save decoding of original sim_toolkit_ind signal processed by stk_proactive_polling */ offset = (long)found_tag[2] - (long)sim_toolkit_ind; sim_data.context_switch_ptr->tag2 = (UBYTE *)(sim_data.context_switch_ptr->sig_ptr) + offset; sim_data.context_switch_ptr->fl_len = fl_len; offset = (long)p_cmd - (long)sim_toolkit_ind; sim_data.context_switch_ptr->p_cmd = (UBYTE *)(sim_data.context_switch_ptr->sig_ptr) + offset; sim_data.context_switch_ptr->cmd_len = cmd_len; sim_data.context_switch_ptr->res_code[0]=res_code[0]; sim_data.context_switch_ptr->res_code[1]=res_code[1]; #ifdef TI_PS_FF_AT_P_CMD_CUST /* functionality is depended on cust_mode */ if (sim_data.cust_mode EQ 1) { /* Cust_Mode 1 operation */ sim_data.user_confirmation_expected = TRUE; /*send Refresh command to MMI*/ PSENDX (MMI, sim_toolkit_ind); /*REFRESH COMMAND NOW PROCESSED IN stk_sim_refresh_user_res() */ /*note: stk_sim_refresh_user_res() will call process_sim_refresh() */ /* This routine will free sig_ptr, user must free the */ /* context_switch. */ /*Terminate Command */ break; } else { /* sim_data.cust_mode EQ 0 */ /* process default operation */ if ( process_sim_refresh(sim_data.context_switch_ptr) ) { /* processed a SIM_RESET. Special case - exit stk_proactive_polling()*/ MFREE(p); /* this thread has finished with the context switch so free it. */ MFREE (sim_data.context_switch_ptr); sim_data.context_switch_ptr = NULL; return; /* exit from stk_proactive_polling() */ } else { /* process_sim_refresh() frees signal sim_data.context_switch_ptr->sig_ptr */ /* this thread has finished with the context switch so free it. */ MFREE (sim_data.context_switch_ptr); sim_data.context_switch_ptr = NULL; } } #else /* sim_data.cust_mode EQ 0 */ /* process default operation */ if ( process_sim_refresh(sim_data.context_switch_ptr) ) { /* processed a SIM_RESET. Special case - exit stk_proactive_polling()*/ MFREE(p); /* this thread has finished with the context switch so free it. */ MFREE (sim_data.context_switch_ptr); sim_data.context_switch_ptr = NULL; return; /* exit from stk_proactive_polling() */ } else { /* process_sim_refresh() frees signal sim_data.context_switch_ptr->sig_ptr */ /* this thread has finished with the context switch so free it. */ MFREE (sim_data.context_switch_ptr); sim_data.context_switch_ptr = NULL; } #endif /* TI_PS_FF_AT_P_CMD_CUST */ break; //end of case STK_REFRESH #ifdef FF_SAT_E case STK_RECEIVE_DATA: if (stk_dti_bip_receive_data((T_sdu*)&sim_toolkit_ind->stk_cmd, res_code[0])) { /* * message contains TLV elements to display * so forward it to MMI */ sim_data.sat_session = TRUE; TRACE_EVENT("SEND receive data indication to MMI"); PSENDX (MMI, sim_toolkit_ind); TRACE_EVENT("receive data indication to MMI sent"); } else { PFREE (sim_toolkit_ind); } break; case STK_SEND_DATA: if (stk_dti_bip_send_data((T_sdu*)&sim_toolkit_ind->stk_cmd)) { /* * message contains TLV elements to display * so forward it to MMI */ sim_data.sat_session = TRUE; PSENDX (MMI, sim_toolkit_ind); } else { PFREE (sim_toolkit_ind); } break; #endif /* FF_SAT_E */ case STK_TIMER_MANAGEMENT: { UBYTE tag_tv[8] = {STK_TIMER_ID_TAG|STK_COMPREHENSION_REQUIRED, STK_TIMER_ID_LEN, 0, STK_TIMER_VALUE_TAG, STK_TIMER_VALUE_LEN, 0, 0 ,0}; tag_tv[2] = found_tag[2][0]; if ((unsigned)(i = (int)found_tag[2][0] - 1) >= MAX_SAT_TIMER) { /* invalid Timer identifier */ res_code[0] = STK_RES_ERR_CMD_DATA; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("stk_build_response No 17"); #endif } /* * Process TIMER MANAGEMENT according to command qualifier */ else switch (cmd_qual) { case 0: if (found_tag[3] EQ NULL) { res_code[0] = STK_RES_ERR_MISS_VALUE; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("TM: missing parameters"); #endif break; } /* translate timer value into unit of seconds */ tm_val = ((T_TIME)BCD2INT(found_tag[3][0]) * 60 + // hours (T_TIME)BCD2INT(found_tag[3][1])) * 60 + // minutes (T_TIME)BCD2INT(found_tag[3][2]); // seconds //TISH, patch for OMAPS00115011, 2007-02-12 //start if (tm_val==0) { res_code[0] = STK_RES_ERR_CMD_DATA; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); TRACE_ERROR("TR wrong tm value"); break; } //end sim_data.timer[i].hour = found_tag[3][0]; sim_data.timer[i].minute = found_tag[3][1]; sim_data.timer[i].second = found_tag[3][2]; TIMER_START (sim_handle, (USHORT)(i+1), (tm_val * 1000 - TIMER_LATENCY)); sim_data.timer[i].active = TRUE; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, tag_tv, 3); #ifdef _SIMULATION_ TRACE_EVENT("TM: set timer"); #endif break; case 1: case 2: if (sim_data.timer[i].active) { TIMER_STATUS (sim_handle, (USHORT)(i + 1), &tm_val); tm_val /= 1000; tag_tv[7] = INT2BCD((UBYTE)(tm_val % 60)); tm_val /= 60; tag_tv[6] = INT2BCD((UBYTE)(tm_val % 60)); tm_val /= 60; if (tm_val < 24) tag_tv[5] = INT2BCD((UBYTE)tm_val); else { // set maximum value tag_tv[7] = tag_tv[6] = 0x59; tag_tv[5] = 0x23; } if (cmd_qual EQ 1) /* timer to be stopped */ { TIMER_STOP (sim_handle, (USHORT)(i + 1)); sim_data.timer[i].active = FALSE; } resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, tag_tv, 8); #ifdef _SIMULATION_ TRACE_EVENT("TM: stop/query timer"); #endif } else { res_code[0] = STK_RES_BUSY_TIMER_STATE; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("TM: timer not running"); #endif } break; default: /* * not supported information request */ res_code[0] = STK_RES_ERR_CMD_TYPE; resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("TM: unknown qualifier"); #endif break; } FKT_TerminalResponse (p->response, (USHORT)resp_len); PFREE (sim_toolkit_ind); break; } default: result = FALSE; res_code[0] = STK_RES_ERR_CMD_TYPE; /* command type not understood */ /* * fall through */ case STK_POLLING_OFF: /* * SIM toolkit cancels the effect of a previous * POLL INTERVAL command. Polling of the SIM * toolkit is done now all thirty seconds with the * STATUS command, when in a call. */ if (result AND SIM_IS_FLAG_SET (PRO_ACTIVE_SIM)) { TRACE_EVENT("Idle Polling switched off"); sim_data.idle_polling = FALSE; /* there will not be a status command after this .so just force one status command to enable sleep */ startTimerPollOff = TRUE; if(SIM_IS_FLAG_SET(CALL_ACTIVE)) { /* SIM_TIMER will be restarted, during a call, for * presence detection */ sim_data.status_time = THIRTY_SECONDS; } else { /* timer value set to max, for with a '0', the timer would run for about one tick */ sim_data.status_time = 0xFFFFFFFF; TIMER_STOP (sim_handle, SIM_TIMER); } } case STK_MORE_TIME: /* * SIM toolkit needs more time. * SIM application sends terminal response with OK. */ resp_len = stk_build_response (p->response, p_cmd, cmd_len, res_code, 1, NULL, 0); #ifdef _SIMULATION_ TRACE_EVENT("MORE_TIME"); #endif FKT_TerminalResponse (p->response, (USHORT)resp_len); PFREE (sim_toolkit_ind); break; } } else { #ifdef _SIMULATION_ TRACE_EVENT("FKT_Fetch_NEQ_SIM_NO_ERROR"); #endif PFREE (sim_toolkit_ind); } MFREE (p); if (vsi_c_status (VSI_CALLER &in_queue, &out_queue) EQ VSI_OK) { if (in_queue > 0) break; /* break while */ } } /* END while */ /* * send end of SAT session indicator */ TRACE_FUNCTION ("stk_proactive_polling() send end of SAT session indicator"); if ((sim_data.term_resp_sent) AND (sim_data.sat_session)) { PALLOC (sim_toolkit_ind, SIM_TOOLKIT_IND); memset (sim_toolkit_ind, 0, sizeof (T_SIM_TOOLKIT_IND)); #ifdef _SIMULATION_ TRACE_EVENT("SAT session ended"); #endif sim_data.sat_session = FALSE; sim_data.term_resp_sent = FALSE; PSENDX (MMI, sim_toolkit_ind); } } TRACE_FUNCTION ("stk_proactive_polling() exited"); } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_stop_all_sat_timers | +--------------------------------------------------------------------+ PURPOSE : Stop all timers started by TIMER MANAGEMENT, if running. Required when card is deactivated or reset. */ GLOBAL void stk_stop_all_sat_timers (void) { int i; for (i = 0; i < MAX_SAT_TIMER; i++) { if (sim_data.timer[i].active) { TIMER_STOP (sim_handle, (USHORT)(i + 1)); sim_data.timer[i].active = FALSE; } } } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_start_timer_and_poll | +--------------------------------------------------------------------+ PURPOSE : Start STATUS timer and control SIM Toolkit Polling */ LOCAL void stk_start_timer_and_poll (void) { T_TIME t_val; /* * start status timer if SIM is inserted * for periodic status polling of SIM Toolkit */ if ((SIM_IS_FLAG_SET (PRO_ACTIVE_SIM) AND sim_data.idle_polling AND SIM_IS_FLAG_CLEARED (TEST_MODE_POLLING)) OR SIM_IS_FLAG_SET (CALL_ACTIVE)) { /* * Start Status Polling */ t_val = (SIM_IS_FLAG_SET (CALL_ACTIVE) AND sim_data.status_time > THIRTY_SECONDS)? THIRTY_SECONDS: sim_data.status_time; TIMER_PSTART (sim_handle, SIM_TIMER, t_val, t_val); } sim_data.chk_sat_avail = TRUE; // stk_proactive_polling(); } #ifdef FF_SAT_E /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_handle_ccd_error | +--------------------------------------------------------------------+ PURPOSE : Handle CCD error causes. */ LOCAL void stk_handle_ccd_error(UBYTE notOK, UBYTE* status, UBYTE* general_result, UBYTE* add_info_result) { UBYTE ccd_error; USHORT parameter_list[MAX_ERR_PAR]; TRACE_EVENT_P1("stk_handle_ccd_error(): %d", notOK); /* * handle errors in list */ memset (parameter_list, 0, sizeof(parameter_list)); ccd_error = ccd_getFirstError(CCDENT_SAT, parameter_list); if (notOK EQ ccdWarning) { while (ccd_error NEQ ERR_NO_MORE_ERROR) { switch (ccd_error) { case ERR_COMPREH_REQUIRED: /* * comprehension required */ TRACE_ERROR("CCD_WARNING=ERR_COMPREH_REQUIRED"); if (*status EQ SIM_CCD_OK) { *general_result = RSLT_UNKN_DATA; *add_info_result = ADD_NO_CAUSE; *status = SIM_CCD_RETURN; } break; case ERR_IE_NOT_EXPECTED: /* * unexpected element */ TRACE_ERROR("CCD_WARNING=ERR_IE_NOT_EXPECTED"); if (*status EQ SIM_CCD_OK) { *general_result = RSLT_PERF_PART_CMPR; *add_info_result = ADD_NO_CAUSE; } break; case ERR_MAND_ELEM_MISS: /* * mandatory elements missing */ TRACE_ERROR("CCD_WARNING=ERR_MAND_ELEM_MISS"); *general_result = RSLT_ERR_REQ_VAL; *add_info_result = ADD_NO_CAUSE; *status = SIM_CCD_RETURN; break; case ERR_MSG_LEN: TRACE_ERROR("CCD_WARNING=ERR_MSG_LEN (TC_509B)"); *general_result = RSLT_PERF_SUCCESS; *add_info_result = ADD_NO_CAUSE; *status = SIM_CCD_OK; break; default: TRACE_ERROR("CCD_WARNING ignored"); TRACE_EVENT_P1("ccd_error=0x%02X", ccd_error); break; } memset (parameter_list,0, sizeof(parameter_list)); ccd_error = ccd_getNextError (CCDENT_SAT, parameter_list); } } else /* ccdError */ { while (ccd_error NEQ ERR_NO_MORE_ERROR) { switch (ccd_error) { case ERR_COMPREH_REQUIRED: /* * comprehension required */ TRACE_ERROR("CCD_ERROR=ERR_COMPREH_REQUIRED"); if (*status EQ SIM_CCD_OK) { *general_result = RSLT_UNKN_DATA; *add_info_result = ADD_NO_CAUSE; *status = SIM_CCD_RETURN; } break; case ERR_IE_NOT_EXPECTED: /* * unexpected element */ TRACE_ERROR("CCD_ERROR=ERR_IE_NOT_EXPECTED"); if (*status EQ SIM_CCD_OK) { *general_result = RSLT_PERF_PART_CMPR; *add_info_result = ADD_NO_CAUSE; } break; case ERR_MAND_ELEM_MISS: /* * mandatory elements missing */ TRACE_ERROR("CCD_ERROR=ERR_MAND_ELEM_MISS"); *general_result = RSLT_ERR_REQ_VAL; *add_info_result = ADD_NO_CAUSE; *status = SIM_CCD_RETURN; break; case ERR_MSG_LEN: TRACE_ERROR("CCD_ERROR=ERR_MSG_LEN"); *general_result = RSLT_UNKN_DATA; *add_info_result = ADD_NO_CAUSE; *status = SIM_CCD_DISCARD; break; default: TRACE_ERROR("CCD_ERROR ignored"); TRACE_EVENT_P1("ccd_error=0x%02X", ccd_error); break; } memset (parameter_list,0, sizeof(parameter_list)); ccd_error = ccd_getNextError (CCDENT_SAT, parameter_list); } } } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_bip_decode_stk_command | +--------------------------------------------------------------------+ PURPOSE : First step of decoding SEND DATA and RECEIVE DATA message. */ LOCAL void stk_bip_decode_stk_command(T_sdu* message, T_cmd_details* cmd_details, UBYTE* status, UBYTE* general_result, UBYTE* add_info_result) { UBYTE ccdRet; MCAST(stk_cmd, STK_CMD); memset(stk_cmd, 0, sizeof(T_STK_CMD)); /* * decode toolkit command */ ccdRet = ccd_decodeMsg (CCDENT_SAT, DOWNLINK, (T_MSGBUF *) message, (UBYTE *) _decodedMsg, STK_CMD); #ifdef _SIMULATION_ TRACE_EVENT_P1("ccdRet@bip_decode_stk_command: %d", ccdRet); #endif if ((stk_cmd->v_pas_cmd) AND (stk_cmd->pas_cmd.v_cmd_details)) { /* * store command details */ *cmd_details = stk_cmd->pas_cmd.cmd_details; if (ccdRet NEQ ccdOK) { /* * handle errors in list */ stk_handle_ccd_error(ccdRet, status, general_result, add_info_result); } if ((stk_cmd->pas_cmd.v_cmd_prms EQ FALSE) OR (stk_cmd->pas_cmd.v_dev_ids EQ FALSE)) { /* * no Channel Data (Length) element present or * no Device Identities element present * Error, required values are missing */ *general_result = RSLT_ERR_REQ_VAL; *add_info_result = ADD_NO_CAUSE; *status = SIM_CCD_RETURN; } if (*status EQ SIM_CCD_OK) { int i = 0; while (tbl_device_src_id[i]) { if (stk_cmd->pas_cmd.dev_ids.dest_dev EQ tbl_device_src_id[i]) break; i++; } if (tbl_device_src_id[i] EQ 0) { /* * device id not valid */ *general_result = RSLT_UNKN_DATA; *add_info_result = ADD_NO_CAUSE; *status = SIM_CCD_RETURN; } else if /* norm sim_data.bip_ch_id for 11.14/12.7 conform */ ((sim_data.bip_state EQ SIM_BIP_CLOSED) OR (stk_cmd->pas_cmd.dev_ids.dest_dev NEQ (sim_data.bip_ch_id | 0x020))) { /* * BIP error, Channel identifier not valid */ *general_result = RSLT_BEARIND_PERR; *add_info_result = ADD_BIP_CHANID_NT_VLD; *status = SIM_CCD_RETURN; } else { /* * store command parameters */ sim_data.bip_cmd_prms = stk_cmd->pas_cmd.cmd_prms; } } } else { /* * incomplete message */ *general_result = RSLT_ERR_REQ_VAL; *add_info_result = ADD_NO_CAUSE; *status = SIM_CCD_RETURN; /* * set Command Details object values to 0x00 */ cmd_details->cmd_nr = 0; cmd_details->cmd_typ = 0; cmd_details->cmd_qlf = 0; } } /* stk_bip_decode_stk_command */ /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_dti_send_data | +-----------------------------------------------------------------------------+ PURPOSE : Create and send a DTI Data primitive. */ LOCAL void stk_dti_send_data() { T_desc2* temp_desc1; /* * send DTI data primitive */ PALLOC_DESC2 (dti_data_ind, DTI2_DATA_IND); #ifdef _SIMULATION_ dti_data_ind->parameters.p_id = DTI_PID_UOS; dti_data_ind->parameters.st_lines.st_flow = DTI_FLOW_ON; dti_data_ind->parameters.st_lines.st_line_sa = DTI_SA_ON; dti_data_ind->parameters.st_lines.st_line_sb = DTI_SB_ON; dti_data_ind->parameters.st_lines.st_break_len = DTI_BREAK_OFF; #endif /* _SIMULATION_ */ switch(sim_data.con_type) { case SIM_CON_TYPE_UDP: MALLOC(temp_desc1, (USHORT)(sizeof(T_desc2) - 1 + sizeof(T_SRC_DES))); memcpy(temp_desc1->buffer, &sim_data.udp_parameters, sizeof(T_SRC_DES));/*lint !e419 Apparent data overrun for function*/ temp_desc1->len = sizeof(T_SRC_DES); temp_desc1->next = sim_data.data_to_send.first; sim_data.data_to_send.first = (ULONG)temp_desc1; sim_data.data_to_send.list_len+= temp_desc1->len; break; case SIM_CON_TYPE_IP: dti_data_ind->parameters.p_id = DTI_PID_IP; break; case SIM_CON_TYPE_SERIAL: dti_data_ind->parameters.st_lines.st_flow = DTI_FLOW_ON; dti_data_ind->parameters.st_lines.st_line_sa = DTI_SA_ON; dti_data_ind->parameters.st_lines.st_line_sb = DTI_SB_ON; dti_data_ind->parameters.st_lines.st_break_len = DTI_BREAK_OFF; break; } dti_data_ind->desc_list2 = sim_data.data_to_send; sim_data.data_to_send.first = (ULONG)NULL; sim_data.data_to_send.list_len = 0; dti_send_data(sim_data.hDTI, 0, 0, 0, dti_data_ind); } /* stk_dti_send_data() */ /* +-----------------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_bip_send_data_terminal_response | +-----------------------------------------------------------------------------+ PURPOSE : Send a Terminal Response message for a SEND DATA message. */ LOCAL void stk_bip_send_data_terminal_response(UBYTE general_result, UBYTE add_info_result) { T_sdu* temp_sdu; UBYTE ccdRet; /* * memory for encoded message */ MALLOC(temp_sdu, (USHORT)(sizeof(T_sdu) - 1 + MAX_STK_CMD)); temp_sdu->l_buf = MAX_STK_CMD << 3; temp_sdu->o_buf = 0; CCD_START; { MCAST(term_resp, TERM_RESP); memset(term_resp, 0, sizeof(T_TERM_RESP)); /* * set Command details */ term_resp->v_cmd_details = TRUE; term_resp->cmd_details = sim_data.bip_tx_cmd_details; /* * set set Device identities */ term_resp->v_dev_ids = TRUE; term_resp->dev_ids.src_dev = DEV_SRC_ME; term_resp->dev_ids.dest_dev = DEV_DST_SIM; /* * set Result */ term_resp->v_res = TRUE; term_resp->res.gen = general_result; switch(general_result) { case RSLT_ME_UNAB_PROC: case RSLT_NTW_UNAB_PROC: case RSLT_LABRWS_GENERIC: case RSLT_SS_ERR: case RSLT_SMS_ERR: case RSLT_USSD_ERR: case RSLT_CC_SIM_PRM: case RSLT_BEARIND_PERR: /* * one byte for additional information */ term_resp->res.v_add = TRUE; term_resp->res.add.l_add = 1 << 3; term_resp->res.add.o_add = 0; term_resp->res.add.b_add[0] = add_info_result; break; } /* * set Channel Data Length */ term_resp->v_chan_dat_lth = TRUE; if ((SIM_CLASS_E_BUFFER_SIZE - sim_data.data_to_send.list_len) > 255) { /* * more than 255 bytes are available in TX buffer */ term_resp->chan_dat_lth = 0xff; } else { term_resp->chan_dat_lth = SIM_CLASS_E_BUFFER_SIZE - sim_data.data_to_send.list_len; } /* * encode message */ ccdRet = ccd_codeMsg (CCDENT_SAT, UPLINK, (T_MSGBUF *) temp_sdu, (UBYTE *) _decodedMsg, TERM_RESP); } CCD_END; /* * send Terminal Response */ if (ccdRet NEQ ccdOK) { TRACE_EVENT_P1("SEND DATA: CCD Coding Error: %d",ccdRet ); } else { FKT_TerminalResponse (temp_sdu->buf, (USHORT)(temp_sdu->l_buf >> 3)); } MFREE(temp_sdu); } /* stk_bip_send_data_terminal_response() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_close_dti_connection | +--------------------------------------------------------------------+ PURPOSE : Close DTI connection. PARAMETER: The open parameter indicates if the DTI connection is still open or already closed. */ LOCAL void stk_close_dti_connection(UBYTE close) { #ifdef _SIMULATION_ TRACE_FUNCTION ("stk_close_dti_connection()"); #endif #ifdef _SIMULATION_ TRACE_EVENT_P1(" |___-->: sim_data.dti_connection_state=0x%02X", sim_data.dti_connection_state); #endif if (sim_data.dti_connection_state NEQ SIM_DTI_CONNECTION_CLOSED) { /* * close UDP connection */ if (sim_data.con_type EQ SIM_CON_TYPE_UDP) { /* * close currently used port */ if ((sim_data.dti_connection_state NEQ SIM_DTI_CONNECTION_BIND) AND (hCommUDP >= VSI_OK)) { PALLOC(udp_closeport_req, UDP_CLOSEPORT_REQ); udp_closeport_req->port = sim_data.udp_parameters.src_port[0]; udp_closeport_req->port = (udp_closeport_req->port << 8); udp_closeport_req->port+= sim_data.udp_parameters.src_port[1]; PSEND(hCommUDP, udp_closeport_req); } /* * release VSI channel */ if (hCommUDP >= VSI_OK) { vsi_c_close (VSI_CALLER hCommUDP); } hCommUDP = VSI_ERROR; } /* * disconnect BIP channel */ if (sim_data.bip_state EQ SIM_BIP_CONNECTED) { sim_data.bip_state = SIM_BIP_OPEN; /* * stop release timer */ if (sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) { if (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START) { #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: stopped"); #endif TIMER_STOP (sim_handle, SIM_BIP_TIMER); } sim_data.bip_timer_state = SIM_BIP_TIMER_DISCONNECTED; } } /* * close DTI connection */ sim_data.dti_connection_state = SIM_DTI_CONNECTION_CLOSED; if (close) { dti_close(sim_data.hDTI, 0, 0, 0, FALSE); } #ifdef _SIMULATION_ TRACE_EVENT_P1("stk_close_dti_connection: sim_data.dti_connection_state=0x%02X", sim_data.dti_connection_state); #endif } } /* stk_close_dti_connection() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_close_bip_channel | +--------------------------------------------------------------------+ PURPOSE : Close Bearer Independent Protocol Channel. */ LOCAL void stk_close_bip_channel(UBYTE general_result, UBYTE add_info_result) { T_desc2* temp_desc1; T_desc2* temp_desc2; if (sim_data.bip_state NEQ SIM_BIP_CLOSED) { /* * set BIP to close */ sim_data.bip_state = SIM_BIP_CLOSED; /* * stop release timer if used */ if (sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) { if (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START) { #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: stopped"); #endif TIMER_STOP (sim_handle, SIM_BIP_TIMER); } sim_data.bip_timer_state = SIM_BIP_TIMER_DISCONNECTED; } /* * send Terminal Response */ if (sim_data.bip_tx_state EQ SIM_BIP_TX_SEND) { stk_bip_send_data_terminal_response(general_result, add_info_result); /* * close channel is not triggered from a proactive_polling function call * so adjust timer and start polling */ stk_start_timer_and_poll(); } /* * release suspension */ sim_data.bip_suspend = FALSE; /* * release RX and TX buffer */ temp_desc1 = (T_desc2*)sim_data.data_to_send.first; while(temp_desc1) { temp_desc2 = temp_desc1; temp_desc1 = (T_desc2*)temp_desc1->next; MFREE(temp_desc2); } sim_data.data_to_send.first = (ULONG)NULL; sim_data.data_to_send.list_len = 0; sim_data.prev_data_to_send.first = (ULONG)NULL; sim_data.prev_data_to_send.list_len = 0; #ifdef _SIMULATION_ TRACE_EVENT("bip_tx_state = IDLE"); #endif sim_data.bip_tx_state = SIM_BIP_TX_IDLE; temp_desc1 = (T_desc2*)sim_data.received_data.first; while(temp_desc1) { temp_desc2 = temp_desc1; temp_desc1 = (T_desc2*)temp_desc1->next; MFREE(temp_desc2); } sim_data.received_data.first = (ULONG)NULL; sim_data.received_data.list_len = 0; sim_data.received_data_pos = 0; #ifdef _SIMULATION_ TRACE_EVENT("bip_rx_state = IDLE"); #endif sim_data.bip_rx_state = SIM_BIP_RX_IDLE; } } /* stk_close_bip_channel() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_dti_inform_mmi | +--------------------------------------------------------------------+ PURPOSE : Informs MMI if the requested actions have been done. */ LOCAL void stk_dti_inform_mmi (UBYTE dti_conn_req, UBYTE bip_conn_req ) { UBYTE dti_conn_done; UBYTE bip_conn_done; #if _SIMULATION_ TRACE_FUNCTION ("stk_dti_inform_mmi()"); #endif /* * set stutus */ dti_conn_done = 0; bip_conn_done = 0; if (sim_data.bip_state EQ SIM_BIP_CLOSED) { bip_conn_done |= SIM_BIP_CLOSE_CHANNEL; } else { bip_conn_done |= SIM_BIP_OPEN_CHANNEL; /* * Only if BIP channel not closed, deal with its suspension state */ if (sim_data.bip_suspend) { bip_conn_done |= SIM_BIP_CHANNEL_SUSPENDED; } else { bip_conn_done |= SIM_BIP_CHANNEL_RESUMED; } } if (sim_data.dti_connection_state EQ SIM_DTI_CONNECTION_OPEN) { dti_conn_done = SIM_DTI_CONNECT; } if (sim_data.dti_connection_state EQ SIM_DTI_CONNECTION_CLOSED) { dti_conn_done = SIM_DTI_DISCONNECT; } /* * inform MMI */ if ( /* reguirements REALLY fullfilled in code, then confirm or indicate */ /* else no reaction at all !! */ (((bip_conn_req & bip_conn_done) EQ bip_conn_req) OR (bip_conn_req EQ SIM_BIP_UNKNOWN)) AND ((dti_conn_req EQ dti_conn_done)OR (dti_conn_req EQ SIM_DTI_UNKNOWN)) AND (~(UCHAR)((dti_conn_req EQ SIM_DTI_UNKNOWN) AND (bip_conn_req EQ SIM_BIP_UNKNOWN))) ) { if (sim_data.sim_dti_req) { /* * send DTI confirm primitive */ PALLOC(sim_dti_cnf, SIM_DTI_CNF); sim_dti_cnf->link_id = sim_data.sim_dti_req->link_id; sim_dti_cnf->dti_conn = dti_conn_done; PSEND(hCommMMI, sim_dti_cnf); /* * free request primitive */ PFREE(sim_data.sim_dti_req); sim_data.sim_dti_req = NULL; } else if (sim_data.sim_bip_req) { /* * send BIP confirm primitive */ PALLOC(sim_bip_cnf, SIM_BIP_CNF); sim_bip_cnf->bip_ch_id = sim_data.sim_bip_req->bip_ch_id; sim_bip_cnf->bip_conn = bip_conn_done; PSEND(hCommMMI, sim_bip_cnf); /* * free request primitive */ PFREE(sim_data.sim_bip_req); sim_data.sim_bip_req = NULL; } else { /* * send indication primitive */ PALLOC(sim_dti_bip_ind, SIM_DTI_BIP_IND); sim_dti_bip_ind->link_id = sim_data.link_id; sim_dti_bip_ind->dti_conn = dti_conn_done; sim_dti_bip_ind->bip_ch_id = sim_data.bip_ch_id; sim_dti_bip_ind->bip_conn = bip_conn_done; PSEND(hCommMMI, sim_dti_bip_ind); } } /* and if (requirements fullfilled) */ } /* stk_dti_inform_mmi() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_dti_connection_opened | +--------------------------------------------------------------------+ PURPOSE : Handle CONNECTION OPENED signal from DTILIB. */ GLOBAL void stk_dti_connection_opened (void) { TRACE_FUNCTION ("stk_dti_connection_opened()"); switch(sim_data.dti_connection_state) { case SIM_DTI_CONNECTION_SETUP: sim_data.dti_connection_state = SIM_DTI_CONNECTION_OPEN; /* * send confirm primitive */ stk_dti_inform_mmi(sim_data.sim_dti_req->dti_conn, (UBYTE) SIM_BIP_UNKNOWN); /* fall through */ case SIM_DTI_CONNECTION_OPEN: /* * connect BIP with DTI */ if (sim_data.bip_state EQ SIM_BIP_OPEN) { sim_data.bip_state = SIM_BIP_CONNECTED; if (sim_data.bip_release_time EQ SIM_NO_AUTO_RELEASE) { sim_data.bip_timer_state = SIM_BIP_TIMER_NOT_USED; } else if (sim_data.bip_suspend) { sim_data.bip_timer_state = SIM_BIP_TIMER_SUSPENDED; } else { sim_data.bip_timer_state = SIM_BIP_TIMER_STOPPED; } } break; default: /* * wrong state * so close DTI connection * and inform MMI */ TRACE_ERROR("DTI_CONNECTION_OPENED in wrong state"); stk_close_dti_connection(TRUE); stk_dti_inform_mmi(SIM_DTI_DISCONNECT, (UBYTE) SIM_BIP_UNKNOWN); break; } /* * reset RX, TX and timer */ sim_data.dti_rx_state = SIM_DTI_RX_IDLE; sim_data.dti_tx_state = SIM_DTI_TX_IDLE; #if 0 /*###jk:OK?*/ if (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START) { sim_data.bip_timer_state = SIM_BIP_TIMER_STOPPED; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: stopped"); #endif TIMER_STOP (sim_handle, SIM_BIP_TIMER); } /* * update timer and DTI states */ if ((sim_data.bip_state EQ SIM_BIP_CONNECTED) AND (sim_data.bip_rx_state EQ SIM_BIP_RX_IDLE)) { /* * start reception */ sim_data.dti_rx_state = SIM_DTI_RX_READY; dti_start(sim_data.hDTI, 0, 0, 0); /* * start timer */ if ((sim_data.bip_timer_state EQ SIM_BIP_TIMER_STOPPED) AND (sim_data.bip_tx_state EQ SIM_BIP_TX_IDLE) ) { /* * no data trafic on the BIP channel, * so use the timer */ sim_data.bip_timer_state = SIM_BIP_TIMER_START; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: start in stk_dti_connection_opened()"); #endif TIMER_START (sim_handle, SIM_BIP_TIMER, sim_data.bip_release_time); } } #endif /*###jk:OK?*/ } /* stk_dti_connection_opened() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_dti_connection_closed | +--------------------------------------------------------------------+ PURPOSE : Handle CONNECTION CLOSED signal from DTILIB. */ GLOBAL void stk_dti_connection_closed (void) { TRACE_FUNCTION ("stk_dti_connection_closed()"); /* * close DTI connection */ stk_close_dti_connection(FALSE); /* * inform MMI about disconnection */ stk_dti_inform_mmi(SIM_DTI_DISCONNECT, (UBYTE) SIM_BIP_UNKNOWN); } /* stk_dti_connection_closed() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_dti_data_received | +--------------------------------------------------------------------+ PURPOSE : Deal with incomming DTI data primitives. */ GLOBAL void stk_dti_data_received (T_DTI2_DATA_IND* dti_data_ind) { UBYTE ccdRet; UBYTE dummy[4]; T_sdu* temp_sdu; T_desc2* temp_desc1; T_desc2* temp_desc2; TRACE_FUNCTION ("stk_dti_data_received()"); #ifdef _SIMULATION_ /* * copy bytes of T_SRC_DES struct in extra descriptor * this is a requirement of the DTI communication with UDP, but it should * not have any side affects in case of communication with other entities */ if ((dti_data_ind->desc_list2.first NEQ (ULONG)NULL) AND (dti_data_ind->desc_list2.list_len >= sizeof(T_SRC_DES))) { T_desc2 *old_desc, *addr_desc, *test_desc, *next_desc; USHORT i, j; old_desc = (T_desc2 *)dti_data_ind->desc_list2.first; /* * build the T_SRC_DES for IP-addresses and ports */ MALLOC(addr_desc, (USHORT)(sizeof(T_desc2) - 1 + sizeof(T_SRC_DES))); addr_desc->len = sizeof(T_SRC_DES); j = 0; for(i=0; i < addr_desc->len; i++) { while(j >= old_desc->len) { next_desc = (T_desc2*)old_desc->next; MFREE(old_desc); old_desc = next_desc; j = 0; } addr_desc->buffer[i] = old_desc->buffer[j]; j++; } /* * Build the desc for the data */ if (j < old_desc->len) { MALLOC(test_desc, (USHORT)(sizeof(T_desc2) - 1 + old_desc->len - j)); test_desc->len = old_desc->len - j; test_desc->next = old_desc->next; for(i=0; i < test_desc->len; i++) { test_desc->buffer[i] = old_desc->buffer[j]; j++; } } else { test_desc = (T_desc2*)old_desc->next; } MFREE(old_desc); dti_data_ind->desc_list2.first = (ULONG)addr_desc; addr_desc->next = (ULONG)test_desc; } #endif /* _SIMULATION_ */ /* * take data */ temp_desc1 = (T_desc2*)dti_data_ind->desc_list2.first; /* * free primitive */ PFREE(dti_data_ind); /* * */ switch(sim_data.con_type) { /* --67 asd; ###jk test error for image*/ case SIM_CON_TYPE_UDP: /* * free first descriptor */ temp_desc2 = temp_desc1; if (temp_desc1) { temp_desc1 = (T_desc2*)temp_desc1->next; } MFREE(temp_desc2); /* fall through */ case SIM_CON_TYPE_IP: /* * store data */ if (sim_data.received_data.first EQ (ULONG)NULL) { sim_data.received_data.first = (ULONG)temp_desc1; } else { /* * error * free received data */ TRACE_ERROR("DTI data received, but still data in RX buffer"); while(temp_desc1) { temp_desc2 = temp_desc1; temp_desc1 = (T_desc2*)temp_desc1->next; MFREE(temp_desc2); } temp_desc1 = NULL; } break; case SIM_CON_TYPE_SERIAL: /* * store data */ if (sim_data.received_data.first EQ (ULONG)NULL) { sim_data.received_data.first = (ULONG)temp_desc1; } else { /* * error, but concatinate data */ TRACE_ERROR("DTI data received, but still data in RX buffer"); /* * find last descriptor */ temp_desc2 = (T_desc2*)sim_data.received_data.first; while(temp_desc2->next NEQ (ULONG)NULL) { temp_desc2 = (T_desc2*)temp_desc2->next; } temp_desc2->next = (ULONG)temp_desc1; } break; } /* * update list length */ while(temp_desc1) { sim_data.received_data.list_len+= temp_desc1->len; temp_desc1 = (T_desc2*)temp_desc1->next; } if (sim_data.received_data.list_len) { /* * change state of BIP RX and stop timer */ #ifdef _SIMULATION_ TRACE_EVENT("bip_rx_state = DATA"); #endif sim_data.bip_rx_state = SIM_BIP_RX_DATA; if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START) ) { sim_data.bip_timer_state = SIM_BIP_TIMER_STOPPED; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: stopped"); #endif TIMER_STOP (sim_handle, SIM_BIP_TIMER); } /* * stop reception */ sim_data.dti_rx_state = SIM_DTI_RX_IDLE; dti_stop(sim_data.hDTI, 0, 0, 0); /* * inform SIM card */ if (sim_data.event_data_avail EQ SIM_EVENT_ENABLE) { CCD_START; { MCAST(env_cmd, ENV_CMD); memset(env_cmd, 0, sizeof(T_ENV_CMD)); /* * Event Download Command */ env_cmd->v_evd_cmd = TRUE; /* * Event List */ env_cmd->evd_cmd.v_ev_list = TRUE; env_cmd->evd_cmd.ev_list.c_event = 1; env_cmd->evd_cmd.ev_list.event[0] = EVENT_DATA_AVAIL; /* * Device Identities */ env_cmd->evd_cmd.v_dev_ids = TRUE; env_cmd->evd_cmd.dev_ids.src_dev = DEV_SRC_ME; env_cmd->evd_cmd.dev_ids.dest_dev = DEV_DST_SIM; /* * Channel Satus */ env_cmd->evd_cmd.v_chan_stat = TRUE; env_cmd->evd_cmd.chan_stat.chan_id = sim_data.bip_ch_id & 0x07; env_cmd->evd_cmd.chan_stat.chan_stat_inf1 = 0; env_cmd->evd_cmd.chan_stat.chan_stat_link = LINK_ESTABL; env_cmd->evd_cmd.chan_stat.chan_stat_inf2 = NO_FURTH_INFO; /* * Channel Data Length */ env_cmd->evd_cmd.v_chan_dat_lth = TRUE; if (sim_data.received_data.list_len > 255) { /* * more than 255 bytes are available in RX buffer */ env_cmd->evd_cmd.chan_dat_lth = 0xff; } else { env_cmd->evd_cmd.chan_dat_lth = (UBYTE)sim_data.received_data.list_len; } /* * encode message */ MALLOC(temp_sdu, (USHORT)(sizeof(T_sdu) - 1 + MAX_STK_CMD)); temp_sdu->l_buf = MAX_STK_CMD << 3; temp_sdu->o_buf = 0; ccdRet = ccd_codeMsg (CCDENT_SAT, UPLINK, (T_MSGBUF *) temp_sdu, (UBYTE *) _decodedMsg, ENV_CMD); } CCD_END; if ( ccdRet NEQ ccdOK ) { TRACE_EVENT_P1("Data Available: CCD Coding Error: %d",ccdRet ); } else { FKT_Envelope (dummy, temp_sdu->buf, (USHORT)(temp_sdu->l_buf >> 3), 0); /* * data received is not triggered from a proactive_polling * function call, so adjust timer and start polling */ stk_start_timer_and_poll(); } MFREE(temp_sdu); } } } /* stk_dti_data_received() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_dti_tx_buffer_full | +--------------------------------------------------------------------+ PURPOSE : Handle TX BUFFER FULL signal from DTILIB. */ GLOBAL void stk_dti_tx_buffer_full (void) { TRACE_FUNCTION ("stk_dti_tx_buffer_full()"); /* * set new DTI state */ sim_data.dti_tx_state = SIM_DTI_TX_IDLE; } /* stk_dti_tx_buffer_full() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_dti_tx_buffer_ready | +--------------------------------------------------------------------+ PURPOSE : Handle TX BUFFER READY signal from DTILIB. */ GLOBAL void stk_dti_tx_buffer_ready (void) { TRACE_FUNCTION ("stk_dti_tx_buffer_ready()"); /* * set new DTI state */ sim_data.dti_tx_state = SIM_DTI_TX_READY; /* * send confirm primitive to MMI: out of here; actually as the response for * SIM_BIP_CONFIG_REQ */ if ((sim_data.sim_bip_config_req) AND (sim_data.dti_connection_state EQ SIM_DTI_CONNECTION_SETUP)) { PALLOC(sim_bip_config_cnf, SIM_BIP_CONFIG_CNF); PSEND(hCommMMI, sim_bip_config_cnf); /* primitive no longer needed.. */ /* and so avoid the second confirm in stk_udp_bind_cnf() */ PFREE(sim_data.sim_bip_config_req); sim_data.sim_bip_config_req = NULL; /* * set the open state: the connection is now truly opened */ sim_data.dti_connection_state = SIM_DTI_CONNECTION_OPEN; } /* * send data */ if (sim_data.bip_tx_state EQ SIM_BIP_TX_SEND) { /* * set new BIP state */ #ifdef _SIMULATION_ TRACE_EVENT("bip_tx_state = IDLE"); #endif sim_data.bip_tx_state = SIM_BIP_TX_IDLE; /* * send DTI data primitive */ stk_dti_send_data(); /* * send Terminal Response */ stk_bip_send_data_terminal_response(sim_data.bip_general_result, sim_data.bip_add_info_result); /* * buffer ready is not triggered from a proactive_polling function call * so adjust timer and start polling */ stk_start_timer_and_poll(); /* * (re)start release timer */ if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_rx_state EQ SIM_BIP_RX_IDLE) AND ((sim_data.bip_timer_state EQ SIM_BIP_TIMER_STOPPED) OR (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START))) { sim_data.bip_timer_state = SIM_BIP_TIMER_START; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: start in stk_dti_tx_buffer_ready()"); #endif TIMER_START (sim_handle, SIM_BIP_TIMER, sim_data.bip_release_time); } } } /* stk_dti_tx_buffer_ready() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_dti_bip_send_data | +--------------------------------------------------------------------+ PURPOSE : Handle SEND DATA message from SIM card. */ GLOBAL UBYTE stk_dti_bip_send_data (T_sdu* message) { UBYTE ccdRet; UBYTE func_ret; UBYTE general_result; UBYTE add_info_result; UBYTE status; T_desc2* temp_desc1; T_desc2* temp_desc2; USHORT temp_pos; USHORT temp_len; TRACE_FUNCTION ("stk_dti_bip_send_data()"); /* * initialize value; */ ccdRet = ccdOK; func_ret = FALSE; general_result = RSLT_PERF_SUCCESS; add_info_result = ADD_NO_CAUSE; status = SIM_CCD_OK; CCD_START; /* * decode SIM Toolkit Command */ stk_bip_decode_stk_command(message, &sim_data.bip_tx_cmd_details, &status, &general_result, &add_info_result); /* * decode SEND DATA message */ if (status EQ SIM_CCD_OK) { MCAST(send_data, SEND_DATA); memset(send_data, 0, sizeof(T_SEND_DATA)); /* * decode SEND DATA message */ ccdRet = ccd_decodeMsg (CCDENT_SAT, DOWNLINK, (T_MSGBUF *) &sim_data.bip_cmd_prms, (UBYTE *) _decodedMsg, SEND_DATA); if (ccdRet NEQ ccdOK) { /* * handle errors in list */ stk_handle_ccd_error(ccdRet, &status, &general_result, &add_info_result); } if (send_data->v_chan_data EQ FALSE) { /* * no Channel Data element present * Error, required values are missing */ general_result = RSLT_ERR_REQ_VAL; add_info_result = ADD_NO_CAUSE; status = SIM_CCD_RETURN; } if (status EQ SIM_CCD_OK) { if (sim_data.bip_suspend) { /* * channel suspended * ME currently unable to process command */ general_result = sim_data.bip_general_result; add_info_result = sim_data.bip_add_info_result; status = SIM_CCD_RETURN; } else if ((SIM_CLASS_E_BUFFER_SIZE - sim_data.data_to_send.list_len) < send_data->chan_data.c_ch_dat_str) { /* * not enough space in tx buffer * BIP error */ general_result = RSLT_BEARIND_PERR; add_info_result = ADD_BIP_BUF_SIZ_NAVAIL; status = SIM_CCD_RETURN; } else { /* * concatenate Channel data */ sim_data.prev_data_to_send = sim_data.data_to_send; if (sim_data.data_to_send.first NEQ (ULONG)NULL) { /* * find last descriptor */ temp_desc1 = (T_desc2*)sim_data.data_to_send.first; while(temp_desc1->next NEQ (ULONG)NULL) { temp_desc1 = (T_desc2*)temp_desc1->next; } } else if (send_data->chan_data.c_ch_dat_str) { /* * allocate a new descriptor */ MALLOC(temp_desc1, (USHORT)(sizeof(T_desc2) - 1 + SIM_BIP_TX_DESC_SIZE)); temp_desc1->next = (ULONG)NULL; temp_desc1->offset= 0; /*###jk:OK*/ temp_desc1->len = 0; temp_desc1->size= 0; /*###jk:OK*/ sim_data.data_to_send.first = (ULONG)temp_desc1; sim_data.data_to_send.list_len = 0; } temp_pos = 0; while(temp_pos < send_data->chan_data.c_ch_dat_str) { if (temp_desc1->len >= SIM_BIP_TX_DESC_SIZE) /*lint !e644 temp_desc1 may not have been initialized*/ { /* * allocate new desriptor */ temp_desc2 = temp_desc1; MALLOC(temp_desc1, (USHORT)(sizeof(T_desc2) - 1 + SIM_BIP_TX_DESC_SIZE)); temp_desc1->next = (ULONG)NULL; temp_desc1->offset = 0; /*###jk:OK*/ temp_desc1->len = 0; temp_desc1->size = 0; /*###jk:OK*/ temp_desc2->next = (ULONG)temp_desc1; } /* * calculate length */ temp_len = send_data->chan_data.c_ch_dat_str - temp_pos; if (temp_len > (SIM_BIP_TX_DESC_SIZE - temp_desc1->len)) { temp_len = SIM_BIP_TX_DESC_SIZE - temp_desc1->len; TRACE_EVENT("sdbsd_5: if (temp_desc1->len >= SIM_BIP_TX_DESC_SIZE)"); /*###jk:tbd*/ TRACE_EVENT_P1("sdbsd_5: temp_len = %d", temp_len); /*###jk:tbd*/ } /* * copy data */ memcpy(&temp_desc1->buffer[temp_desc1->len], &send_data->chan_data.ch_dat_str[temp_pos], temp_len); temp_pos += temp_len; temp_desc1->len += temp_len; temp_desc1->size += temp_len; /*###jk:OK?*/ sim_data.data_to_send.list_len+= temp_len; } } /* * process SEND DATA message */ if (status EQ SIM_CCD_OK) { /* * if alpha identifier or icon identifier is present then * forward message to MMI */ if ((send_data->v_alpha_id) OR (send_data->v_icon)) { func_ret = TRUE; } /* * check for immediate/store bit */ if (sim_data.bip_tx_cmd_details.cmd_qlf & SIM_QLF_SEND_DATA_1) { /* * send data immediately */ if (sim_data.bip_state EQ SIM_BIP_CONNECTED) { /* * DTI connected */ if (sim_data.dti_tx_state EQ SIM_DTI_TX_READY) { /* * send DTI data primitive */ #ifdef _SIMULATION_ TRACE_EVENT("bip_tx_state = IDLE"); #endif sim_data.bip_tx_state = SIM_BIP_TX_IDLE; status = SIM_CCD_RETURN; stk_dti_send_data(); } else { TRACE_EVENT("stbsd_8: else (sim_data.dti_tx_state EQ SIM_DTI_TX_READY)"); /*###jk:tbd*/ /* * start release timer */ #ifdef _SIMULATION_ TRACE_EVENT("bip_tx_state = SEND"); #endif sim_data.bip_tx_state = SIM_BIP_TX_SEND; status = SIM_CCD_DISCARD; if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START) ) { sim_data.bip_timer_state = SIM_BIP_TIMER_STOPPED; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: stopped"); #endif TIMER_STOP (sim_handle, SIM_BIP_TIMER); } } /* * (re)start release timer */ if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_rx_state EQ SIM_BIP_RX_IDLE) AND (sim_data.bip_tx_state EQ SIM_BIP_TX_IDLE) AND ((sim_data.bip_timer_state EQ SIM_BIP_TIMER_STOPPED) OR (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START))) { sim_data.bip_timer_state = SIM_BIP_TIMER_START; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: start in stk_dti_bip_send_data(..)"); #endif TIMER_START (sim_handle, SIM_BIP_TIMER, sim_data.bip_release_time); } } else { TRACE_EVENT("stbsd_7: else (sim_data.bip_state EQ SIM_BIP_CONNECTED)"); /*###jk:tbd*/ /* * wait for DTI connection */ #ifdef _SIMULATION_ TRACE_EVENT("bip_tx_state = SEND"); #endif sim_data.bip_tx_state = SIM_BIP_TX_SEND; status = SIM_CCD_DISCARD; if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START) ) { sim_data.bip_timer_state = SIM_BIP_TIMER_STOPPED; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: stopped"); #endif TIMER_STOP (sim_handle, SIM_BIP_TIMER); } if (sim_data.dti_connection_state EQ SIM_DTI_CONNECTION_CLOSED) { /* * on demand link establishment * so forward message to MMI */ func_ret = TRUE; } } } else { /* * store data */ TRACE_EVENT("bip_tx_state = STORE"); sim_data.bip_tx_state = SIM_BIP_TX_STORE; status = SIM_CCD_RETURN; /* * stop release timer if used */ if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START) ) { sim_data.bip_timer_state = SIM_BIP_TIMER_STOPPED; TRACE_EVENT("SIM_BIP_TIMER: stopped"); TIMER_STOP (sim_handle, SIM_BIP_TIMER); } } } } } /* if (status EQ SIM_CCD_OK) */ CCD_END; /* * send TERMINAL RESPONSE message */ if (status EQ SIM_CCD_RETURN) { stk_bip_send_data_terminal_response(general_result, add_info_result); } else { /* * store result codes */ sim_data.bip_general_result = general_result; sim_data.bip_add_info_result = add_info_result; } /* * send return value */ return func_ret; } /* stk_dti_bip_send_data() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (6302) MODULE : SIM_STK | | STATE : code ROUTINE : stk_dti_bip_receive_data | +--------------------------------------------------------------------+ PURPOSE : Handle RECEIVE DATA message from SIM card. */ LOCAL UBYTE stk_dti_bip_receive_data (T_sdu* message, UBYTE result_code) { UBYTE ccdRet; UBYTE func_ret; UBYTE general_result; UBYTE add_info_result; UBYTE status; UBYTE chan_dat_lth; T_desc2* temp_desc1; T_desc2* temp_desc2; T_sdu* temp_sdu; USHORT temp_pos; USHORT temp_len; /* * initialize value; */ ccdRet = ccdOK; func_ret = FALSE; general_result = RSLT_PERF_SUCCESS; add_info_result = ADD_NO_CAUSE; status = SIM_CCD_OK; chan_dat_lth = 0; /* jk: initialized because of LINT error */ CCD_START; /* * decode SIM Toolkit Command */ stk_bip_decode_stk_command(message, &sim_data.bip_rx_cmd_details, &status, &general_result, &add_info_result); /* * decode RECEIVE DATA message */ if (status EQ SIM_CCD_OK) { MCAST(receive_data, RECEIVE_DATA); memset(receive_data, 0, sizeof(T_RECEIVE_DATA)); ccdRet = ccd_decodeMsg (CCDENT_SAT, DOWNLINK, (T_MSGBUF *) &sim_data.bip_cmd_prms, (UBYTE *) _decodedMsg, RECEIVE_DATA); #ifdef _SIMULATION_ TRACE_EVENT_P1("ccdRet@dti_bip_receive_data: %d", ccdRet); #endif if (ccdRet NEQ ccdOK) { /* * handle errors in list */ stk_handle_ccd_error(ccdRet, &status, &general_result, &add_info_result); } if (receive_data->v_chan_dat_lth EQ FALSE) { /* * no Channel Data Length element present * Error, required values are missing */ general_result = RSLT_ERR_REQ_VAL; add_info_result = ADD_NO_CAUSE; status = SIM_CCD_RETURN; } /* * process RECEIVE DATA message */ if (status EQ SIM_CCD_OK) { /* * if alpha identifier or icon identifier is present then * forward message to MMI */ if ((receive_data->v_alpha_id) OR (receive_data->v_icon)) { func_ret = TRUE; } if (sim_data.received_data.list_len < receive_data->chan_dat_lth) { /* * can not fill the complete buffer */ general_result = RSLT_PERF_MISS_INFO; add_info_result = ADD_NO_CAUSE; } status = SIM_CCD_RETURN; /* * store Channel Data Length */ chan_dat_lth = receive_data->chan_dat_lth; } } CCD_END; /* * send TERMINAL RESPONSE message */ if (status EQ SIM_CCD_RETURN) { CCD_START; { MCAST(term_resp, TERM_RESP); memset(term_resp, 0, sizeof(T_TERM_RESP)); /* * set Command details */ term_resp->v_cmd_details = TRUE; term_resp->cmd_details = sim_data.bip_rx_cmd_details; /* * set set Device identities */ term_resp->v_dev_ids = TRUE; term_resp->dev_ids.src_dev = DEV_SRC_ME; term_resp->dev_ids.dest_dev = DEV_DST_SIM; /* * set Result */ term_resp->v_res = TRUE; term_resp->res.gen = general_result; switch (general_result) { case RSLT_ME_UNAB_PROC: case RSLT_NTW_UNAB_PROC: case RSLT_LABRWS_GENERIC: case RSLT_SS_ERR: case RSLT_SMS_ERR: case RSLT_USSD_ERR: case RSLT_CC_SIM_PRM: /* * one byte for additional information */ term_resp->res.v_add = TRUE; term_resp->res.add.l_add = 1 << 3; term_resp->res.add.o_add = 0; term_resp->res.add.b_add[0] = add_info_result; break; case RSLT_BEARIND_PERR: /* * one byte for additional information */ term_resp->res.v_add = TRUE; term_resp->res.add.l_add = 1 << 3; term_resp->res.add.o_add = 0; term_resp->res.add.b_add[0] = add_info_result-1; break; default: if (result_code) term_resp->res.gen = RSLT_PERF_PART_CMPR; break; } /* * set Channel Data */ term_resp->v_chan_data = TRUE; switch (general_result) { case RSLT_PERF_SUCCESS: case RSLT_PERF_PART_CMPR: case RSLT_PERF_MISS_INFO: case RSLT_PERF_MDFY_SIM: case RSLT_PERF_MDFIED: /* * calculate Channel Data String length */ term_resp->chan_data.c_ch_dat_str = SIM_TERM_RESP_MAX_CHANNEL_DATA; if (sim_data.received_data.list_len < term_resp->chan_data.c_ch_dat_str) { term_resp->chan_data.c_ch_dat_str = (UBYTE)sim_data.received_data.list_len; } if (chan_dat_lth < term_resp->chan_data.c_ch_dat_str) { term_resp->chan_data.c_ch_dat_str = chan_dat_lth; } /* * copy data */ temp_desc1 = (T_desc2*)sim_data.received_data.first; temp_pos = 0; while (temp_pos < term_resp->chan_data.c_ch_dat_str) { /* * calculate length */ temp_len = term_resp->chan_data.c_ch_dat_str - temp_pos; if (temp_len > (temp_desc1->len - sim_data.received_data_pos)) { temp_len = temp_desc1->len - sim_data.received_data_pos; } /* * copy data */ memcpy(&term_resp->chan_data.ch_dat_str[temp_pos], &temp_desc1->buffer[sim_data.received_data_pos], temp_len); /* * updata length and position values and descriptors */ temp_pos += temp_len; sim_data.received_data_pos += temp_len; sim_data.received_data.list_len-= temp_len; if (sim_data.received_data_pos >= temp_desc1->len) { temp_desc2 = temp_desc1; temp_desc1 = (T_desc2*)temp_desc1->next; sim_data.received_data_pos = 0; sim_data.received_data.first = (ULONG)temp_desc1; MFREE(temp_desc2); } } break; case RSLT_BEARIND_PERR: case RSLT_UNKN_DATA: term_resp->v_chan_data = FALSE; term_resp->chan_data.c_ch_dat_str = 0; break; default: /* * if an error is occured then do not provide data */ term_resp->chan_data.c_ch_dat_str = 0; break; } switch (general_result) { case RSLT_BEARIND_PERR: case RSLT_UNKN_DATA: break; default: /* * set Channel Data Length */ term_resp->v_chan_dat_lth = TRUE; if (sim_data.received_data.list_len > 255) { /* * more than 255 bytes are available in RX buffer */ term_resp->chan_dat_lth = 0xff; } else { term_resp->chan_dat_lth = (UBYTE)sim_data.received_data.list_len; } break; } /* * send Terminal Response */ MALLOC(temp_sdu, (USHORT)(sizeof(T_sdu) - 1 + MAX_STK_CMD)); temp_sdu->l_buf = MAX_STK_CMD << 3; temp_sdu->o_buf = 0; ccdRet = ccd_codeMsg (CCDENT_SAT, UPLINK, (T_MSGBUF *) temp_sdu, (UBYTE *) _decodedMsg, TERM_RESP); } CCD_END; if ( ccdRet NEQ ccdOK ) { TRACE_EVENT_P1("CCD Coding Error: %d",ccdRet ); } else { FKT_TerminalResponse (temp_sdu->buf, (USHORT)(temp_sdu->l_buf >> 3)); } MFREE(temp_sdu); } if ((sim_data.bip_rx_state EQ SIM_BIP_RX_DATA) AND (sim_data.received_data.list_len EQ 0)) { /* * set new BIP state */ #ifdef _SIMULATION_ TRACE_EVENT("bip_rx_state = IDLE"); #endif sim_data.bip_rx_state = SIM_BIP_RX_IDLE; /* * start DTI reception */ if ((sim_data.bip_state EQ SIM_BIP_CONNECTED) AND (sim_data.dti_rx_state EQ SIM_DTI_RX_IDLE)) { sim_data.dti_rx_state = SIM_DTI_RX_READY; dti_start(sim_data.hDTI, 0, 0, 0); } /* * start release timer */ if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_timer_state EQ SIM_BIP_TIMER_STOPPED) AND (sim_data.bip_tx_state EQ SIM_BIP_TX_IDLE) ) { /* * no data trafic on the BIP channel, * so use the timer */ sim_data.bip_timer_state = SIM_BIP_TIMER_START; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: start in stk_dti_bip_receive_data(..)"); #endif TIMER_START (sim_handle, SIM_BIP_TIMER, sim_data.bip_release_time); } } /* * send return value */ return func_ret; } /* stk_dti_bip_receive_data() */ #ifdef FF_SAT_E /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_udp_bind_cnf | +--------------------------------------------------------------------+ PURPOSE : Process the primitive UDP_BIND_CNF. */ GLOBAL void stk_udp_bind_cnf (T_UDP_BIND_CNF* udp_bind_cnf) { TRACE_FUNCTION ("stk_udp_bind_cnf()"); if (sim_data.dti_connection_state EQ SIM_DTI_CONNECTION_BIND) { if (udp_bind_cnf->err EQ UDP_BIND_NOERROR) { /* * store source port */ sim_data.udp_parameters.src_port[0] = ((udp_bind_cnf->port >> 8) & 0x00ff); sim_data.udp_parameters.src_port[1] = ((udp_bind_cnf->port) & 0x00ff); /* * make sending of the confirmation primitive (SIM_BIP_CONFIG_CNF) * out of "stk_dti_buffer_ready()" possible */ sim_data.dti_connection_state = SIM_DTI_CONNECTION_SETUP; /*###jk:OK? moved & changed from "stk_dti_connection_open()" */ if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START) ) { sim_data.bip_timer_state = SIM_BIP_TIMER_STOPPED; #ifdef _SIMULATION_ TRACE_EVENT("stk_udp_bind_cnf(): SIM_BIP_TIMER: stopped"); #endif TIMER_STOP (sim_handle, SIM_BIP_TIMER); } /* * update timer (if used) and DTI states */ if ((sim_data.bip_state EQ SIM_BIP_CONNECTED) AND (sim_data.bip_rx_state EQ SIM_BIP_RX_IDLE)) { /* * start reception */ sim_data.dti_rx_state = SIM_DTI_RX_READY; dti_start(sim_data.hDTI, 0, 0, 0); /* * start timer if used */ if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_tx_state EQ SIM_BIP_TX_IDLE) AND (sim_data.bip_timer_state EQ SIM_BIP_TIMER_STOPPED) ) { /* * no data trafic on the BIP channel, * so use the timer */ sim_data.bip_timer_state = SIM_BIP_TIMER_START; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: start in stk_dti_connection_opened()"); #endif TIMER_START (sim_handle, SIM_BIP_TIMER, sim_data.bip_release_time); } } /*###jk:OK?*/ } else { /* * can not open UDP port * so disconnect BIP channel from DTI */ stk_close_dti_connection(TRUE); /* * send confirm primitive */ stk_dti_inform_mmi(SIM_DTI_DISCONNECT, (UBYTE) SIM_BIP_UNKNOWN); } } /* * free primitive */ PFREE(udp_bind_cnf); } /* stk_udp_bind_cnf() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_udp_closeport_cnf | +--------------------------------------------------------------------+ PURPOSE : Process the primitive UDP_CLOSEPORT_CNF. */ GLOBAL void stk_udp_closeport_cnf ( T_UDP_CLOSEPORT_CNF* udp_closeport_cnf) { TRACE_FUNCTION ("stk_udp_closeport_cnf()"); /* * free primitive */ PFREE(udp_closeport_cnf); } /* stk_udp_closeport_cnf() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_udp_error_ind | +--------------------------------------------------------------------+ PURPOSE : Process the primitive UDP_ERROR_IND. */ GLOBAL void stk_udp_error_ind (T_UDP_ERROR_IND* udp_error_ind) { TRACE_FUNCTION ("stk_udp_error_ind()"); /* * free primitive */ PFREE(udp_error_ind); /* * generate error response */ { PALLOC(udp_error_res, UDP_ERROR_RES); PSEND(hCommUDP, udp_error_res); } } /* stk_udp_error_ind() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_udp_shutdown_ind | +--------------------------------------------------------------------+ PURPOSE : Process the primitive UDP_SHUTDOWN_IND. */ GLOBAL void stk_udp_shutdown_ind (T_UDP_SHUTDOWN_IND* udp_shutdown_ind) { TRACE_FUNCTION ("stk_udp_shutdown_ind()"); /* * send confirm primitive */ if (hCommUDP < VSI_OK) { hCommUDP = vsi_c_open (VSI_CALLER UDP_NAME); } if (hCommUDP >= VSI_OK) { PALLOC(udp_shutdown_res, UDP_SHUTDOWN_RES); PSEND(hCommUDP, udp_shutdown_res); /* * release VSI channel */ vsi_c_close (VSI_CALLER hCommUDP); hCommUDP = VSI_ERROR; } /* * close DTI connection */ if (sim_data.con_type EQ SIM_CON_TYPE_UDP) { switch(sim_data.dti_connection_state) { case SIM_DTI_CONNECTION_OPEN: case SIM_DTI_CONNECTION_SETUP: stk_close_dti_connection(TRUE); break; default: stk_close_dti_connection(FALSE); break; } /* * inform ACI about disconnection */ stk_dti_inform_mmi(SIM_DTI_DISCONNECT, (UBYTE) SIM_BIP_UNKNOWN); } } /* stk_udp_shutdown_ind() */ #endif /* FF_SAT_E */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_sim_dti_req | +--------------------------------------------------------------------+ PURPOSE : Process the primitive SIM_DTI_REQ. */ GLOBAL void stk_sim_dti_req (T_SIM_DTI_REQ* sim_dti_req) { UBYTE dti_conn; TRACE_FUNCTION ("stk_sim_dti_req()"); #ifdef _SIMULATION_ /* * set entity_name parameter */ switch(sim_dti_req->entity_name) { case 1: sim_dti_req->entity_name = (ULONG)("UDP"); break; case 2: sim_dti_req->entity_name = (ULONG)("SND"); break; case 3: sim_dti_req->entity_name = (ULONG)("L2R"); break; default: sim_dti_req->entity_name = (ULONG)(NULL); break; } #endif /* _SIMULATION_ */ /* * store the received primitive */ if (sim_data.sim_dti_req) { /* * free previous primitive before store the new one */ PFREE(sim_data.sim_dti_req); } sim_data.sim_dti_req = sim_dti_req; /* * store requested operations */ dti_conn = sim_dti_req->dti_conn; switch (dti_conn) { case SIM_DTI_DISCONNECT: /* close DTI connection */ { switch(sim_data.dti_connection_state) { case SIM_DTI_CONNECTION_OPEN: case SIM_DTI_CONNECTION_SETUP: stk_close_dti_connection(TRUE); break; default: stk_close_dti_connection(FALSE); break; } break; } case SIM_DTI_CONNECT: /* open DTI connection */ { /* * if a new DTI connection is requested close the old one before */ switch(sim_data.dti_connection_state) { case SIM_DTI_CONNECTION_OPEN: case SIM_DTI_CONNECTION_SETUP: stk_close_dti_connection(TRUE); break; default: stk_close_dti_connection(FALSE); } /* * store relevant data */ sim_data.link_id = sim_dti_req->link_id; sim_data.dti_connection_state = SIM_DTI_CONNECTION_SETUP; dti_open(sim_data.hDTI, /* DTI handle */ 0, /* instance */ 0, /* interface */ 0, /* channel */ 0, /* queue size */ sim_dti_req->dti_direction, /* direction */ FLOW_CNTRL_ENABLED, /* comm_type */ DTI_VERSION_10, /* version */ (UBYTE*)sim_dti_req->entity_name, /* entity name */ sim_dti_req->link_id); /* link identifier */ } } /* * send confirm primitive */ if (sim_data.sim_dti_req) { TRACE_EVENT_P1("if(sim_data.sim_dti_req): ~ ->dti_conn=0x%02X", sim_dti_req->dti_conn); stk_dti_inform_mmi(sim_dti_req->dti_conn, (UBYTE)SIM_BIP_UNKNOWN); } } /* stk_sim_dti_req() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_sim_bip_req | +--------------------------------------------------------------------+ PURPOSE : Process the primitive SIM_BIP_REQ. */ GLOBAL void stk_sim_bip_req (T_SIM_BIP_REQ* sim_bip_req) { UBYTE bip_conn; USHORT temp_len; T_desc2* temp_desc1; T_desc2* temp_desc2; TRACE_FUNCTION ("stk_sim_bip_req()"); /* * store the received primitive */ if (sim_data.sim_bip_req) { /* * free previous primitive before store the new one */ PFREE(sim_data.sim_bip_req); } sim_data.sim_bip_req = sim_bip_req; /* * store requested operations */ bip_conn = sim_bip_req->bip_conn; /* * resume BIP channel */ if (bip_conn & SIM_BIP_CHANNEL_RESUMED) { /* * set new suspension state */ sim_data.bip_suspend = FALSE; /* * restart timer if used */ if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_timer_state EQ SIM_BIP_TIMER_SUSPENDED) ) { sim_data.bip_timer_state = SIM_BIP_TIMER_STOPPED; if ((sim_data.bip_rx_state EQ SIM_BIP_RX_IDLE) AND (sim_data.bip_tx_state EQ SIM_BIP_TX_IDLE) ) { /* * no data trafic on the BIP channel, * so use the timer */ sim_data.bip_timer_state = SIM_BIP_TIMER_START; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: start in stk_sim_bip_req(..)"); #endif TIMER_START (sim_handle, SIM_BIP_TIMER, sim_data.bip_release_time); } } } /* * close BIP channel */ if (bip_conn & SIM_BIP_CLOSE_CHANNEL) { stk_close_bip_channel(sim_bip_req->general_result, sim_bip_req->add_info_result); } /* * open BIP channel */ if ((bip_conn & SIM_BIP_OPEN_CHANNEL) AND (sim_data.bip_state EQ SIM_BIP_CLOSED)) { /* * set new BIP state and * store BIP channel identifier */ sim_data.bip_state = SIM_BIP_OPEN; sim_data.bip_ch_id = sim_bip_req->bip_ch_id; /* * store relevant data */ sim_data.bip_release_time = (T_TIME)sim_bip_req->release_time * 100; /* convert to msec. */ if (sim_bip_req->general_result NEQ RSLT_PERF_SUCCESS) { sim_data.bip_general_result = sim_bip_req->general_result; sim_data.bip_add_info_result = sim_bip_req->add_info_result; } } /* * suspend BIP channel */ if (bip_conn & SIM_BIP_CHANNEL_SUSPENDED) { /* * set new suspension state */ sim_data.bip_suspend = TRUE; /* * stop timer if timer is used */ if (sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) { if (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START) { sim_data.bip_timer_state = SIM_BIP_TIMER_SUSPENDED; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: stopped"); #endif TIMER_STOP (sim_handle, SIM_BIP_TIMER); } else if (sim_data.bip_timer_state EQ SIM_BIP_TIMER_STOPPED) { sim_data.bip_timer_state = SIM_BIP_TIMER_SUSPENDED; } } /* * store result codes */ sim_data.bip_general_result = sim_bip_req->general_result; sim_data.bip_add_info_result = sim_bip_req->add_info_result; /* * inform SIM card */ if (sim_data.bip_tx_state EQ SIM_BIP_TX_SEND) { /* * free data of last SEND DATA message */ temp_desc1 = (T_desc2*)sim_data.data_to_send.first; sim_data.data_to_send = sim_data.prev_data_to_send; if (sim_data.data_to_send.first EQ (ULONG)NULL) { while(temp_desc1) { temp_desc2 = temp_desc1; temp_desc1 = (T_desc2*)temp_desc1->next; MFREE(temp_desc2); } } else { temp_len = 0; while((temp_desc1) AND (temp_len + temp_desc1->len) < sim_data.data_to_send.list_len) { temp_len += temp_desc1->len; temp_desc1 = (T_desc2*)temp_desc1->next; } if (temp_desc1) { temp_desc1->len = sim_data.data_to_send.list_len - temp_len; temp_desc2 = (T_desc2*)temp_desc1->next; temp_desc1->next = (ULONG)NULL; temp_desc1 = temp_desc2; while(temp_desc1) { temp_desc2 = temp_desc1; temp_desc1 = (T_desc2*)temp_desc1->next; MFREE(temp_desc2); } } } /* * set new BIP TX state */ if (sim_data.data_to_send.first EQ (ULONG)NULL) { #ifdef _SIMULATION_ TRACE_EVENT("bip_tx_state = IDLE"); #endif sim_data.bip_tx_state = SIM_BIP_TX_IDLE; } else { #ifdef _SIMULATION_ TRACE_EVENT("bip_tx_state = STORE"); #endif sim_data.bip_tx_state = SIM_BIP_TX_STORE; if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START) ) { sim_data.bip_timer_state = SIM_BIP_TIMER_STOPPED; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: stopped"); #endif TIMER_STOP (sim_handle, SIM_BIP_TIMER); } } /* * send Terminal Response */ stk_bip_send_data_terminal_response(sim_bip_req->general_result, sim_bip_req->add_info_result); /* * suspension is not triggered from a proactive_polling function call * so adjust timer and start polling */ stk_start_timer_and_poll(); } } /* * send confirm primitive */ if (sim_data.sim_bip_req) { stk_dti_inform_mmi((UBYTE) SIM_DTI_UNKNOWN, sim_bip_req->bip_conn); } } /* stk_sim_bip_req() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_sim_bip_config_req | +--------------------------------------------------------------------+ PURPOSE : Process the primitive SIM_BIP_CONFIG_REQ. */ GLOBAL void stk_sim_bip_config_req(T_SIM_BIP_CONFIG_REQ * sim_bip_config_req) { TRACE_FUNCTION ("stk_sim_bip_config_req()"); /* * store the received primitive */ if (sim_data.sim_bip_config_req) { /* * free previous primitive before store the new one */ PFREE(sim_data.sim_bip_config_req); } sim_data.sim_bip_config_req = sim_bip_config_req; /* * store relevant data */ sim_data.con_type = sim_bip_config_req->con_type; #if 0 sim_data.udp_parameters.src_ip[0] = (UBYTE)((sim_bip_config_req->local_ip >> 24) & 0x000000ff); sim_data.udp_parameters.src_ip[1] = (UBYTE)((sim_bip_config_req->local_ip >> 16) & 0x000000ff); sim_data.udp_parameters.src_ip[2] = (UBYTE)((sim_bip_config_req->local_ip >> 8) & 0x000000ff); sim_data.udp_parameters.src_ip[3] = (UBYTE)((sim_bip_config_req->local_ip) & 0x000000ff); #else sim_data.udp_parameters.src_ip[3] = (UBYTE)((sim_bip_config_req->local_ip >> 24) & 0x000000ff); sim_data.udp_parameters.src_ip[2] = (UBYTE)((sim_bip_config_req->local_ip >> 16) & 0x000000ff); sim_data.udp_parameters.src_ip[1] = (UBYTE)((sim_bip_config_req->local_ip >> 8) & 0x000000ff); sim_data.udp_parameters.src_ip[0] = (UBYTE)((sim_bip_config_req->local_ip) & 0x000000ff); #endif sim_data.udp_parameters.des_ip[0] = (UBYTE)((sim_bip_config_req->destination_ip >> 24) & 0x000000ff); sim_data.udp_parameters.des_ip[1] = (UBYTE)((sim_bip_config_req->destination_ip >> 16) & 0x000000ff); sim_data.udp_parameters.des_ip[2] = (UBYTE)((sim_bip_config_req->destination_ip >> 8) & 0x000000ff); sim_data.udp_parameters.des_ip[3] = (UBYTE)((sim_bip_config_req->destination_ip) & 0x000000ff); #if 0 sim_data.udp_parameters.des_port[0] = (UBYTE)((sim_bip_config_req->destination_port >> 8) & 0x00ff); sim_data.udp_parameters.des_port[1] = (UBYTE)((sim_bip_config_req->destination_port) & 0x00ff); #else sim_data.udp_parameters.des_port[1] = (UBYTE)((sim_bip_config_req->destination_port >> 8) & 0x00ff); sim_data.udp_parameters.des_port[0] = (UBYTE)((sim_bip_config_req->destination_port) & 0x00ff); #endif /* * UDP connection */ if (sim_data.con_type EQ SIM_CON_TYPE_UDP) { /* * open VSI channel to UDP */ if (hCommUDP < VSI_OK) { TRACE_EVENT("if(hCommUDP < VSI_OK)"); if ((hCommUDP = vsi_c_open (VSI_CALLER UDP_NAME)) < VSI_OK) { TRACE_EVENT("if ((hCommUDP = vsi_c_open (VSI_CALLER UDP_NAME)) < VSI_OK)"); /* * can not open VSI channel * so act as if DTI close was requested */ sim_data.sim_dti_req->dti_conn = SIM_DTI_DISCONNECT; TRACE_EVENT_P1(": sim_data.sim_dti_req->dti_conn=0x%02X", sim_data.sim_dti_req->dti_conn); } } /* * send UDP_BIND_REQ */ if (hCommUDP >= VSI_OK) { PALLOC(udp_bind_req, UDP_BIND_REQ); sim_data.dti_connection_state = SIM_DTI_CONNECTION_BIND; udp_bind_req->port = UDP_AUTOASSIGN_PORT; PSEND(hCommUDP, udp_bind_req); /* * send confirm primitive: * will be sent out of "stk_dti_buffer_ready()" function in case of success, * in case of udp_bind_cnf failure only indication primitive will be sent * out of "stk_udp_bind_cnf" to MMI instead. */ } } /* * bearer level connection */ else { /*###jk:OK? moved & changed from "stk_dti_connection_open()" */ if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_timer_state EQ SIM_BIP_TIMER_START) ) { sim_data.bip_timer_state = SIM_BIP_TIMER_STOPPED; #ifdef _SIMULATION_ TRACE_EVENT("stk_udp_bind_cnf(): SIM_BIP_TIMER: stopped"); #endif TIMER_STOP (sim_handle, SIM_BIP_TIMER); } /* * update timer (if used) and DTI states */ if ((sim_data.bip_state EQ SIM_BIP_CONNECTED) AND (sim_data.bip_rx_state EQ SIM_BIP_RX_IDLE)) { /* * start reception */ sim_data.dti_rx_state = SIM_DTI_RX_READY; dti_start(sim_data.hDTI, 0, 0, 0); /* * start timer if used */ if ((sim_data.bip_timer_state NEQ SIM_BIP_TIMER_NOT_USED) AND (sim_data.bip_tx_state EQ SIM_BIP_TX_IDLE) AND (sim_data.bip_timer_state EQ SIM_BIP_TIMER_STOPPED) ) { /* * no data trafic on the BIP channel, * so use the timer */ sim_data.bip_timer_state = SIM_BIP_TIMER_START; #ifdef _SIMULATION_ TRACE_EVENT("SIM_BIP_TIMER: start in stk_dti_connection_opened()"); #endif TIMER_START (sim_handle, SIM_BIP_TIMER, sim_data.bip_release_time); } } /*###jk:OK?*/ /* * make sending of the confirmation primitive (SIM_BIP_CONFIG_CNF) * out of "stk_dti_buffer_ready()" possible */ sim_data.dti_connection_state = SIM_DTI_CONNECTION_SETUP; /* * send confirm primitive: * will be sent out of "stk_dti_buffer_ready()" function in order to * prevent sending of the data to connected neighbour DTI-entity before * this entity is ready to receive data. */ #if 0 /*###jk:OK? -> */ /* * send confirm primitive */ if (sim_data.sim_bip_config_req) { PALLOC(sim_bip_config_cnf, SIM_BIP_CONFIG_CNF); PSEND(hCommMMI, sim_bip_config_cnf); } #endif } } /* stk_bip_config_req() */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_sim_eventlist_req | +--------------------------------------------------------------------+ PURPOSE : Process the primitive SIM_EVENTLIST_REQ. */ GLOBAL void stk_sim_eventlist_req ( T_SIM_EVENTLIST_REQ* sim_eventlist_req) { TRACE_FUNCTION ("stk_sim_eventlist_req()"); /* * store new state of Data available event */ sim_data.event_data_avail = sim_eventlist_req->event_data_avail; /* * free primitive */ PFREE(sim_eventlist_req); /* * send confirm primitive */ { PALLOC(sim_eventlist_cnf, SIM_EVENTLIST_CNF); sim_eventlist_cnf->event_data_avail = sim_data.event_data_avail; PSEND(hCommMMI, sim_eventlist_cnf); } } /* stk_sim_eventlist_req() */ #endif /* FF_SAT_E */ /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_sim_toolkit_res | +--------------------------------------------------------------------+ PURPOSE : Process the primitive SIM_TOOLKIT_RES. MMI sends a Terminal Response Message to SIM toolkit. */ GLOBAL void stk_sim_toolkit_res (T_SIM_TOOLKIT_RES * sim_toolkit_res) { TRACE_FUNCTION ("stk_sim_toolkit_res()"); if (sim_data.ext_sat_cmd) { FKT_TerminalResponse (sim_toolkit_res->stk_cmd.cmd, (USHORT)(sim_toolkit_res->stk_cmd.l_cmd>>3)); PFREE (sim_toolkit_res); sim_data.ext_sat_cmd = FALSE; stk_start_timer_and_poll(); } else { TRACE_EVENT("no outstanding TR"); } } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_sim_toolkit_req | +--------------------------------------------------------------------+ PURPOSE : Process the primitive SIM_TOOLKIT_REQ. MMI sends an Envelope Message to SIM toolkit. */ GLOBAL void stk_sim_toolkit_req (T_SIM_TOOLKIT_REQ * sim_toolkit_req) { PALLOC (sim_toolkit_cnf, SIM_TOOLKIT_CNF); sim_toolkit_cnf->stk_cmd.o_cmd = 0; sim_toolkit_cnf->req_id = sim_toolkit_req->req_id; TRACE_FUNCTION ("stk_sim_toolkit_req()"); switch (sim_toolkit_req->source) // check valid source { case SRC_MMI: case SRC_SMS: break; default: TRACE_EVENT ("SIM_TOOLKIT_REQ: invalid source"); PFREE (sim_toolkit_cnf); PFREE (sim_toolkit_req); return; } /* * Forward envelope command to SIM toolkit */ sim_toolkit_cnf->cause = FKT_Envelope (sim_toolkit_cnf->stk_cmd.cmd, sim_toolkit_req->stk_cmd.cmd, (USHORT)(sim_toolkit_req->stk_cmd.l_cmd >> 3), NOT_PRESENT_16BIT); /* * Special treatment of SW1 is required for Call Control/MO-SM Control by SIM */ if (sim_toolkit_cnf->cause NEQ SIM_NO_ERROR) { switch (sim_toolkit_req->stk_cmd.cmd[0]) /* ENVELOPE tag */ { case 0xD4: /* Call Control */ case 0xD5: /* MO-SM Control */ if (sim_data.sw1 EQ 0x6F OR sim_data.sw1 EQ 0x92) /* Card problem */ sim_toolkit_cnf->cause = SIM_NO_ERROR; break; default: break; } } if (sim_data.sim_data_len > 0) { sim_toolkit_cnf->stk_cmd.l_cmd = stk_l_cmd; stk_l_cmd = 0; } else sim_toolkit_cnf->stk_cmd.l_cmd = 0; /* * send confirmation to requesting entity */ if (sim_toolkit_req->source EQ SRC_MMI) { PSENDX (MMI, sim_toolkit_cnf); } else { PSENDX (SMS, sim_toolkit_cnf); } PFREE (sim_toolkit_req); stk_start_timer_and_poll(); } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_file_update_res | +--------------------------------------------------------------------+ PURPOSE : Process the primitive SIM_FILE_UPDATE_RES. A File Change Notification is responded. */ GLOBAL void stk_file_update_res (T_SIM_FILE_UPDATE_RES * file_update_res) { UBYTE *p_res; // access to result code TRACE_FUNCTION ("stk_file_update_res()"); if (file_update_res->source EQ SRC_MMI) sim_data.file_change_resp &= ~1; else if (file_update_res->source EQ SRC_SMS) sim_data.file_change_resp &= ~2; else if (file_update_res->source EQ SRC_MM) sim_data.file_change_resp &= ~4; else { PFREE (file_update_res); return; } p_res = &sim_data.stk_response[sim_data.stk_response[1] + 8]; if (file_update_res->fu_rsc EQ SIM_FU_SUCC_ADD) *p_res = STK_RES_SUCC_ADD_EF_READ; if (sim_data.file_change_resp EQ 0) { FKT_TerminalResponse (sim_data.stk_response, (USHORT)sim_data.stk_resp_len); stk_start_timer_and_poll(); } PFREE (file_update_res); } /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_timeout | +--------------------------------------------------------------------+ PURPOSE : Process the timeout of a timer established by the SAT command TIMER MANAGEMENT. */ static const UBYTE timer_env[] = { STK_TIMER_EXPIRATION_TAG, STK_DEVICE_IDENTITY_LEN+STK_TIMER_ID_LEN+STK_TIMER_VALUE_LEN+6, STK_DEVICE_IDENTITY_TAG|STK_COMPREHENSION_REQUIRED, STK_DEVICE_IDENTITY_LEN, 0x82, 0x81, STK_TIMER_ID_TAG|STK_COMPREHENSION_REQUIRED, STK_TIMER_ID_LEN, 0, STK_TIMER_VALUE_TAG|STK_COMPREHENSION_REQUIRED, STK_TIMER_VALUE_LEN, 0, 0, 0 }; extern UBYTE pending_timers[]; extern UBYTE next_pos_to_fill; extern UBYTE next_pos_to_send; GLOBAL void stk_timeout (USHORT index) { UBYTE env[sizeof(timer_env)]; UBYTE dummy[4]; USHORT error; TRACE_FUNCTION("stk_timeout"); #ifdef FF_SAT_E /* * handle BIP timeout */ if (index EQ SIM_BIP_TIMER) { /* * close DTI connection */ switch(sim_data.dti_connection_state) { case SIM_DTI_CONNECTION_OPEN: case SIM_DTI_CONNECTION_SETUP: stk_close_dti_connection(TRUE); break; default: stk_close_dti_connection(FALSE); break; } /* * close BIP channel and inform MMI */ stk_close_bip_channel(RSLT_BEARIND_PERR, ADD_BIP_CHAN_CLOSD); stk_dti_inform_mmi(SIM_DTI_DISCONNECT, SIM_BIP_CLOSE_CHANNEL); return; } #endif /* FF_SAT_E */ if ((unsigned)(--index) >= MAX_SAT_TIMER) /* index range 0..7 */ return; if (sim_data.timer[index].active) { memcpy (env, timer_env, sizeof(timer_env)); env[8] = (UBYTE)(index + 1); /* Timer number range is 1..8 */ env[11] = sim_data.timer[index].hour; env[12] = sim_data.timer[index].minute; env[13] = sim_data.timer[index].second; error = FKT_Envelope (dummy, env, sizeof(timer_env), 0); //TISH, test patch for OMAPS00179771: SATK: Timer Expiration, modified by Jinshu Wang, 2008-09-01 //start #if 0 if (error EQ SIM_NO_ERROR) stk_start_timer_and_poll(); #else if (error EQ SIM_NO_ERROR) { stk_start_timer_and_poll(); sim_data.timer[index].active = FALSE; //modified by Jinshu Wang, 2008-09-04 if(sim_data.chk_sat_avail) { sim_data.chk_sat_avail = FALSE; stk_proactive_polling(); } return; //modified by Jinshu Wang, 2008-09-04 } #endif //end /* * If SIM response is busy(9300), we have to once retry sending * timer-expiry envelope after SIM becomes OK on getting a TR */ if (error EQ SIM_CAUSE_SAT_BUSY) { pending_timers[next_pos_to_fill] = (UBYTE) index; if (8 == next_pos_to_fill) { next_pos_to_fill = 0; } else { next_pos_to_fill++; } } sim_data.timer[index].active = FALSE; } } #ifdef TI_PS_FF_AT_P_CMD_CUST /* +--------------------------------------------------------------------+ | PROJECT : GSM-PS (8419) MODULE : SIM_STK | | STATE : code ROUTINE : stk_sim_refresh_user_res | +--------------------------------------------------------------------+ PURPOSE : Process the primitive SIM_REFRESH_USER_RES. MMI sends a Response Message to SIM_TOOLKIT_IND. This Primitive should not occur when cust_mode is 0 */ GLOBAL void stk_sim_refresh_user_res (T_SIM_REFRESH_USER_RES * sim_refresh_user_res) { TRACE_FUNCTION ("stk_sim_refresh_user_res()"); /* primitive should not occur for cust_mode 0 */ if (sim_data.cust_mode EQ 0) { //Handle error condition TRACE_FUNCTION_P1 ("!!!!Incorrect operation: Unexpected mode: cust_mode = %d !!!!",sim_data.cust_mode); TRACE_FUNCTION ("Primitive should not occur if cust_mode = 0"); //MFREE (sim_data.context_switch_ptr); Don't free context as it should be freed by other thread. i.e. stk_proactive_polling() PFREE (sim_refresh_user_res); return; } /* check that we are expecting primitive */ if ( sim_data.user_confirmation_expected EQ FALSE ) { //Handle error condition TRACE_FUNCTION ("!!!!Incorrect operation: user_confirmation_expected is FALSE !!!!"); TRACE_FUNCTION ("Primitive should only occur if user_confirmation_expected is TRUE"); //MFREE (sim_data.context_switch_ptr); Don't free context switch as setup can not be guaranteed to be correct. PFREE (sim_refresh_user_res); return; } else { //Reset user_confirmation_expected sim_data.user_confirmation_expected = FALSE; } if ((sim_refresh_user_res->user_accepts) EQ FALSE) { TRACE_FUNCTION ("User REJECTS Refresh Request"); FKT_TerminalResponse (sim_refresh_user_res->stk_cmd.cmd, (USHORT)(sim_refresh_user_res->stk_cmd.l_cmd>>3)); sim_data.ext_sat_cmd = FALSE; stk_start_timer_and_poll(); //process_sim_refresh() automatically frees the signal. //As we are not calling process_sim_refresh() we need to handle the freeing TRACE_ASSERT(sim_data.context_switch_ptr->sig_ptr); PFREE (sim_data.context_switch_ptr->sig_ptr); } else { TRACE_FUNCTION ("User ACCEPTS Refresh Request"); process_sim_refresh(sim_data.context_switch_ptr); TRACE_FUNCTION ("stk_sim_refresh_user_res() send end of SAT session indicator"); if ((sim_data.term_resp_sent) AND (sim_data.sat_session)) { PALLOC (sim_toolkit_ind, SIM_TOOLKIT_IND); memset (sim_toolkit_ind, 0, sizeof (T_SIM_TOOLKIT_IND)); #ifdef _SIMULATION_ TRACE_EVENT("SAT session ended"); #endif sim_data.sat_session = FALSE; sim_data.term_resp_sent = FALSE; PSENDX (MMI, sim_toolkit_ind); } } TRACE_ASSERT(sim_data.context_switch_ptr); MFREE (sim_data.context_switch_ptr); sim_data.context_switch_ptr = NULL; PFREE (sim_refresh_user_res); TRACE_FUNCTION ("stk_sim_refresh_user_res() exited"); } #endif /* TI_PS_FF_AT_P_CMD_CUST */ #endif