Skip to content
Snippets Groups Projects
dats_main.c 24.2 KiB
Newer Older
  • Learn to ignore specific revisions
  • /*************************************************************************************************/
    /*!
     *  \file
     *
     *  \brief  Data transmitter sample application.
     *
     *  Copyright (c) 2012-2018 Arm Ltd. All Rights Reserved.
     *  ARM Ltd. confidential and proprietary.
     *
     *  IMPORTANT.  Your use of this file is governed by a Software License Agreement
     *  ("Agreement") that must be accepted in order to download or otherwise receive a
     *  copy of this file.  You may not use or copy this file for any purpose other than
     *  as described in the Agreement.  If you do not agree to all of the terms of the
     *  Agreement do not use this file and delete all copies in your possession or control;
     *  if you do not have a copy of the Agreement, you must contact ARM Ltd. prior
     *  to any use, copying or further distribution of this software.
     */
    /*************************************************************************************************/
    
    #include <stdio.h>
    #include <string.h>
    #include "wsf_types.h"
    #include "util/bstream.h"
    #include "wsf_msg.h"
    #include "wsf_trace.h"
    #include "wsf_buf.h"
    #include "wsf_assert.h"
    #include "hci_api.h"
    
    #include "hci_vs.h"
    
    #include "sec_api.h"
    #include "dm_api.h"
    #include "smp_api.h"
    #include "att_api.h"
    #include "app_api.h"
    #include "app_main.h"
    #include "app_db.h"
    #include "app_ui.h"
    #include "svc_ch.h"
    #include "svc_core.h"
    #include "svc_wp.h"
    #include "util/calc128.h"
    #include "dats/dats_api.h"
    
    /**************************************************************************************************
      Macros
    **************************************************************************************************/
    
    /*! Enumeration of client characteristic configuration descriptors */
    enum
    {
      DATS_GATT_SC_CCC_IDX,           /*! GATT service, service changed characteristic */
      DATS_WP_DAT_CCC_IDX,            /*! ARM Ltd. proprietary service, data transfer characteristic */
      DATS_NUM_CCC_IDX
    };
    
    /**************************************************************************************************
      Configurable Parameters
    **************************************************************************************************/
    
    /*! configurable parameters for advertising */
    /* These intervals directly impact energy usage during the non-connected/advertising mode */
    
    #ifdef BTLE_APP_USE_LEGACY_API
    
    static const appAdvCfg_t datsAdvCfg =
    {
      { 1000,     0,     0},                  /*! Advertising durations in ms */
      {   96,   200,     0}                   /*! Advertising intervals in 0.625 ms units */
    };
    
    #else /* BTLE_APP_USE_LEGACY_API */
    static const appExtAdvCfg_t datsExtAdvCfg =
    {
      {    0,     0,     0}, /*! Advertising durations for extended advertising in ms */
      {   96,   200,     0}, /*! Advertising intervals for extended advertising in 0.625 ms units (20 ms to 10.24 s). */
      {    0,     0,     0}, /*! Maximum number of extended advertising events controller will send prior to terminating extended advertising */
      {    0,     0,     0}, /*! Whether to use legacy advertising PDUs with extended advertising. If set to TRUE then length of advertising data cannot exceed 31 octets. */
      {    6,     0,     0}  /*! Advertising intervals for periodic advertising in 1.25 ms units (7.5 ms to 81.91875 s). */
    };
    
    #endif /* BTLE_APP_USE_LEGACY_API */
    
    
    /*! configurable parameters for slave */
    static const appSlaveCfg_t datsSlaveCfg =
    {
      1,                                      /*! Maximum connections */
    };
    
    /*! configurable parameters for security */
    static const appSecCfg_t datsSecCfg =
    {
      DM_AUTH_BOND_FLAG,                                         /*! Authentication and bonding flags */
      DM_KEY_DIST_LTK | DM_KEY_DIST_IRK | DM_KEY_DIST_CSRK,      /*! Initiator key distribution flags */
      DM_KEY_DIST_LTK | DM_KEY_DIST_IRK | DM_KEY_DIST_CSRK,      /*! Responder key distribution flags */
      FALSE,                                                     /*! TRUE if Out-of-band pairing data is present */
      FALSE                                                      /*! TRUE to initiate security upon connection */
    };
    
    /*! TRUE if Out-of-band pairing data is to be sent */
    static const bool_t datsSendOobData = FALSE;
    
    /*! SMP security parameter configuration */
    static const smpCfg_t datsSmpCfg =
    {
      3000,                                   /*! 'Repeated attempts' timeout in msec */
      SMP_IO_NO_IN_NO_OUT,                    /*! I/O Capability */
      7,                                      /*! Minimum encryption key length */
      16,                                     /*! Maximum encryption key length */
      3,                                      /*! Attempts to trigger 'repeated attempts' timeout */
      0,                                      /*! Device authentication requirements */
    };
    
    /*! configurable parameters for connection parameter update */
    static const appUpdateCfg_t datsUpdateCfg =
    {
      0,                                      /*! Connection idle period in ms before attempting
                                                  connection parameter update; set to zero to disable */
      640,                                    /*! Minimum connection interval in 1.25ms units */
      800,                                    /*! Maximum connection interval in 1.25ms units */
      3,                                      /*! Connection latency */
      900,                                    /*! Supervision timeout in 10ms units */
      5                                       /*! Number of update attempts before giving up */
    };
    
    /*! ATT configurable parameters (increase MTU) */
    static const attCfg_t datsAttCfg =
    {
      15,                               /* ATT server service discovery connection idle timeout in seconds */
      241,                              /* desired ATT MTU */
      ATT_MAX_TRANS_TIMEOUT,            /* transcation timeout in seconds */
      4                                 /* number of queued prepare writes supported by server */
    };
    
    /*! local IRK */
    static uint8_t localIrk[] =
    {
      0x95, 0xC8, 0xEE, 0x6F, 0xC5, 0x0D, 0xEF, 0x93, 0x35, 0x4E, 0x7C, 0x57, 0x08, 0xE2, 0xA3, 0x85
    };
    
    /**************************************************************************************************
      Advertising Data
    **************************************************************************************************/
    
    
    #ifdef BTLE_APP_USE_LEGACY_API
    
    /*! advertising data, discoverable mode */
    static const uint8_t datsAdvDataDisc[] =
    {
      /*! flags */
      2,                                      /*! length */
      DM_ADV_TYPE_FLAGS,                      /*! AD type */
      DM_FLAG_LE_GENERAL_DISC |               /*! flags */
      DM_FLAG_LE_BREDR_NOT_SUP,
    
      /*! manufacturer specific data */
      3,                                      /*! length */
      DM_ADV_TYPE_MANUFACTURER,               /*! AD type */
      UINT16_TO_BYTES(HCI_ID_ARM)             /*! company ID */
    };
    
    /*! scan data, discoverable mode */
    static const uint8_t datsScanDataDisc[] =
    {
      /*! device name */
      8,                                      /*! length */
      DM_ADV_TYPE_LOCAL_NAME,                 /*! AD type */
      'D',
      'a',
      't',
    
      's',
      ' ',
      'T',
      'X'
    };
    
    #else /* BTLE_APP_USE_LEGACY_API */
    /*! extended advertising and scan data, discoverable mode */
    static const uint8_t datsExtAdvDataDisc[] =
    {
      /*! flags */
      2,                                      /*! length */
      DM_ADV_TYPE_FLAGS,                      /*! AD type */
      DM_FLAG_LE_GENERAL_DISC |               /*! flags */
      DM_FLAG_LE_BREDR_NOT_SUP,
    
      /*! manufacturer specific data */
      3,                                      /*! length */
      DM_ADV_TYPE_MANUFACTURER,               /*! AD type */
      UINT16_TO_BYTES(HCI_ID_ARM),            /*! company ID */
    
      /*! device name */
      8,                                     /*! length */
      DM_ADV_TYPE_LOCAL_NAME,                 /*! AD type */
      'D',
      'a',
      't',
      's',
      ' ',
      'T',
      'X'
    };
    
    static const uint8_t datsExtScanDataDisc[] =
    {
      /*! device name */
      8,                                     /*! length */
      DM_ADV_TYPE_LOCAL_NAME,                 /*! AD type */
      'D',
    
      'a',
    
    #endif /* BTLE_APP_USE_LEGACY_API */
    
    
    /**************************************************************************************************
      Client Characteristic Configuration Descriptors
    **************************************************************************************************/
    
    /*! client characteristic configuration descriptors settings, indexed by above enumeration */
    static const attsCccSet_t datsCccSet[DATS_NUM_CCC_IDX] =
    {
      /* cccd handle          value range               security level */
      {GATT_SC_CH_CCC_HDL,    ATT_CLIENT_CFG_INDICATE,  DM_SEC_LEVEL_NONE},   /* DATS_GATT_SC_CCC_IDX */
      {WP_DAT_CH_CCC_HDL,     ATT_CLIENT_CFG_NOTIFY,    DM_SEC_LEVEL_NONE}    /* DATS_WP_DAT_CCC_IDX */
    };
    
    /**************************************************************************************************
      Local Variables
    **************************************************************************************************/
    
    /*! application control block */
    static struct
    {
      wsfHandlerId_t    handlerId;        /* WSF handler ID */
    } datsCb;
    
    /* LESC OOB configuration */
    static dmSecLescOobCfg_t *datsOobCfg;
    
    /*************************************************************************************************/
    /*!
     *  \brief  Send notification containing data.
     *
     *  \param  connId      DM connection ID.
     *
     *  \return None.
     */
    /*************************************************************************************************/
    static void datsSendData(dmConnId_t connId)
    {
      uint8_t str[] = "hello back";
    
      if (AttsCccEnabled(connId, DATS_WP_DAT_CCC_IDX))
      {
        /* send notification */
        AttsHandleValueNtf(connId, WP_DAT_HDL, sizeof(str), str);
      }
    }
    
    /*************************************************************************************************/
    /*!
     *  \brief  Application DM callback.
     *
     *  \param  pDmEvt  DM callback event
     *
     *  \return None.
     */
    /*************************************************************************************************/
    static void datsDmCback(dmEvt_t *pDmEvt)
    {
      dmEvt_t   *pMsg;
      uint16_t  len;
    
      if (pDmEvt->hdr.event == DM_SEC_ECC_KEY_IND)
      {
        DmSecSetEccKey(&pDmEvt->eccMsg.data.key);
    
        // If the local device sends OOB data.
        if (datsSendOobData)
        {
          uint8_t oobLocalRandom[SMP_RAND_LEN];
          SecRand(oobLocalRandom, SMP_RAND_LEN);
          DmSecCalcOobReq(oobLocalRandom, pDmEvt->eccMsg.data.key.pubKey_x);
        }
      }
      else if (pDmEvt->hdr.event == DM_SEC_CALC_OOB_IND)
      {
        if (datsOobCfg == NULL)
        {
          datsOobCfg = WsfBufAlloc(sizeof(dmSecLescOobCfg_t));
        }
    
        if (datsOobCfg)
        {
          Calc128Cpy(datsOobCfg->localConfirm, pDmEvt->oobCalcInd.confirm);
          Calc128Cpy(datsOobCfg->localRandom, pDmEvt->oobCalcInd.random);
        }
      }
      else
      {
        len = DmSizeOfEvt(pDmEvt);
    
        if ((pMsg = WsfMsgAlloc(len)) != NULL)
        {
          memcpy(pMsg, pDmEvt, len);
          WsfMsgSend(datsCb.handlerId, pMsg);
        }
      }
    }
    
    /*************************************************************************************************/
    /*!
     *  \brief  Application ATT callback.
     *
     *  \param  pEvt    ATT callback event
     *
     *  \return None.
     */
    /*************************************************************************************************/
    static void datsAttCback(attEvt_t *pEvt)
    {
    }
    
    /*************************************************************************************************/
    /*!
     *  \brief  Application ATTS client characteristic configuration callback.
     *
     *  \param  pDmEvt  DM callback event
     *
     *  \return None.
     */
    /*************************************************************************************************/
    static void datsCccCback(attsCccEvt_t *pEvt)
    {
      appDbHdl_t    dbHdl;
    
      /* if CCC not set from initialization and there's a device record */
      if ((pEvt->handle != ATT_HANDLE_NONE) &&
          ((dbHdl = AppDbGetHdl((dmConnId_t) pEvt->hdr.param)) != APP_DB_HDL_NONE))
      {
        /* store value in device database */
        AppDbSetCccTblValue(dbHdl, pEvt->idx, pEvt->value);
      }
    }
    
    /*************************************************************************************************/
    /*!
     *  \brief  ATTS write callback for proprietary data service.
     *
     *  \return ATT status.
     */
    /*************************************************************************************************/
    uint8_t datsWpWriteCback(dmConnId_t connId, uint16_t handle, uint8_t operation,
                              uint16_t offset, uint16_t len, uint8_t *pValue, attsAttr_t *pAttr)
    {
      /* print received data */
      printf("%s\n", pValue);
    
      /* send back some data */
      datsSendData(connId);
    
      return ATT_SUCCESS;
    }
    
    /*************************************************************************************************/
    /*!
    *
    *  \brief  Get peer key from a device database record.
    *
    *  \param  pMsg    Pointer to DM callback event message.
    *
    *  \return Pointer to peer key if key is valid or NULL if not valid.
    */
    /*************************************************************************************************/
    static dmSecKey_t *datsGetPeerKey(dmEvt_t *pMsg)
    {
      appConnCb_t *pCb;
    
      /* look up app connection control block from DM connection ID */
      pCb = &appConnCb[pMsg->hdr.param - 1];
    
      /* if database record handle valid */
      if (pCb->dbHdl != APP_DB_HDL_NONE)
      {
        return AppDbGetKey(pCb->dbHdl, DM_KEY_IRK, NULL);
      }
    
      return NULL;
    }
    
    /*************************************************************************************************/
    /*!
    *
    *  \brief  Handle add device to resolving list indication.
    *
    *  \param  pMsg    Pointer to DM callback event message.
    *
    *  \return None.
    */
    /*************************************************************************************************/
    static void datsPrivAddDevToResListInd(dmEvt_t *pMsg)
    {
    
    #ifndef BTLE_APP_USE_LEGACY_API
      uint8_t addrZeros[BDA_ADDR_LEN] = { 0 };
    #endif /* BTLE_APP_USE_LEGACY_API */
    
    
      dmSecKey_t *pPeerKey;
    
      /* if peer IRK present */
      if ((pPeerKey = datsGetPeerKey(pMsg)) != NULL)
      {
        /* set advertising peer address */
    
    #ifdef BTLE_APP_USE_LEGACY_API
    
        AppSetAdvPeerAddr(pPeerKey->irk.addrType, pPeerKey->irk.bdAddr);
    
    #else /* BTLE_APP_USE_LEGACY_API */
        AppExtSetAdvPeerAddr(DM_ADV_HANDLE_DEFAULT, HCI_ADDR_TYPE_PUBLIC, addrZeros);
    #endif /* BTLE_APP_USE_LEGACY_API */
    
      }
    }
    
    /*************************************************************************************************/
    /*!
    *
    *  \brief  Handle remove device from resolving list indication.
    *
    *  \param  pMsg    Pointer to DM callback event message.
    *
    *  \return None.
    */
    /*************************************************************************************************/
    static void datsPrivRemDevFromResListInd(dmEvt_t *pMsg)
    {
      if (pMsg->hdr.status == HCI_SUCCESS)
      {
        appConnCb_t *pCb;
    
        /* look up app connection control block from DM connection ID */
        pCb = &appConnCb[pMsg->hdr.param - 1];
    
        if (pCb->dbHdl != APP_DB_HDL_NONE)
        {
          uint8_t addrZeros[BDA_ADDR_LEN] = { 0 };
    
          /* clear advertising peer address and its type */
    
    #ifdef BTLE_APP_USE_LEGACY_API
    
          AppSetAdvPeerAddr(HCI_ADDR_TYPE_PUBLIC, addrZeros);
    
    #else /* BTLE_APP_USE_LEGACY_API */
          AppExtSetAdvPeerAddr(DM_ADV_HANDLE_DEFAULT, HCI_ADDR_TYPE_PUBLIC, addrZeros);
    #endif /* BTLE_APP_USE_LEGACY_API */
    
        }
      }
    }
    
    /*************************************************************************************************/
    /*!
     *
     *  \brief  Display stack version.
     *
     *  \param  version    version number.
     *
     *  \return None.
     */
    /*************************************************************************************************/
    void datsDisplayStackVersion(const char *pVersion)
    {
      APP_TRACE_INFO1("Stack Version: %s", pVersion);
    }
    
    /*************************************************************************************************/
    /*!
     *  \brief  Set up advertising and other procedures that need to be performed after
     *          device reset.
     *
     *  \param  pMsg    Pointer to message.
     *
     *  \return None.
     */
    /*************************************************************************************************/
    static void datsSetup(dmEvt_t *pMsg)
    {
    
    #ifndef BTLE_APP_USE_LEGACY_API
      uint8_t advHandle;
    #endif /* BTLE_APP_USE_LEGACY_API */
    
      /* Scan responses and connections limited to the EvKit dats peer */
    
      DmDevWhiteListAdd(DM_ADDR_PUBLIC, (bdAddr_t){0x02, 0x01, 0x44, 0x8B, 0x05, 0x00});
      DmDevSetFilterPolicy(DM_FILT_POLICY_MODE_ADV, HCI_FILT_WHITE_LIST);
    
      /* set advertising and scan response data for discoverable mode */
    
    #ifdef BTLE_APP_USE_LEGACY_API
    
      AppAdvSetData(APP_ADV_DATA_DISCOVERABLE, sizeof(datsAdvDataDisc), (uint8_t *) datsAdvDataDisc);
      AppAdvSetData(APP_SCAN_DATA_DISCOVERABLE, sizeof(datsScanDataDisc), (uint8_t *) datsScanDataDisc);
    
    #else /* BTLE_APP_USE_LEGACY_API */
      AppExtAdvSetData(DM_ADV_HANDLE_DEFAULT, APP_ADV_DATA_DISCOVERABLE, sizeof(datsExtAdvDataDisc), (uint8_t *) datsExtAdvDataDisc, HCI_EXT_ADV_DATA_LEN);
      AppExtAdvSetData(DM_ADV_HANDLE_DEFAULT, APP_SCAN_DATA_DISCOVERABLE, sizeof(datsExtScanDataDisc), (uint8_t *) datsExtScanDataDisc, HCI_EXT_ADV_DATA_LEN);
    #endif /* BTLE_APP_USE_LEGACY_API */
     
    
    
      /* set advertising and scan response data for connectable mode */
    
    #ifdef BTLE_APP_USE_LEGACY_API
    
      AppAdvSetData(APP_ADV_DATA_CONNECTABLE, sizeof(datsAdvDataDisc), (uint8_t *) datsAdvDataDisc);
      AppAdvSetData(APP_SCAN_DATA_CONNECTABLE, sizeof(datsScanDataDisc), (uint8_t *) datsScanDataDisc);
    
    #else /* BTLE_APP_USE_LEGACY_API */
      AppExtAdvSetData(DM_ADV_HANDLE_DEFAULT, APP_ADV_DATA_CONNECTABLE, sizeof(datsExtAdvDataDisc), (uint8_t *) datsExtAdvDataDisc, HCI_EXT_ADV_DATA_LEN);
      AppExtAdvSetData(DM_ADV_HANDLE_DEFAULT, APP_SCAN_DATA_CONNECTABLE, sizeof(datsExtScanDataDisc), (uint8_t *) datsExtScanDataDisc, HCI_EXT_ADV_DATA_LEN);
    #endif /* BTLE_APP_USE_LEGACY_API */
    
      DmSetDefaultPhy(0, HCI_PHY_LE_1M_BIT | HCI_PHY_LE_2M_BIT, HCI_PHY_LE_1M_BIT | HCI_PHY_LE_2M_BIT);
    
    
      /* start advertising; automatically set connectable/discoverable mode and bondable mode */
    
    #ifndef BTLE_APP_USE_LEGACY_API
      advHandle = DM_ADV_HANDLE_DEFAULT;
    #endif /* BTLE_APP_USE_LEGACY_API */
    
    #ifdef BTLE_APP_USE_LEGACY_API
    
      AppAdvStart(APP_MODE_AUTO_INIT);
    
    #else /* BTLE_APP_USE_LEGACY_API */
      AppExtAdvStart(1, &advHandle, APP_MODE_AUTO_INIT);
    #endif /* BTLE_APP_USE_LEGACY_API */
    
    }
    
    /*************************************************************************************************/
    /*!
     *  \brief  Process messages from the event handler.
     *
     *  \param  pMsg    Pointer to message.
     *
     *  \return None.
     */
    /*************************************************************************************************/
    static void datsProcMsg(dmEvt_t *pMsg)
    {
      uint8_t uiEvent = APP_UI_NONE;
    
      switch(pMsg->hdr.event)
      {
        case DM_RESET_CMPL_IND:
          DmSecGenerateEccKeyReq();
          datsSetup(pMsg);
          uiEvent = APP_UI_RESET_CMPL;
          break;
    
        case DM_ADV_START_IND:
          uiEvent = APP_UI_ADV_START;
          break;
    
    
    #ifndef BTLE_APP_IGNORE_EXT_EVENTS
        case DM_ADV_SET_START_IND:
          uiEvent = APP_UI_ADV_SET_START_IND;
          break;
    #endif /* BTLE_APP_IGNORE_EXT_EVENTS */
    
    
        case DM_ADV_STOP_IND:
          uiEvent = APP_UI_ADV_STOP;
          break;
    
    
    #ifndef BTLE_APP_IGNORE_EXT_EVENTS
         case DM_ADV_SET_STOP_IND:
          uiEvent = APP_UI_ADV_SET_STOP_IND;
          break;
    #endif /* BTLE_APP_IGNORE_EXT_EVENTS */
    
    
        case DM_CONN_OPEN_IND:
          uiEvent = APP_UI_CONN_OPEN;
          break;
    
        case DM_CONN_CLOSE_IND:
          printf("Connection closed status 0x%x, reason 0x%x", pMsg->connClose.status, pMsg->connClose.reason);
          switch (pMsg->connClose.reason)
          {
            case HCI_ERR_CONN_TIMEOUT:      printf(" TIMEOUT\n");         break;
            case HCI_ERR_LOCAL_TERMINATED:  printf(" LOCAL TERM\n");      break;
            case HCI_ERR_REMOTE_TERMINATED: printf(" REMOTE TERM\n");     break;
            case HCI_ERR_CONN_FAIL:         printf(" FAIL ESTABLISH\n");  break;
            case HCI_ERR_MIC_FAILURE:       printf(" MIC FAILURE\n");     break;
          }
    #if 0
          WSF_ASSERT(0);
    #endif
          uiEvent = APP_UI_CONN_CLOSE;
          break;
    
        case DM_SEC_PAIR_CMPL_IND:
          uiEvent = APP_UI_SEC_PAIR_CMPL;
          break;
    
        case DM_SEC_PAIR_FAIL_IND:
          uiEvent = APP_UI_SEC_PAIR_FAIL;
          break;
    
        case DM_SEC_ENCRYPT_IND:
          uiEvent = APP_UI_SEC_ENCRYPT;
          break;
    
        case DM_SEC_ENCRYPT_FAIL_IND:
          uiEvent = APP_UI_SEC_ENCRYPT_FAIL;
          break;
    
        case DM_SEC_AUTH_REQ_IND:
    
          if (pMsg->authReq.oob)
          {
            dmConnId_t connId = (dmConnId_t) pMsg->hdr.param;
    
            /* TODO: Perform OOB Exchange with the peer. */
    
    
            /* TODO: Fill datsOobCfg peerConfirm and peerRandom with value passed out of band */
    
            if (datsOobCfg != NULL)
            {
              DmSecSetOob(connId, datsOobCfg);
            }
    
            DmSecAuthRsp(connId, 0, NULL);
          }
          else
          {
            AppHandlePasskey(&pMsg->authReq);
          }
          break;
    
        case DM_SEC_COMPARE_IND:
          AppHandleNumericComparison(&pMsg->cnfInd);
          break;
    
        case DM_PRIV_ADD_DEV_TO_RES_LIST_IND:
          datsPrivAddDevToResListInd(pMsg);
          break;
    
        case DM_PRIV_REM_DEV_FROM_RES_LIST_IND:
          datsPrivRemDevFromResListInd(pMsg);
          break;
    
        case DM_ADV_NEW_ADDR_IND:
          break;
    
    
        case DM_VENDOR_SPEC_IND:
          break;
    
    
        default:
          break;
      }
    
      if (uiEvent != APP_UI_NONE)
      {
        AppUiAction(uiEvent);
      }
    }
    
    /*************************************************************************************************/
    /*!
     *  \brief  Application handler init function called during system initialization.
     *
     *  \param  handlerID  WSF handler ID.
     *
     *  \return None.
     */
    /*************************************************************************************************/
    void DatsHandlerInit(wsfHandlerId_t handlerId)
    {
      APP_TRACE_INFO0("DatsHandlerInit");
    
      /* store handler ID */
      datsCb.handlerId = handlerId;
    
      /* Set configuration pointers */
      pAppSlaveCfg = (appSlaveCfg_t *) &datsSlaveCfg;
    
    #ifdef BTLE_APP_USE_LEGACY_API
    
      pAppAdvCfg = (appAdvCfg_t *) &datsAdvCfg;
    
    #else /* BTLE_APP_USE_LEGACY_API */
      pAppExtAdvCfg = (appExtAdvCfg_t *) &datsExtAdvCfg;
    #endif /* BTLE_APP_USE_LEGACY_API */
    
      pAppSecCfg = (appSecCfg_t *) &datsSecCfg;
      pAppUpdateCfg = (appUpdateCfg_t *) &datsUpdateCfg;
      pSmpCfg = (smpCfg_t *) &datsSmpCfg;
      pAttCfg = (attCfg_t *) &datsAttCfg;
    
      /* Initialize application framework */
      AppSlaveInit();
    
      /* Set IRK for the local device */
      DmSecSetLocalIrk(localIrk);
    }
    
    /*************************************************************************************************/
    /*!
     *  \brief  Callback for WSF buffer diagnostic messages.
     *
     *  \param  pInfo     Diagnostics message
     *
     *  \return None.
     */
    /*************************************************************************************************/
    static void datsWsfBufDiagnostics(WsfBufDiag_t *pInfo)
    {
      if (pInfo->type == WSF_BUF_ALLOC_FAILED)
      {
        APP_TRACE_INFO2("Dats got WSF Buffer Allocation Failure - Task: %d Len: %d",
                         pInfo->param.alloc.taskId, pInfo->param.alloc.len);
      }
    }
    
    /*************************************************************************************************/
    /*!
     *  \brief  WSF event handler for application.
     *
     *  \param  event   WSF event mask.
     *  \param  pMsg    WSF message.
     *
     *  \return None.
     */
    /*************************************************************************************************/
    void DatsHandler(wsfEventMask_t event, wsfMsgHdr_t *pMsg)
    {
      if (pMsg != NULL)
      {
        APP_TRACE_INFO1("Dats got evt 0x%x", pMsg->event);
    
        if (pMsg->event >= DM_CBACK_START && pMsg->event <= DM_CBACK_END)
        {
          /* process advertising and connection-related messages */
          AppSlaveProcDmMsg((dmEvt_t *) pMsg);
    
          /* process security-related messages */
          AppSlaveSecProcDmMsg((dmEvt_t *) pMsg);
        }
    
        /* perform profile and user interface-related operations */
        datsProcMsg((dmEvt_t *) pMsg);
      }
    }
    
    /*************************************************************************************************/
    /*!
     *  \brief  Start the application.
     *
     *  \return None.
     */
    /*************************************************************************************************/
    void DatsStart(void)
    {
      /* Register for stack callbacks */
      DmRegister(datsDmCback);
      DmConnRegister(DM_CLIENT_ID_APP, datsDmCback);
      AttRegister(datsAttCback);
      AttConnRegister(AppServerConnCback);
      AttsCccRegister(DATS_NUM_CCC_IDX, (attsCccSet_t *) datsCccSet, datsCccCback);
    
      /* Initialize attribute server database */
      SvcCoreAddGroup();
      SvcWpCbackRegister(NULL, datsWpWriteCback);
      SvcWpAddGroup();
    
      WsfBufDiagRegister(datsWsfBufDiagnostics);
    
      /* Reset the device */
      DmDevReset();
    }