Home | History | Annotate | Download | only in libloc_api
      1 /******************************************************************************
      2   @file:  loc_eng_ni.cpp
      3   @brief:  module for network initiated interactions
      4 
      5   DESCRIPTION
      6      LOC_API network initiated operation support
      7 
      8   INITIALIZATION AND SEQUENCING REQUIREMENTS
      9 
     10   -----------------------------------------------------------------------------
     11   Copyright (c) 2009 QUALCOMM Incorporated.
     12   All Rights Reserved. QUALCOMM Proprietary and Confidential.
     13   -----------------------------------------------------------------------------
     14 ******************************************************************************/
     15 
     16 /*=====================================================================
     17                         EDIT HISTORY FOR MODULE
     18 
     19   This section contains comments describing changes made to the module.
     20   Notice that changes are listed in reverse chronological order.
     21 
     22 when       who      what, where, why
     23 --------   ---      -------------------------------------------------------
     24 07/30/09   dx       Initial version
     25 
     26 $Id:
     27 ======================================================================*/
     28 
     29 #define LOG_NDDEBUG 0
     30 #define LOG_NIDEBUG 0
     31 
     32 #include <stdio.h>
     33 #include <stdlib.h>
     34 #include <pthread.h>
     35 #include <string.h>
     36 #include <ctype.h>
     37 #include <unistd.h>
     38 #include <time.h>
     39 
     40 #include <rpc/rpc.h>
     41 #include <loc_api_rpc_glue.h>
     42 #include <loc_eng.h>
     43 #include <loc_eng_ni.h>
     44 
     45 #define LOG_TAG "lib_locapi"
     46 #include <utils/Log.h>
     47 
     48 // comment this out to enable logging
     49 // #undef ALOGD
     50 // #define ALOGD(...) {}
     51 
     52 /*=============================================================================
     53  *
     54  *                             DATA DECLARATION
     55  *
     56  *============================================================================*/
     57 
     58 const GpsNiInterface sLocEngNiInterface =
     59 {
     60     sizeof(GpsNiInterface),
     61     loc_eng_ni_init,
     62     loc_eng_ni_respond,
     63 };
     64 
     65 boolean loc_eng_ni_data_init = FALSE;
     66 loc_eng_ni_data_s_type loc_eng_ni_data;
     67 
     68 extern loc_eng_data_s_type loc_eng_data;
     69 
     70 /*=============================================================================
     71  *
     72  *                             FUNCTION DECLARATIONS
     73  *
     74  *============================================================================*/
     75 
     76 /*===========================================================================
     77 
     78 FUNCTION respond_from_enum
     79 
     80 DESCRIPTION
     81    Returns the name of the response
     82 
     83 RETURN VALUE
     84    response name string
     85 
     86 ===========================================================================*/
     87 static const char* respond_from_enum(rpc_loc_ni_user_resp_e_type resp)
     88 {
     89     switch (resp)
     90     {
     91     case RPC_LOC_NI_LCS_NOTIFY_VERIFY_ACCEPT:
     92         return "accept";
     93     case RPC_LOC_NI_LCS_NOTIFY_VERIFY_DENY:
     94         return "deny";
     95     case RPC_LOC_NI_LCS_NOTIFY_VERIFY_NORESP:
     96         return "no response";
     97     default:
     98         return NULL;
     99     }
    100 }
    101 
    102 /*===========================================================================
    103 
    104 FUNCTION loc_ni_respond
    105 
    106 DESCRIPTION
    107    Displays the NI request and awaits user input. If a previous request is
    108    in session, the new one is handled using sys.ni_default_response (if exists);
    109    otherwise, it is denied.
    110 
    111 DEPENDENCY
    112    Do not lock the data by mutex loc_ni_lock
    113 
    114 RETURN VALUE
    115    none
    116 
    117 ===========================================================================*/
    118 static void loc_ni_respond(rpc_loc_ni_user_resp_e_type resp,
    119                     const rpc_loc_ni_event_s_type *request_pass_back
    120 )
    121 {
    122     ALOGD("Sending NI response: %s\n", respond_from_enum(resp));
    123 
    124     rpc_loc_ioctl_data_u_type data;
    125     rpc_loc_ioctl_callback_s_type callback_payload;
    126 
    127     memcpy(&data.rpc_loc_ioctl_data_u_type_u.user_verify_resp.ni_event_pass_back,
    128             request_pass_back, sizeof (rpc_loc_ni_event_s_type));
    129     data.rpc_loc_ioctl_data_u_type_u.user_verify_resp.user_resp = resp;
    130 
    131     loc_eng_ioctl(
    132             loc_eng_data.client_handle,
    133             RPC_LOC_IOCTL_INFORM_NI_USER_RESPONSE,
    134             &data,
    135             LOC_IOCTL_DEFAULT_TIMEOUT,
    136             &callback_payload
    137     );
    138 }
    139 
    140 /*===========================================================================
    141 
    142 FUNCTION loc_ni_fill_notif_verify_type
    143 
    144 DESCRIPTION
    145    Fills need_notify, need_verify, etc.
    146 
    147 RETURN VALUE
    148    none
    149 
    150 ===========================================================================*/
    151 static boolean loc_ni_fill_notif_verify_type(GpsNiNotification *notif,
    152       rpc_loc_ni_notify_verify_e_type notif_priv)
    153 {
    154     notif->notify_flags       = 0;
    155     notif->default_response   = GPS_NI_RESPONSE_NORESP;
    156 
    157     switch (notif_priv)
    158     {
    159     case RPC_LOC_NI_USER_NO_NOTIFY_NO_VERIFY:
    160         notif->notify_flags = 0;
    161         break;
    162 
    163     case RPC_LOC_NI_USER_NOTIFY_ONLY:
    164         notif->notify_flags = GPS_NI_NEED_NOTIFY;
    165         break;
    166 
    167     case RPC_LOC_NI_USER_NOTIFY_VERIFY_ALLOW_NO_RESP:
    168         notif->notify_flags = GPS_NI_NEED_NOTIFY | GPS_NI_NEED_VERIFY;
    169         notif->default_response = GPS_NI_RESPONSE_ACCEPT;
    170         break;
    171 
    172     case RPC_LOC_NI_USER_NOTIFY_VERIFY_NOT_ALLOW_NO_RESP:
    173         notif->notify_flags = GPS_NI_NEED_NOTIFY | GPS_NI_NEED_VERIFY;
    174         notif->default_response = GPS_NI_RESPONSE_DENY;
    175         break;
    176 
    177     case RPC_LOC_NI_USER_PRIVACY_OVERRIDE:
    178         notif->notify_flags = GPS_NI_PRIVACY_OVERRIDE;
    179         break;
    180 
    181     default:
    182         return FALSE;
    183     }
    184 
    185     return TRUE;
    186 }
    187 
    188 /*===========================================================================
    189 
    190 FUNCTION hexcode
    191 
    192 DESCRIPTION
    193    Converts a binary array into a Hex string. E.g., 1F 00 3F --> "1F003F"
    194 
    195 RETURN VALUE
    196    bytes encoded
    197 
    198 ===========================================================================*/
    199 static int hexcode(char *hexstring, int string_size, const char *data, int data_size)
    200 {
    201     int i;
    202     for (i = 0; i < data_size; i++)
    203     {
    204         char ch = data[i];
    205         if (i*2 + 3 <= string_size)
    206         {
    207             snprintf(&hexstring[i*2], 3, "%02X", ch);
    208         }
    209         else {
    210             break;
    211         }
    212     }
    213     return i;
    214 }
    215 
    216 static GpsNiEncodingType convert_encoding_type(int loc_encoding)
    217 {
    218     GpsNiEncodingType enc = GPS_ENC_UNKNOWN;
    219 
    220     switch (loc_encoding)
    221     {
    222     case RPC_LOC_NI_SUPL_UTF8:
    223         enc = GPS_ENC_SUPL_UTF8;
    224         break;
    225     case RPC_LOC_NI_SUPL_UCS2:
    226         enc = GPS_ENC_SUPL_UCS2;
    227         break;
    228     case RPC_LOC_NI_SUPL_GSM_DEFAULT:
    229         enc = GPS_ENC_SUPL_GSM_DEFAULT;
    230         break;
    231     default:
    232         break;
    233     }
    234 
    235     return enc;
    236 }
    237 
    238 /*===========================================================================
    239 
    240 FUNCTION loc_ni_request_handler
    241 
    242 DESCRIPTION
    243    Displays the NI request and awaits user input. If a previous request is
    244    in session, it is ignored.
    245 
    246 RETURN VALUE
    247    none
    248 
    249 ===========================================================================*/
    250 static void loc_ni_request_handler(const char *msg, const rpc_loc_ni_event_s_type *ni_req)
    251 {
    252     GpsNiNotification notif;
    253     notif.size = sizeof(notif);
    254     strlcpy(notif.text, "[text]", sizeof notif.text);    // defaults
    255     strlcpy(notif.requestor_id, "[requestor id]", sizeof notif.requestor_id);
    256 
    257     /* If busy, use default or deny */
    258     if (loc_eng_ni_data.notif_in_progress)
    259     {
    260 #if 0
    261         /* Cannot be here because the current thread is in RPC client */
    262         /* XXX Consider adding an event queue to process overlapped NI requests */
    263         loc_ni_user_resp_e_type response =
    264             sys.ni_default_resp == 1 /* accept */ ?
    265                     LOC_NI_LCS_NOTIFY_VERIFY_ACCEPT :
    266                     LOC_NI_LCS_NOTIFY_VERIFY_DENY;
    267 
    268         loc_ni_respond(response, ni_req); */
    269 #endif
    270         ALOGW("loc_ni_request_handler, notification in progress, new NI request ignored, type: %d",
    271                 ni_req->event);
    272     }
    273     else {
    274         /* Print notification */
    275         ALOGD("NI Notification: %s, event: %d", msg, ni_req->event);
    276 
    277         pthread_mutex_lock(&loc_eng_ni_data.loc_ni_lock);
    278 
    279         /* Save request */
    280         memcpy(&loc_eng_ni_data.loc_ni_request, ni_req, sizeof loc_eng_ni_data.loc_ni_request);
    281 
    282         /* Set up NI response waiting */
    283         loc_eng_ni_data.notif_in_progress = TRUE;
    284         loc_eng_ni_data.current_notif_id = abs(rand());
    285 
    286         /* Fill in notification */
    287         notif.notification_id = loc_eng_ni_data.current_notif_id;
    288 
    289         const rpc_loc_ni_vx_notify_verify_req_s_type *vx_req;
    290         const rpc_loc_ni_supl_notify_verify_req_s_type *supl_req;
    291         const rpc_loc_ni_umts_cp_notify_verify_req_s_type *umts_cp_req;
    292 
    293         switch (ni_req->event)
    294         {
    295             case RPC_LOC_NI_EVENT_VX_NOTIFY_VERIFY_REQ:
    296                 vx_req = &ni_req->payload.rpc_loc_ni_event_payload_u_type_u.vx_req;
    297                 notif.ni_type     = GPS_NI_TYPE_VOICE;
    298                 notif.timeout     = LOC_NI_NO_RESPONSE_TIME; // vx_req->user_resp_timer_val;
    299                 memset(notif.extras, 0, sizeof notif.extras);
    300                 memset(notif.text, 0, sizeof notif.text);
    301                 memset(notif.requestor_id, 0, sizeof notif.requestor_id);
    302 
    303                 // Requestor ID
    304                 hexcode(notif.requestor_id, sizeof notif.requestor_id,
    305                         vx_req->requester_id.requester_id,
    306                         vx_req->requester_id.requester_id_length);
    307 
    308                 notif.text_encoding = 0; // No text and no encoding
    309                 notif.requestor_id_encoding = convert_encoding_type(vx_req->encoding_scheme);
    310 
    311                 // Set default_response & notify_flags
    312                 loc_ni_fill_notif_verify_type(&notif, vx_req->notification_priv_type);
    313 
    314                 break;
    315 
    316             case RPC_LOC_NI_EVENT_UMTS_CP_NOTIFY_VERIFY_REQ:
    317                 umts_cp_req = &ni_req->payload.rpc_loc_ni_event_payload_u_type_u.umts_cp_req;
    318                 notif.ni_type     = GPS_NI_TYPE_UMTS_CTRL_PLANE;
    319                 notif.timeout     = LOC_NI_NO_RESPONSE_TIME; // umts_cp_req->user_response_timer;
    320                 memset(notif.extras, 0, sizeof notif.extras);
    321                 memset(notif.text, 0, sizeof notif.text);
    322                 memset(notif.requestor_id, 0, sizeof notif.requestor_id);
    323 
    324                 // Stores notification text
    325                 hexcode(notif.text, sizeof notif.text,
    326 #if (AMSS_VERSION==3200)
    327                         umts_cp_req->notification_text.notification_text_val,
    328 #else
    329                         umts_cp_req->notification_text,
    330 #endif
    331                         umts_cp_req->notification_length);
    332 
    333                 // Stores requestor ID
    334                 hexcode(notif.requestor_id, sizeof notif.requestor_id,
    335 #if (AMSS_VERSION==3200)
    336                         umts_cp_req->requestor_id.requestor_id_string.requestor_id_string_val,
    337 #else
    338                         umts_cp_req->requestor_id.requestor_id_string,
    339 #endif
    340                         umts_cp_req->requestor_id.string_len);
    341 
    342                 notif.text_encoding = convert_encoding_type(umts_cp_req->datacoding_scheme);
    343                 notif.requestor_id_encoding = convert_encoding_type(umts_cp_req->datacoding_scheme);
    344 
    345                 // Set default_response & notify_flags
    346                 loc_ni_fill_notif_verify_type(&notif, umts_cp_req->notification_priv_type);
    347 
    348                 break;
    349 
    350             case RPC_LOC_NI_EVENT_SUPL_NOTIFY_VERIFY_REQ:
    351                 supl_req = &ni_req->payload.rpc_loc_ni_event_payload_u_type_u.supl_req;
    352                 notif.ni_type     = GPS_NI_TYPE_UMTS_SUPL;
    353                 notif.timeout     = LOC_NI_NO_RESPONSE_TIME; // supl_req->user_response_timer;
    354                 memset(notif.extras, 0, sizeof notif.extras);
    355                 memset(notif.text, 0, sizeof notif.text);
    356                 memset(notif.requestor_id, 0, sizeof notif.requestor_id);
    357 
    358                 // Client name
    359                 if (supl_req->flags & RPC_LOC_NI_CLIENT_NAME_PRESENT)
    360                 {
    361                     hexcode(notif.text, sizeof notif.text,
    362 #if (AMSS_VERSION==3200)
    363                             supl_req->client_name.client_name_string.client_name_string_val,   /* buffer */
    364 #else
    365                             supl_req->client_name.client_name_string,                          /* buffer */
    366 #endif
    367                             supl_req->client_name.string_len                                   /* length */
    368                     );
    369                     ALOGD("SUPL NI: client_name: %s len=%d", notif.text, supl_req->client_name.string_len);
    370                 } else {
    371                     ALOGD("SUPL NI: client_name not present.");
    372                 }
    373 
    374                 // Requestor ID
    375                 if (supl_req->flags & RPC_LOC_NI_REQUESTOR_ID_PRESENT)
    376                 {
    377                     hexcode(notif.requestor_id, sizeof notif.requestor_id,
    378 #if (AMSS_VERSION==3200)
    379                             supl_req->requestor_id.requestor_id_string.requestor_id_string_val,  /* buffer */
    380 #else
    381                             supl_req->requestor_id.requestor_id_string,                          /* buffer */
    382 #endif
    383                             supl_req->requestor_id.string_len                                    /* length */
    384                     );
    385                     ALOGD("SUPL NI: requestor_id: %s len=%d", notif.requestor_id, supl_req->requestor_id.string_len);
    386                 } else {
    387                     ALOGD("SUPL NI: requestor_id not present.");
    388                 }
    389 
    390                 // Encoding type
    391                 if (supl_req->flags & RPC_LOC_NI_ENCODING_TYPE_PRESENT)
    392                 {
    393                     notif.text_encoding = convert_encoding_type(supl_req->datacoding_scheme);
    394                     notif.requestor_id_encoding = convert_encoding_type(supl_req->datacoding_scheme);
    395                 } else {
    396                     notif.text_encoding = notif.requestor_id_encoding = GPS_ENC_UNKNOWN;
    397                 }
    398 
    399                 // Set default_response & notify_flags
    400                 loc_ni_fill_notif_verify_type(&notif, ni_req->payload.rpc_loc_ni_event_payload_u_type_u.supl_req.notification_priv_type);
    401 
    402                 break;
    403 
    404             default:
    405                 ALOGE("loc_ni_request_handler, unknown request event: %d", ni_req->event);
    406                 return;
    407         }
    408 
    409         /* Log requestor ID and text for debugging */
    410         ALOGI("Notification: notif_type: %d, timeout: %d, default_resp: %d", notif.ni_type, notif.timeout, notif.default_response);
    411         ALOGI("              requestor_id: %s (encoding: %d)", notif.requestor_id, notif.requestor_id_encoding);
    412         ALOGI("              text: %s text (encoding: %d)", notif.text, notif.text_encoding);
    413 
    414         /* For robustness, always sets a timeout to clear up the notification status, even though
    415         * the OEM layer in java does not do so.
    416         **/
    417         loc_eng_ni_data.response_time_left = 5 + (notif.timeout != 0 ? notif.timeout : LOC_NI_NO_RESPONSE_TIME);
    418         ALOGI("Automatically sends 'no response' in %d seconds (to clear status)\n", loc_eng_ni_data.response_time_left);
    419 
    420         pthread_mutex_unlock(&loc_eng_ni_data.loc_ni_lock);
    421 
    422         /* Notify callback */
    423         if (loc_eng_data.ni_notify_cb != NULL)
    424         {
    425             loc_eng_data.ni_notify_cb(&notif);
    426         }
    427     }
    428 }
    429 
    430 /*===========================================================================
    431 
    432 FUNCTION loc_ni_process_user_response
    433 
    434 DESCRIPTION
    435    Handles user input from the UI
    436 
    437 RETURN VALUE
    438    error code (0 for successful, -1 for error)
    439 
    440 ===========================================================================*/
    441 int loc_ni_process_user_response(GpsUserResponseType userResponse)
    442 {
    443     ALOGD("NI response from UI: %d", userResponse);
    444 
    445     rpc_loc_ni_user_resp_e_type resp;
    446     switch (userResponse)
    447     {
    448         case GPS_NI_RESPONSE_ACCEPT:
    449             resp = RPC_LOC_NI_LCS_NOTIFY_VERIFY_ACCEPT;
    450             break;
    451         case GPS_NI_RESPONSE_DENY:
    452             resp = RPC_LOC_NI_LCS_NOTIFY_VERIFY_DENY;
    453             break;
    454         case GPS_NI_RESPONSE_NORESP:
    455             resp = RPC_LOC_NI_LCS_NOTIFY_VERIFY_NORESP;
    456             break;
    457         default:
    458             return -1;
    459     }
    460 
    461     loc_ni_respond(resp, &loc_eng_ni_data.loc_ni_request);
    462 
    463     /* Make the NI respond */
    464     pthread_mutex_lock(&loc_eng_ni_data.loc_ni_lock);
    465     loc_eng_ni_data.notif_in_progress = FALSE;
    466     loc_eng_ni_data.response_time_left = 0;
    467     loc_eng_ni_data.current_notif_id = -1;
    468     pthread_mutex_unlock(&loc_eng_ni_data.loc_ni_lock);
    469 
    470     return 0;
    471 }
    472 
    473 /*===========================================================================
    474 
    475 FUNCTION loc_eng_ni_callback
    476 
    477 DESCRIPTION
    478    Loc API callback handler
    479 
    480 RETURN VALUE
    481    error code (0 for success)
    482 
    483 ===========================================================================*/
    484 int loc_eng_ni_callback (
    485       rpc_loc_event_mask_type               loc_event,              /* event mask           */
    486       const rpc_loc_event_payload_u_type*   loc_event_payload       /* payload              */
    487 )
    488 {
    489     int rc = 0;
    490     const rpc_loc_ni_event_s_type *ni_req = &loc_event_payload->rpc_loc_event_payload_u_type_u.ni_request;
    491     if (loc_event == RPC_LOC_EVENT_NI_NOTIFY_VERIFY_REQUEST)
    492     {
    493         switch (ni_req->event)
    494         {
    495             case RPC_LOC_NI_EVENT_VX_NOTIFY_VERIFY_REQ:
    496                 ALOGI("VX Notification");
    497                 loc_ni_request_handler("VX Notify", ni_req);
    498                 break;
    499 
    500             case RPC_LOC_NI_EVENT_UMTS_CP_NOTIFY_VERIFY_REQ:
    501                 ALOGI("UMTS CP Notification\n");
    502                 loc_ni_request_handler("UMTS CP Notify", ni_req);
    503                 break;
    504 
    505             case RPC_LOC_NI_EVENT_SUPL_NOTIFY_VERIFY_REQ:
    506                 ALOGI("SUPL Notification\n");
    507                 loc_ni_request_handler("SUPL Notify", ni_req);
    508                 break;
    509 
    510             default:
    511                 ALOGE("Unknown NI event: %x\n", (int) ni_req->event);
    512                 break;
    513         }
    514     }
    515     return rc;
    516 }
    517 
    518 /*===========================================================================
    519 
    520 FUNCTION loc_ni_thread_proc
    521 
    522 ===========================================================================*/
    523 static void loc_ni_thread_proc(void *unused)
    524 {
    525     ALOGI("Starting Loc NI thread...\n");
    526 
    527     while (1)
    528     {
    529         /* wakes up every second to check timed out requests */
    530         sleep(1);
    531 
    532         pthread_mutex_lock(&loc_eng_ni_data.loc_ni_lock);
    533 
    534         if (loc_eng_ni_data.notif_in_progress && loc_eng_ni_data.response_time_left > 0)
    535         {
    536             loc_eng_ni_data.response_time_left--;
    537             if (loc_eng_ni_data.response_time_left <= 0)
    538             {
    539                 loc_ni_respond(RPC_LOC_NI_LCS_NOTIFY_VERIFY_NORESP, &loc_eng_ni_data.loc_ni_request);
    540                 loc_eng_ni_data.notif_in_progress = FALSE;
    541             }
    542         }
    543 
    544         pthread_mutex_unlock(&loc_eng_ni_data.loc_ni_lock);
    545     } /* while (1) */
    546 }
    547 
    548 /*===========================================================================
    549 FUNCTION    loc_eng_ni_init
    550 
    551 DESCRIPTION
    552    This function initializes the NI interface
    553 
    554 DEPENDENCIES
    555    NONE
    556 
    557 RETURN VALUE
    558    None
    559 
    560 SIDE EFFECTS
    561    N/A
    562 
    563 ===========================================================================*/
    564 void loc_eng_ni_init(GpsNiCallbacks *callbacks)
    565 {
    566     ALOGD("loc_eng_ni_init: entered.");
    567 
    568     if (!loc_eng_ni_data_init)
    569     {
    570         pthread_mutex_init(&loc_eng_ni_data.loc_ni_lock, NULL);
    571         callbacks->create_thread_cb("loc_api_ni", loc_ni_thread_proc, NULL);
    572         loc_eng_ni_data_init = TRUE;
    573     }
    574 
    575     loc_eng_ni_data.notif_in_progress = FALSE;
    576     loc_eng_ni_data.current_notif_id = -1;
    577     loc_eng_ni_data.response_time_left = 0;
    578 
    579     srand(time(NULL));
    580     loc_eng_data.ni_notify_cb = callbacks->notify_cb;
    581 }
    582 
    583 /*===========================================================================
    584 FUNCTION    loc_eng_ni_respond
    585 
    586 DESCRIPTION
    587    This function sends an NI respond to the modem processor
    588 
    589 DEPENDENCIES
    590    NONE
    591 
    592 RETURN VALUE
    593    None
    594 
    595 SIDE EFFECTS
    596    N/A
    597 
    598 ===========================================================================*/
    599 void loc_eng_ni_respond(int notif_id, GpsUserResponseType user_response)
    600 {
    601     if (notif_id == loc_eng_ni_data.current_notif_id && loc_eng_ni_data.notif_in_progress)
    602     {
    603         ALOGI("loc_eng_ni_respond: send user response %d for notif %d", user_response, notif_id);
    604         loc_ni_process_user_response(user_response);
    605     } else {
    606         ALOGE("loc_eng_ni_respond: notif_id %d mismatch or notification not in progress, response: %d",
    607             notif_id, user_response);
    608     }
    609 }
    610