| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686 | /**************************************************************************************************   Phyplus Microelectronics Limited confidential and proprietary.   All rights reserved.  IMPORTANT: All rights of this software belong to Phyplus Microelectronics   Limited ("Phyplus"). Your use of this Software is limited to those   specific rights granted under  the terms of the business contract, the   confidential agreement, the non-disclosure agreement and any other forms   of agreements as a customer or a partner of Phyplus. You may not use this   Software unless you agree to abide by the terms of these agreements.   You acknowledge that the Software may not be modified, copied,   distributed or disclosed unless embedded on a Phyplus Bluetooth Low Energy   (BLE) integrated circuit, either as a product or is integrated into your   products.  Other than for the aforementioned purposes, you may not use,   reproduce, copy, prepare derivative works of, modify, distribute, perform,   display or sell this Software and/or its documentation for any purposes.  YOU FURTHER ACKNOWLEDGE AND AGREE THAT THE SOFTWARE AND DOCUMENTATION ARE  PROVIDED AS IS WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESS OR IMPLIED,  INCLUDING WITHOUT LIMITATION, ANY WARRANTY OF MERCHANTABILITY, TITLE,  NON-INFRINGEMENT AND FITNESS FOR A PARTICULAR PURPOSE. IN NO EVENT SHALL  PHYPLUS OR ITS SUBSIDIARIES BE LIABLE OR OBLIGATED UNDER CONTRACT,  NEGLIGENCE, STRICT LIABILITY, CONTRIBUTION, BREACH OF WARRANTY, OR OTHER  LEGAL EQUITABLE THEORY ANY DIRECT OR INDIRECT DAMAGES OR EXPENSES  INCLUDING BUT NOT LIMITED TO ANY INCIDENTAL, SPECIAL, INDIRECT, PUNITIVE  OR CONSEQUENTIAL DAMAGES, LOST PROFITS OR LOST DATA, COST OF PROCUREMENT  OF SUBSTITUTE GOODS, TECHNOLOGY, SERVICES, OR ANY CLAIMS BY THIRD PARTIES  (INCLUDING BUT NOT LIMITED TO ANY DEFENSE THEREOF), OR OTHER SIMILAR COSTS.  **************************************************************************************************//********************************************************************* * INCLUDES */#include "bcomdef.h"#include "OSAL.h"#include "osal_cbTimer.h"#include "osal_snv.h"#include "hci_tl.h"#include "l2cap.h"#include "linkdb.h"#include "gap.h"#include "gapbondmgr.h"#include "central.h"/********************************************************************* * MACROS *//********************************************************************* * CONSTANTS */// Profile Events#define START_ADVERTISING_EVT         0x0001#define RSSI_READ_EVT                 0x0002#define UPDATE_PARAMS_TIMEOUT_EVT     0x0004// Profile OSAL Message IDs#define GAPCENTRALROLE_RSSI_MSG_EVT   0xE0/********************************************************************* * TYPEDEFS */// RSSI read data structuretypedef struct{  uint16        period;  uint16        connHandle;  uint8         timerId;} gapCentralRoleRssi_t;// OSAL event structure for RSSI timer eventstypedef struct{  osal_event_hdr_t      hdr;  gapCentralRoleRssi_t  *pRssi;} gapCentralRoleRssiEvent_t;/********************************************************************* * GLOBAL VARIABLES *//********************************************************************* * EXTERNAL VARIABLES *//********************************************************************* * EXTERNAL FUNCTIONS *//********************************************************************* * LOCAL VARIABLES */// Task IDstatic uint8 gapCentralRoleTaskId;// App callbacksstatic gapCentralRoleCB_t *pGapCentralRoleCB;// Array of RSSI read structuresstatic gapCentralRoleRssi_t gapCentralRoleRssi[GAPCENTRALROLE_NUM_RSSI_LINKS];/********************************************************************* * Profile Parameters - reference GAPCENTRALROLE_PROFILE_PARAMETERS for * descriptions */static uint8  gapCentralRoleIRK[KEYLEN];static uint8  gapCentralRoleSRK[KEYLEN];static uint32 gapCentralRoleSignCounter;static uint8  gapCentralRoleBdAddr[B_ADDR_LEN];static uint8  gapCentralRoleMaxScanRes = 0;/********************************************************************* * LOCAL FUNCTIONS */static void gapCentralRole_ProcessOSALMsg( osal_event_hdr_t *pMsg );static void gapCentralRole_ProcessGAPMsg( gapEventHdr_t *pMsg );static gapCentralRoleRssi_t *gapCentralRole_RssiAlloc( uint16 connHandle );static gapCentralRoleRssi_t *gapCentralRole_RssiFind( uint16 connHandle );static void gapCentralRole_RssiFree( uint16 connHandle );static void gapCentralRole_timerCB( uint8 *pData );/********************************************************************* * PUBLIC FUNCTIONS *//** * @brief   Start the device in Central role.  This function is typically *          called once during system startup. * * Public function defined in central.h. */bStatus_t GAPCentralRole_StartDevice( gapCentralRoleCB_t *pAppCallbacks ){  if ( pAppCallbacks )  {    pGapCentralRoleCB = pAppCallbacks;  }  return GAP_DeviceInit( gapCentralRoleTaskId, GAP_PROFILE_CENTRAL,                         gapCentralRoleMaxScanRes, gapCentralRoleIRK,                         gapCentralRoleSRK, &gapCentralRoleSignCounter );}/** * @brief   Set a parameter in the Central Profile. * * Public function defined in central.h. */bStatus_t GAPCentralRole_SetParameter( uint16 param, uint8 len, void *pValue ){  bStatus_t ret = SUCCESS;  switch ( param )  {    case GAPCENTRALROLE_IRK:      if ( len == KEYLEN )      {        VOID osal_memcpy( gapCentralRoleIRK, pValue, KEYLEN ) ;      }      else      {        ret = bleInvalidRange;      }      break;    case GAPCENTRALROLE_SRK:      if ( len == KEYLEN )      {        VOID osal_memcpy( gapCentralRoleSRK, pValue, KEYLEN ) ;      }      else      {        ret = bleInvalidRange;      }      break;    case GAPCENTRALROLE_SIGNCOUNTER:      if ( len == sizeof ( uint32 ) )      {        gapCentralRoleSignCounter = *((uint32*)pValue);      }      else      {        ret = bleInvalidRange;      }      break;    case GAPCENTRALROLE_MAX_SCAN_RES:      if ( len == sizeof ( uint8 ) )      {        gapCentralRoleMaxScanRes = *((uint8*)pValue);      }      else      {        ret = bleInvalidRange;      }      break;    default:      ret = INVALIDPARAMETER;      break;  }  return ret;}/** * @brief   Get a parameter in the Central Profile. * * Public function defined in central.h. */bStatus_t GAPCentralRole_GetParameter( uint16 param, void *pValue ){  bStatus_t ret = SUCCESS;  switch ( param )  {    case GAPCENTRALROLE_IRK:      VOID osal_memcpy( pValue, gapCentralRoleIRK, KEYLEN ) ;      break;    case GAPCENTRALROLE_SRK:      VOID osal_memcpy( pValue, gapCentralRoleSRK, KEYLEN ) ;      break;    case GAPCENTRALROLE_SIGNCOUNTER:      *((uint32*)pValue) = gapCentralRoleSignCounter;      break;    case GAPCENTRALROLE_BD_ADDR:      VOID osal_memcpy( pValue, gapCentralRoleBdAddr, B_ADDR_LEN ) ;      break;    case GAPCENTRALROLE_MAX_SCAN_RES:      *((uint8*)pValue) = gapCentralRoleMaxScanRes;      break;    default:      ret = INVALIDPARAMETER;      break;  }  return ret;}/** * @brief   Terminate a link. * * Public function defined in central.h. */bStatus_t GAPCentralRole_TerminateLink( uint16 connHandle ){  return GAP_TerminateLinkReq( gapCentralRoleTaskId, connHandle, HCI_DISCONNECT_REMOTE_USER_TERM ) ;}/** * @brief   Establish a link to a peer device. * * Public function defined in central.h. */bStatus_t GAPCentralRole_EstablishLink( uint8 highDutyCycle, uint8 whiteList,                                        uint8 addrTypePeer, uint8 *peerAddr ){  gapEstLinkReq_t params;  params.taskID = gapCentralRoleTaskId;  params.highDutyCycle = highDutyCycle;  params.whiteList = whiteList;  params.addrTypePeer = addrTypePeer;  VOID osal_memcpy( params.peerAddr, peerAddr, B_ADDR_LEN );  return GAP_EstablishLinkReq( ¶ms );}/** * @brief   Update the link connection parameters. * * Public function defined in central.h. */bStatus_t GAPCentralRole_UpdateLink( uint16 connHandle, uint16 connIntervalMin,                                     uint16 connIntervalMax, uint16 connLatency,                                     uint16 connTimeout ){  return (bStatus_t) HCI_LE_ConnUpdateCmd( connHandle, connIntervalMin,                                            connIntervalMax, connLatency,                                            connTimeout, 0, 0 );}/** * @brief   Start a device discovery scan. * * Public function defined in central.h. */bStatus_t GAPCentralRole_StartDiscovery( uint8 mode, uint8 activeScan, uint8 whiteList ){  gapDevDiscReq_t params;  params.taskID = gapCentralRoleTaskId;  params.mode = mode;  params.activeScan = activeScan;  params.whiteList = whiteList;  return GAP_DeviceDiscoveryRequest( ¶ms );}/** * @brief   Cancel a device discovery scan. * * Public function defined in central.h. */bStatus_t GAPCentralRole_CancelDiscovery( void ){  return GAP_DeviceDiscoveryCancel( gapCentralRoleTaskId );}/** * @brief   Start periodic RSSI reads on a link. * * Public function defined in central.h. */bStatus_t GAPCentralRole_StartRssi( uint16 connHandle, uint16 period ){  gapCentralRoleRssi_t  *pRssi;  // Verify link is up  if (!linkDB_Up(connHandle))  {    return bleIncorrectMode;  }  // If already allocated  if ((pRssi = gapCentralRole_RssiFind( connHandle )) != NULL)  {    // Stop timer    osal_CbTimerStop( pRssi->timerId );  }  // Allocate structure  else if ((pRssi = gapCentralRole_RssiAlloc( connHandle )) != NULL)  {    pRssi->period = period;  }  // Allocate failed  else  {    return bleNoResources;  }  // Start timer  osal_CbTimerStart( gapCentralRole_timerCB, (uint8 *) pRssi,                     period, &pRssi->timerId );  return SUCCESS;}/** * @brief   Cancel periodic RSSI reads on a link. * * Public function defined in central.h. */bStatus_t GAPCentralRole_CancelRssi(uint16 connHandle ){  gapCentralRoleRssi_t  *pRssi;  if ((pRssi = gapCentralRole_RssiFind( connHandle )) != NULL)  {    // Stop timer    osal_CbTimerStop( pRssi->timerId );    // Free RSSI structure    gapCentralRole_RssiFree( connHandle );    return SUCCESS;  }  // Not found  return bleIncorrectMode;}/** * @brief   Central Profile Task initialization function. * * @param   taskId - Task ID. * * @return  void */void GAPCentralRole_Init( uint8 taskId ){  uint8 i;  gapCentralRoleTaskId = taskId;  // Initialize internal data  for ( i = 0; i < GAPCENTRALROLE_NUM_RSSI_LINKS; i++ )  {    gapCentralRoleRssi[i].connHandle = GAP_CONNHANDLE_ALL;    gapCentralRoleRssi[i].timerId = INVALID_TIMER_ID;  }  // Initialize parameters  // Retore items from NV//  VOID osal_snv_read( BLE_NVID_IRK, KEYLEN, gapCentralRoleIRK );//  VOID osal_snv_read( BLE_NVID_CSRK, KEYLEN, gapCentralRoleSRK );//  VOID osal_snv_read( BLE_NVID_SIGNCOUNTER, sizeof( uint32 ), &gapCentralRoleSignCounter );  // Register for HCI messages (for RSSI)  GAP_RegisterForHCIMsgs( taskId );}/** * @brief   Central Profile Task event processing function. * * @param   taskId - Task ID * @param   events - Events. * * @return  events not processed */uint16 GAPCentralRole_ProcessEvent( uint8 taskId, uint16 events ){  if ( events & SYS_EVENT_MSG )  {    uint8 *pMsg;    if ( (pMsg = osal_msg_receive( gapCentralRoleTaskId )) != NULL )    {      gapCentralRole_ProcessOSALMsg( (osal_event_hdr_t *) pMsg );      // Release the OSAL message      VOID osal_msg_deallocate( pMsg );    }    // return unprocessed events    return (events ^ SYS_EVENT_MSG);  }  if ( events & GAP_EVENT_SIGN_COUNTER_CHANGED )  {    // Sign counter changed, save it to NV  //  VOID osal_snv_write( BLE_NVID_SIGNCOUNTER, sizeof( uint32 ), &gapCentralRoleSignCounter );    return ( events ^ GAP_EVENT_SIGN_COUNTER_CHANGED );  }  // Discard unknown events  return 0;}/********************************************************************* * @fn      gapCentralRole_ProcessOSALMsg * * @brief   Process an incoming task message. * * @param   pMsg - message to process * * @return  none */static void gapCentralRole_ProcessOSALMsg( osal_event_hdr_t *pMsg ){  switch ( pMsg->event )  {    case HCI_GAP_EVENT_EVENT:      if ( pMsg->status == HCI_COMMAND_COMPLETE_EVENT_CODE )      {        hciEvt_CmdComplete_t *pPkt = (hciEvt_CmdComplete_t *) pMsg;        if ( pPkt->cmdOpcode == HCI_READ_RSSI )        {          uint16 connHandle = BUILD_UINT16( pPkt->pReturnParam[1], pPkt->pReturnParam[2] );          int8 rssi = (int8) pPkt->pReturnParam[3];          // Report RSSI to app          if ( pGapCentralRoleCB && pGapCentralRoleCB->rssiCB )          {            pGapCentralRoleCB->rssiCB( connHandle, rssi );          }        }      }      break;    case GAP_MSG_EVENT:      gapCentralRole_ProcessGAPMsg( (gapEventHdr_t *) pMsg );      break;    case GAPCENTRALROLE_RSSI_MSG_EVT:      {        gapCentralRoleRssi_t *pRssi = ((gapCentralRoleRssiEvent_t *) pMsg)->pRssi;        // If link is up and RSSI reads active        if (pRssi->connHandle != GAP_CONNHANDLE_ALL &&            linkDB_Up(pRssi->connHandle))        {          // Restart timer          osal_CbTimerStart( gapCentralRole_timerCB, (uint8 *) pRssi,                             pRssi->period, &pRssi->timerId );          // Read RSSI          VOID HCI_ReadRssiCmd( pRssi->connHandle );        }      }      break;    default:      break;  }}/********************************************************************* * @fn      gapCentralRole_ProcessGAPMsg * * @brief   Process an incoming task message from GAP. * * @param   pMsg - message to process * * @return  none */static void gapCentralRole_ProcessGAPMsg( gapEventHdr_t *pMsg ){  switch ( pMsg->opcode )  {    case GAP_DEVICE_INIT_DONE_EVENT:      {        gapDeviceInitDoneEvent_t *pPkt = (gapDeviceInitDoneEvent_t *) pMsg;        if ( pPkt->hdr.status == SUCCESS )        {          // Save off the generated keys         // VOID osal_snv_write( BLE_NVID_IRK, KEYLEN, gapCentralRoleIRK );         // VOID osal_snv_write( BLE_NVID_CSRK, KEYLEN, gapCentralRoleSRK );          // Save off the information          VOID osal_memcpy( gapCentralRoleBdAddr, pPkt->devAddr, B_ADDR_LEN );        }      }      break;    case GAP_LINK_ESTABLISHED_EVENT:      {        gapEstLinkReqEvent_t *pPkt = (gapEstLinkReqEvent_t *) pMsg;        if (pPkt->hdr.status == SUCCESS)        {          // Notify the Bond Manager of the connection          VOID GAPBondMgr_LinkEst( pPkt->devAddrType, pPkt->devAddr,                                   pPkt->connectionHandle, GAP_PROFILE_CENTRAL );        }      }      break;    case GAP_LINK_TERMINATED_EVENT:      {        uint16 connHandle = ((gapTerminateLinkEvent_t *) pMsg)->connectionHandle;        VOID GAPBondMgr_ProcessGAPMsg( (gapEventHdr_t *)pMsg );        // Cancel RSSI reads        GAPCentralRole_CancelRssi( connHandle );      }      break;    // temporary workaround    case GAP_SLAVE_REQUESTED_SECURITY_EVENT:      VOID GAPBondMgr_ProcessGAPMsg( pMsg );      break;    default:      break;  }  // Pass event to app  if ( pGapCentralRoleCB && pGapCentralRoleCB->eventCB )  {    pGapCentralRoleCB->eventCB( (gapCentralRoleEvent_t *) pMsg );  }}/********************************************************************* * @fn      gapCentralRole_RssiAlloc * * @brief   Allocate an RSSI structure. * * @param   connHandle - Connection handle * * @return  pointer to structure or NULL if allocation failed. */static gapCentralRoleRssi_t *gapCentralRole_RssiAlloc( uint16 connHandle ){  uint8 i;  // Find free RSSI structure  for ( i = 0; i < GAPCENTRALROLE_NUM_RSSI_LINKS; i++ )  {    if ( gapCentralRoleRssi[i].connHandle == GAP_CONNHANDLE_ALL )    {      gapCentralRoleRssi[i].connHandle = connHandle;      return &gapCentralRoleRssi[i];    }  }  // No free structure found  return NULL;}/********************************************************************* * @fn      gapCentralRole_RssiFind * * @brief   Find an RSSI structure. * * @param   connHandle - Connection handle * * @return  pointer to structure or NULL if not found. */static gapCentralRoleRssi_t *gapCentralRole_RssiFind( uint16 connHandle ){  uint8 i;  // Find free RSSI structure  for ( i = 0; i < GAPCENTRALROLE_NUM_RSSI_LINKS; i++ )  {    if ( gapCentralRoleRssi[i].connHandle == connHandle )    {      return &gapCentralRoleRssi[i];    }  }  // Not found  return NULL;}/********************************************************************* * @fn      gapCentralRole_RssiFree * * @brief   Free an RSSI structure. * * @param   connHandle - Connection handle * * @return  none */static void gapCentralRole_RssiFree( uint16 connHandle ){  uint8 i;  // Find RSSI structure  for ( i = 0; i < GAPCENTRALROLE_NUM_RSSI_LINKS; i++ )  {    if ( gapCentralRoleRssi[i].connHandle == connHandle )    {      gapCentralRoleRssi[i].connHandle = GAP_CONNHANDLE_ALL;      break;    }  }}/********************************************************************* * @fn      gapCentralRole_timerCB * * @brief   OSAL timer callback function * * @param   pData - Data pointer * * @return  none */static void gapCentralRole_timerCB( uint8 *pData ){  gapCentralRoleRssiEvent_t *pMsg;  // Timer has expired so clear timer ID  ((gapCentralRoleRssi_t *) pData)->timerId = INVALID_TIMER_ID;  // Send OSAL message  pMsg = (gapCentralRoleRssiEvent_t *) osal_msg_allocate( sizeof(gapCentralRoleRssiEvent_t) );  if ( pMsg )  {    pMsg->hdr.event = GAPCENTRALROLE_RSSI_MSG_EVT;    pMsg->pRssi = (gapCentralRoleRssi_t *) pData;    osal_msg_send ( gapCentralRoleTaskId, (uint8 *) pMsg );  }}/******************************************************************************************************************************************/
 |