xref: /btstack/src/classic/hfp_gsm_model.c (revision 20b2edb6e64d1c94b04514e8c185411d5eb4c044)
174386ee0SMatthias Ringwald /*
274386ee0SMatthias Ringwald  * Copyright (C) 2014 BlueKitchen GmbH
374386ee0SMatthias Ringwald  *
474386ee0SMatthias Ringwald  * Redistribution and use in source and binary forms, with or without
574386ee0SMatthias Ringwald  * modification, are permitted provided that the following conditions
674386ee0SMatthias Ringwald  * are met:
774386ee0SMatthias Ringwald  *
874386ee0SMatthias Ringwald  * 1. Redistributions of source code must retain the above copyright
974386ee0SMatthias Ringwald  *    notice, this list of conditions and the following disclaimer.
1074386ee0SMatthias Ringwald  * 2. Redistributions in binary form must reproduce the above copyright
1174386ee0SMatthias Ringwald  *    notice, this list of conditions and the following disclaimer in the
1274386ee0SMatthias Ringwald  *    documentation and/or other materials provided with the distribution.
1374386ee0SMatthias Ringwald  * 3. Neither the name of the copyright holders nor the names of
1474386ee0SMatthias Ringwald  *    contributors may be used to endorse or promote products derived
1574386ee0SMatthias Ringwald  *    from this software without specific prior written permission.
1674386ee0SMatthias Ringwald  * 4. Any redistribution, use, or modification is done solely for
1774386ee0SMatthias Ringwald  *    personal benefit and not for any commercial purpose or for
1874386ee0SMatthias Ringwald  *    monetary gain.
1974386ee0SMatthias Ringwald  *
2074386ee0SMatthias Ringwald  * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS
2174386ee0SMatthias Ringwald  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
2274386ee0SMatthias Ringwald  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
2374386ee0SMatthias Ringwald  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS
2474386ee0SMatthias Ringwald  * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
2574386ee0SMatthias Ringwald  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
2674386ee0SMatthias Ringwald  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
2774386ee0SMatthias Ringwald  * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
2874386ee0SMatthias Ringwald  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
2974386ee0SMatthias Ringwald  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
3074386ee0SMatthias Ringwald  * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
3174386ee0SMatthias Ringwald  * SUCH DAMAGE.
3274386ee0SMatthias Ringwald  *
3374386ee0SMatthias Ringwald  * Please inquire about commercial licensing options at
3474386ee0SMatthias Ringwald  * [email protected]
3574386ee0SMatthias Ringwald  *
3674386ee0SMatthias Ringwald  */
37ab2c6ae4SMatthias Ringwald 
38e501bae0SMatthias Ringwald #define BTSTACK_FILE__ "hfp_gsm_model.c"
3974386ee0SMatthias Ringwald 
4074386ee0SMatthias Ringwald // *****************************************************************************
4174386ee0SMatthias Ringwald //
429cae807eSMatthias Ringwald // GSM Model
4374386ee0SMatthias Ringwald //
4474386ee0SMatthias Ringwald // *****************************************************************************
4574386ee0SMatthias Ringwald 
467907f069SMatthias Ringwald #include "btstack_config.h"
4774386ee0SMatthias Ringwald 
4874386ee0SMatthias Ringwald #include <stdint.h>
4974386ee0SMatthias Ringwald #include <string.h>
5074386ee0SMatthias Ringwald 
5174386ee0SMatthias Ringwald #include "btstack_memory.h"
5259c6af15SMatthias Ringwald #include "classic/core.h"
5374386ee0SMatthias Ringwald #include "classic/hfp.h"
5474386ee0SMatthias Ringwald #include "classic/hfp_gsm_model.h"
55746ccb7eSMatthias Ringwald #include "classic/sdp_server.h"
56efda0b48SMatthias Ringwald #include "classic/sdp_client_rfcomm.h"
5716ece135SMatthias Ringwald #include "btstack_debug.h"
5874386ee0SMatthias Ringwald #include "hci.h"
5956042629SMatthias Ringwald #include "hci_cmd.h"
6074386ee0SMatthias Ringwald #include "hci_dump.h"
6174386ee0SMatthias Ringwald #include "l2cap.h"
6282636622SMatthias Ringwald #include "btstack_run_loop.h"
6374386ee0SMatthias Ringwald 
6474386ee0SMatthias Ringwald #define HFP_GSM_MAX_NR_CALLS 3
659cae807eSMatthias Ringwald #define HFP_GSM_MAX_CALL_NUMBER_SIZE 25
6674386ee0SMatthias Ringwald 
6774386ee0SMatthias Ringwald static hfp_gsm_call_t gsm_calls[HFP_GSM_MAX_NR_CALLS];
68*20b2edb6SMatthias Ringwald static hfp_callsetup_status_t callsetup_status;
6974386ee0SMatthias Ringwald 
70d0c20769SMatthias Ringwald static uint8_t clip_type;
719cae807eSMatthias Ringwald static char clip_number[HFP_GSM_MAX_CALL_NUMBER_SIZE];
729cae807eSMatthias Ringwald static char last_dialed_number[HFP_GSM_MAX_CALL_NUMBER_SIZE];
73d0c20769SMatthias Ringwald 
7466a048abSMatthias Ringwald static inline int get_number_active_calls(void);
7566a048abSMatthias Ringwald 
7666a048abSMatthias Ringwald static void set_callsetup_status(hfp_callsetup_status_t status){
7766a048abSMatthias Ringwald     callsetup_status = status;
7866a048abSMatthias Ringwald     if (callsetup_status != HFP_CALLSETUP_STATUS_OUTGOING_CALL_SETUP_IN_ALERTING_STATE) return;
7966a048abSMatthias Ringwald 
8066a048abSMatthias Ringwald     int i ;
8166a048abSMatthias Ringwald     for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
8266a048abSMatthias Ringwald         if (gsm_calls[i].direction == HFP_ENHANCED_CALL_DIR_OUTGOING){
8366a048abSMatthias Ringwald             gsm_calls[i].enhanced_status = HFP_ENHANCED_CALL_STATUS_OUTGOING_ALERTING;
8466a048abSMatthias Ringwald         }
8566a048abSMatthias Ringwald     }
8666a048abSMatthias Ringwald }
8766a048abSMatthias Ringwald 
8866a048abSMatthias Ringwald static inline void set_enhanced_call_status_active(int index_in_table){
89a9a12719SMatthias Ringwald     if ((index_in_table < 0) || (index_in_table > HFP_GSM_MAX_NR_CALLS)) return;
9066a048abSMatthias Ringwald     gsm_calls[index_in_table].enhanced_status = HFP_ENHANCED_CALL_STATUS_ACTIVE;
9166a048abSMatthias Ringwald     gsm_calls[index_in_table].used_slot = 1;
9266a048abSMatthias Ringwald }
9366a048abSMatthias Ringwald 
9466a048abSMatthias Ringwald static inline void set_enhanced_call_status_held(int index_in_table){
95a9a12719SMatthias Ringwald     if ((index_in_table < 0) || (index_in_table > HFP_GSM_MAX_NR_CALLS)) return;
9666a048abSMatthias Ringwald     gsm_calls[index_in_table].enhanced_status = HFP_ENHANCED_CALL_STATUS_HELD;
9766a048abSMatthias Ringwald     gsm_calls[index_in_table].used_slot = 1;
9866a048abSMatthias Ringwald }
9966a048abSMatthias Ringwald 
10066a048abSMatthias Ringwald static inline void set_enhanced_call_status_response_hold(int index_in_table){
101a9a12719SMatthias Ringwald     if ((index_in_table < 0) || (index_in_table > HFP_GSM_MAX_NR_CALLS)) return;
10266a048abSMatthias Ringwald     gsm_calls[index_in_table].enhanced_status = HFP_ENHANCED_CALL_STATUS_CALL_HELD_BY_RESPONSE_AND_HOLD;
10366a048abSMatthias Ringwald     gsm_calls[index_in_table].used_slot = 1;
10466a048abSMatthias Ringwald }
10566a048abSMatthias Ringwald 
10666a048abSMatthias Ringwald static inline void set_enhanced_call_status_initiated(int index_in_table){
107a9a12719SMatthias Ringwald     if ((index_in_table < 0) || (index_in_table > HFP_GSM_MAX_NR_CALLS)) return;
10866a048abSMatthias Ringwald     if (gsm_calls[index_in_table].direction == HFP_ENHANCED_CALL_DIR_OUTGOING){
10966a048abSMatthias Ringwald         gsm_calls[index_in_table].enhanced_status = HFP_ENHANCED_CALL_STATUS_OUTGOING_DIALING;
11066a048abSMatthias Ringwald     } else {
11166a048abSMatthias Ringwald         if (get_number_active_calls() > 0){
11266a048abSMatthias Ringwald             gsm_calls[index_in_table].enhanced_status = HFP_ENHANCED_CALL_STATUS_INCOMING_WAITING;
11366a048abSMatthias Ringwald         } else {
11466a048abSMatthias Ringwald             gsm_calls[index_in_table].enhanced_status = HFP_ENHANCED_CALL_STATUS_INCOMING;
11566a048abSMatthias Ringwald         }
11666a048abSMatthias Ringwald     }
11766a048abSMatthias Ringwald     gsm_calls[index_in_table].used_slot = 1;
11866a048abSMatthias Ringwald }
11966a048abSMatthias Ringwald 
12066a048abSMatthias Ringwald static int get_enhanced_call_status(int index_in_table){
121a9a12719SMatthias Ringwald     if ((index_in_table < 0) || (index_in_table > HFP_GSM_MAX_NR_CALLS)) return -1;
12266a048abSMatthias Ringwald     if (!gsm_calls[index_in_table].used_slot) return -1;
12366a048abSMatthias Ringwald     return gsm_calls[index_in_table].enhanced_status;
12466a048abSMatthias Ringwald }
12566a048abSMatthias Ringwald 
12666a048abSMatthias Ringwald static inline int is_enhanced_call_status_active(int index_in_table){
127a9a12719SMatthias Ringwald     if ((index_in_table < 0) || (index_in_table > HFP_GSM_MAX_NR_CALLS)) return 0;
12866a048abSMatthias Ringwald     return get_enhanced_call_status(index_in_table) == HFP_ENHANCED_CALL_STATUS_ACTIVE;
12966a048abSMatthias Ringwald }
13066a048abSMatthias Ringwald 
13166a048abSMatthias Ringwald static inline int is_enhanced_call_status_initiated(int index_in_table){
132a9a12719SMatthias Ringwald     if ((index_in_table < 0) || (index_in_table > HFP_GSM_MAX_NR_CALLS)) return 0;
13366a048abSMatthias Ringwald     switch (get_enhanced_call_status(index_in_table)){
13466a048abSMatthias Ringwald         case HFP_ENHANCED_CALL_STATUS_OUTGOING_DIALING:
13566a048abSMatthias Ringwald         case HFP_ENHANCED_CALL_STATUS_OUTGOING_ALERTING:
13666a048abSMatthias Ringwald         case HFP_ENHANCED_CALL_STATUS_INCOMING:
13766a048abSMatthias Ringwald         case HFP_ENHANCED_CALL_STATUS_INCOMING_WAITING:
13866a048abSMatthias Ringwald             return 1;
13966a048abSMatthias Ringwald         default:
14066a048abSMatthias Ringwald             return 0;
14166a048abSMatthias Ringwald     }
14266a048abSMatthias Ringwald }
14366a048abSMatthias Ringwald 
14466a048abSMatthias Ringwald static void free_call_slot(int index_in_table){
145a9a12719SMatthias Ringwald     if ((index_in_table < 0) || (index_in_table > HFP_GSM_MAX_NR_CALLS)) return;
14666a048abSMatthias Ringwald     gsm_calls[index_in_table].used_slot = 0;
14766a048abSMatthias Ringwald }
148d0c20769SMatthias Ringwald 
149d210d9c4SMatthias Ringwald void hfp_gsm_init(void){
150*20b2edb6SMatthias Ringwald     callsetup_status = HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS;
151d0c20769SMatthias Ringwald     clip_type = 0;
152d0c20769SMatthias Ringwald     memset(clip_number, 0, sizeof(clip_number));
1539cae807eSMatthias Ringwald     memset(last_dialed_number, 0, sizeof(last_dialed_number));
154d210d9c4SMatthias Ringwald     memset(gsm_calls, 0, sizeof(gsm_calls));
155d210d9c4SMatthias Ringwald     int i;
156d210d9c4SMatthias Ringwald     for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
15766a048abSMatthias Ringwald         free_call_slot(i);
158d210d9c4SMatthias Ringwald     }
159d210d9c4SMatthias Ringwald }
16074386ee0SMatthias Ringwald 
161*20b2edb6SMatthias Ringwald void hfp_gsm_deinit(void){
162*20b2edb6SMatthias Ringwald }
163*20b2edb6SMatthias Ringwald 
16466a048abSMatthias Ringwald static int get_number_calls_with_enhanced_status(hfp_enhanced_call_status_t enhanced_status){
16574386ee0SMatthias Ringwald     int i, count = 0;
16674386ee0SMatthias Ringwald     for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
167fdfcab43SMatthias Ringwald         if (get_enhanced_call_status(i) == (int) enhanced_status) count++;
16874386ee0SMatthias Ringwald     }
16974386ee0SMatthias Ringwald     return count;
17074386ee0SMatthias Ringwald }
17174386ee0SMatthias Ringwald 
17266a048abSMatthias Ringwald static int get_call_index_with_enhanced_status(hfp_enhanced_call_status_t enhanced_status){
17374386ee0SMatthias Ringwald     int i ;
17474386ee0SMatthias Ringwald     for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
175fdfcab43SMatthias Ringwald         if (get_enhanced_call_status(i) == (int) enhanced_status) return i;
17666a048abSMatthias Ringwald     }
17766a048abSMatthias Ringwald     return -1;
17866a048abSMatthias Ringwald }
17966a048abSMatthias Ringwald 
18066a048abSMatthias Ringwald static inline int get_initiated_call_index(void){
18166a048abSMatthias Ringwald     int i ;
18266a048abSMatthias Ringwald     for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
18366a048abSMatthias Ringwald         if (is_enhanced_call_status_initiated(i)) return i;
18474386ee0SMatthias Ringwald     }
18574386ee0SMatthias Ringwald     return -1;
18674386ee0SMatthias Ringwald }
18774386ee0SMatthias Ringwald 
1889cae807eSMatthias Ringwald static inline int get_next_free_slot(void){
18966a048abSMatthias Ringwald     int i ;
19066a048abSMatthias Ringwald     for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
19166a048abSMatthias Ringwald         if (!gsm_calls[i].used_slot) return i;
19266a048abSMatthias Ringwald     }
19366a048abSMatthias Ringwald     return -1;
19474386ee0SMatthias Ringwald }
19574386ee0SMatthias Ringwald 
1969cae807eSMatthias Ringwald static inline int get_active_call_index(void){
19766a048abSMatthias Ringwald     return get_call_index_with_enhanced_status(HFP_ENHANCED_CALL_STATUS_ACTIVE);
198d210d9c4SMatthias Ringwald }
199d210d9c4SMatthias Ringwald 
2009cae807eSMatthias Ringwald static inline int get_held_call_index(void){
20166a048abSMatthias Ringwald     return get_call_index_with_enhanced_status(HFP_ENHANCED_CALL_STATUS_HELD);
202d210d9c4SMatthias Ringwald }
203d210d9c4SMatthias Ringwald 
2049cae807eSMatthias Ringwald static inline int get_response_held_call_index(void){
20566a048abSMatthias Ringwald     return get_call_index_with_enhanced_status(HFP_ENHANCED_CALL_STATUS_CALL_HELD_BY_RESPONSE_AND_HOLD);
206d210d9c4SMatthias Ringwald }
207d210d9c4SMatthias Ringwald 
2089cae807eSMatthias Ringwald static inline int get_number_none_calls(void){
20966a048abSMatthias Ringwald     int i, count = 0;
21066a048abSMatthias Ringwald     for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
21166a048abSMatthias Ringwald         if (!gsm_calls[i].used_slot) count++;
21266a048abSMatthias Ringwald     }
21366a048abSMatthias Ringwald     return count;
214d210d9c4SMatthias Ringwald }
21574386ee0SMatthias Ringwald 
2169cae807eSMatthias Ringwald static inline int get_number_active_calls(void){
21766a048abSMatthias Ringwald     return get_number_calls_with_enhanced_status(HFP_ENHANCED_CALL_STATUS_ACTIVE);
21874386ee0SMatthias Ringwald }
21974386ee0SMatthias Ringwald 
2209cae807eSMatthias Ringwald static inline int get_number_held_calls(void){
22166a048abSMatthias Ringwald     return get_number_calls_with_enhanced_status(HFP_ENHANCED_CALL_STATUS_HELD);
22274386ee0SMatthias Ringwald }
22374386ee0SMatthias Ringwald 
2249cae807eSMatthias Ringwald static inline int get_number_response_held_calls(void){
22566a048abSMatthias Ringwald     return get_number_calls_with_enhanced_status(HFP_ENHANCED_CALL_STATUS_CALL_HELD_BY_RESPONSE_AND_HOLD);
226d210d9c4SMatthias Ringwald }
227d210d9c4SMatthias Ringwald 
2289cae807eSMatthias Ringwald static int next_call_index(void){
229d0c20769SMatthias Ringwald     return HFP_GSM_MAX_NR_CALLS + 1 - get_number_none_calls();
230d0c20769SMatthias Ringwald }
231d0c20769SMatthias Ringwald 
232d0c20769SMatthias Ringwald static void hfp_gsm_set_clip(int index_in_table, uint8_t type, const char * number){
23321791470SMilanka Ringwald     uint16_t number_str_len = (uint16_t) strlen(number);
23421791470SMilanka Ringwald     if (number_str_len == 0) return;
2359cae807eSMatthias Ringwald 
236d0c20769SMatthias Ringwald     gsm_calls[index_in_table].clip_type = type;
23721791470SMilanka Ringwald     int clip_number_size = btstack_min(number_str_len, HFP_GSM_MAX_CALL_NUMBER_SIZE - 1);
238d0c20769SMatthias Ringwald     strncpy(gsm_calls[index_in_table].clip_number, number, clip_number_size);
2399cae807eSMatthias Ringwald     gsm_calls[index_in_table].clip_number[clip_number_size] = '\0';
2409cae807eSMatthias Ringwald     strncpy(last_dialed_number, number, clip_number_size);
2419cae807eSMatthias Ringwald     last_dialed_number[clip_number_size] = '\0';
2429cae807eSMatthias Ringwald 
2439cae807eSMatthias Ringwald     clip_type = 0;
2449cae807eSMatthias Ringwald     memset(clip_number, 0, sizeof(clip_number));
245d0c20769SMatthias Ringwald }
246d0c20769SMatthias Ringwald 
247d0c20769SMatthias Ringwald static void delete_call(int delete_index_in_table){
248d0c20769SMatthias Ringwald     int i ;
249d0c20769SMatthias Ringwald     for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
250d0c20769SMatthias Ringwald         if (gsm_calls[i].index > gsm_calls[delete_index_in_table].index){
251d0c20769SMatthias Ringwald             gsm_calls[i].index--;
252d0c20769SMatthias Ringwald         }
253d0c20769SMatthias Ringwald     }
25466a048abSMatthias Ringwald     free_call_slot(delete_index_in_table);
255d0c20769SMatthias Ringwald 
256d0c20769SMatthias Ringwald     gsm_calls[delete_index_in_table].clip_type = 0;
257d0c20769SMatthias Ringwald     gsm_calls[delete_index_in_table].index = 0;
258d0c20769SMatthias Ringwald     gsm_calls[delete_index_in_table].clip_number[0] = '\0';
2599cae807eSMatthias Ringwald     gsm_calls[delete_index_in_table].mpty = HFP_ENHANCED_CALL_MPTY_NOT_A_CONFERENCE_CALL;
260d0c20769SMatthias Ringwald }
261d0c20769SMatthias Ringwald 
2629cae807eSMatthias Ringwald 
2639cae807eSMatthias Ringwald static void create_call(hfp_enhanced_call_dir_t direction){
264d0c20769SMatthias Ringwald     int next_free_slot = get_next_free_slot();
2659cae807eSMatthias Ringwald     gsm_calls[next_free_slot].direction = direction;
266d0c20769SMatthias Ringwald     gsm_calls[next_free_slot].index = next_call_index();
26766a048abSMatthias Ringwald     set_enhanced_call_status_initiated(next_free_slot);
268d0c20769SMatthias Ringwald     gsm_calls[next_free_slot].clip_type = 0;
269d0c20769SMatthias Ringwald     gsm_calls[next_free_slot].clip_number[0] = '\0';
2709cae807eSMatthias Ringwald     gsm_calls[next_free_slot].mpty = HFP_ENHANCED_CALL_MPTY_NOT_A_CONFERENCE_CALL;
271d0c20769SMatthias Ringwald 
272d0c20769SMatthias Ringwald     hfp_gsm_set_clip(next_free_slot, clip_type, clip_number);
273d0c20769SMatthias Ringwald }
274d0c20769SMatthias Ringwald 
2759cae807eSMatthias Ringwald 
2769cae807eSMatthias Ringwald int hfp_gsm_get_number_of_calls(void){
2779cae807eSMatthias Ringwald     return HFP_GSM_MAX_NR_CALLS - get_number_none_calls();
2789cae807eSMatthias Ringwald }
2799cae807eSMatthias Ringwald 
2809cae807eSMatthias Ringwald void hfp_gsm_clear_last_dialed_number(void){
2819cae807eSMatthias Ringwald     memset(last_dialed_number, 0, sizeof(last_dialed_number));
2829cae807eSMatthias Ringwald }
2839cae807eSMatthias Ringwald 
2849cae807eSMatthias Ringwald char * hfp_gsm_last_dialed_number(void){
2859cae807eSMatthias Ringwald     return &last_dialed_number[0];
2869cae807eSMatthias Ringwald }
2879cae807eSMatthias Ringwald 
2889cae807eSMatthias Ringwald hfp_gsm_call_t * hfp_gsm_call(int call_index){
2899cae807eSMatthias Ringwald     int i;
2909cae807eSMatthias Ringwald 
2919cae807eSMatthias Ringwald     for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
2929cae807eSMatthias Ringwald         hfp_gsm_call_t * call = &gsm_calls[i];
2939cae807eSMatthias Ringwald         if (call->index != call_index) continue;
2949cae807eSMatthias Ringwald         return call;
2959cae807eSMatthias Ringwald     }
2969cae807eSMatthias Ringwald     return NULL;
2979cae807eSMatthias Ringwald }
2989cae807eSMatthias Ringwald 
2999cae807eSMatthias Ringwald uint8_t hfp_gsm_clip_type(void){
300d0c20769SMatthias Ringwald     if (clip_type != 0) return clip_type;
301d0c20769SMatthias Ringwald 
302d0c20769SMatthias Ringwald     int initiated_call_index = get_initiated_call_index();
303d0c20769SMatthias Ringwald     if (initiated_call_index != -1){
304d0c20769SMatthias Ringwald         if (gsm_calls[initiated_call_index].clip_type != 0) {
305d0c20769SMatthias Ringwald             return gsm_calls[initiated_call_index].clip_type;
306d0c20769SMatthias Ringwald         }
307d0c20769SMatthias Ringwald     }
308d0c20769SMatthias Ringwald 
309d0c20769SMatthias Ringwald     int active_call_index = get_active_call_index();
310d0c20769SMatthias Ringwald     if (active_call_index != -1){
311d0c20769SMatthias Ringwald         if (gsm_calls[active_call_index].clip_type != 0) {
312d0c20769SMatthias Ringwald             return gsm_calls[active_call_index].clip_type;
313d0c20769SMatthias Ringwald         }
314d0c20769SMatthias Ringwald     }
315d0c20769SMatthias Ringwald     return 0;
316d0c20769SMatthias Ringwald }
317d0c20769SMatthias Ringwald 
3189cae807eSMatthias Ringwald char *  hfp_gsm_clip_number(void){
3199cae807eSMatthias Ringwald     if (strlen(clip_number) != 0) return clip_number;
320d0c20769SMatthias Ringwald 
321d0c20769SMatthias Ringwald     int initiated_call_index = get_initiated_call_index();
322d0c20769SMatthias Ringwald     if (initiated_call_index != -1){
323d0c20769SMatthias Ringwald         if (gsm_calls[initiated_call_index].clip_type != 0) {
324d0c20769SMatthias Ringwald             return gsm_calls[initiated_call_index].clip_number;
325d0c20769SMatthias Ringwald         }
326d0c20769SMatthias Ringwald     }
327d0c20769SMatthias Ringwald 
328d0c20769SMatthias Ringwald     int active_call_index = get_active_call_index();
329d0c20769SMatthias Ringwald     if (active_call_index != -1){
330d0c20769SMatthias Ringwald         if (gsm_calls[active_call_index].clip_type != 0) {
331d0c20769SMatthias Ringwald             return gsm_calls[active_call_index].clip_number;
332d0c20769SMatthias Ringwald         }
333d0c20769SMatthias Ringwald     }
334d0c20769SMatthias Ringwald     clip_number[0] = 0;
335d0c20769SMatthias Ringwald     return clip_number;
336d0c20769SMatthias Ringwald }
337d0c20769SMatthias Ringwald 
3389cae807eSMatthias Ringwald hfp_call_status_t hfp_gsm_call_status(void){
339d210d9c4SMatthias Ringwald     if (get_number_active_calls() + get_number_held_calls() + get_number_response_held_calls()){
34074386ee0SMatthias Ringwald         return HFP_CALL_STATUS_ACTIVE_OR_HELD_CALL_IS_PRESENT;
34174386ee0SMatthias Ringwald     }
34274386ee0SMatthias Ringwald     return HFP_CALL_STATUS_NO_HELD_OR_ACTIVE_CALLS;
34374386ee0SMatthias Ringwald }
34474386ee0SMatthias Ringwald 
3459cae807eSMatthias Ringwald hfp_callheld_status_t hfp_gsm_callheld_status(void){
34674386ee0SMatthias Ringwald     // @note: order is important
34774386ee0SMatthias Ringwald     if (get_number_held_calls() == 0){
34874386ee0SMatthias Ringwald         return HFP_CALLHELD_STATUS_NO_CALLS_HELD;
34974386ee0SMatthias Ringwald     }
35074386ee0SMatthias Ringwald     if (get_number_active_calls() == 0) {
35174386ee0SMatthias Ringwald         return HFP_CALLHELD_STATUS_CALL_ON_HOLD_AND_NO_ACTIVE_CALLS;
35274386ee0SMatthias Ringwald     }
35374386ee0SMatthias Ringwald     return HFP_CALLHELD_STATUS_CALL_ON_HOLD_OR_SWAPPED;
35474386ee0SMatthias Ringwald }
35574386ee0SMatthias Ringwald 
3569cae807eSMatthias Ringwald hfp_callsetup_status_t hfp_gsm_callsetup_status(void){
35774386ee0SMatthias Ringwald     return callsetup_status;
35874386ee0SMatthias Ringwald }
35974386ee0SMatthias Ringwald 
3609cae807eSMatthias Ringwald static int hfp_gsm_response_held_active(void){
361d210d9c4SMatthias Ringwald     return get_response_held_call_index() != -1 ;
362d210d9c4SMatthias Ringwald }
363d210d9c4SMatthias Ringwald 
364d210d9c4SMatthias Ringwald int hfp_gsm_call_possible(void){
365d210d9c4SMatthias Ringwald     return get_number_none_calls() > 0;
366d210d9c4SMatthias Ringwald }
367d210d9c4SMatthias Ringwald 
368f8737b81SMatthias Ringwald void hfp_gsm_handler(hfp_ag_call_event_t event, uint8_t index, uint8_t type, const char * number){
369d0c20769SMatthias Ringwald     int next_free_slot = get_next_free_slot();
37074386ee0SMatthias Ringwald     int current_call_index = get_active_call_index();
371d210d9c4SMatthias Ringwald     int initiated_call_index = get_initiated_call_index();
372d210d9c4SMatthias Ringwald     int held_call_index = get_held_call_index();
373d0c20769SMatthias Ringwald     int i;
374d0c20769SMatthias Ringwald 
37574386ee0SMatthias Ringwald     switch (event){
37674386ee0SMatthias Ringwald         case HFP_AG_OUTGOING_CALL_INITIATED:
37774386ee0SMatthias Ringwald         case HFP_AG_OUTGOING_REDIAL_INITIATED:
37874386ee0SMatthias Ringwald             if (next_free_slot == -1){
379d210d9c4SMatthias Ringwald                 log_error("gsm: max call nr exceeded");
38074386ee0SMatthias Ringwald                 return;
38174386ee0SMatthias Ringwald             }
3829cae807eSMatthias Ringwald             create_call(HFP_ENHANCED_CALL_DIR_OUTGOING);
383d210d9c4SMatthias Ringwald             break;
38474386ee0SMatthias Ringwald 
385d210d9c4SMatthias Ringwald         case HFP_AG_OUTGOING_CALL_REJECTED:
386d210d9c4SMatthias Ringwald             if (current_call_index != -1){
387d0c20769SMatthias Ringwald                 delete_call(current_call_index);
388d210d9c4SMatthias Ringwald             }
38966a048abSMatthias Ringwald             set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
390d210d9c4SMatthias Ringwald             break;
391d210d9c4SMatthias Ringwald 
392d210d9c4SMatthias Ringwald         case HFP_AG_OUTGOING_CALL_ACCEPTED:
39374386ee0SMatthias Ringwald             if (current_call_index != -1){
39466a048abSMatthias Ringwald                 set_enhanced_call_status_held(current_call_index);
39574386ee0SMatthias Ringwald             }
39666a048abSMatthias Ringwald             set_callsetup_status(HFP_CALLSETUP_STATUS_OUTGOING_CALL_SETUP_IN_DIALING_STATE);
39774386ee0SMatthias Ringwald             break;
398d210d9c4SMatthias Ringwald 
39974386ee0SMatthias Ringwald         case HFP_AG_OUTGOING_CALL_RINGING:
400d210d9c4SMatthias Ringwald             if (current_call_index == -1){
401d210d9c4SMatthias Ringwald                 log_error("gsm: no active call");
402d210d9c4SMatthias Ringwald                 return;
403d210d9c4SMatthias Ringwald             }
40466a048abSMatthias Ringwald             set_callsetup_status(HFP_CALLSETUP_STATUS_OUTGOING_CALL_SETUP_IN_ALERTING_STATE);
40574386ee0SMatthias Ringwald             break;
40674386ee0SMatthias Ringwald         case HFP_AG_OUTGOING_CALL_ESTABLISHED:
40766a048abSMatthias Ringwald             set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
40866a048abSMatthias Ringwald             set_enhanced_call_status_active(initiated_call_index);
40974386ee0SMatthias Ringwald             break;
410d210d9c4SMatthias Ringwald 
411d210d9c4SMatthias Ringwald         case HFP_AG_INCOMING_CALL:
412d210d9c4SMatthias Ringwald             if (hfp_gsm_callsetup_status() != HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS) break;
41366a048abSMatthias Ringwald             set_callsetup_status(HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS);
4149cae807eSMatthias Ringwald             create_call(HFP_ENHANCED_CALL_DIR_INCOMING);
415d210d9c4SMatthias Ringwald             break;
416d210d9c4SMatthias Ringwald 
417d210d9c4SMatthias Ringwald         case HFP_AG_INCOMING_CALL_ACCEPTED_BY_AG:
418d210d9c4SMatthias Ringwald             if (hfp_gsm_callsetup_status() != HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS) break;
41966a048abSMatthias Ringwald             set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
420d210d9c4SMatthias Ringwald 
421d210d9c4SMatthias Ringwald             if (hfp_gsm_call_status() == HFP_CALL_STATUS_ACTIVE_OR_HELD_CALL_IS_PRESENT){
42266a048abSMatthias Ringwald                 set_enhanced_call_status_held(current_call_index);
423d210d9c4SMatthias Ringwald             }
42466a048abSMatthias Ringwald             set_enhanced_call_status_active(initiated_call_index);
425d210d9c4SMatthias Ringwald             break;
426d210d9c4SMatthias Ringwald 
427d210d9c4SMatthias Ringwald         case HFP_AG_HELD_CALL_JOINED_BY_AG:
428d210d9c4SMatthias Ringwald             if (hfp_gsm_call_status() != HFP_CALL_STATUS_ACTIVE_OR_HELD_CALL_IS_PRESENT) break;
429d210d9c4SMatthias Ringwald 
430d210d9c4SMatthias Ringwald             // TODO: is following condition correct? Can we join incoming call before it is answered?
431d210d9c4SMatthias Ringwald             if (callsetup_status == HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS){
43266a048abSMatthias Ringwald                 set_enhanced_call_status_active(initiated_call_index);
43366a048abSMatthias Ringwald                 set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
4349cae807eSMatthias Ringwald             } else if (hfp_gsm_callheld_status() == HFP_CALLHELD_STATUS_CALL_ON_HOLD_OR_SWAPPED) {
43566a048abSMatthias Ringwald                 set_enhanced_call_status_active(held_call_index);
436d210d9c4SMatthias Ringwald             }
437d210d9c4SMatthias Ringwald 
4389cae807eSMatthias Ringwald             for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
43966a048abSMatthias Ringwald                 if (is_enhanced_call_status_active(i)){
4409cae807eSMatthias Ringwald                     gsm_calls[i].mpty = HFP_ENHANCED_CALL_MPTY_CONFERENCE_CALL;
4419cae807eSMatthias Ringwald                 }
442d210d9c4SMatthias Ringwald             }
443d210d9c4SMatthias Ringwald             break;
444d210d9c4SMatthias Ringwald 
445d210d9c4SMatthias Ringwald         case HFP_AG_INCOMING_CALL_ACCEPTED_BY_HF:
446d210d9c4SMatthias Ringwald             if (hfp_gsm_callsetup_status() != HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS) break;
447d210d9c4SMatthias Ringwald             if (hfp_gsm_call_status() != HFP_CALL_STATUS_NO_HELD_OR_ACTIVE_CALLS) break;
44866a048abSMatthias Ringwald             set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
44966a048abSMatthias Ringwald             set_enhanced_call_status_active(initiated_call_index);
450d210d9c4SMatthias Ringwald             break;
451d210d9c4SMatthias Ringwald 
452d210d9c4SMatthias Ringwald         case HFP_AG_RESPONSE_AND_HOLD_ACCEPT_INCOMING_CALL_BY_AG:
453d210d9c4SMatthias Ringwald         case HFP_AG_RESPONSE_AND_HOLD_ACCEPT_INCOMING_CALL_BY_HF:
454d210d9c4SMatthias Ringwald             if (hfp_gsm_callsetup_status() != HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS) break;
455d210d9c4SMatthias Ringwald             if (hfp_gsm_call_status() != HFP_CALL_STATUS_NO_HELD_OR_ACTIVE_CALLS) break;
45666a048abSMatthias Ringwald             set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
45766a048abSMatthias Ringwald             set_enhanced_call_status_response_hold(initiated_call_index);
458d210d9c4SMatthias Ringwald             break;
459d210d9c4SMatthias Ringwald 
460d210d9c4SMatthias Ringwald         case HFP_AG_RESPONSE_AND_HOLD_ACCEPT_HELD_CALL_BY_AG:
461d210d9c4SMatthias Ringwald         case HFP_AG_RESPONSE_AND_HOLD_ACCEPT_HELD_CALL_BY_HF:
462d210d9c4SMatthias Ringwald             if (!hfp_gsm_response_held_active()) break;
46366a048abSMatthias Ringwald             set_enhanced_call_status_active(get_response_held_call_index());
464d210d9c4SMatthias Ringwald             break;
465d210d9c4SMatthias Ringwald 
466d210d9c4SMatthias Ringwald         case HFP_AG_RESPONSE_AND_HOLD_REJECT_HELD_CALL_BY_AG:
467d210d9c4SMatthias Ringwald         case HFP_AG_RESPONSE_AND_HOLD_REJECT_HELD_CALL_BY_HF:
468d210d9c4SMatthias Ringwald             if (!hfp_gsm_response_held_active()) break;
469d0c20769SMatthias Ringwald             delete_call(get_response_held_call_index());
470d210d9c4SMatthias Ringwald             break;
471d210d9c4SMatthias Ringwald 
472d210d9c4SMatthias Ringwald 
473d210d9c4SMatthias Ringwald         case HFP_AG_TERMINATE_CALL_BY_HF:
474d210d9c4SMatthias Ringwald             switch (hfp_gsm_call_status()){
475d210d9c4SMatthias Ringwald                 case HFP_CALL_STATUS_NO_HELD_OR_ACTIVE_CALLS:
47666a048abSMatthias Ringwald                     set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
477d210d9c4SMatthias Ringwald                     break;
478d210d9c4SMatthias Ringwald                 case HFP_CALL_STATUS_ACTIVE_OR_HELD_CALL_IS_PRESENT:
479d0c20769SMatthias Ringwald                     delete_call(current_call_index);
480d210d9c4SMatthias Ringwald                     break;
4817bbeb3adSMilanka Ringwald                 default:
4827bbeb3adSMilanka Ringwald                     break;
483d210d9c4SMatthias Ringwald             }
484d210d9c4SMatthias Ringwald             break;
485d210d9c4SMatthias Ringwald 
486d210d9c4SMatthias Ringwald         case HFP_AG_TERMINATE_CALL_BY_AG:
487d210d9c4SMatthias Ringwald             switch (hfp_gsm_call_status()){
488d210d9c4SMatthias Ringwald                 case HFP_CALL_STATUS_NO_HELD_OR_ACTIVE_CALLS:
489d210d9c4SMatthias Ringwald                     if (hfp_gsm_callsetup_status() != HFP_CALLSETUP_STATUS_INCOMING_CALL_SETUP_IN_PROGRESS) break;
49066a048abSMatthias Ringwald                     set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
491d210d9c4SMatthias Ringwald                     break;
492d210d9c4SMatthias Ringwald                 case HFP_CALL_STATUS_ACTIVE_OR_HELD_CALL_IS_PRESENT:
49366a048abSMatthias Ringwald                     set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
494d0c20769SMatthias Ringwald                     delete_call(current_call_index);
495d210d9c4SMatthias Ringwald                     break;
496d210d9c4SMatthias Ringwald                 default:
497d210d9c4SMatthias Ringwald                     break;
498d210d9c4SMatthias Ringwald             }
499d210d9c4SMatthias Ringwald             break;
500d210d9c4SMatthias Ringwald 
501d0c20769SMatthias Ringwald         case HFP_AG_CALL_DROPPED:
50266a048abSMatthias Ringwald             set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
503d210d9c4SMatthias Ringwald             if (hfp_gsm_call_status() != HFP_CALL_STATUS_ACTIVE_OR_HELD_CALL_IS_PRESENT) break;
504d210d9c4SMatthias Ringwald 
505d210d9c4SMatthias Ringwald             for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
506d0c20769SMatthias Ringwald                 delete_call(i);
507d210d9c4SMatthias Ringwald             }
508d210d9c4SMatthias Ringwald             break;
509d0c20769SMatthias Ringwald 
510d210d9c4SMatthias Ringwald         case HFP_AG_CALL_HOLD_USER_BUSY:
511d210d9c4SMatthias Ringwald             // Held or waiting call gets active,
51266a048abSMatthias Ringwald             set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
51366a048abSMatthias Ringwald             free_call_slot(initiated_call_index);
51466a048abSMatthias Ringwald             set_enhanced_call_status_active(held_call_index);
515d210d9c4SMatthias Ringwald             break;
516d210d9c4SMatthias Ringwald 
517d0c20769SMatthias Ringwald         case HFP_AG_CALL_HOLD_RELEASE_ACTIVE_ACCEPT_HELD_OR_WAITING_CALL:
518c1ab6cc1SMatthias Ringwald             if ((index != 0) && (index <= HFP_GSM_MAX_NR_CALLS) ){
519d0c20769SMatthias Ringwald                 for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
520d0c20769SMatthias Ringwald                     if (gsm_calls[i].index == index){
521d0c20769SMatthias Ringwald                         delete_call(i);
522d0c20769SMatthias Ringwald                         continue;
523d0c20769SMatthias Ringwald                     }
524d0c20769SMatthias Ringwald                 }
525d0c20769SMatthias Ringwald             } else {
526d210d9c4SMatthias Ringwald                 for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
52766a048abSMatthias Ringwald                     if (is_enhanced_call_status_active(i)){
528d0c20769SMatthias Ringwald                         delete_call(i);
529d0c20769SMatthias Ringwald                     }
530d210d9c4SMatthias Ringwald                 }
531d210d9c4SMatthias Ringwald             }
532d210d9c4SMatthias Ringwald 
533d210d9c4SMatthias Ringwald             if (callsetup_status != HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS){
53466a048abSMatthias Ringwald                 set_enhanced_call_status_active(initiated_call_index);
535d210d9c4SMatthias Ringwald             } else {
53666a048abSMatthias Ringwald                 set_enhanced_call_status_active(held_call_index);
537d210d9c4SMatthias Ringwald             }
538d0c20769SMatthias Ringwald 
53966a048abSMatthias Ringwald             set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
540d210d9c4SMatthias Ringwald             break;
541d0c20769SMatthias Ringwald 
542d0c20769SMatthias Ringwald         case HFP_AG_CALL_HOLD_PARK_ACTIVE_ACCEPT_HELD_OR_WAITING_CALL:
543d210d9c4SMatthias Ringwald             for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
544c1ab6cc1SMatthias Ringwald                 if (is_enhanced_call_status_active(i) && (gsm_calls[i].index != index)){
54566a048abSMatthias Ringwald                     set_enhanced_call_status_held(i);
546d210d9c4SMatthias Ringwald                 }
547d210d9c4SMatthias Ringwald             }
548d210d9c4SMatthias Ringwald 
549d210d9c4SMatthias Ringwald             if (callsetup_status != HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS){
55066a048abSMatthias Ringwald                 set_enhanced_call_status_active(initiated_call_index);
551d210d9c4SMatthias Ringwald             } else {
55266a048abSMatthias Ringwald                 set_enhanced_call_status_active(held_call_index);
553d210d9c4SMatthias Ringwald             }
55466a048abSMatthias Ringwald             set_callsetup_status(HFP_CALLSETUP_STATUS_NO_CALL_SETUP_IN_PROGRESS);
555d210d9c4SMatthias Ringwald             break;
556d0c20769SMatthias Ringwald 
557d0c20769SMatthias Ringwald         case HFP_AG_CALL_HOLD_ADD_HELD_CALL:
558d210d9c4SMatthias Ringwald             if (hfp_gsm_callheld_status() != HFP_CALLHELD_STATUS_NO_CALLS_HELD){
559d210d9c4SMatthias Ringwald                 for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
56066a048abSMatthias Ringwald                     if (gsm_calls[i].used_slot){
56166a048abSMatthias Ringwald                         set_enhanced_call_status_active(i);
5629cae807eSMatthias Ringwald                         gsm_calls[i].mpty = HFP_ENHANCED_CALL_MPTY_CONFERENCE_CALL;
563d210d9c4SMatthias Ringwald                     }
564d210d9c4SMatthias Ringwald                 }
565d210d9c4SMatthias Ringwald             }
566d210d9c4SMatthias Ringwald             break;
567d0c20769SMatthias Ringwald 
568d0c20769SMatthias Ringwald         case HFP_AG_CALL_HOLD_EXIT_AND_JOIN_CALLS:
569d210d9c4SMatthias Ringwald             for (i = 0; i < HFP_GSM_MAX_NR_CALLS; i++){
570d0c20769SMatthias Ringwald                 delete_call(i);
571d210d9c4SMatthias Ringwald             }
572d0c20769SMatthias Ringwald             break;
573d210d9c4SMatthias Ringwald 
574d0c20769SMatthias Ringwald         case HFP_AG_SET_CLIP:
575d0c20769SMatthias Ringwald             if (initiated_call_index != -1){
576d0c20769SMatthias Ringwald                 hfp_gsm_set_clip(initiated_call_index, type, number);
577d210d9c4SMatthias Ringwald                 break;
578d210d9c4SMatthias Ringwald             }
5799cae807eSMatthias Ringwald 
580d0c20769SMatthias Ringwald             clip_type = type;
581d0c20769SMatthias Ringwald             strncpy(clip_number, number, sizeof(clip_number));
582d0c20769SMatthias Ringwald             clip_number[sizeof(clip_number)-1] = '\0';
583d0c20769SMatthias Ringwald 
584d0c20769SMatthias Ringwald             break;
58574386ee0SMatthias Ringwald         default:
58674386ee0SMatthias Ringwald             break;
58774386ee0SMatthias Ringwald     }
58874386ee0SMatthias Ringwald }
589