• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /******************************************************************************
2  *
3  *  Copyright (C) 1999-2014 Broadcom Corporation
4  *
5  *  Licensed under the Apache License, Version 2.0 (the "License");
6  *  you may not use this file except in compliance with the License.
7  *  You may obtain a copy of the License at:
8  *
9  *  http://www.apache.org/licenses/LICENSE-2.0
10  *
11  *  Unless required by applicable law or agreed to in writing, software
12  *  distributed under the License is distributed on an "AS IS" BASIS,
13  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  *  See the License for the specific language governing permissions and
15  *  limitations under the License.
16  *
17  ******************************************************************************/
18 
19 /******************************************************************************
20  *
21  *  This file contains functions that handle inquiries. These include
22  *  setting discoverable mode, controlling the mode of the Baseband, and
23  *  maintaining a small database of inquiry responses, with API for people
24  *  to browse it.
25  *
26  ******************************************************************************/
27 
28 #include <stdlib.h>
29 #include <string.h>
30 #include <stdio.h>
31 #include <stddef.h>
32 
33 #include "device/include/controller.h"
34 #include "osi/include/time.h"
35 
36 #include "bt_types.h"
37 #include "bt_common.h"
38 #include "hcimsgs.h"
39 #include "btu.h"
40 #include "btm_api.h"
41 #include "btm_int.h"
42 #include "hcidefs.h"
43 
44 /* 3 second timeout waiting for responses */
45 #define BTM_INQ_REPLY_TIMEOUT_MS (3 * 1000)
46 
47 /* TRUE to enable DEBUG traces for btm_inq */
48 #ifndef BTM_INQ_DEBUG
49 #define BTM_INQ_DEBUG   FALSE
50 #endif
51 
52 extern fixed_queue_t *btu_general_alarm_queue;
53 
54 /********************************************************************************/
55 /*                 L O C A L    D A T A    D E F I N I T I O N S                */
56 /********************************************************************************/
57 static const LAP general_inq_lap = {0x9e,0x8b,0x33};
58 static const LAP limited_inq_lap = {0x9e,0x8b,0x00};
59 
60 const UINT16 BTM_EIR_UUID_LKUP_TBL[BTM_EIR_MAX_SERVICES] =
61 {
62     UUID_SERVCLASS_SERVICE_DISCOVERY_SERVER,
63 /*    UUID_SERVCLASS_BROWSE_GROUP_DESCRIPTOR,   */
64 /*    UUID_SERVCLASS_PUBLIC_BROWSE_GROUP,       */
65     UUID_SERVCLASS_SERIAL_PORT,
66     UUID_SERVCLASS_LAN_ACCESS_USING_PPP,
67     UUID_SERVCLASS_DIALUP_NETWORKING,
68     UUID_SERVCLASS_IRMC_SYNC,
69     UUID_SERVCLASS_OBEX_OBJECT_PUSH,
70     UUID_SERVCLASS_OBEX_FILE_TRANSFER,
71     UUID_SERVCLASS_IRMC_SYNC_COMMAND,
72     UUID_SERVCLASS_HEADSET,
73     UUID_SERVCLASS_CORDLESS_TELEPHONY,
74     UUID_SERVCLASS_AUDIO_SOURCE,
75     UUID_SERVCLASS_AUDIO_SINK,
76     UUID_SERVCLASS_AV_REM_CTRL_TARGET,
77 /*    UUID_SERVCLASS_ADV_AUDIO_DISTRIBUTION,    */
78     UUID_SERVCLASS_AV_REMOTE_CONTROL,
79 /*    UUID_SERVCLASS_VIDEO_CONFERENCING,        */
80     UUID_SERVCLASS_INTERCOM,
81     UUID_SERVCLASS_FAX,
82     UUID_SERVCLASS_HEADSET_AUDIO_GATEWAY,
83 /*    UUID_SERVCLASS_WAP,                       */
84 /*    UUID_SERVCLASS_WAP_CLIENT,                */
85     UUID_SERVCLASS_PANU,
86     UUID_SERVCLASS_NAP,
87     UUID_SERVCLASS_GN,
88     UUID_SERVCLASS_DIRECT_PRINTING,
89 /*    UUID_SERVCLASS_REFERENCE_PRINTING,        */
90     UUID_SERVCLASS_IMAGING,
91     UUID_SERVCLASS_IMAGING_RESPONDER,
92     UUID_SERVCLASS_IMAGING_AUTO_ARCHIVE,
93     UUID_SERVCLASS_IMAGING_REF_OBJECTS,
94     UUID_SERVCLASS_HF_HANDSFREE,
95     UUID_SERVCLASS_AG_HANDSFREE,
96     UUID_SERVCLASS_DIR_PRT_REF_OBJ_SERVICE,
97 /*    UUID_SERVCLASS_REFLECTED_UI,              */
98     UUID_SERVCLASS_BASIC_PRINTING,
99     UUID_SERVCLASS_PRINTING_STATUS,
100     UUID_SERVCLASS_HUMAN_INTERFACE,
101     UUID_SERVCLASS_CABLE_REPLACEMENT,
102     UUID_SERVCLASS_HCRP_PRINT,
103     UUID_SERVCLASS_HCRP_SCAN,
104 /*    UUID_SERVCLASS_COMMON_ISDN_ACCESS,        */
105 /*    UUID_SERVCLASS_VIDEO_CONFERENCING_GW,     */
106 /*    UUID_SERVCLASS_UDI_MT,                    */
107 /*    UUID_SERVCLASS_UDI_TA,                    */
108 /*    UUID_SERVCLASS_VCP,                       */
109     UUID_SERVCLASS_SAP,
110     UUID_SERVCLASS_PBAP_PCE,
111     UUID_SERVCLASS_PBAP_PSE,
112     UUID_SERVCLASS_PHONE_ACCESS,
113     UUID_SERVCLASS_HEADSET_HS,
114     UUID_SERVCLASS_PNP_INFORMATION,
115 /*    UUID_SERVCLASS_GENERIC_NETWORKING,        */
116 /*    UUID_SERVCLASS_GENERIC_FILETRANSFER,      */
117 /*    UUID_SERVCLASS_GENERIC_AUDIO,             */
118 /*    UUID_SERVCLASS_GENERIC_TELEPHONY,         */
119 /*    UUID_SERVCLASS_UPNP_SERVICE,              */
120 /*    UUID_SERVCLASS_UPNP_IP_SERVICE,           */
121 /*    UUID_SERVCLASS_ESDP_UPNP_IP_PAN,          */
122 /*    UUID_SERVCLASS_ESDP_UPNP_IP_LAP,          */
123 /*    UUID_SERVCLASS_ESDP_UPNP_IP_L2CAP,        */
124     UUID_SERVCLASS_VIDEO_SOURCE,
125     UUID_SERVCLASS_VIDEO_SINK,
126 /*    UUID_SERVCLASS_VIDEO_DISTRIBUTION         */
127     UUID_SERVCLASS_MESSAGE_ACCESS,
128     UUID_SERVCLASS_MESSAGE_NOTIFICATION,
129     UUID_SERVCLASS_HDP_SOURCE,
130     UUID_SERVCLASS_HDP_SINK
131 };
132 
133 /********************************************************************************/
134 /*              L O C A L    F U N C T I O N     P R O T O T Y P E S            */
135 /********************************************************************************/
136 static void         btm_initiate_inquiry (tBTM_INQUIRY_VAR_ST *p_inq);
137 static tBTM_STATUS  btm_set_inq_event_filter (UINT8 filter_cond_type, tBTM_INQ_FILT_COND *p_filt_cond);
138 static void         btm_clr_inq_result_flt (void);
139 
140 static UINT8        btm_convert_uuid_to_eir_service( UINT16 uuid16 );
141 static void         btm_set_eir_uuid( UINT8 *p_eir, tBTM_INQ_RESULTS *p_results );
142 static UINT8       *btm_eir_get_uuid_list( UINT8 *p_eir, UINT8 uuid_size,
143                                            UINT8 *p_num_uuid, UINT8 *p_uuid_list_type );
144 static UINT16       btm_convert_uuid_to_uuid16( UINT8 *p_uuid, UINT8 uuid_size );
145 
146 /*******************************************************************************
147 **
148 ** Function         BTM_SetDiscoverability
149 **
150 ** Description      This function is called to set the device into or out of
151 **                  discoverable mode. Discoverable mode means inquiry
152 **                  scans are enabled.  If a value of '0' is entered for window or
153 **                  interval, the default values are used.
154 **
155 ** Returns          BTM_SUCCESS if successful
156 **                  BTM_BUSY if a setting of the filter is already in progress
157 **                  BTM_NO_RESOURCES if couldn't get a memory pool buffer
158 **                  BTM_ILLEGAL_VALUE if a bad parameter was detected
159 **                  BTM_WRONG_MODE if the device is not up.
160 **
161 *******************************************************************************/
BTM_SetDiscoverability(UINT16 inq_mode,UINT16 window,UINT16 interval)162 tBTM_STATUS BTM_SetDiscoverability (UINT16 inq_mode, UINT16 window, UINT16 interval)
163 {
164     UINT8        scan_mode = 0;
165     UINT16       service_class;
166     UINT8       *p_cod;
167     UINT8        major, minor;
168     DEV_CLASS    cod;
169     LAP          temp_lap[2];
170     BOOLEAN      is_limited;
171     BOOLEAN      cod_limited;
172 
173     BTM_TRACE_API ("BTM_SetDiscoverability");
174 #if (BLE_INCLUDED == TRUE && BLE_INCLUDED == TRUE)
175     if (controller_get_interface()->supports_ble())
176     {
177         if (btm_ble_set_discoverability((UINT16)(inq_mode))
178                             == BTM_SUCCESS)
179         {
180             btm_cb.btm_inq_vars.discoverable_mode &= (~BTM_BLE_DISCOVERABLE_MASK);
181             btm_cb.btm_inq_vars.discoverable_mode |= (inq_mode & BTM_BLE_DISCOVERABLE_MASK);
182         }
183     }
184     inq_mode &= ~BTM_BLE_DISCOVERABLE_MASK;
185 #endif
186 
187     /*** Check mode parameter ***/
188     if (inq_mode > BTM_MAX_DISCOVERABLE)
189         return (BTM_ILLEGAL_VALUE);
190 
191     /* Make sure the controller is active */
192     if (!controller_get_interface()->get_is_ready())
193         return (BTM_DEV_RESET);
194 
195     /* If the window and/or interval is '0', set to default values */
196     if (!window)
197         window = BTM_DEFAULT_DISC_WINDOW;
198 
199     if (!interval)
200         interval = BTM_DEFAULT_DISC_INTERVAL;
201 
202     BTM_TRACE_API ("BTM_SetDiscoverability: mode %d [NonDisc-0, Lim-1, Gen-2], window 0x%04x, interval 0x%04x",
203                         inq_mode, window, interval);
204 
205     /*** Check for valid window and interval parameters ***/
206     /*** Only check window and duration if mode is connectable ***/
207     if (inq_mode != BTM_NON_DISCOVERABLE)
208     {
209         /* window must be less than or equal to interval */
210         if (window < HCI_MIN_INQUIRYSCAN_WINDOW     ||
211             window > HCI_MAX_INQUIRYSCAN_WINDOW     ||
212             interval < HCI_MIN_INQUIRYSCAN_INTERVAL ||
213             interval > HCI_MAX_INQUIRYSCAN_INTERVAL ||
214             window > interval)
215         {
216             return (BTM_ILLEGAL_VALUE);
217         }
218     }
219 
220     /* Set the IAC if needed */
221     if (inq_mode != BTM_NON_DISCOVERABLE)
222     {
223         if (inq_mode & BTM_LIMITED_DISCOVERABLE)
224         {
225             /* Use the GIAC and LIAC codes for limited discoverable mode */
226             memcpy (temp_lap[0], limited_inq_lap, LAP_LEN);
227             memcpy (temp_lap[1], general_inq_lap, LAP_LEN);
228 
229             if (!btsnd_hcic_write_cur_iac_lap (2, (LAP * const) temp_lap))
230                 return (BTM_NO_RESOURCES);  /* Cannot continue */
231         }
232         else
233         {
234             if (!btsnd_hcic_write_cur_iac_lap (1, (LAP * const) &general_inq_lap))
235                 return (BTM_NO_RESOURCES);  /* Cannot continue */
236         }
237 
238         scan_mode |= HCI_INQUIRY_SCAN_ENABLED;
239     }
240 
241     /* Send down the inquiry scan window and period if changed */
242     if ((window != btm_cb.btm_inq_vars.inq_scan_window) ||
243         (interval != btm_cb.btm_inq_vars.inq_scan_period))
244     {
245         if (btsnd_hcic_write_inqscan_cfg (interval, window))
246         {
247             btm_cb.btm_inq_vars.inq_scan_window = window;
248             btm_cb.btm_inq_vars.inq_scan_period = interval;
249         }
250         else
251             return (BTM_NO_RESOURCES);
252     }
253 
254     if (btm_cb.btm_inq_vars.connectable_mode & BTM_CONNECTABLE_MASK)
255         scan_mode |= HCI_PAGE_SCAN_ENABLED;
256 
257     if (btsnd_hcic_write_scan_enable (scan_mode))
258     {
259         btm_cb.btm_inq_vars.discoverable_mode &= (~BTM_DISCOVERABLE_MASK);
260         btm_cb.btm_inq_vars.discoverable_mode |= inq_mode;
261     }
262     else
263         return (BTM_NO_RESOURCES);
264 
265     /* Change the service class bit if mode has changed */
266     p_cod = BTM_ReadDeviceClass();
267     BTM_COD_SERVICE_CLASS(service_class, p_cod);
268     is_limited = (inq_mode & BTM_LIMITED_DISCOVERABLE) ? TRUE : FALSE;
269     cod_limited = (service_class & BTM_COD_SERVICE_LMTD_DISCOVER) ? TRUE : FALSE;
270     if (is_limited ^ cod_limited)
271     {
272         BTM_COD_MINOR_CLASS(minor, p_cod );
273         BTM_COD_MAJOR_CLASS(major, p_cod );
274         if (is_limited)
275             service_class |= BTM_COD_SERVICE_LMTD_DISCOVER;
276         else
277             service_class &= ~BTM_COD_SERVICE_LMTD_DISCOVER;
278 
279         FIELDS_TO_COD(cod, minor, major, service_class);
280         (void) BTM_SetDeviceClass (cod);
281     }
282 
283     return (BTM_SUCCESS);
284 }
285 
286 /*******************************************************************************
287 **
288 ** Function         BTM_SetInquiryScanType
289 **
290 ** Description      This function is called to set the iquiry scan-type to
291 **                  standard or interlaced.
292 **
293 ** Returns          BTM_SUCCESS if successful
294 **                  BTM_MODE_UNSUPPORTED if not a 1.2 device
295 **                  BTM_WRONG_MODE if the device is not up.
296 **
297 *******************************************************************************/
BTM_SetInquiryScanType(UINT16 scan_type)298 tBTM_STATUS BTM_SetInquiryScanType (UINT16 scan_type)
299 {
300 
301     BTM_TRACE_API ("BTM_SetInquiryScanType");
302     if (scan_type != BTM_SCAN_TYPE_STANDARD && scan_type != BTM_SCAN_TYPE_INTERLACED)
303         return (BTM_ILLEGAL_VALUE);
304 
305     /* whatever app wants if device is not 1.2 scan type should be STANDARD */
306     if (!controller_get_interface()->supports_interlaced_inquiry_scan())
307      return (BTM_MODE_UNSUPPORTED);
308 
309     /* Check for scan type if configuration has been changed */
310     if (scan_type != btm_cb.btm_inq_vars.inq_scan_type)
311     {
312         if (BTM_IsDeviceUp())
313         {
314             if (btsnd_hcic_write_inqscan_type ((UINT8)scan_type))
315                 btm_cb.btm_inq_vars.inq_scan_type = scan_type;
316             else
317                 return (BTM_NO_RESOURCES);
318         }
319         else return (BTM_WRONG_MODE);
320     }
321     return (BTM_SUCCESS);
322 }
323 
324 /*******************************************************************************
325 **
326 ** Function         BTM_SetPageScanType
327 **
328 ** Description      This function is called to set the page scan-type to
329 **                  standard or interlaced.
330 **
331 ** Returns          BTM_SUCCESS if successful
332 **                  BTM_MODE_UNSUPPORTED if not a 1.2 device
333 **                  BTM_WRONG_MODE if the device is not up.
334 **
335 *******************************************************************************/
BTM_SetPageScanType(UINT16 scan_type)336 tBTM_STATUS BTM_SetPageScanType (UINT16 scan_type)
337 {
338     BTM_TRACE_API ("BTM_SetPageScanType");
339     if (scan_type != BTM_SCAN_TYPE_STANDARD && scan_type != BTM_SCAN_TYPE_INTERLACED)
340         return (BTM_ILLEGAL_VALUE);
341 
342     /* whatever app wants if device is not 1.2 scan type should be STANDARD */
343     if (!controller_get_interface()->supports_interlaced_inquiry_scan())
344      return (BTM_MODE_UNSUPPORTED);
345 
346     /* Check for scan type if configuration has been changed */
347     if (scan_type != btm_cb.btm_inq_vars.page_scan_type)
348     {
349         if (BTM_IsDeviceUp())
350         {
351             if (btsnd_hcic_write_pagescan_type ((UINT8)scan_type))
352                 btm_cb.btm_inq_vars.page_scan_type  = scan_type;
353             else
354                 return (BTM_NO_RESOURCES);
355         }
356         else return (BTM_WRONG_MODE);
357     }
358     return (BTM_SUCCESS);
359 }
360 
361 
362 /*******************************************************************************
363 **
364 ** Function         BTM_SetInquiryMode
365 **
366 ** Description      This function is called to set standard or with RSSI
367 **                  mode of the inquiry for local device.
368 **
369 ** Output Params:   mode - standard, with RSSI, extended
370 **
371 ** Returns          BTM_SUCCESS if successful
372 **                  BTM_NO_RESOURCES if couldn't get a memory pool buffer
373 **                  BTM_ILLEGAL_VALUE if a bad parameter was detected
374 **                  BTM_WRONG_MODE if the device is not up.
375 **
376 *******************************************************************************/
BTM_SetInquiryMode(UINT8 mode)377 tBTM_STATUS BTM_SetInquiryMode (UINT8 mode)
378 {
379     const controller_t *controller = controller_get_interface();
380     BTM_TRACE_API ("BTM_SetInquiryMode");
381     if (mode == BTM_INQ_RESULT_STANDARD)
382     {
383         /* mandatory mode */
384     }
385     else if (mode == BTM_INQ_RESULT_WITH_RSSI)
386     {
387         if (!controller->supports_rssi_with_inquiry_results())
388             return (BTM_MODE_UNSUPPORTED);
389     }
390     else if (mode == BTM_INQ_RESULT_EXTENDED)
391     {
392         if (!controller->supports_extended_inquiry_response())
393             return (BTM_MODE_UNSUPPORTED);
394     }
395     else
396         return (BTM_ILLEGAL_VALUE);
397 
398     if (!BTM_IsDeviceUp())
399         return (BTM_WRONG_MODE);
400 
401     if (!btsnd_hcic_write_inquiry_mode (mode))
402         return (BTM_NO_RESOURCES);
403 
404     return (BTM_SUCCESS);
405 }
406 
407 /*******************************************************************************
408 **
409 ** Function         BTM_ReadDiscoverability
410 **
411 ** Description      This function is called to read the current discoverability
412 **                  mode of the device.
413 **
414 ** Output Params:   p_window - current inquiry scan duration
415 **                  p_interval - current inquiry scan interval
416 **
417 ** Returns          BTM_NON_DISCOVERABLE, BTM_LIMITED_DISCOVERABLE, or
418 **                  BTM_GENERAL_DISCOVERABLE
419 **
420 *******************************************************************************/
BTM_ReadDiscoverability(UINT16 * p_window,UINT16 * p_interval)421 UINT16 BTM_ReadDiscoverability (UINT16 *p_window, UINT16 *p_interval)
422 {
423     BTM_TRACE_API ("BTM_ReadDiscoverability");
424     if (p_window)
425         *p_window = btm_cb.btm_inq_vars.inq_scan_window;
426 
427     if (p_interval)
428         *p_interval = btm_cb.btm_inq_vars.inq_scan_period;
429 
430     return (btm_cb.btm_inq_vars.discoverable_mode);
431 }
432 
433 
434 /*******************************************************************************
435 **
436 ** Function         BTM_SetPeriodicInquiryMode
437 **
438 ** Description      This function is called to set the device periodic inquiry mode.
439 **                  If the duration is zero, the periodic inquiry mode is cancelled.
440 **
441 **                  Note: We currently do not allow concurrent inquiry and periodic inquiry.
442 **
443 ** Parameters:      p_inqparms - pointer to the inquiry information
444 **                      mode - GENERAL or LIMITED inquiry
445 **                      duration - length in 1.28 sec intervals (If '0', the inquiry is CANCELLED)
446 **                      max_resps - maximum amount of devices to search for before ending the inquiry
447 **                      filter_cond_type - BTM_CLR_INQUIRY_FILTER, BTM_FILTER_COND_DEVICE_CLASS, or
448 **                                         BTM_FILTER_COND_BD_ADDR
449 **                      filter_cond - value for the filter (based on filter_cond_type)
450 **
451 **                  max_delay - maximum amount of time between successive inquiries
452 **                  min_delay - minimum amount of time between successive inquiries
453 **                  p_results_cb - callback returning pointer to results (tBTM_INQ_RESULTS)
454 **
455 ** Returns          BTM_CMD_STARTED if successfully started
456 **                  BTM_ILLEGAL_VALUE if a bad parameter is detected
457 **                  BTM_NO_RESOURCES if could not allocate a message buffer
458 **                  BTM_SUCCESS - if cancelling the periodic inquiry
459 **                  BTM_BUSY - if an inquiry is already active
460 **                  BTM_WRONG_MODE if the device is not up.
461 **
462 *******************************************************************************/
BTM_SetPeriodicInquiryMode(tBTM_INQ_PARMS * p_inqparms,UINT16 max_delay,UINT16 min_delay,tBTM_INQ_RESULTS_CB * p_results_cb)463 tBTM_STATUS BTM_SetPeriodicInquiryMode (tBTM_INQ_PARMS *p_inqparms, UINT16 max_delay,
464                                         UINT16 min_delay, tBTM_INQ_RESULTS_CB *p_results_cb)
465 {
466     tBTM_STATUS  status;
467     tBTM_INQUIRY_VAR_ST *p_inq = &btm_cb.btm_inq_vars;
468 
469     BTM_TRACE_API ("BTM_SetPeriodicInquiryMode: mode: %d, dur: %d, rsps: %d, flt: %d, min: %d, max: %d",
470         p_inqparms->mode, p_inqparms->duration, p_inqparms->max_resps,
471         p_inqparms->filter_cond_type, min_delay, max_delay);
472 
473     /*** Make sure the device is ready ***/
474     if (!BTM_IsDeviceUp())
475         return (BTM_WRONG_MODE);
476 
477     /* Only one active inquiry is allowed in this implementation.
478        Also do not allow an inquiry if the inquiry filter is being updated */
479     if (p_inq->inq_active || p_inq->inqfilt_active)
480         return (BTM_BUSY);
481 
482     /* If illegal parameters return FALSE */
483     if (p_inqparms->mode != BTM_GENERAL_INQUIRY &&
484         p_inqparms->mode != BTM_LIMITED_INQUIRY)
485         return (BTM_ILLEGAL_VALUE);
486 
487     /* Verify the parameters for this command */
488     if (p_inqparms->duration < BTM_MIN_INQUIRY_LEN     ||
489         p_inqparms->duration > BTM_MAX_INQUIRY_LENGTH  ||
490         min_delay <= p_inqparms->duration              ||
491         min_delay < BTM_PER_INQ_MIN_MIN_PERIOD         ||
492         min_delay > BTM_PER_INQ_MAX_MIN_PERIOD         ||
493         max_delay <= min_delay                         ||
494         max_delay < BTM_PER_INQ_MIN_MAX_PERIOD)
495  /*       max_delay > BTM_PER_INQ_MAX_MAX_PERIOD)*/
496  /*  BTM_PER_INQ_MAX_MAX_PERIOD set to 1's in all bits. Condition resulting in false always*/
497     {
498         return (BTM_ILLEGAL_VALUE);
499     }
500 
501     /* Save the inquiry parameters to be used upon the completion of setting/clearing the inquiry filter */
502     p_inq->inqparms = *p_inqparms;
503     p_inq->per_min_delay = min_delay;
504     p_inq->per_max_delay = max_delay;
505     p_inq->inq_cmpl_info.num_resp = 0;         /* Clear the results counter */
506     p_inq->p_inq_results_cb = p_results_cb;
507 
508     p_inq->inq_active = (UINT8)((p_inqparms->mode == BTM_LIMITED_INQUIRY) ?
509                             (BTM_LIMITED_INQUIRY_ACTIVE | BTM_PERIODIC_INQUIRY_ACTIVE) :
510                             (BTM_GENERAL_INQUIRY_ACTIVE | BTM_PERIODIC_INQUIRY_ACTIVE));
511 
512     /* If a filter is specified, then save it for later and clear the current filter.
513        The setting of the filter is done upon completion of clearing of the previous
514        filter.
515     */
516     if (p_inqparms->filter_cond_type != BTM_CLR_INQUIRY_FILTER)
517     {
518         p_inq->state = BTM_INQ_CLR_FILT_STATE;
519         p_inqparms->filter_cond_type = BTM_CLR_INQUIRY_FILTER;
520     }
521     else    /* The filter is not being used so simply clear it; the inquiry can start after this operation */
522         p_inq->state = BTM_INQ_SET_FILT_STATE;
523 
524     /* Before beginning the inquiry the current filter must be cleared, so initiate the command */
525     if ((status = btm_set_inq_event_filter (p_inqparms->filter_cond_type, &p_inqparms->filter_cond)) != BTM_CMD_STARTED)
526     {
527         /* If set filter command is not succesful reset the state */
528         p_inq->p_inq_results_cb = NULL;
529         p_inq->state = BTM_INQ_INACTIVE_STATE;
530 
531     }
532 
533     return (status);
534 }
535 
536 
537 /*******************************************************************************
538 **
539 ** Function         BTM_CancelPeriodicInquiry
540 **
541 ** Description      This function cancels a periodic inquiry
542 **
543 ** Returns
544 **                  BTM_NO_RESOURCES if could not allocate a message buffer
545 **                  BTM_SUCCESS - if cancelling the periodic inquiry
546 **                  BTM_WRONG_MODE if the device is not up.
547 **
548 *******************************************************************************/
BTM_CancelPeriodicInquiry(void)549 tBTM_STATUS BTM_CancelPeriodicInquiry(void)
550 {
551     tBTM_INQUIRY_VAR_ST *p_inq = &btm_cb.btm_inq_vars;
552     tBTM_STATUS          status = BTM_SUCCESS;
553     BTM_TRACE_API ("BTM_CancelPeriodicInquiry called");
554 
555     /*** Make sure the device is ready ***/
556     if (!BTM_IsDeviceUp())
557         return (BTM_WRONG_MODE);
558 
559     /* Only cancel if one is active */
560     if (btm_cb.btm_inq_vars.inq_active & BTM_PERIODIC_INQUIRY_ACTIVE)
561     {
562         btm_cb.btm_inq_vars.inq_active = BTM_INQUIRY_INACTIVE;
563         btm_cb.btm_inq_vars.p_inq_results_cb = (tBTM_INQ_RESULTS_CB *) NULL;
564 
565         if (!btsnd_hcic_exit_per_inq ())
566             status = BTM_NO_RESOURCES;
567 
568         /* If the event filter is in progress, mark it so that the processing of the return
569            event will be ignored */
570         if(p_inq->inqfilt_active)
571             p_inq->pending_filt_complete_event++;
572 
573         p_inq->inqfilt_active = FALSE;
574         p_inq->inq_counter++;
575     }
576 
577     return (status);
578 }
579 
580 
581 /*******************************************************************************
582 **
583 ** Function         BTM_SetConnectability
584 **
585 ** Description      This function is called to set the device into or out of
586 **                  connectable mode. Discoverable mode means page scans enabled.
587 **
588 ** Returns          BTM_SUCCESS if successful
589 **                  BTM_ILLEGAL_VALUE if a bad parameter is detected
590 **                  BTM_NO_RESOURCES if could not allocate a message buffer
591 **                  BTM_WRONG_MODE if the device is not up.
592 **
593 *******************************************************************************/
BTM_SetConnectability(UINT16 page_mode,UINT16 window,UINT16 interval)594 tBTM_STATUS BTM_SetConnectability (UINT16 page_mode, UINT16 window, UINT16 interval)
595 {
596     UINT8    scan_mode = 0;
597     tBTM_INQUIRY_VAR_ST *p_inq = &btm_cb.btm_inq_vars;
598 
599     BTM_TRACE_API ("BTM_SetConnectability");
600 
601 #if (BLE_INCLUDED == TRUE && BLE_INCLUDED == TRUE)
602     if (controller_get_interface()->supports_ble())
603     {
604         if (btm_ble_set_connectability(page_mode) != BTM_SUCCESS)
605         {
606             return BTM_NO_RESOURCES;
607         }
608         p_inq->connectable_mode &= (~BTM_BLE_CONNECTABLE_MASK);
609         p_inq->connectable_mode |= (page_mode & BTM_BLE_CONNECTABLE_MASK);
610     }
611     page_mode &= ~BTM_BLE_CONNECTABLE_MASK;
612 #endif
613 
614     /*** Check mode parameter ***/
615     if (page_mode != BTM_NON_CONNECTABLE && page_mode != BTM_CONNECTABLE)
616         return (BTM_ILLEGAL_VALUE);
617 
618     /* Make sure the controller is active */
619     if (!controller_get_interface()->get_is_ready())
620         return (BTM_DEV_RESET);
621 
622     /* If the window and/or interval is '0', set to default values */
623     if (!window)
624         window = BTM_DEFAULT_CONN_WINDOW;
625 
626     if (!interval)
627         interval = BTM_DEFAULT_CONN_INTERVAL;
628 
629     BTM_TRACE_API ("BTM_SetConnectability: mode %d [NonConn-0, Conn-1], window 0x%04x, interval 0x%04x",
630                         page_mode, window, interval);
631 
632     /*** Check for valid window and interval parameters ***/
633     /*** Only check window and duration if mode is connectable ***/
634     if (page_mode == BTM_CONNECTABLE)
635     {
636         /* window must be less than or equal to interval */
637         if (window < HCI_MIN_PAGESCAN_WINDOW     ||
638             window > HCI_MAX_PAGESCAN_WINDOW     ||
639             interval < HCI_MIN_PAGESCAN_INTERVAL ||
640             interval > HCI_MAX_PAGESCAN_INTERVAL ||
641             window > interval)
642         {
643             return (BTM_ILLEGAL_VALUE);
644         }
645 
646         scan_mode |= HCI_PAGE_SCAN_ENABLED;
647     }
648 
649     if ((window != p_inq->page_scan_window) ||
650         (interval != p_inq->page_scan_period))
651     {
652         p_inq->page_scan_window = window;
653         p_inq->page_scan_period = interval;
654         if (!btsnd_hcic_write_pagescan_cfg (interval, window))
655             return (BTM_NO_RESOURCES);
656     }
657 
658     /* Keep the inquiry scan as previouosly set */
659     if (p_inq->discoverable_mode & BTM_DISCOVERABLE_MASK)
660         scan_mode |= HCI_INQUIRY_SCAN_ENABLED;
661 
662     if (btsnd_hcic_write_scan_enable (scan_mode))
663     {
664         p_inq->connectable_mode &= (~BTM_CONNECTABLE_MASK);
665         p_inq->connectable_mode |= page_mode;
666 
667         return (BTM_SUCCESS);
668     }
669 
670     return (BTM_NO_RESOURCES);
671 }
672 
673 
674 /*******************************************************************************
675 **
676 ** Function         BTM_ReadConnectability
677 **
678 ** Description      This function is called to read the current discoverability
679 **                  mode of the device.
680 ** Output Params    p_window - current page scan duration
681 **                  p_interval - current time between page scans
682 **
683 ** Returns          BTM_NON_CONNECTABLE or BTM_CONNECTABLE
684 **
685 *******************************************************************************/
BTM_ReadConnectability(UINT16 * p_window,UINT16 * p_interval)686 UINT16 BTM_ReadConnectability (UINT16 *p_window, UINT16 *p_interval)
687 {
688     BTM_TRACE_API ("BTM_ReadConnectability");
689     if (p_window)
690         *p_window = btm_cb.btm_inq_vars.page_scan_window;
691 
692     if (p_interval)
693         *p_interval = btm_cb.btm_inq_vars.page_scan_period;
694 
695     return (btm_cb.btm_inq_vars.connectable_mode);
696 }
697 
698 
699 
700 /*******************************************************************************
701 **
702 ** Function         BTM_IsInquiryActive
703 **
704 ** Description      This function returns a bit mask of the current inquiry state
705 **
706 ** Returns          BTM_INQUIRY_INACTIVE if inactive (0)
707 **                  BTM_LIMITED_INQUIRY_ACTIVE if a limted inquiry is active
708 **                  BTM_GENERAL_INQUIRY_ACTIVE if a general inquiry is active
709 **                  BTM_PERIODIC_INQUIRY_ACTIVE if a periodic inquiry is active
710 **
711 *******************************************************************************/
BTM_IsInquiryActive(void)712 UINT16 BTM_IsInquiryActive (void)
713 {
714     BTM_TRACE_API ("BTM_IsInquiryActive");
715 
716     return(btm_cb.btm_inq_vars.inq_active);
717 }
718 
719 
720 
721 /*******************************************************************************
722 **
723 ** Function         BTM_CancelInquiry
724 **
725 ** Description      This function cancels an inquiry if active
726 **
727 ** Returns          BTM_SUCCESS if successful
728 **                  BTM_NO_RESOURCES if could not allocate a message buffer
729 **                  BTM_WRONG_MODE if the device is not up.
730 **
731 *******************************************************************************/
BTM_CancelInquiry(void)732 tBTM_STATUS BTM_CancelInquiry(void)
733 {
734     tBTM_STATUS           status = BTM_SUCCESS;
735     tBTM_INQUIRY_VAR_ST *p_inq = &btm_cb.btm_inq_vars;
736 #if (defined(BTA_HOST_INTERLEAVE_SEARCH) && BTA_HOST_INTERLEAVE_SEARCH == TRUE)
737     UINT8 active_mode=p_inq->inq_active;
738 #endif
739     BTM_TRACE_API ("BTM_CancelInquiry called");
740 
741     /*** Make sure the device is ready ***/
742     if (!BTM_IsDeviceUp())
743         return (BTM_WRONG_MODE);
744 
745     /* Only cancel if not in periodic mode, otherwise the caller should call BTM_CancelPeriodicMode */
746     if ((p_inq->inq_active &BTM_INQUIRY_ACTIVE_MASK) != 0 &&
747         (!(p_inq->inq_active & BTM_PERIODIC_INQUIRY_ACTIVE)))
748     {
749         p_inq->inq_active = BTM_INQUIRY_INACTIVE;
750         p_inq->state = BTM_INQ_INACTIVE_STATE;
751         p_inq->p_inq_results_cb = (tBTM_INQ_RESULTS_CB *) NULL; /* Do not notify caller anymore */
752         p_inq->p_inq_cmpl_cb = (tBTM_CMPL_CB *) NULL;    /* Do not notify caller anymore */
753 
754         /* If the event filter is in progress, mark it so that the processing of the return
755             event will be ignored */
756         if (p_inq->inqfilt_active)
757         {
758             p_inq->inqfilt_active = FALSE;
759             p_inq->pending_filt_complete_event++;
760         }
761          /* Initiate the cancel inquiry */
762         else
763         {
764             if (((p_inq->inqparms.mode & BTM_BR_INQUIRY_MASK) != 0)
765 #if (defined(BTA_HOST_INTERLEAVE_SEARCH) && BTA_HOST_INTERLEAVE_SEARCH == TRUE)
766             &&(active_mode & BTM_BR_INQUIRY_MASK)
767 #endif
768             )
769             {
770                 if (!btsnd_hcic_inq_cancel())
771                     status = BTM_NO_RESOURCES;
772             }
773 #if BLE_INCLUDED == TRUE
774             if (((p_inq->inqparms.mode & BTM_BLE_INQUIRY_MASK) != 0)
775 #if (defined(BTA_HOST_INTERLEAVE_SEARCH) && BTA_HOST_INTERLEAVE_SEARCH == TRUE)
776             &&(active_mode & BTM_BLE_INQ_ACTIVE_MASK)
777 #endif
778             )
779                 btm_ble_stop_inquiry();
780 #endif
781         }
782 
783         /* Do not send the BUSY_LEVEL event yet. Wait for the cancel_complete event
784          * and then send the BUSY_LEVEL event
785          * btm_acl_update_busy_level (BTM_BLI_INQ_DONE_EVT);
786          */
787 
788          p_inq->inq_counter++;
789          btm_clr_inq_result_flt();
790     }
791 
792     return (status);
793 }
794 
795 
796 /*******************************************************************************
797 **
798 ** Function         BTM_StartInquiry
799 **
800 ** Description      This function is called to start an inquiry.
801 **
802 ** Parameters:      p_inqparms - pointer to the inquiry information
803 **                      mode - GENERAL or LIMITED inquiry, BR/LE bit mask seperately
804 **                      duration - length in 1.28 sec intervals (If '0', the inquiry is CANCELLED)
805 **                      max_resps - maximum amount of devices to search for before ending the inquiry
806 **                      filter_cond_type - BTM_CLR_INQUIRY_FILTER, BTM_FILTER_COND_DEVICE_CLASS, or
807 **                                         BTM_FILTER_COND_BD_ADDR
808 **                      filter_cond - value for the filter (based on filter_cond_type)
809 **
810 **                  p_results_cb   - Pointer to the callback routine which gets called
811 **                                upon receipt of an inquiry result. If this field is
812 **                                NULL, the application is not notified.
813 **
814 **                  p_cmpl_cb   - Pointer to the callback routine which gets called
815 **                                upon completion.  If this field is NULL, the
816 **                                application is not notified when completed.
817 ** Returns          tBTM_STATUS
818 **                  BTM_CMD_STARTED if successfully initiated
819 **                  BTM_BUSY if already in progress
820 **                  BTM_ILLEGAL_VALUE if parameter(s) are out of range
821 **                  BTM_NO_RESOURCES if could not allocate resources to start the command
822 **                  BTM_WRONG_MODE if the device is not up.
823 **
824 *******************************************************************************/
BTM_StartInquiry(tBTM_INQ_PARMS * p_inqparms,tBTM_INQ_RESULTS_CB * p_results_cb,tBTM_CMPL_CB * p_cmpl_cb)825 tBTM_STATUS BTM_StartInquiry (tBTM_INQ_PARMS *p_inqparms, tBTM_INQ_RESULTS_CB *p_results_cb,
826                               tBTM_CMPL_CB *p_cmpl_cb)
827 {
828     tBTM_STATUS  status = BTM_CMD_STARTED;
829     tBTM_INQUIRY_VAR_ST *p_inq = &btm_cb.btm_inq_vars;
830 
831     BTM_TRACE_API ("BTM_StartInquiry: mode: %d, dur: %d, rsps: %d, flt: %d",
832                         p_inqparms->mode, p_inqparms->duration, p_inqparms->max_resps,
833                         p_inqparms->filter_cond_type);
834 
835     /* Only one active inquiry is allowed in this implementation.
836        Also do not allow an inquiry if the inquiry filter is being updated */
837     if (p_inq->inq_active || p_inq->inqfilt_active)
838     {
839 #if (defined BLE_INCLUDED && BLE_INCLUDED == TRUE)
840         /*check if LE observe is already running*/
841         if(p_inq->scan_type==INQ_LE_OBSERVE && p_inq->p_inq_ble_results_cb!=NULL)
842         {
843             BTM_TRACE_API("BTM_StartInquiry: LE observe in progress");
844             p_inq->scan_type = INQ_GENERAL;
845             p_inq->inq_active = BTM_INQUIRY_INACTIVE;
846             btm_cb.ble_ctr_cb.inq_var.scan_type = BTM_BLE_SCAN_MODE_NONE;
847             btsnd_hcic_ble_set_scan_enable (BTM_BLE_SCAN_DISABLE, BTM_BLE_DUPLICATE_ENABLE);
848         }
849         else
850 #endif
851         {
852             return (BTM_BUSY);
853             BTM_TRACE_API("BTM_StartInquiry: return BUSY");
854         }
855     }
856     else
857         p_inq->scan_type = INQ_GENERAL;
858 
859         /*** Make sure the device is ready ***/
860     if (!BTM_IsDeviceUp())
861         return (BTM_WRONG_MODE);
862 
863     if ((p_inqparms->mode & BTM_BR_INQUIRY_MASK)!= BTM_GENERAL_INQUIRY &&
864         (p_inqparms->mode & BTM_BR_INQUIRY_MASK)!= BTM_LIMITED_INQUIRY
865 #if (BLE_INCLUDED == TRUE)
866         && (p_inqparms->mode & BTM_BLE_INQUIRY_MASK)!= BTM_BLE_GENERAL_INQUIRY
867         && (p_inqparms->mode & BTM_BLE_INQUIRY_MASK)!= BTM_BLE_LIMITED_INQUIRY
868 #endif
869         )
870         return (BTM_ILLEGAL_VALUE);
871 
872 #if (defined(BTA_HOST_INTERLEAVE_SEARCH) && BTA_HOST_INTERLEAVE_SEARCH == TRUE)
873         if(p_inq->next_state==BTM_FINISH)
874             return BTM_ILLEGAL_VALUE;
875 #endif
876 
877 
878     /* Save the inquiry parameters to be used upon the completion of setting/clearing the inquiry filter */
879     p_inq->inqparms = *p_inqparms;
880 
881     /* Initialize the inquiry variables */
882     p_inq->state = BTM_INQ_ACTIVE_STATE;
883     p_inq->p_inq_cmpl_cb = p_cmpl_cb;
884     p_inq->p_inq_results_cb = p_results_cb;
885     p_inq->inq_cmpl_info.num_resp = 0;         /* Clear the results counter */
886     p_inq->inq_active = p_inqparms->mode;
887 
888     BTM_TRACE_DEBUG("BTM_StartInquiry: p_inq->inq_active = 0x%02x", p_inq->inq_active);
889 
890 /* interleave scan minimal conditions */
891 #if (BLE_INCLUDED==TRUE && (defined(BTA_HOST_INTERLEAVE_SEARCH) && BTA_HOST_INTERLEAVE_SEARCH == TRUE))
892 
893     /* check if both modes are present */
894     if((p_inqparms->mode & BTM_BLE_INQUIRY_MASK) && (p_inqparms->mode & BTM_BR_INQUIRY_MASK))
895     {
896         BTM_TRACE_API("BTM:Interleave Inquiry Mode Set");
897         p_inqparms->duration=p_inqparms->intl_duration[p_inq->next_state];
898         p_inq->inqparms.duration=p_inqparms->duration;
899     }
900     else
901     {
902         BTM_TRACE_API("BTM:Single Mode: No interleaving, Mode:0x%02x", p_inqparms->mode);
903         p_inq->next_state=BTM_NO_INTERLEAVING;
904     }
905 #endif
906 
907 
908 
909 /* start LE inquiry here if requested */
910 #if BLE_INCLUDED == TRUE
911     if ((p_inqparms->mode & BTM_BLE_INQUIRY_MASK)
912 #if (defined(BTA_HOST_INTERLEAVE_SEARCH) && BTA_HOST_INTERLEAVE_SEARCH == TRUE)
913         &&(p_inq->next_state==BTM_BLE_ONE || p_inq->next_state==BTM_BLE_TWO ||
914            p_inq->next_state==BTM_NO_INTERLEAVING)
915 #endif
916         )
917 
918     {
919 #if (defined(BTA_HOST_INTERLEAVE_SEARCH) && BTA_HOST_INTERLEAVE_SEARCH == TRUE)
920         p_inq->inq_active = (p_inqparms->mode & BTM_BLE_INQUIRY_MASK);
921         BTM_TRACE_API("BTM:Starting LE Scan with duration %d and activeMode:0x%02x",
922                        p_inqparms->duration, (p_inqparms->mode & BTM_BLE_INQUIRY_MASK));
923 #endif
924         if (!controller_get_interface()->supports_ble())
925         {
926             p_inq->inqparms.mode &= ~ BTM_BLE_INQUIRY_MASK;
927             status = BTM_ILLEGAL_VALUE;
928         }
929         /* BLE for now does not support filter condition for inquiry */
930         else if ((status = btm_ble_start_inquiry((UINT8)(p_inqparms->mode & BTM_BLE_INQUIRY_MASK),
931                                             p_inqparms->duration)) != BTM_CMD_STARTED)
932         {
933             BTM_TRACE_ERROR("Err Starting LE Inquiry.");
934             p_inq->inqparms.mode &= ~ BTM_BLE_INQUIRY_MASK;
935         }
936 #if (!defined(BTA_HOST_INTERLEAVE_SEARCH) || BTA_HOST_INTERLEAVE_SEARCH == FALSE)
937         p_inqparms->mode &= ~BTM_BLE_INQUIRY_MASK;
938 #endif
939 
940 #if (defined(BTA_HOST_INTERLEAVE_SEARCH) && BTA_HOST_INTERLEAVE_SEARCH == TRUE)
941         if(p_inq->next_state==BTM_NO_INTERLEAVING)
942         {
943             p_inq->next_state=BTM_FINISH;
944         }
945         else
946         {
947             BTM_TRACE_API("BTM:Interleaving: started LE scan, Advancing to next state: %d",
948                            p_inq->next_state+1);
949             p_inq->next_state+=1;
950         }
951         /* reset next_state if status <> BTM_Started */
952         if(status!=BTM_CMD_STARTED)
953             p_inq->next_state=BTM_BR_ONE;
954 
955         /* if interleave scan..return here */
956         return status;
957 #endif
958 
959 
960         BTM_TRACE_DEBUG("BTM_StartInquiry: mode = %02x", p_inqparms->mode);
961     }
962 #endif /* end of BLE_INCLUDED */
963 
964     /* we're done with this routine if BR/EDR inquiry is not desired. */
965     if ((p_inqparms->mode & BTM_BR_INQUIRY_MASK) == BTM_INQUIRY_NONE)
966         return status;
967 
968     /* BR/EDR inquiry portion */
969 #if (defined(BTA_HOST_INTERLEAVE_SEARCH) && BTA_HOST_INTERLEAVE_SEARCH == TRUE)
970     if((p_inq->next_state==BTM_BR_ONE || p_inq->next_state==BTM_BR_TWO ||
971         p_inq->next_state==BTM_NO_INTERLEAVING ))
972     {
973         p_inq->inq_active = (p_inqparms->mode & BTM_BR_INQUIRY_MASK);
974 #endif
975     /* If a filter is specified, then save it for later and clear the current filter.
976        The setting of the filter is done upon completion of clearing of the previous
977        filter.
978     */
979     switch (p_inqparms->filter_cond_type)
980     {
981     case BTM_CLR_INQUIRY_FILTER:
982         p_inq->state = BTM_INQ_SET_FILT_STATE;
983         break;
984 
985     case BTM_FILTER_COND_DEVICE_CLASS:
986     case BTM_FILTER_COND_BD_ADDR:
987         /* The filter is not being used so simply clear it;
988             the inquiry can start after this operation */
989         p_inq->state = BTM_INQ_CLR_FILT_STATE;
990         p_inqparms->filter_cond_type = BTM_CLR_INQUIRY_FILTER;
991         /* =============>>>> adding LE filtering here ????? */
992         break;
993 
994     default:
995         return (BTM_ILLEGAL_VALUE);
996     }
997 
998     /* Before beginning the inquiry the current filter must be cleared, so initiate the command */
999     if ((status = btm_set_inq_event_filter (p_inqparms->filter_cond_type,
1000                                             &p_inqparms->filter_cond)) != BTM_CMD_STARTED)
1001         p_inq->state = BTM_INQ_INACTIVE_STATE;
1002 
1003 #if (defined(BTA_HOST_INTERLEAVE_SEARCH) && BTA_HOST_INTERLEAVE_SEARCH == TRUE)
1004         if (p_inq->next_state==BTM_NO_INTERLEAVING)
1005             p_inq->next_state=BTM_FINISH;
1006         else
1007         {
1008             BTM_TRACE_API("BTM:Interleaving: Started BTM inq, Advancing to next state: %d",
1009                            p_inq->next_state+1);
1010             p_inq->next_state+=1;
1011         }
1012      }
1013      if (status!=BTM_CMD_STARTED)
1014      {
1015          /* Some error beginning the scan process.
1016             Reset the next_state parameter.. Do we need to reset the inq_active also?
1017          */
1018         BTM_TRACE_API("BTM:Interleaving: Error in Starting inquiry, status: 0x%02x", status);
1019         p_inq->next_state=BTM_BR_ONE;
1020      }
1021 #endif
1022 
1023 
1024     return (status);
1025 }
1026 
1027 
1028 /*******************************************************************************
1029 **
1030 ** Function         BTM_ReadRemoteDeviceName
1031 **
1032 ** Description      This function initiates a remote device HCI command to the
1033 **                  controller and calls the callback when the process has completed.
1034 **
1035 ** Input Params:    remote_bda      - device address of name to retrieve
1036 **                  p_cb            - callback function called when BTM_CMD_STARTED
1037 **                                    is returned.
1038 **                                    A pointer to tBTM_REMOTE_DEV_NAME is passed to the
1039 **                                    callback.
1040 **
1041 ** Returns
1042 **                  BTM_CMD_STARTED is returned if the request was successfully sent
1043 **                                  to HCI.
1044 **                  BTM_BUSY if already in progress
1045 **                  BTM_UNKNOWN_ADDR if device address is bad
1046 **                  BTM_NO_RESOURCES if could not allocate resources to start the command
1047 **                  BTM_WRONG_MODE if the device is not up.
1048 **
1049 *******************************************************************************/
BTM_ReadRemoteDeviceName(BD_ADDR remote_bda,tBTM_CMPL_CB * p_cb,tBT_TRANSPORT transport)1050 tBTM_STATUS  BTM_ReadRemoteDeviceName (BD_ADDR remote_bda, tBTM_CMPL_CB *p_cb
1051                                                 ,tBT_TRANSPORT transport)
1052 {
1053     tBTM_INQ_INFO   *p_cur = NULL;
1054     tINQ_DB_ENT     *p_i;
1055 
1056     BTM_TRACE_API ("BTM_ReadRemoteDeviceName: bd addr [%02x%02x%02x%02x%02x%02x]",
1057                remote_bda[0], remote_bda[1], remote_bda[2],
1058                remote_bda[3], remote_bda[4], remote_bda[5]);
1059 
1060     /* Use the remote device's clock offset if it is in the local inquiry database */
1061     if ((p_i = btm_inq_db_find (remote_bda)) != NULL)
1062     {
1063         p_cur = &p_i->inq_info;
1064     }
1065     BTM_TRACE_API ("no device found in inquiry db");
1066 
1067 #if (BLE_INCLUDED == TRUE)
1068     if (transport == BT_TRANSPORT_LE)
1069     {
1070         return btm_ble_read_remote_name(remote_bda, p_cur, p_cb);
1071     }
1072     else
1073 #endif
1074 
1075     return (btm_initiate_rem_name (remote_bda, p_cur, BTM_RMT_NAME_EXT,
1076                                    BTM_EXT_RMT_NAME_TIMEOUT_MS, p_cb));
1077 }
1078 
1079 /*******************************************************************************
1080 **
1081 ** Function         BTM_CancelRemoteDeviceName
1082 **
1083 ** Description      This function initiates the cancel request for the specified
1084 **                  remote device.
1085 **
1086 ** Input Params:    None
1087 **
1088 ** Returns
1089 **                  BTM_CMD_STARTED is returned if the request was successfully sent
1090 **                                  to HCI.
1091 **                  BTM_NO_RESOURCES if could not allocate resources to start the command
1092 **                  BTM_WRONG_MODE if there is not an active remote name request.
1093 **
1094 *******************************************************************************/
BTM_CancelRemoteDeviceName(void)1095 tBTM_STATUS  BTM_CancelRemoteDeviceName (void)
1096 {
1097     tBTM_INQUIRY_VAR_ST *p_inq = &btm_cb.btm_inq_vars;
1098 
1099     BTM_TRACE_API ("BTM_CancelRemoteDeviceName()");
1100 
1101     /* Make sure there is not already one in progress */
1102     if (p_inq->remname_active)
1103     {
1104 #if BLE_INCLUDED == TRUE
1105         if (BTM_UseLeLink(p_inq->remname_bda))
1106         {
1107             if (btm_ble_cancel_remote_name(p_inq->remname_bda))
1108                 return (BTM_CMD_STARTED);
1109             else
1110                 return (BTM_UNKNOWN_ADDR);
1111         }
1112         else
1113 #endif
1114         if (btsnd_hcic_rmt_name_req_cancel (p_inq->remname_bda))
1115             return (BTM_CMD_STARTED);
1116         else
1117             return (BTM_NO_RESOURCES);
1118     }
1119     else
1120         return (BTM_WRONG_MODE);
1121 }
1122 
1123 /*******************************************************************************
1124 **
1125 ** Function         BTM_InqDbRead
1126 **
1127 ** Description      This function looks through the inquiry database for a match
1128 **                  based on Bluetooth Device Address. This is the application's
1129 **                  interface to get the inquiry details of a specific BD address.
1130 **
1131 ** Returns          pointer to entry, or NULL if not found
1132 **
1133 *******************************************************************************/
BTM_InqDbRead(const BD_ADDR p_bda)1134 tBTM_INQ_INFO *BTM_InqDbRead (const BD_ADDR p_bda)
1135 {
1136     BTM_TRACE_API ("BTM_InqDbRead: bd addr [%02x%02x%02x%02x%02x%02x]",
1137                p_bda[0], p_bda[1], p_bda[2], p_bda[3], p_bda[4], p_bda[5]);
1138 
1139     tINQ_DB_ENT *p_ent = btm_inq_db_find(p_bda);
1140     if (!p_ent)
1141       return NULL;
1142 
1143     return &p_ent->inq_info;
1144 }
1145 
1146 
1147 /*******************************************************************************
1148 **
1149 ** Function         BTM_InqDbFirst
1150 **
1151 ** Description      This function looks through the inquiry database for the first
1152 **                  used entry, and returns that. This is used in conjunction with
1153 **                  BTM_InqDbNext by applications as a way to walk through the
1154 **                  inquiry database.
1155 **
1156 ** Returns          pointer to first in-use entry, or NULL if DB is empty
1157 **
1158 *******************************************************************************/
BTM_InqDbFirst(void)1159 tBTM_INQ_INFO *BTM_InqDbFirst (void)
1160 {
1161     UINT16       xx;
1162     tINQ_DB_ENT  *p_ent = btm_cb.btm_inq_vars.inq_db;
1163 
1164     for (xx = 0; xx < BTM_INQ_DB_SIZE; xx++, p_ent++)
1165     {
1166         if (p_ent->in_use)
1167             return (&p_ent->inq_info);
1168     }
1169 
1170     /* If here, no used entry found */
1171     return ((tBTM_INQ_INFO *)NULL);
1172 }
1173 
1174 
1175 /*******************************************************************************
1176 **
1177 ** Function         BTM_InqDbNext
1178 **
1179 ** Description      This function looks through the inquiry database for the next
1180 **                  used entry, and returns that.  If the input parameter is NULL,
1181 **                  the first entry is returned.
1182 **
1183 ** Returns          pointer to next in-use entry, or NULL if no more found.
1184 **
1185 *******************************************************************************/
BTM_InqDbNext(tBTM_INQ_INFO * p_cur)1186 tBTM_INQ_INFO *BTM_InqDbNext (tBTM_INQ_INFO *p_cur)
1187 {
1188     tINQ_DB_ENT  *p_ent;
1189     UINT16        inx;
1190 
1191     if (p_cur)
1192     {
1193         p_ent = (tINQ_DB_ENT *) ((UINT8 *)p_cur - offsetof (tINQ_DB_ENT, inq_info));
1194         inx = (UINT16)((p_ent - btm_cb.btm_inq_vars.inq_db) + 1);
1195 
1196         for (p_ent = &btm_cb.btm_inq_vars.inq_db[inx]; inx < BTM_INQ_DB_SIZE; inx++, p_ent++)
1197         {
1198             if (p_ent->in_use)
1199                 return (&p_ent->inq_info);
1200         }
1201 
1202         /* If here, more entries found */
1203         return ((tBTM_INQ_INFO *)NULL);
1204     }
1205     else
1206         return (BTM_InqDbFirst());
1207 }
1208 
1209 
1210 /*******************************************************************************
1211 **
1212 ** Function         BTM_ClearInqDb
1213 **
1214 ** Description      This function is called to clear out a device or all devices
1215 **                  from the inquiry database.
1216 **
1217 ** Parameter        p_bda - (input) BD_ADDR ->  Address of device to clear
1218 **                                              (NULL clears all entries)
1219 **
1220 ** Returns          BTM_BUSY if an inquiry, get remote name, or event filter
1221 **                          is active, otherwise BTM_SUCCESS
1222 **
1223 *******************************************************************************/
BTM_ClearInqDb(BD_ADDR p_bda)1224 tBTM_STATUS BTM_ClearInqDb (BD_ADDR p_bda)
1225 {
1226     tBTM_INQUIRY_VAR_ST     *p_inq = &btm_cb.btm_inq_vars;
1227 
1228     /* If an inquiry or remote name is in progress return busy */
1229     if (p_inq->inq_active != BTM_INQUIRY_INACTIVE ||
1230         p_inq->inqfilt_active)
1231         return (BTM_BUSY);
1232 
1233     btm_clr_inq_db(p_bda);
1234 
1235     return (BTM_SUCCESS);
1236 }
1237 
1238 /*******************************************************************************
1239 **
1240 ** Function         BTM_ReadInquiryRspTxPower
1241 **
1242 ** Description      This command will read the inquiry Transmit Power level used
1243 **                  to transmit the FHS and EIR data packets.
1244 **                  This can be used directly in the Tx Power Level EIR data type.
1245 **
1246 ** Returns          BTM_SUCCESS if successful
1247 **
1248 *******************************************************************************/
BTM_ReadInquiryRspTxPower(tBTM_CMPL_CB * p_cb)1249 tBTM_STATUS BTM_ReadInquiryRspTxPower (tBTM_CMPL_CB *p_cb)
1250 {
1251     if (btm_cb.devcb.p_inq_tx_power_cmpl_cb)
1252         return (BTM_BUSY);
1253 
1254     btm_cb.devcb.p_inq_tx_power_cmpl_cb = p_cb;
1255     alarm_set_on_queue(btm_cb.devcb.read_inq_tx_power_timer,
1256                        BTM_INQ_REPLY_TIMEOUT_MS,
1257                        btm_read_inq_tx_power_timeout, NULL,
1258                        btu_general_alarm_queue);
1259 
1260     if (!btsnd_hcic_read_inq_tx_power ())
1261     {
1262         btm_cb.devcb.p_inq_tx_power_cmpl_cb = NULL;
1263         alarm_cancel(btm_cb.devcb.read_inq_tx_power_timer);
1264         return (BTM_NO_RESOURCES);
1265     }
1266     else
1267         return (BTM_CMD_STARTED);
1268 }
1269 
1270 /*********************************************************************************
1271 **********************************************************************************
1272 **                                                                              **
1273 **                      BTM Internal Inquiry Functions                          **
1274 **                                                                              **
1275 **********************************************************************************
1276 *********************************************************************************/
1277 /*******************************************************************************
1278 **
1279 ** Function         btm_inq_db_reset
1280 **
1281 ** Description      This function is called at at reset to clear the inquiry
1282 **                  database & pending callback.
1283 **
1284 ** Returns          void
1285 **
1286 *******************************************************************************/
btm_inq_db_reset(void)1287 void btm_inq_db_reset (void)
1288 {
1289     tBTM_REMOTE_DEV_NAME     rem_name;
1290     tBTM_INQUIRY_VAR_ST     *p_inq = &btm_cb.btm_inq_vars;
1291     UINT8                    num_responses;
1292     UINT8                    temp_inq_active;
1293     tBTM_STATUS              status;
1294 
1295     /* If an inquiry or periodic inquiry is active, reset the mode to inactive */
1296     if (p_inq->inq_active != BTM_INQUIRY_INACTIVE)
1297     {
1298         temp_inq_active = p_inq->inq_active;    /* Save so state can change BEFORE
1299                                                        callback is called */
1300         p_inq->inq_active = BTM_INQUIRY_INACTIVE;
1301 
1302         /* If not a periodic inquiry, the complete callback must be called to notify caller */
1303         if (temp_inq_active == BTM_LIMITED_INQUIRY_ACTIVE ||
1304             temp_inq_active == BTM_GENERAL_INQUIRY_ACTIVE)
1305         {
1306             if (p_inq->p_inq_cmpl_cb)
1307             {
1308                 num_responses = 0;
1309                 (*p_inq->p_inq_cmpl_cb)(&num_responses);
1310             }
1311         }
1312     }
1313 
1314     /* Cancel a remote name request if active, and notify the caller (if waiting) */
1315     if (p_inq->remname_active )
1316     {
1317         alarm_cancel(p_inq->remote_name_timer);
1318         p_inq->remname_active = FALSE;
1319         memset(p_inq->remname_bda, 0, BD_ADDR_LEN);
1320 
1321         if (p_inq->p_remname_cmpl_cb)
1322         {
1323             rem_name.status = BTM_DEV_RESET;
1324 
1325             (*p_inq->p_remname_cmpl_cb)(&rem_name);
1326             p_inq->p_remname_cmpl_cb = NULL;
1327         }
1328     }
1329 
1330     /* Cancel an inquiry filter request if active, and notify the caller (if waiting) */
1331     if (p_inq->inqfilt_active)
1332     {
1333         p_inq->inqfilt_active = FALSE;
1334 
1335         if (p_inq->p_inqfilter_cmpl_cb)
1336         {
1337             status = BTM_DEV_RESET;
1338             (*p_inq->p_inqfilter_cmpl_cb)(&status);
1339         }
1340     }
1341 
1342     p_inq->state = BTM_INQ_INACTIVE_STATE;
1343     p_inq->pending_filt_complete_event = 0;
1344     p_inq->p_inq_results_cb = NULL;
1345     btm_clr_inq_db(NULL);   /* Clear out all the entries in the database */
1346     btm_clr_inq_result_flt();
1347 
1348     p_inq->discoverable_mode = BTM_NON_DISCOVERABLE;
1349     p_inq->connectable_mode  = BTM_NON_CONNECTABLE;
1350     p_inq->page_scan_type    = BTM_SCAN_TYPE_STANDARD;
1351     p_inq->inq_scan_type     = BTM_SCAN_TYPE_STANDARD;
1352 
1353 #if BLE_INCLUDED == TRUE
1354     p_inq->discoverable_mode |= BTM_BLE_NON_DISCOVERABLE;
1355     p_inq->connectable_mode  |= BTM_BLE_NON_CONNECTABLE;
1356 #endif
1357     return;
1358 }
1359 
1360 
1361 /*********************************************************************************
1362 **
1363 ** Function         btm_inq_db_init
1364 **
1365 ** Description      This function is called at startup to initialize the inquiry
1366 **                  database.
1367 **
1368 ** Returns          void
1369 **
1370 *******************************************************************************/
btm_inq_db_init(void)1371 void btm_inq_db_init (void)
1372 {
1373 #if 0  /* cleared in btm_init; put back in if called from anywhere else! */
1374     memset (&btm_cb.btm_inq_vars, 0, sizeof (tBTM_INQUIRY_VAR_ST));
1375 #endif
1376     alarm_free(btm_cb.btm_inq_vars.remote_name_timer);
1377     btm_cb.btm_inq_vars.remote_name_timer =
1378         alarm_new("btm_inq.remote_name_timer");
1379     btm_cb.btm_inq_vars.no_inc_ssp = BTM_NO_SSP_ON_INQUIRY;
1380 }
1381 
1382 /*********************************************************************************
1383 **
1384 ** Function         btm_inq_stop_on_ssp
1385 **
1386 ** Description      This function is called on incoming SSP
1387 **
1388 ** Returns          void
1389 **
1390 *******************************************************************************/
btm_inq_stop_on_ssp(void)1391 void btm_inq_stop_on_ssp(void)
1392 {
1393     UINT8 normal_active = (BTM_GENERAL_INQUIRY_ACTIVE|BTM_LIMITED_INQUIRY_ACTIVE);
1394 
1395 #if (BTM_INQ_DEBUG == TRUE)
1396     BTM_TRACE_DEBUG ("btm_inq_stop_on_ssp: no_inc_ssp=%d inq_active:0x%x state:%d inqfilt_active:%d",
1397         btm_cb.btm_inq_vars.no_inc_ssp, btm_cb.btm_inq_vars.inq_active, btm_cb.btm_inq_vars.state, btm_cb.btm_inq_vars.inqfilt_active);
1398 #endif
1399     if (btm_cb.btm_inq_vars.no_inc_ssp)
1400     {
1401         if (btm_cb.btm_inq_vars.state == BTM_INQ_ACTIVE_STATE)
1402         {
1403             if (btm_cb.btm_inq_vars.inq_active & BTM_PERIODIC_INQUIRY_ACTIVE)
1404             {
1405                 BTM_CancelPeriodicInquiry();
1406             }
1407             else if (btm_cb.btm_inq_vars.inq_active & normal_active)
1408             {
1409                 /* can not call BTM_CancelInquiry() here. We need to report inquiry complete evt */
1410                 btsnd_hcic_inq_cancel();
1411             }
1412         }
1413         /* do not allow inquiry to start */
1414         btm_cb.btm_inq_vars.inq_active |= BTM_SSP_INQUIRY_ACTIVE;
1415     }
1416 }
1417 
1418 /*********************************************************************************
1419 **
1420 ** Function         btm_inq_clear_ssp
1421 **
1422 ** Description      This function is called when pairing_state becomes idle
1423 **
1424 ** Returns          void
1425 **
1426 *******************************************************************************/
btm_inq_clear_ssp(void)1427 void btm_inq_clear_ssp(void)
1428 {
1429     btm_cb.btm_inq_vars.inq_active &= ~BTM_SSP_INQUIRY_ACTIVE;
1430 }
1431 
1432 /*********************************************************************************
1433 **
1434 ** Function         btm_clr_inq_db
1435 **
1436 ** Description      This function is called to clear out a device or all devices
1437 **                  from the inquiry database.
1438 **
1439 ** Parameter        p_bda - (input) BD_ADDR ->  Address of device to clear
1440 **                                              (NULL clears all entries)
1441 **
1442 ** Returns          void
1443 **
1444 *******************************************************************************/
btm_clr_inq_db(BD_ADDR p_bda)1445 void btm_clr_inq_db (BD_ADDR p_bda)
1446 {
1447     tBTM_INQUIRY_VAR_ST     *p_inq = &btm_cb.btm_inq_vars;
1448     tINQ_DB_ENT             *p_ent = p_inq->inq_db;
1449     UINT16                   xx;
1450 
1451 #if (BTM_INQ_DEBUG == TRUE)
1452     BTM_TRACE_DEBUG ("btm_clr_inq_db: inq_active:0x%x state:%d",
1453         btm_cb.btm_inq_vars.inq_active, btm_cb.btm_inq_vars.state);
1454 #endif
1455     for (xx = 0; xx < BTM_INQ_DB_SIZE; xx++, p_ent++)
1456     {
1457         if (p_ent->in_use)
1458         {
1459             /* If this is the specified BD_ADDR or clearing all devices */
1460             if (p_bda == NULL ||
1461                 (!memcmp (p_ent->inq_info.results.remote_bd_addr, p_bda, BD_ADDR_LEN)))
1462             {
1463                 p_ent->in_use = FALSE;
1464             }
1465         }
1466     }
1467 #if (BTM_INQ_DEBUG == TRUE)
1468     BTM_TRACE_DEBUG ("inq_active:0x%x state:%d",
1469         btm_cb.btm_inq_vars.inq_active, btm_cb.btm_inq_vars.state);
1470 #endif
1471 }
1472 
1473 
1474 /*******************************************************************************
1475 **
1476 ** Function         btm_clr_inq_result_flt
1477 **
1478 ** Description      This function looks through the bdaddr database for a match
1479 **                  based on Bluetooth Device Address
1480 **
1481 ** Returns          TRUE if found, else FALSE (new entry)
1482 **
1483 *******************************************************************************/
btm_clr_inq_result_flt(void)1484 static void btm_clr_inq_result_flt (void)
1485 {
1486     tBTM_INQUIRY_VAR_ST *p_inq = &btm_cb.btm_inq_vars;
1487 
1488     osi_free_and_reset((void **)&p_inq->p_bd_db);
1489     p_inq->num_bd_entries = 0;
1490     p_inq->max_bd_entries = 0;
1491 }
1492 
1493 /*******************************************************************************
1494 **
1495 ** Function         btm_inq_find_bdaddr
1496 **
1497 ** Description      This function looks through the bdaddr database for a match
1498 **                  based on Bluetooth Device Address
1499 **
1500 ** Returns          TRUE if found, else FALSE (new entry)
1501 **
1502 *******************************************************************************/
btm_inq_find_bdaddr(BD_ADDR p_bda)1503 BOOLEAN btm_inq_find_bdaddr (BD_ADDR p_bda)
1504 {
1505     tBTM_INQUIRY_VAR_ST *p_inq = &btm_cb.btm_inq_vars;
1506     tINQ_BDADDR         *p_db = &p_inq->p_bd_db[0];
1507     UINT16       xx;
1508 
1509     /* Don't bother searching, database doesn't exist or periodic mode */
1510     if ((p_inq->inq_active & BTM_PERIODIC_INQUIRY_ACTIVE) || !p_db)
1511         return (FALSE);
1512 
1513     for (xx = 0; xx < p_inq->num_bd_entries; xx++, p_db++)
1514     {
1515         if (!memcmp(p_db->bd_addr, p_bda, BD_ADDR_LEN)
1516             && p_db->inq_count == p_inq->inq_counter)
1517             return (TRUE);
1518     }
1519 
1520     if (xx < p_inq->max_bd_entries)
1521     {
1522         p_db->inq_count = p_inq->inq_counter;
1523         memcpy(p_db->bd_addr, p_bda, BD_ADDR_LEN);
1524         p_inq->num_bd_entries++;
1525     }
1526 
1527     /* If here, New Entry */
1528     return (FALSE);
1529 }
1530 
1531 /*******************************************************************************
1532 **
1533 ** Function         btm_inq_db_find
1534 **
1535 ** Description      This function looks through the inquiry database for a match
1536 **                  based on Bluetooth Device Address
1537 **
1538 ** Returns          pointer to entry, or NULL if not found
1539 **
1540 *******************************************************************************/
btm_inq_db_find(const BD_ADDR p_bda)1541 tINQ_DB_ENT *btm_inq_db_find (const BD_ADDR p_bda)
1542 {
1543     UINT16       xx;
1544     tINQ_DB_ENT  *p_ent = btm_cb.btm_inq_vars.inq_db;
1545 
1546     for (xx = 0; xx < BTM_INQ_DB_SIZE; xx++, p_ent++)
1547     {
1548         if ((p_ent->in_use) && (!memcmp (p_ent->inq_info.results.remote_bd_addr, p_bda, BD_ADDR_LEN)))
1549             return (p_ent);
1550     }
1551 
1552     /* If here, not found */
1553     return (NULL);
1554 }
1555 
1556 
1557 /*******************************************************************************
1558 **
1559 ** Function         btm_inq_db_new
1560 **
1561 ** Description      This function looks through the inquiry database for an unused
1562 **                  entry. If no entry is free, it allocates the oldest entry.
1563 **
1564 ** Returns          pointer to entry
1565 **
1566 *******************************************************************************/
btm_inq_db_new(BD_ADDR p_bda)1567 tINQ_DB_ENT *btm_inq_db_new (BD_ADDR p_bda)
1568 {
1569     UINT16       xx;
1570     tINQ_DB_ENT  *p_ent = btm_cb.btm_inq_vars.inq_db;
1571     tINQ_DB_ENT  *p_old = btm_cb.btm_inq_vars.inq_db;
1572     UINT32       ot = 0xFFFFFFFF;
1573 
1574     for (xx = 0; xx < BTM_INQ_DB_SIZE; xx++, p_ent++)
1575     {
1576         if (!p_ent->in_use)
1577         {
1578             memset (p_ent, 0, sizeof (tINQ_DB_ENT));
1579             memcpy (p_ent->inq_info.results.remote_bd_addr, p_bda, BD_ADDR_LEN);
1580             p_ent->in_use = TRUE;
1581 
1582             return (p_ent);
1583         }
1584 
1585         if (p_ent->time_of_resp < ot)
1586         {
1587             p_old = p_ent;
1588             ot    = p_ent->time_of_resp;
1589         }
1590     }
1591 
1592     /* If here, no free entry found. Return the oldest. */
1593 
1594     memset (p_old, 0, sizeof (tINQ_DB_ENT));
1595     memcpy (p_old->inq_info.results.remote_bd_addr, p_bda, BD_ADDR_LEN);
1596     p_old->in_use = TRUE;
1597 
1598     return (p_old);
1599 }
1600 
1601 
1602 /*******************************************************************************
1603 **
1604 ** Function         btm_set_inq_event_filter
1605 **
1606 ** Description      This function is called to set the inquiry event filter.
1607 **                  It is called by either internally, or by the external API function
1608 **                  (BTM_SetInqEventFilter).  It is used internally as part of the
1609 **                  inquiry processing.
1610 **
1611 ** Input Params:
1612 **                  filter_cond_type - this is the type of inquiry filter to apply:
1613 **                          BTM_FILTER_COND_DEVICE_CLASS,
1614 **                          BTM_FILTER_COND_BD_ADDR, or
1615 **                          BTM_CLR_INQUIRY_FILTER
1616 **
1617 **                  p_filt_cond - this is either a BD_ADDR or DEV_CLASS depending on the
1618 **                          filter_cond_type  (See section 4.7.3 of Core Spec 1.0b).
1619 **
1620 ** Returns          BTM_CMD_STARTED if successfully initiated
1621 **                  BTM_NO_RESOURCES if couldn't get a memory pool buffer
1622 **                  BTM_ILLEGAL_VALUE if a bad parameter was detected
1623 **
1624 *******************************************************************************/
btm_set_inq_event_filter(UINT8 filter_cond_type,tBTM_INQ_FILT_COND * p_filt_cond)1625 static tBTM_STATUS btm_set_inq_event_filter (UINT8 filter_cond_type,
1626                                              tBTM_INQ_FILT_COND *p_filt_cond)
1627 {
1628     UINT8    condition_length = DEV_CLASS_LEN * 2;
1629     UINT8    condition_buf[DEV_CLASS_LEN * 2];
1630     UINT8   *p_cond = condition_buf;                    /* points to the condition to pass to HCI */
1631 
1632 #if (BTM_INQ_DEBUG == TRUE)
1633     BTM_TRACE_DEBUG ("btm_set_inq_event_filter: filter type %d [Clear-0, COD-1, BDADDR-2]",
1634         filter_cond_type);
1635     BTM_TRACE_DEBUG ("                       condition [%02x%02x%02x %02x%02x%02x]",
1636                p_filt_cond->bdaddr_cond[0], p_filt_cond->bdaddr_cond[1], p_filt_cond->bdaddr_cond[2],
1637                p_filt_cond->bdaddr_cond[3], p_filt_cond->bdaddr_cond[4], p_filt_cond->bdaddr_cond[5]);
1638 #endif
1639 
1640     /* Load the correct filter condition to pass to the lower layer */
1641     switch (filter_cond_type)
1642     {
1643     case BTM_FILTER_COND_DEVICE_CLASS:
1644         /* copy the device class and device class fields into contiguous memory to send to HCI */
1645         memcpy (condition_buf, p_filt_cond->cod_cond.dev_class, DEV_CLASS_LEN);
1646         memcpy (&condition_buf[DEV_CLASS_LEN],
1647                 p_filt_cond->cod_cond.dev_class_mask, DEV_CLASS_LEN);
1648 
1649         /* condition length should already be set as the default */
1650         break;
1651 
1652     case BTM_FILTER_COND_BD_ADDR:
1653         p_cond = p_filt_cond->bdaddr_cond;
1654 
1655         /* condition length should already be set as the default */
1656         break;
1657 
1658     case BTM_CLR_INQUIRY_FILTER:
1659         condition_length = 0;
1660         break;
1661 
1662     default:
1663         return (BTM_ILLEGAL_VALUE);     /* Bad parameter was passed in */
1664     }
1665 
1666     btm_cb.btm_inq_vars.inqfilt_active = TRUE;
1667 
1668     /* Filter the inquiry results for the specified condition type and value */
1669     if (btsnd_hcic_set_event_filter(HCI_FILTER_INQUIRY_RESULT, filter_cond_type,
1670                                     p_cond, condition_length))
1671 
1672         return (BTM_CMD_STARTED);
1673     else
1674         return (BTM_NO_RESOURCES);
1675 }
1676 
1677 
1678 /*******************************************************************************
1679 **
1680 ** Function         btm_event_filter_complete
1681 **
1682 ** Description      This function is called when a set event filter has completed.
1683 **                  Note: This routine currently only handles inquiry filters.
1684 **                      Connection filters are ignored for now.
1685 **
1686 ** Returns          void
1687 **
1688 *******************************************************************************/
btm_event_filter_complete(UINT8 * p)1689 void btm_event_filter_complete (UINT8 *p)
1690 {
1691     UINT8            hci_status;
1692     tBTM_STATUS      status;
1693     tBTM_INQUIRY_VAR_ST *p_inq = &btm_cb.btm_inq_vars;
1694     tBTM_CMPL_CB   *p_cb = p_inq->p_inqfilter_cmpl_cb;
1695 
1696 #if (BTM_INQ_DEBUG == TRUE)
1697     BTM_TRACE_DEBUG ("btm_event_filter_complete: inq_active:0x%x state:%d inqfilt_active:%d",
1698         btm_cb.btm_inq_vars.inq_active, btm_cb.btm_inq_vars.state, btm_cb.btm_inq_vars.inqfilt_active);
1699 #endif
1700     /* If the filter complete event is from an old or cancelled request, ignore it */
1701     if(p_inq->pending_filt_complete_event)
1702     {
1703         p_inq->pending_filt_complete_event--;
1704         return;
1705     }
1706 
1707     /* Only process the inquiry filter; Ignore the connection filter until it
1708        is used by the upper layers */
1709     if (p_inq->inqfilt_active == TRUE )
1710     {
1711         /* Extract the returned status from the buffer */
1712         STREAM_TO_UINT8 (hci_status, p);
1713         if (hci_status != HCI_SUCCESS)
1714         {
1715             /* If standalone operation, return the error status; if embedded in the inquiry, continue the inquiry */
1716             BTM_TRACE_WARNING ("BTM Warning: Set Event Filter Failed (HCI returned 0x%x)", hci_status);
1717             status = BTM_ERR_PROCESSING;
1718         }
1719         else
1720             status = BTM_SUCCESS;
1721 
1722         /* If the set filter was initiated externally (via BTM_SetInqEventFilter), call the
1723            callback function to notify the initiator that it has completed */
1724         if (p_inq->state == BTM_INQ_INACTIVE_STATE)
1725         {
1726             p_inq->inqfilt_active = FALSE;
1727             if (p_cb)
1728                 (*p_cb) (&status);
1729         }
1730         else    /* An inquiry is active (the set filter command was internally generated),
1731                    process the next state of the process (Set a new filter or start the inquiry). */
1732         {
1733             if(status != BTM_SUCCESS)
1734             {
1735                 /* Process the inquiry complete (Error Status) */
1736                 btm_process_inq_complete (BTM_ERR_PROCESSING, (UINT8)(p_inq->inqparms.mode & BTM_BR_INQUIRY_MASK));
1737 
1738                 /* btm_process_inq_complete() does not restore the following settings on periodic inquiry */
1739                 p_inq->inqfilt_active = FALSE;
1740                 p_inq->inq_active = BTM_INQUIRY_INACTIVE;
1741                 p_inq->state = BTM_INQ_INACTIVE_STATE;
1742 
1743                 return;
1744             }
1745 
1746             /* Check to see if a new filter needs to be set up */
1747             if (p_inq->state == BTM_INQ_CLR_FILT_STATE)
1748             {
1749                 if ((status = btm_set_inq_event_filter (p_inq->inqparms.filter_cond_type, &p_inq->inqparms.filter_cond)) == BTM_CMD_STARTED)
1750                 {
1751                     p_inq->state = BTM_INQ_SET_FILT_STATE;
1752                 }
1753                 else    /* Error setting the filter: Call the initiator's callback function to indicate a failure */
1754                 {
1755                     p_inq->inqfilt_active = FALSE;
1756 
1757                     /* Process the inquiry complete (Error Status) */
1758                     btm_process_inq_complete (BTM_ERR_PROCESSING, (UINT8)(p_inq->inqparms.mode & BTM_BR_INQUIRY_MASK));
1759                 }
1760             }
1761             else    /* Initiate the Inquiry or Periodic Inquiry */
1762             {
1763                 p_inq->state = BTM_INQ_ACTIVE_STATE;
1764                 p_inq->inqfilt_active = FALSE;
1765                 btm_initiate_inquiry (p_inq);
1766             }
1767         }
1768     }
1769 }
1770 
1771 
1772 /*******************************************************************************
1773 **
1774 ** Function         btm_initiate_inquiry
1775 **
1776 ** Description      This function is called to start an inquiry or periodic inquiry
1777 **                  upon completion of the setting and/or clearing of the inquiry filter.
1778 **
1779 ** Inputs:          p_inq (btm_cb.btm_inq_vars) - pointer to saved inquiry information
1780 **                      mode - GENERAL or LIMITED inquiry
1781 **                      duration - length in 1.28 sec intervals (If '0', the inquiry is CANCELLED)
1782 **                      max_resps - maximum amount of devices to search for before ending the inquiry
1783 **                      filter_cond_type - BTM_CLR_INQUIRY_FILTER, BTM_FILTER_COND_DEVICE_CLASS, or
1784 **                                         BTM_FILTER_COND_BD_ADDR
1785 **                      filter_cond - value for the filter (based on filter_cond_type)
1786 **
1787 ** Returns          If an error occurs the initiator's callback is called with the error status.
1788 **
1789 *******************************************************************************/
btm_initiate_inquiry(tBTM_INQUIRY_VAR_ST * p_inq)1790 static void btm_initiate_inquiry (tBTM_INQUIRY_VAR_ST *p_inq)
1791 {
1792     const LAP       *lap;
1793     tBTM_INQ_PARMS  *p_inqparms = &p_inq->inqparms;
1794 
1795 #if (BTM_INQ_DEBUG == TRUE)
1796     BTM_TRACE_DEBUG ("btm_initiate_inquiry: inq_active:0x%x state:%d inqfilt_active:%d",
1797         btm_cb.btm_inq_vars.inq_active, btm_cb.btm_inq_vars.state, btm_cb.btm_inq_vars.inqfilt_active);
1798 #endif
1799     btm_acl_update_busy_level (BTM_BLI_INQ_EVT);
1800 
1801     if (p_inq->inq_active & BTM_SSP_INQUIRY_ACTIVE)
1802     {
1803         btm_process_inq_complete (BTM_NO_RESOURCES, (UINT8)(p_inqparms->mode & BTM_BR_INQUIRY_MASK));
1804         return;
1805     }
1806 
1807     /* Make sure the number of responses doesn't overflow the database configuration */
1808     p_inqparms->max_resps = (UINT8)((p_inqparms->max_resps <= BTM_INQ_DB_SIZE) ? p_inqparms->max_resps : BTM_INQ_DB_SIZE);
1809 
1810     lap = (p_inq->inq_active & BTM_LIMITED_INQUIRY_ACTIVE) ? &limited_inq_lap : &general_inq_lap;
1811 
1812     if (p_inq->inq_active & BTM_PERIODIC_INQUIRY_ACTIVE)
1813     {
1814         if (!btsnd_hcic_per_inq_mode (p_inq->per_max_delay,
1815                                       p_inq->per_min_delay,
1816                                       *lap, p_inqparms->duration,
1817                                       p_inqparms->max_resps))
1818             btm_process_inq_complete (BTM_NO_RESOURCES, (UINT8)(p_inqparms->mode & BTM_BR_INQUIRY_MASK));
1819     }
1820     else
1821     {
1822         btm_clr_inq_result_flt();
1823 
1824         /* Allocate memory to hold bd_addrs responding */
1825         p_inq->p_bd_db = (tINQ_BDADDR *)osi_calloc(BT_DEFAULT_BUFFER_SIZE);
1826         p_inq->max_bd_entries = (UINT16)(BT_DEFAULT_BUFFER_SIZE / sizeof(tINQ_BDADDR));
1827 
1828         if (!btsnd_hcic_inquiry(*lap, p_inqparms->duration, 0))
1829             btm_process_inq_complete (BTM_NO_RESOURCES, (UINT8)(p_inqparms->mode & BTM_BR_INQUIRY_MASK));
1830     }
1831 }
1832 
1833 /*******************************************************************************
1834 **
1835 ** Function         btm_process_inq_results
1836 **
1837 ** Description      This function is called when inquiry results are received from
1838 **                  the device. It updates the inquiry database. If the inquiry
1839 **                  database is full, the oldest entry is discarded.
1840 **
1841 ** Parameters       inq_res_mode - BTM_INQ_RESULT_STANDARD
1842 **                                 BTM_INQ_RESULT_WITH_RSSI
1843 **                                 BTM_INQ_RESULT_EXTENDED
1844 **
1845 ** Returns          void
1846 **
1847 *******************************************************************************/
btm_process_inq_results(UINT8 * p,UINT8 inq_res_mode)1848 void btm_process_inq_results (UINT8 *p, UINT8 inq_res_mode)
1849 {
1850     UINT8            num_resp, xx;
1851     BD_ADDR          bda;
1852     tINQ_DB_ENT     *p_i;
1853     tBTM_INQ_RESULTS *p_cur=NULL;
1854     BOOLEAN          is_new = TRUE;
1855     BOOLEAN          update = FALSE;
1856     INT8             i_rssi;
1857     tBTM_INQUIRY_VAR_ST *p_inq = &btm_cb.btm_inq_vars;
1858     tBTM_INQ_RESULTS_CB *p_inq_results_cb = p_inq->p_inq_results_cb;
1859     UINT8            page_scan_rep_mode = 0;
1860     UINT8            page_scan_per_mode = 0;
1861     UINT8            page_scan_mode = 0;
1862     UINT8            rssi = 0;
1863     DEV_CLASS        dc;
1864     UINT16           clock_offset;
1865     UINT8            *p_eir_data = NULL;
1866 
1867 #if (BTM_INQ_DEBUG == TRUE)
1868     BTM_TRACE_DEBUG ("btm_process_inq_results inq_active:0x%x state:%d inqfilt_active:%d",
1869         btm_cb.btm_inq_vars.inq_active, btm_cb.btm_inq_vars.state, btm_cb.btm_inq_vars.inqfilt_active);
1870 #endif
1871     /* Only process the results if the BR inquiry is still active */
1872     if (!(p_inq->inq_active & BTM_BR_INQ_ACTIVE_MASK))
1873         return;
1874 
1875     STREAM_TO_UINT8 (num_resp, p);
1876 
1877     if (inq_res_mode == BTM_INQ_RESULT_EXTENDED && (num_resp > 1)) {
1878         BTM_TRACE_ERROR ("btm_process_inq_results() extended results (%d) > 1",
1879                          num_resp);
1880         return;
1881     }
1882 
1883     for (xx = 0; xx < num_resp; xx++)
1884     {
1885         update = FALSE;
1886         /* Extract inquiry results */
1887         STREAM_TO_BDADDR   (bda, p);
1888         STREAM_TO_UINT8    (page_scan_rep_mode, p);
1889         STREAM_TO_UINT8    (page_scan_per_mode, p);
1890 
1891         if (inq_res_mode == BTM_INQ_RESULT_STANDARD)
1892         {
1893             STREAM_TO_UINT8(page_scan_mode, p);
1894         }
1895 
1896         STREAM_TO_DEVCLASS (dc, p);
1897         STREAM_TO_UINT16   (clock_offset, p);
1898         if (inq_res_mode != BTM_INQ_RESULT_STANDARD)
1899         {
1900             STREAM_TO_UINT8(rssi, p);
1901         }
1902 
1903         p_i = btm_inq_db_find (bda);
1904 
1905         /* Only process the num_resp is smaller than max_resps.
1906            If results are queued to BTU task while canceling inquiry,
1907            or when more than one result is in this response, > max_resp
1908            responses could be processed which can confuse some apps
1909         */
1910         if (p_inq->inqparms.max_resps &&
1911             p_inq->inq_cmpl_info.num_resp >= p_inq->inqparms.max_resps
1912 #if BLE_INCLUDED == TRUE
1913             /* new device response */
1914             && ( p_i == NULL ||
1915                 /* exisiting device with BR/EDR info */
1916                 (p_i && (p_i->inq_info.results.device_type & BT_DEVICE_TYPE_BREDR) != 0)
1917                )
1918 #endif
1919 
1920             )
1921         {
1922 /*            BTM_TRACE_WARNING("INQ RES: Extra Response Received...ignoring"); */
1923             return;
1924         }
1925 
1926         /* Check if this address has already been processed for this inquiry */
1927         if (btm_inq_find_bdaddr(bda))
1928         {
1929 /*             BTM_TRACE_DEBUG("BDA seen before [%02x%02x %02x%02x %02x%02x]",
1930                              bda[0], bda[1], bda[2], bda[3], bda[4], bda[5]);*/
1931              /* By default suppose no update needed */
1932             i_rssi = (INT8)rssi;
1933 
1934             /* If this new RSSI is higher than the last one */
1935             if(p_inq->inqparms.report_dup && (rssi != 0) &&
1936                p_i && (i_rssi > p_i->inq_info.results.rssi || p_i->inq_info.results.rssi == 0
1937 #if BLE_INCLUDED == TRUE
1938                /* BR/EDR inquiry information update */
1939                        || (p_i->inq_info.results.device_type & BT_DEVICE_TYPE_BREDR) != 0
1940 #endif
1941                        ))
1942             {
1943                 p_cur = &p_i->inq_info.results;
1944                 BTM_TRACE_DEBUG("update RSSI new:%d, old:%d", i_rssi, p_cur->rssi);
1945                 p_cur->rssi = i_rssi;
1946                 update = TRUE;
1947             }
1948             /* If we received a second Extended Inq Event for an already */
1949             /* discovered device, this is because for the first one EIR was not received */
1950             else if ((inq_res_mode == BTM_INQ_RESULT_EXTENDED) && (p_i))
1951             {
1952                 p_cur = &p_i->inq_info.results;
1953                 update = TRUE;
1954             }
1955             /* If no update needed continue with next response (if any) */
1956             else
1957                 continue;
1958         }
1959 
1960         /* If existing entry, use that, else get a new one (possibly reusing the oldest) */
1961         if (p_i == NULL)
1962         {
1963             p_i = btm_inq_db_new (bda);
1964             is_new = TRUE;
1965         }
1966 
1967         /* If an entry for the device already exists, overwrite it ONLY if it is from
1968            a previous inquiry. (Ignore it if it is a duplicate response from the same
1969            inquiry.
1970         */
1971         else if (p_i->inq_count == p_inq->inq_counter
1972 #if (BLE_INCLUDED == TRUE )
1973             && (p_i->inq_info.results.device_type == BT_DEVICE_TYPE_BREDR)
1974 #endif
1975             )
1976             is_new = FALSE;
1977 
1978         /* keep updating RSSI to have latest value */
1979         if( inq_res_mode != BTM_INQ_RESULT_STANDARD )
1980             p_i->inq_info.results.rssi = (INT8)rssi;
1981         else
1982             p_i->inq_info.results.rssi = BTM_INQ_RES_IGNORE_RSSI;
1983 
1984         if (is_new == TRUE)
1985         {
1986             /* Save the info */
1987             p_cur = &p_i->inq_info.results;
1988             p_cur->page_scan_rep_mode = page_scan_rep_mode;
1989             p_cur->page_scan_per_mode = page_scan_per_mode;
1990             p_cur->page_scan_mode     = page_scan_mode;
1991             p_cur->dev_class[0]       = dc[0];
1992             p_cur->dev_class[1]       = dc[1];
1993             p_cur->dev_class[2]       = dc[2];
1994             p_cur->clock_offset       = clock_offset  | BTM_CLOCK_OFFSET_VALID;
1995 
1996             p_i->time_of_resp = time_get_os_boottime_ms();
1997 
1998             if (p_i->inq_count != p_inq->inq_counter)
1999                 p_inq->inq_cmpl_info.num_resp++;       /* A new response was found */
2000 
2001 #if (defined BLE_INCLUDED && BLE_INCLUDED == TRUE)
2002             p_cur->inq_result_type    = BTM_INQ_RESULT_BR;
2003             if (p_i->inq_count != p_inq->inq_counter)
2004             {
2005                 p_cur->device_type  = BT_DEVICE_TYPE_BREDR;
2006                 p_i->scan_rsp       = FALSE;
2007             }
2008             else
2009                 p_cur->device_type    |= BT_DEVICE_TYPE_BREDR;
2010 #endif
2011                 p_i->inq_count = p_inq->inq_counter;   /* Mark entry for current inquiry */
2012 
2013             /* If the number of responses found and not unlimited, issue a cancel inquiry */
2014             if (!(p_inq->inq_active & BTM_PERIODIC_INQUIRY_ACTIVE) &&
2015                 p_inq->inqparms.max_resps &&
2016                 p_inq->inq_cmpl_info.num_resp == p_inq->inqparms.max_resps
2017 #if BLE_INCLUDED == TRUE
2018                 /* BLE scanning is active and received adv */
2019                 && ((((p_inq->inqparms.mode & BTM_BLE_INQUIRY_MASK) != 0) &&
2020                      p_cur->device_type == BT_DEVICE_TYPE_DUMO && p_i->scan_rsp) ||
2021                     (p_inq->inqparms.mode & BTM_BLE_INQUIRY_MASK) == 0)
2022 #endif
2023                 )
2024             {
2025 /*                BTM_TRACE_DEBUG("BTMINQ: Found devices, cancelling inquiry..."); */
2026                 btsnd_hcic_inq_cancel();
2027 
2028 #if BLE_INCLUDED == TRUE
2029                 if ((p_inq->inqparms.mode & BTM_BLE_INQUIRY_MASK) != 0)
2030                     btm_ble_stop_inquiry();
2031 #endif
2032                 btm_acl_update_busy_level (BTM_BLI_INQ_DONE_EVT);
2033             }
2034             /* Initialize flag to FALSE. This flag is set/used by application */
2035             p_i->inq_info.appl_knows_rem_name = FALSE;
2036         }
2037 
2038         if (is_new || update)
2039         {
2040             if( inq_res_mode == BTM_INQ_RESULT_EXTENDED )
2041             {
2042                 memset( p_cur->eir_uuid, 0,
2043                         BTM_EIR_SERVICE_ARRAY_SIZE * (BTM_EIR_ARRAY_BITS/8));
2044                 /* set bit map of UUID list from received EIR */
2045                 btm_set_eir_uuid( p, p_cur );
2046                 p_eir_data = p;
2047             }
2048             else
2049                 p_eir_data = NULL;
2050 
2051             /* If a callback is registered, call it with the results */
2052             if (p_inq_results_cb)
2053                 (p_inq_results_cb)((tBTM_INQ_RESULTS *) p_cur, p_eir_data);
2054         }
2055     }
2056 }
2057 
2058 /*******************************************************************************
2059 **
2060 ** Function         btm_sort_inq_result
2061 **
2062 ** Description      This function is called when inquiry complete is received
2063 **                  from the device to sort inquiry results based on rssi.
2064 **
2065 ** Returns          void
2066 **
2067 *******************************************************************************/
btm_sort_inq_result(void)2068 void btm_sort_inq_result(void)
2069 {
2070     UINT8 xx, yy, num_resp;
2071     tINQ_DB_ENT *p_ent  = btm_cb.btm_inq_vars.inq_db;
2072     tINQ_DB_ENT *p_next = btm_cb.btm_inq_vars.inq_db+1;
2073     int size;
2074     tINQ_DB_ENT *p_tmp = (tINQ_DB_ENT *)osi_malloc(sizeof(tINQ_DB_ENT));
2075 
2076     num_resp = (btm_cb.btm_inq_vars.inq_cmpl_info.num_resp<BTM_INQ_DB_SIZE)?
2077                 btm_cb.btm_inq_vars.inq_cmpl_info.num_resp: BTM_INQ_DB_SIZE;
2078 
2079     size = sizeof(tINQ_DB_ENT);
2080     for (xx = 0; xx < num_resp-1; xx++, p_ent++) {
2081         for (yy = xx+1, p_next = p_ent+1; yy < num_resp; yy++, p_next++) {
2082             if (p_ent->inq_info.results.rssi < p_next->inq_info.results.rssi) {
2083                 memcpy(p_tmp, p_next, size);
2084                 memcpy(p_next, p_ent,  size);
2085                 memcpy(p_ent, p_tmp,  size);
2086             }
2087         }
2088     }
2089 
2090     osi_free(p_tmp);
2091 }
2092 
2093 /*******************************************************************************
2094 **
2095 ** Function         btm_process_inq_complete
2096 **
2097 ** Description      This function is called when inquiry complete is received
2098 **                  from the device.  Call the callback if not in periodic inquiry
2099 **                  mode AND it is not NULL (The caller wants the event).
2100 **
2101 **                  The callback pass back the status and the number of responses
2102 **
2103 ** Returns          void
2104 **
2105 *******************************************************************************/
btm_process_inq_complete(UINT8 status,UINT8 mode)2106 void btm_process_inq_complete (UINT8 status, UINT8 mode)
2107 {
2108     tBTM_CMPL_CB        *p_inq_cb = btm_cb.btm_inq_vars.p_inq_cmpl_cb;
2109     tBTM_INQUIRY_VAR_ST *p_inq = &btm_cb.btm_inq_vars;
2110 
2111 #if (defined(BTA_HOST_INTERLEAVE_SEARCH) && BTA_HOST_INTERLEAVE_SEARCH == TRUE)
2112     /* inquiry inactive case happens when inquiry is cancelled.
2113        Make mode 0 for no further inquiries from the current inquiry process
2114     */
2115     if(status!=HCI_SUCCESS || p_inq->next_state==BTM_FINISH || !p_inq->inq_active)
2116     {
2117         /* re-initialize for next inquiry request */
2118         p_inq->next_state=BTM_BR_ONE;
2119         /* make the mode 0 here */
2120         p_inq->inqparms.mode &= ~(p_inq->inqparms.mode);
2121 
2122     }
2123 #endif
2124 
2125 #if (!defined(BTA_HOST_INTERLEAVE_SEARCH) || BTA_HOST_INTERLEAVE_SEARCH == FALSE)
2126     p_inq->inqparms.mode &= ~(mode);
2127 #endif
2128 
2129     if(p_inq->scan_type == INQ_LE_OBSERVE && !p_inq->inq_active)
2130     {
2131         /*end of LE observe*/
2132         p_inq->p_inq_ble_results_cb = (tBTM_INQ_RESULTS_CB *) NULL;
2133         p_inq->p_inq_ble_cmpl_cb = (tBTM_CMPL_CB *) NULL;
2134         p_inq->scan_type=INQ_NONE;
2135     }
2136 
2137 
2138 #if (BTM_INQ_DEBUG == TRUE)
2139     BTM_TRACE_DEBUG ("btm_process_inq_complete inq_active:0x%x state:%d inqfilt_active:%d",
2140         btm_cb.btm_inq_vars.inq_active, btm_cb.btm_inq_vars.state, btm_cb.btm_inq_vars.inqfilt_active);
2141 #endif
2142     btm_acl_update_busy_level (BTM_BLI_INQ_DONE_EVT);
2143     /* Ignore any stray or late complete messages if the inquiry is not active */
2144     if (p_inq->inq_active)
2145     {
2146         p_inq->inq_cmpl_info.status = (tBTM_STATUS)((status == HCI_SUCCESS) ? BTM_SUCCESS : BTM_ERR_PROCESSING);
2147 
2148         /* Notify caller that the inquiry has completed; (periodic inquiries do not send completion events */
2149         if (!(p_inq->inq_active & BTM_PERIODIC_INQUIRY_ACTIVE) && p_inq->inqparms.mode == 0)
2150         {
2151 #if BLE_INCLUDED == TRUE
2152             btm_clear_all_pending_le_entry();
2153 #endif
2154             p_inq->state = BTM_INQ_INACTIVE_STATE;
2155 
2156             /* Increment so the start of a next inquiry has a new count */
2157             p_inq->inq_counter++;
2158 
2159             btm_clr_inq_result_flt();
2160 
2161             if((p_inq->inq_cmpl_info.status == BTM_SUCCESS) &&
2162                 controller_get_interface()->supports_rssi_with_inquiry_results())
2163             {
2164                 btm_sort_inq_result();
2165             }
2166 
2167             /* Clear the results callback if set */
2168             p_inq->p_inq_results_cb = (tBTM_INQ_RESULTS_CB *) NULL;
2169             p_inq->inq_active = BTM_INQUIRY_INACTIVE;
2170             p_inq->p_inq_cmpl_cb = (tBTM_CMPL_CB *) NULL;
2171 
2172             /* If we have a callback registered for inquiry complete, call it */
2173             BTM_TRACE_DEBUG ("BTM Inq Compl Callback: status 0x%02x, num results %d",
2174                         p_inq->inq_cmpl_info.status, p_inq->inq_cmpl_info.num_resp);
2175 
2176             if (p_inq_cb)
2177                 (p_inq_cb)((tBTM_INQUIRY_CMPL *) &p_inq->inq_cmpl_info);
2178         }
2179 #if (defined(BTA_HOST_INTERLEAVE_SEARCH) && BTA_HOST_INTERLEAVE_SEARCH == TRUE)
2180         if(p_inq->inqparms.mode != 0 && !(p_inq->inq_active & BTM_PERIODIC_INQUIRY_ACTIVE))
2181         {
2182             /* make inquiry inactive for next iteration */
2183             p_inq->inq_active = BTM_INQUIRY_INACTIVE;
2184             /* call the inquiry again */
2185             BTM_StartInquiry(&p_inq->inqparms,p_inq->p_inq_results_cb,p_inq->p_inq_cmpl_cb);
2186         }
2187 #endif
2188     }
2189     if(p_inq->inqparms.mode == 0 && p_inq->scan_type == INQ_GENERAL)//this inquiry is complete
2190     {
2191         p_inq->scan_type = INQ_NONE;
2192 #if (defined BLE_INCLUDED && BLE_INCLUDED == TRUE)
2193         /* check if the LE observe is pending */
2194         if(p_inq->p_inq_ble_results_cb != NULL)
2195         {
2196             BTM_TRACE_DEBUG("BTM Inq Compl: resuming a pending LE scan");
2197             BTM_BleObserve(1,0, p_inq->p_inq_ble_results_cb, p_inq->p_inq_ble_cmpl_cb);
2198         }
2199 #endif
2200     }
2201 #if (BTM_INQ_DEBUG == TRUE)
2202     BTM_TRACE_DEBUG ("inq_active:0x%x state:%d inqfilt_active:%d",
2203         btm_cb.btm_inq_vars.inq_active, btm_cb.btm_inq_vars.state, btm_cb.btm_inq_vars.inqfilt_active);
2204 #endif
2205 }
2206 
2207 /*******************************************************************************
2208 **
2209 ** Function         btm_process_cancel_complete
2210 **
2211 ** Description      This function is called when inquiry cancel complete is received
2212 **                  from the device.This function will also call the btm_process_inq_complete
2213 **                  This function is needed to differentiate a cancel_cmpl_evt from the
2214 **                  inq_cmpl_evt
2215 **
2216 ** Returns          void
2217 **
2218 *******************************************************************************/
btm_process_cancel_complete(UINT8 status,UINT8 mode)2219 void btm_process_cancel_complete(UINT8 status, UINT8 mode)
2220 {
2221      btm_acl_update_busy_level (BTM_BLI_INQ_CANCEL_EVT);
2222      btm_process_inq_complete(status, mode);
2223 }
2224 /*******************************************************************************
2225 **
2226 ** Function         btm_initiate_rem_name
2227 **
2228 ** Description      This function looks initiates a remote name request.  It is called
2229 **                  either by GAP or by the API call BTM_ReadRemoteDeviceName.
2230 **
2231 ** Input Params:    p_cur         - pointer to an inquiry result structure (NULL if nonexistent)
2232 **                  p_cb            - callback function called when BTM_CMD_STARTED
2233 **                                    is returned.
2234 **                                    A pointer to tBTM_REMOTE_DEV_NAME is passed to the
2235 **                                    callback.
2236 **
2237 ** Returns
2238 **                  BTM_CMD_STARTED is returned if the request was sent to HCI.
2239 **                  BTM_BUSY if already in progress
2240 **                  BTM_NO_RESOURCES if could not allocate resources to start the command
2241 **                  BTM_WRONG_MODE if the device is not up.
2242 **
2243 *******************************************************************************/
btm_initiate_rem_name(BD_ADDR remote_bda,tBTM_INQ_INFO * p_cur,UINT8 origin,period_ms_t timeout_ms,tBTM_CMPL_CB * p_cb)2244 tBTM_STATUS  btm_initiate_rem_name (BD_ADDR remote_bda, tBTM_INQ_INFO *p_cur,
2245                                     UINT8 origin, period_ms_t timeout_ms,
2246                                     tBTM_CMPL_CB *p_cb)
2247 {
2248     tBTM_INQUIRY_VAR_ST *p_inq = &btm_cb.btm_inq_vars;
2249     BOOLEAN              cmd_ok;
2250 
2251 
2252     /*** Make sure the device is ready ***/
2253     if (!BTM_IsDeviceUp())
2254         return (BTM_WRONG_MODE);
2255 
2256 
2257     if (origin == BTM_RMT_NAME_SEC)
2258     {
2259         cmd_ok = btsnd_hcic_rmt_name_req (remote_bda, HCI_PAGE_SCAN_REP_MODE_R1,
2260                                          HCI_MANDATARY_PAGE_SCAN_MODE, 0);
2261         if (cmd_ok)
2262             return BTM_CMD_STARTED;
2263         else
2264             return BTM_NO_RESOURCES;
2265     }
2266     /* Make sure there are no two remote name requests from external API in progress */
2267     else if (origin == BTM_RMT_NAME_EXT)
2268     {
2269         if (p_inq->remname_active)
2270         {
2271             return (BTM_BUSY);
2272         }
2273         else
2274         {
2275             /* If there is no remote name request running,call the callback function and start timer */
2276             p_inq->p_remname_cmpl_cb = p_cb;
2277             memcpy(p_inq->remname_bda, remote_bda, BD_ADDR_LEN);
2278 
2279             alarm_set_on_queue(p_inq->remote_name_timer, timeout_ms,
2280                                btm_inq_remote_name_timer_timeout, NULL,
2281                                btu_general_alarm_queue);
2282 
2283             /* If the database entry exists for the device, use its clock offset */
2284             if (p_cur)
2285             {
2286                 cmd_ok = btsnd_hcic_rmt_name_req (remote_bda,
2287                                          p_cur->results.page_scan_rep_mode,
2288                                          p_cur->results.page_scan_mode,
2289                                          (UINT16)(p_cur->results.clock_offset |
2290                                                   BTM_CLOCK_OFFSET_VALID));
2291             }
2292             else /* Otherwise use defaults and mark the clock offset as invalid */
2293             {
2294                 cmd_ok = btsnd_hcic_rmt_name_req (remote_bda, HCI_PAGE_SCAN_REP_MODE_R1,
2295                                          HCI_MANDATARY_PAGE_SCAN_MODE, 0);
2296             }
2297             if (cmd_ok)
2298             {
2299                 p_inq->remname_active = TRUE;
2300                 return BTM_CMD_STARTED;
2301             }
2302             else
2303                 return BTM_NO_RESOURCES;
2304         }
2305     }
2306     else
2307     {
2308         return BTM_ILLEGAL_VALUE;
2309     }
2310 }
2311 
2312 /*******************************************************************************
2313 **
2314 ** Function         btm_process_remote_name
2315 **
2316 ** Description      This function is called when a remote name is received from
2317 **                  the device. If remote names are cached, it updates the inquiry
2318 **                  database.
2319 **
2320 ** Returns          void
2321 **
2322 *******************************************************************************/
btm_process_remote_name(BD_ADDR bda,BD_NAME bdn,UINT16 evt_len,UINT8 hci_status)2323 void btm_process_remote_name (BD_ADDR bda, BD_NAME bdn, UINT16 evt_len, UINT8 hci_status)
2324 {
2325     tBTM_REMOTE_DEV_NAME    rem_name;
2326     tBTM_INQUIRY_VAR_ST    *p_inq = &btm_cb.btm_inq_vars;
2327     tBTM_CMPL_CB           *p_cb = p_inq->p_remname_cmpl_cb;
2328     UINT8                  *p_n1;
2329 
2330     UINT16                 temp_evt_len;
2331 
2332     if (bda != NULL)
2333     {
2334         BTM_TRACE_EVENT("BDA %02x:%02x:%02x:%02x:%02x:%02x",bda[0], bda[1],
2335                  bda[2], bda[3],
2336                  bda[4], bda[5]);
2337     }
2338 
2339     BTM_TRACE_EVENT("Inquire BDA %02x:%02x:%02x:%02x:%02x:%02x",p_inq->remname_bda[0], p_inq->remname_bda[1],
2340              p_inq->remname_bda[2], p_inq->remname_bda[3],
2341              p_inq->remname_bda[4], p_inq->remname_bda[5]);
2342 
2343 
2344 
2345     /* If the inquire BDA and remote DBA are the same, then stop the timer and set the active to false */
2346     if ((p_inq->remname_active ==TRUE)&&
2347         (((bda != NULL) &&
2348         (memcmp(bda, p_inq->remname_bda,BD_ADDR_LEN)==0)) || bda == NULL))
2349 
2350     {
2351 #if BLE_INCLUDED == TRUE
2352         if (BTM_UseLeLink(p_inq->remname_bda))
2353         {
2354             if (hci_status == HCI_ERR_UNSPECIFIED)
2355                 btm_ble_cancel_remote_name(p_inq->remname_bda);
2356         }
2357 #endif
2358         alarm_cancel(p_inq->remote_name_timer);
2359         p_inq->remname_active = FALSE;
2360          /* Clean up and return the status if the command was not successful */
2361          /* Note: If part of the inquiry, the name is not stored, and the    */
2362          /*       inquiry complete callback is called.                       */
2363 
2364         if (hci_status == HCI_SUCCESS)
2365         {
2366             /* Copy the name from the data stream into the return structure */
2367             /* Note that even if it is not being returned, it is used as a  */
2368             /*      temporary buffer.                                       */
2369             p_n1 = (UINT8 *)rem_name.remote_bd_name;
2370             rem_name.length = (evt_len < BD_NAME_LEN) ? evt_len : BD_NAME_LEN;
2371             rem_name.remote_bd_name[rem_name.length] = 0;
2372             rem_name.status = BTM_SUCCESS;
2373             temp_evt_len = rem_name.length;
2374 
2375             while (temp_evt_len > 0)
2376             {
2377                 *p_n1++ = *bdn++;
2378                 temp_evt_len--;
2379             }
2380             rem_name.remote_bd_name[rem_name.length] = 0;
2381         }
2382 
2383 
2384         /* If processing a stand alone remote name then report the error in the callback */
2385         else
2386         {
2387             rem_name.status = BTM_BAD_VALUE_RET;
2388             rem_name.length = 0;
2389             rem_name.remote_bd_name[0] = 0;
2390         }
2391         /* Reset the remote BAD to zero and call callback if possible */
2392         memset(p_inq->remname_bda, 0, BD_ADDR_LEN);
2393 
2394         p_inq->p_remname_cmpl_cb = NULL;
2395         if (p_cb)
2396             (p_cb)((tBTM_REMOTE_DEV_NAME *)&rem_name);
2397     }
2398 }
2399 
btm_inq_remote_name_timer_timeout(UNUSED_ATTR void * data)2400 void btm_inq_remote_name_timer_timeout(UNUSED_ATTR void *data)
2401 {
2402     btm_inq_rmt_name_failed();
2403 }
2404 
2405 /*******************************************************************************
2406 **
2407 ** Function         btm_inq_rmt_name_failed
2408 **
2409 ** Description      This function is if timeout expires while getting remote
2410 **                  name.  This is done for devices that incorrectly do not
2411 **                  report operation failure
2412 **
2413 ** Returns          void
2414 **
2415 *******************************************************************************/
btm_inq_rmt_name_failed(void)2416 void btm_inq_rmt_name_failed (void)
2417 {
2418     BTM_TRACE_ERROR ("btm_inq_rmt_name_failed()  remname_active=%d", btm_cb.btm_inq_vars.remname_active);
2419 
2420     if (btm_cb.btm_inq_vars.remname_active)
2421         btm_process_remote_name (btm_cb.btm_inq_vars.remname_bda, NULL, 0, HCI_ERR_UNSPECIFIED);
2422     else
2423         btm_process_remote_name (NULL, NULL, 0, HCI_ERR_UNSPECIFIED);
2424 
2425     btm_sec_rmt_name_request_complete (NULL, NULL, HCI_ERR_UNSPECIFIED);
2426 }
2427 
2428 /*******************************************************************************
2429 **
2430 ** Function         btm_read_inq_tx_power_timeout
2431 **
2432 ** Description      Callback when reading the inquiry tx power times out.
2433 **
2434 ** Returns          void
2435 **
2436 *******************************************************************************/
btm_read_inq_tx_power_timeout(UNUSED_ATTR void * data)2437 void btm_read_inq_tx_power_timeout(UNUSED_ATTR void *data)
2438 {
2439     tBTM_CMPL_CB  *p_cb = btm_cb.devcb.p_inq_tx_power_cmpl_cb;
2440     btm_cb.devcb.p_inq_tx_power_cmpl_cb = NULL;
2441     if (p_cb)
2442         (*p_cb)((void *) NULL);
2443 }
2444 
2445 /*******************************************************************************
2446 **
2447 ** Function         btm_read_inq_tx_power_complete
2448 **
2449 ** Description      read inquiry tx power level complete callback function.
2450 **
2451 ** Returns          void
2452 **
2453 *******************************************************************************/
btm_read_inq_tx_power_complete(UINT8 * p)2454 void btm_read_inq_tx_power_complete(UINT8 *p)
2455 {
2456     tBTM_CMPL_CB                *p_cb = btm_cb.devcb.p_inq_tx_power_cmpl_cb;
2457     tBTM_INQ_TXPWR_RESULTS        results;
2458 
2459     BTM_TRACE_DEBUG("%s", __func__);
2460     alarm_cancel(btm_cb.devcb.read_inq_tx_power_timer);
2461     btm_cb.devcb.p_inq_tx_power_cmpl_cb = NULL;
2462 
2463     /* If there was a registered callback, call it */
2464     if (p_cb)
2465     {
2466         STREAM_TO_UINT8  (results.hci_status, p);
2467 
2468         if (results.hci_status == HCI_SUCCESS)
2469         {
2470             results.status = BTM_SUCCESS;
2471 
2472             STREAM_TO_UINT8 (results.tx_power, p);
2473             BTM_TRACE_EVENT ("BTM INQ TX POWER Complete: tx_power %d, hci status 0x%02x",
2474                               results.tx_power, results.hci_status);
2475         }
2476         else
2477             results.status = BTM_ERR_PROCESSING;
2478 
2479         (*p_cb)(&results);
2480     }
2481 
2482 }
2483 /*******************************************************************************
2484 **
2485 ** Function         BTM_WriteEIR
2486 **
2487 ** Description      This function is called to write EIR data to controller.
2488 **
2489 ** Parameters       p_buff - allocated HCI command buffer including extended
2490 **                           inquriry response
2491 **
2492 ** Returns          BTM_SUCCESS  - if successful
2493 **                  BTM_MODE_UNSUPPORTED - if local device cannot support it
2494 **
2495 *******************************************************************************/
BTM_WriteEIR(BT_HDR * p_buff)2496 tBTM_STATUS BTM_WriteEIR( BT_HDR *p_buff )
2497 {
2498     if (controller_get_interface()->supports_extended_inquiry_response())
2499     {
2500         BTM_TRACE_API("Write Extended Inquiry Response to controller");
2501         btsnd_hcic_write_ext_inquiry_response (p_buff, BTM_EIR_DEFAULT_FEC_REQUIRED);
2502         return BTM_SUCCESS;
2503     }
2504     else
2505     {
2506         osi_free(p_buff);
2507         return BTM_MODE_UNSUPPORTED;
2508     }
2509 }
2510 
2511 /*******************************************************************************
2512 **
2513 ** Function         BTM_CheckEirData
2514 **
2515 ** Description      This function is called to get EIR data from significant part.
2516 **
2517 ** Parameters       p_eir - pointer of EIR significant part
2518 **                  type   - finding EIR data type
2519 **                  p_length - return the length of EIR data not including type
2520 **
2521 ** Returns          pointer of EIR data
2522 **
2523 *******************************************************************************/
BTM_CheckEirData(UINT8 * p_eir,UINT8 type,UINT8 * p_length)2524 UINT8 *BTM_CheckEirData( UINT8 *p_eir, UINT8 type, UINT8 *p_length )
2525 {
2526     UINT8 *p = p_eir;
2527     UINT8 length;
2528     UINT8 eir_type;
2529     BTM_TRACE_API("BTM_CheckEirData type=0x%02X", type);
2530 
2531     STREAM_TO_UINT8(length, p);
2532     while( length && (p - p_eir <= HCI_EXT_INQ_RESPONSE_LEN))
2533     {
2534         STREAM_TO_UINT8(eir_type, p);
2535         if( eir_type == type )
2536         {
2537             /* length doesn't include itself */
2538             *p_length = length - 1; /* minus the length of type */
2539             return p;
2540         }
2541         p += length - 1; /* skip the length of data */
2542         STREAM_TO_UINT8(length, p);
2543     }
2544 
2545     *p_length = 0;
2546     return NULL;
2547 }
2548 
2549 /*******************************************************************************
2550 **
2551 ** Function         btm_convert_uuid_to_eir_service
2552 **
2553 ** Description      This function is called to get the bit position of UUID.
2554 **
2555 ** Parameters       uuid16 - UUID 16-bit
2556 **
2557 ** Returns          BTM EIR service ID if found
2558 **                  BTM_EIR_MAX_SERVICES - if not found
2559 **
2560 *******************************************************************************/
btm_convert_uuid_to_eir_service(UINT16 uuid16)2561 static UINT8 btm_convert_uuid_to_eir_service( UINT16 uuid16 )
2562 {
2563     UINT8 xx;
2564 
2565     for( xx = 0; xx < BTM_EIR_MAX_SERVICES; xx++ )
2566     {
2567         if( uuid16 == BTM_EIR_UUID_LKUP_TBL[xx])
2568         {
2569             return xx;
2570         }
2571     }
2572     return BTM_EIR_MAX_SERVICES;
2573 }
2574 
2575 /*******************************************************************************
2576 **
2577 ** Function         BTM_HasEirService
2578 **
2579 ** Description      This function is called to know if UUID in bit map of UUID.
2580 **
2581 ** Parameters       p_eir_uuid - bit map of UUID list
2582 **                  uuid16 - UUID 16-bit
2583 **
2584 ** Returns          TRUE - if found
2585 **                  FALSE - if not found
2586 **
2587 *******************************************************************************/
BTM_HasEirService(UINT32 * p_eir_uuid,UINT16 uuid16)2588 BOOLEAN BTM_HasEirService( UINT32 *p_eir_uuid, UINT16 uuid16 )
2589 {
2590     UINT8 service_id;
2591 
2592     service_id = btm_convert_uuid_to_eir_service(uuid16);
2593     if( service_id < BTM_EIR_MAX_SERVICES )
2594         return( BTM_EIR_HAS_SERVICE( p_eir_uuid, service_id ));
2595     else
2596         return( FALSE );
2597 }
2598 
2599 /*******************************************************************************
2600 **
2601 ** Function         BTM_HasInquiryEirService
2602 **
2603 ** Description      This function is called to know if UUID in bit map of UUID list.
2604 **
2605 ** Parameters       p_results - inquiry results
2606 **                  uuid16 - UUID 16-bit
2607 **
2608 ** Returns          BTM_EIR_FOUND - if found
2609 **                  BTM_EIR_NOT_FOUND - if not found and it is complete list
2610 **                  BTM_EIR_UNKNOWN - if not found and it is not complete list
2611 **
2612 *******************************************************************************/
BTM_HasInquiryEirService(tBTM_INQ_RESULTS * p_results,UINT16 uuid16)2613 tBTM_EIR_SEARCH_RESULT BTM_HasInquiryEirService( tBTM_INQ_RESULTS *p_results, UINT16 uuid16 )
2614 {
2615     if( BTM_HasEirService( p_results->eir_uuid, uuid16 ))
2616     {
2617         return BTM_EIR_FOUND;
2618     }
2619     else if( p_results->eir_complete_list )
2620     {
2621         return BTM_EIR_NOT_FOUND;
2622     }
2623     else
2624         return BTM_EIR_UNKNOWN;
2625 }
2626 
2627 /*******************************************************************************
2628 **
2629 ** Function         BTM_AddEirService
2630 **
2631 ** Description      This function is called to add a service in bit map of UUID list.
2632 **
2633 ** Parameters       p_eir_uuid - bit mask of UUID list for EIR
2634 **                  uuid16 - UUID 16-bit
2635 **
2636 ** Returns          None
2637 **
2638 *******************************************************************************/
BTM_AddEirService(UINT32 * p_eir_uuid,UINT16 uuid16)2639 void BTM_AddEirService( UINT32 *p_eir_uuid, UINT16 uuid16 )
2640 {
2641     UINT8 service_id;
2642 
2643     service_id = btm_convert_uuid_to_eir_service(uuid16);
2644     if( service_id < BTM_EIR_MAX_SERVICES )
2645         BTM_EIR_SET_SERVICE( p_eir_uuid, service_id );
2646 }
2647 
2648 /*******************************************************************************
2649 **
2650 ** Function         BTM_RemoveEirService
2651 **
2652 ** Description      This function is called to remove a service in bit map of UUID list.
2653 **
2654 ** Parameters       p_eir_uuid - bit mask of UUID list for EIR
2655 **                  uuid16 - UUID 16-bit
2656 **
2657 ** Returns          None
2658 **
2659 *******************************************************************************/
BTM_RemoveEirService(UINT32 * p_eir_uuid,UINT16 uuid16)2660 void BTM_RemoveEirService( UINT32 *p_eir_uuid, UINT16 uuid16 )
2661 {
2662     UINT8 service_id;
2663 
2664     service_id = btm_convert_uuid_to_eir_service(uuid16);
2665     if( service_id < BTM_EIR_MAX_SERVICES )
2666         BTM_EIR_CLR_SERVICE( p_eir_uuid, service_id );
2667 }
2668 
2669 /*******************************************************************************
2670 **
2671 ** Function         BTM_GetEirSupportedServices
2672 **
2673 ** Description      This function is called to get UUID list from bit map of UUID list.
2674 **
2675 ** Parameters       p_eir_uuid - bit mask of UUID list for EIR
2676 **                  p - reference of current pointer of EIR
2677 **                  max_num_uuid16 - max number of UUID can be written in EIR
2678 **                  num_uuid16 - number of UUID have been written in EIR
2679 **
2680 ** Returns          BTM_EIR_MORE_16BITS_UUID_TYPE, if it has more than max
2681 **                  BTM_EIR_COMPLETE_16BITS_UUID_TYPE, otherwise
2682 **
2683 *******************************************************************************/
BTM_GetEirSupportedServices(UINT32 * p_eir_uuid,UINT8 ** p,UINT8 max_num_uuid16,UINT8 * p_num_uuid16)2684 UINT8 BTM_GetEirSupportedServices( UINT32 *p_eir_uuid,    UINT8 **p,
2685                                    UINT8  max_num_uuid16, UINT8 *p_num_uuid16)
2686 {
2687     UINT8 service_index;
2688 
2689     *p_num_uuid16 = 0;
2690 
2691     for(service_index = 0; service_index < BTM_EIR_MAX_SERVICES; service_index++)
2692     {
2693         if( BTM_EIR_HAS_SERVICE( p_eir_uuid, service_index ))
2694         {
2695             if( *p_num_uuid16 < max_num_uuid16 )
2696             {
2697                 UINT16_TO_STREAM(*p, BTM_EIR_UUID_LKUP_TBL[service_index]);
2698                 (*p_num_uuid16)++;
2699             }
2700             /* if max number of UUIDs are stored and found one more */
2701             else
2702             {
2703                 return BTM_EIR_MORE_16BITS_UUID_TYPE;
2704             }
2705         }
2706     }
2707     return BTM_EIR_COMPLETE_16BITS_UUID_TYPE;
2708 }
2709 
2710 /*******************************************************************************
2711 **
2712 ** Function         BTM_GetEirUuidList
2713 **
2714 ** Description      This function parses EIR and returns UUID list.
2715 **
2716 ** Parameters       p_eir - EIR
2717 **                  uuid_size - LEN_UUID_16, LEN_UUID_32, LEN_UUID_128
2718 **                  p_num_uuid - return number of UUID in found list
2719 **                  p_uuid_list - return UUID list
2720 **                  max_num_uuid - maximum number of UUID to be returned
2721 **
2722 ** Returns          0 - if not found
2723 **                  BTM_EIR_COMPLETE_16BITS_UUID_TYPE
2724 **                  BTM_EIR_MORE_16BITS_UUID_TYPE
2725 **                  BTM_EIR_COMPLETE_32BITS_UUID_TYPE
2726 **                  BTM_EIR_MORE_32BITS_UUID_TYPE
2727 **                  BTM_EIR_COMPLETE_128BITS_UUID_TYPE
2728 **                  BTM_EIR_MORE_128BITS_UUID_TYPE
2729 **
2730 *******************************************************************************/
BTM_GetEirUuidList(UINT8 * p_eir,UINT8 uuid_size,UINT8 * p_num_uuid,UINT8 * p_uuid_list,UINT8 max_num_uuid)2731 UINT8 BTM_GetEirUuidList( UINT8 *p_eir, UINT8 uuid_size, UINT8 *p_num_uuid,
2732                             UINT8 *p_uuid_list, UINT8 max_num_uuid)
2733 {
2734     UINT8   *p_uuid_data;
2735     UINT8   type;
2736     UINT8   yy, xx;
2737     UINT16  *p_uuid16 = (UINT16 *)p_uuid_list;
2738     UINT32  *p_uuid32 = (UINT32 *)p_uuid_list;
2739     char    buff[LEN_UUID_128 * 2 + 1];
2740 
2741     p_uuid_data = btm_eir_get_uuid_list( p_eir, uuid_size, p_num_uuid, &type );
2742     if( p_uuid_data == NULL )
2743     {
2744         return 0x00;
2745     }
2746 
2747     if( *p_num_uuid > max_num_uuid )
2748     {
2749         BTM_TRACE_WARNING("BTM_GetEirUuidList number of uuid in EIR = %d, size of uuid list = %d",
2750                            *p_num_uuid, max_num_uuid );
2751         *p_num_uuid = max_num_uuid;
2752     }
2753 
2754     BTM_TRACE_DEBUG("BTM_GetEirUuidList type = %02X, number of uuid = %d", type, *p_num_uuid );
2755 
2756     if( uuid_size == LEN_UUID_16 )
2757     {
2758         for( yy = 0; yy < *p_num_uuid; yy++ )
2759         {
2760             STREAM_TO_UINT16(*(p_uuid16 + yy), p_uuid_data);
2761             BTM_TRACE_DEBUG("                     0x%04X", *(p_uuid16 + yy));
2762         }
2763     }
2764     else if( uuid_size == LEN_UUID_32 )
2765     {
2766         for( yy = 0; yy < *p_num_uuid; yy++ )
2767         {
2768             STREAM_TO_UINT32(*(p_uuid32 + yy), p_uuid_data);
2769             BTM_TRACE_DEBUG("                     0x%08X", *(p_uuid32 + yy));
2770         }
2771     }
2772     else if( uuid_size == LEN_UUID_128 )
2773     {
2774         for( yy = 0; yy < *p_num_uuid; yy++ )
2775         {
2776             STREAM_TO_ARRAY16(p_uuid_list + yy * LEN_UUID_128, p_uuid_data);
2777             for( xx = 0; xx < LEN_UUID_128; xx++ )
2778                 sprintf(buff + xx*2, "%02X", *(p_uuid_list + yy * LEN_UUID_128 + xx));
2779             BTM_TRACE_DEBUG("                     0x%s", buff);
2780         }
2781     }
2782 
2783     return type;
2784 }
2785 
2786 
2787 /*******************************************************************************
2788 **
2789 ** Function         btm_eir_get_uuid_list
2790 **
2791 ** Description      This function searches UUID list in EIR.
2792 **
2793 ** Parameters       p_eir - address of EIR
2794 **                  uuid_size - size of UUID to find
2795 **                  p_num_uuid - number of UUIDs found
2796 **                  p_uuid_list_type - EIR data type
2797 **
2798 ** Returns          NULL - if UUID list with uuid_size is not found
2799 **                  beginning of UUID list in EIR - otherwise
2800 **
2801 *******************************************************************************/
btm_eir_get_uuid_list(UINT8 * p_eir,UINT8 uuid_size,UINT8 * p_num_uuid,UINT8 * p_uuid_list_type)2802 static UINT8 *btm_eir_get_uuid_list( UINT8 *p_eir, UINT8 uuid_size,
2803                                      UINT8 *p_num_uuid, UINT8 *p_uuid_list_type )
2804 {
2805     UINT8   *p_uuid_data;
2806     UINT8   complete_type, more_type;
2807     UINT8   uuid_len;
2808 
2809     switch( uuid_size )
2810     {
2811     case LEN_UUID_16:
2812         complete_type = BTM_EIR_COMPLETE_16BITS_UUID_TYPE;
2813         more_type     = BTM_EIR_MORE_16BITS_UUID_TYPE;
2814         break;
2815     case LEN_UUID_32:
2816         complete_type = BTM_EIR_COMPLETE_32BITS_UUID_TYPE;
2817         more_type     = BTM_EIR_MORE_32BITS_UUID_TYPE;
2818         break;
2819     case LEN_UUID_128:
2820         complete_type = BTM_EIR_COMPLETE_128BITS_UUID_TYPE;
2821         more_type     = BTM_EIR_MORE_128BITS_UUID_TYPE;
2822         break;
2823     default:
2824         *p_num_uuid = 0;
2825         return NULL;
2826         break;
2827     }
2828 
2829     p_uuid_data = BTM_CheckEirData( p_eir, complete_type, &uuid_len );
2830     if(p_uuid_data == NULL)
2831     {
2832         p_uuid_data = BTM_CheckEirData( p_eir, more_type, &uuid_len );
2833         *p_uuid_list_type = more_type;
2834     }
2835     else
2836     {
2837         *p_uuid_list_type = complete_type;
2838     }
2839 
2840     *p_num_uuid = uuid_len / uuid_size;
2841     return p_uuid_data;
2842 }
2843 
2844 /*******************************************************************************
2845 **
2846 ** Function         btm_convert_uuid_to_uuid16
2847 **
2848 ** Description      This function converts UUID to UUID 16-bit.
2849 **
2850 ** Parameters       p_uuid - address of UUID
2851 **                  uuid_size - size of UUID
2852 **
2853 ** Returns          0 - if UUID cannot be converted to UUID 16-bit
2854 **                  UUID 16-bit - otherwise
2855 **
2856 *******************************************************************************/
btm_convert_uuid_to_uuid16(UINT8 * p_uuid,UINT8 uuid_size)2857 static UINT16 btm_convert_uuid_to_uuid16( UINT8 *p_uuid, UINT8 uuid_size )
2858 {
2859     static const UINT8  base_uuid[LEN_UUID_128] = {0xFB, 0x34, 0x9B, 0x5F, 0x80, 0x00, 0x00, 0x80,
2860                                                    0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
2861     UINT16 uuid16 = 0;
2862     UINT32 uuid32;
2863     BOOLEAN is_base_uuid;
2864     UINT8  xx;
2865 
2866     switch (uuid_size)
2867     {
2868     case LEN_UUID_16:
2869         STREAM_TO_UINT16 (uuid16, p_uuid);
2870         break;
2871     case LEN_UUID_32:
2872         STREAM_TO_UINT32 (uuid32, p_uuid);
2873         if (uuid32 < 0x10000)
2874             uuid16 = (UINT16) uuid32;
2875         break;
2876     case LEN_UUID_128:
2877         /* See if we can compress his UUID down to 16 or 32bit UUIDs */
2878         is_base_uuid = TRUE;
2879         for (xx = 0; xx < LEN_UUID_128 - 4; xx++)
2880         {
2881             if (p_uuid[xx] != base_uuid[xx])
2882             {
2883                 is_base_uuid = FALSE;
2884                 break;
2885             }
2886         }
2887         if (is_base_uuid)
2888         {
2889             if ((p_uuid[LEN_UUID_128 - 1] == 0) && (p_uuid[LEN_UUID_128 - 2] == 0))
2890             {
2891                 p_uuid += (LEN_UUID_128 - 4);
2892                 STREAM_TO_UINT16(uuid16, p_uuid);
2893             }
2894         }
2895         break;
2896     default:
2897         BTM_TRACE_WARNING("btm_convert_uuid_to_uuid16 invalid uuid size");
2898         break;
2899     }
2900 
2901     return( uuid16);
2902 }
2903 
2904 /*******************************************************************************
2905 **
2906 ** Function         btm_set_eir_uuid
2907 **
2908 ** Description      This function is called to store received UUID into inquiry result.
2909 **
2910 ** Parameters       p_eir - pointer of EIR significant part
2911 **                  p_results - pointer of inquiry result
2912 **
2913 ** Returns          None
2914 **
2915 *******************************************************************************/
btm_set_eir_uuid(UINT8 * p_eir,tBTM_INQ_RESULTS * p_results)2916 void btm_set_eir_uuid( UINT8 *p_eir, tBTM_INQ_RESULTS *p_results )
2917 {
2918     UINT8   *p_uuid_data;
2919     UINT8   num_uuid;
2920     UINT16  uuid16;
2921     UINT8   yy;
2922     UINT8   type = BTM_EIR_MORE_16BITS_UUID_TYPE;
2923 
2924     p_uuid_data = btm_eir_get_uuid_list( p_eir, LEN_UUID_16, &num_uuid, &type );
2925 
2926     if(type == BTM_EIR_COMPLETE_16BITS_UUID_TYPE)
2927     {
2928         p_results->eir_complete_list = TRUE;
2929     }
2930     else
2931     {
2932         p_results->eir_complete_list = FALSE;
2933     }
2934 
2935     BTM_TRACE_API("btm_set_eir_uuid eir_complete_list=0x%02X", p_results->eir_complete_list);
2936 
2937     if( p_uuid_data )
2938     {
2939         for( yy = 0; yy < num_uuid; yy++ )
2940         {
2941             STREAM_TO_UINT16(uuid16, p_uuid_data);
2942             BTM_AddEirService( p_results->eir_uuid, uuid16 );
2943         }
2944     }
2945 
2946     p_uuid_data = btm_eir_get_uuid_list( p_eir, LEN_UUID_32, &num_uuid, &type );
2947     if( p_uuid_data )
2948     {
2949         for( yy = 0; yy < num_uuid; yy++ )
2950         {
2951             uuid16 = btm_convert_uuid_to_uuid16( p_uuid_data, LEN_UUID_32 );
2952             p_uuid_data += LEN_UUID_32;
2953             if( uuid16 )
2954                 BTM_AddEirService( p_results->eir_uuid, uuid16 );
2955         }
2956     }
2957 
2958     p_uuid_data = btm_eir_get_uuid_list( p_eir, LEN_UUID_128, &num_uuid, &type );
2959     if( p_uuid_data )
2960     {
2961         for( yy = 0; yy < num_uuid; yy++ )
2962         {
2963             uuid16 = btm_convert_uuid_to_uuid16( p_uuid_data, LEN_UUID_128 );
2964             p_uuid_data += LEN_UUID_128;
2965             if( uuid16 )
2966                 BTM_AddEirService( p_results->eir_uuid, uuid16 );
2967         }
2968     }
2969 }
2970