diff --git a/epicardium/ble/app/app_main.c b/epicardium/ble/app/app_main.c
new file mode 100644
index 0000000000000000000000000000000000000000..567dd8a2a567d09fcca1b0f1788823a262bc9dd0
--- /dev/null
+++ b/epicardium/ble/app/app_main.c
@@ -0,0 +1,420 @@
+/*************************************************************************************************/
+/*!
+ *  \file
+ *
+ *  \brief  Application framework main module.
+ *
+ *  Copyright (c) 2011-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.
+ */
+/*************************************************************************************************/
+
+/* card10:
+ * copied from: lib/sdk/Libraries/BTLE/stack/ble-profiles/sources/apps/app/app_main.c
+ *
+ * Reason: we need to correctly implement AppHandleNumericComparison
+ */
+#include <string.h>
+#include "wsf_types.h"
+#include "wsf_msg.h"
+#include "sec_api.h"
+#include "wsf_trace.h"
+#include "wsf_timer.h"
+#include "wsf_assert.h"
+#include "util/bstream.h"
+#include "dm_api.h"
+#include "app_api.h"
+#include "app_main.h"
+#include "app_ui.h"
+
+/**************************************************************************************************
+  Global Variables
+**************************************************************************************************/
+
+/*! Configuration pointer for advertising */
+appAdvCfg_t *pAppAdvCfg;
+
+/*! Configuration pointer for extended and periodic advertising */
+appExtAdvCfg_t *pAppExtAdvCfg;
+
+/*! Configuration pointer for slave */
+appSlaveCfg_t *pAppSlaveCfg;
+
+/*! Configuration pointer for master */
+appMasterCfg_t *pAppMasterCfg;
+
+/*! Configuration pointer for extended master */
+appExtMasterCfg_t *pAppExtMasterCfg;
+
+/*! Configuration pointer for security */
+appSecCfg_t *pAppSecCfg;
+
+/*! Configuration pointer for connection parameter update */
+appUpdateCfg_t *pAppUpdateCfg;
+
+/*! Configuration pointer for discovery */
+appDiscCfg_t *pAppDiscCfg;
+
+/*! Configuration pointer for application */
+appCfg_t *pAppCfg;
+
+/*! Connection control block array */
+appConnCb_t appConnCb[DM_CONN_MAX];
+
+/*! WSF handler ID */
+wsfHandlerId_t appHandlerId;
+
+/*! Main control block */
+appCb_t appCb;
+
+/*! Configuration structure for incoming request actions */
+const appReqActCfg_t appReqActCfg =
+{
+  APP_ACT_ACCEPT        /*! Action for the remote connection parameter request */
+};
+
+/*! Configuration pointer for incoming request actions on master */
+appReqActCfg_t *pAppMasterReqActCfg = (appReqActCfg_t *) &appReqActCfg;
+
+/*! Configurable pointer for incoming request actions on slave */
+appReqActCfg_t *pAppSlaveReqActCfg = (appReqActCfg_t *) &appReqActCfg;
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Process messages from the event handler.
+ *
+ *  \param  pMsg    Pointer to message.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+static void appProcMsg(wsfMsgHdr_t *pMsg)
+{
+  switch(pMsg->event)
+  {
+    case APP_BTN_POLL_IND:
+      appUiBtnPoll();
+      break;
+
+    case APP_UI_TIMER_IND:
+      appUiTimerExpired(pMsg);
+      break;
+
+    default:
+      break;
+  }
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Check the bonded state of a connection.
+ *
+ *  \param  connId      DM connection ID.
+ *
+ *  \return Bonded state.
+ */
+/*************************************************************************************************/
+bool_t appCheckBonded(dmConnId_t connId)
+{
+  WSF_ASSERT((connId > 0) && (connId <= DM_CONN_MAX));
+
+  return appConnCb[connId - 1].bonded;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Check the bond-by-LTK state of a connection.
+ *
+ *  \param  connId      DM connection ID.
+ *
+ *  \return Bond-by-LTK state.
+ */
+/*************************************************************************************************/
+bool_t appCheckBondByLtk(dmConnId_t connId)
+{
+  WSF_ASSERT((connId > 0) && (connId <= DM_CONN_MAX));
+
+  return appConnCb[connId - 1].bondByLtk;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Return the number of existing connections of the given role.
+ *
+ *  \param  role      Connection role
+ *
+ *  \return Number of connections.
+ */
+/*************************************************************************************************/
+uint8_t appNumConns(uint8_t role)
+{
+  appConnCb_t   *pCcb = appConnCb;
+  uint8_t       i, j;
+
+  for (i = DM_CONN_MAX, j = 0; i > 0; i--, pCcb++)
+  {
+    if ((pCcb->connId != DM_CONN_ID_NONE) && (DmConnRole(pCcb->connId) == role))
+    {
+      j++;
+    }
+  }
+
+  return j;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  App framework handler init function called during system initialization.
+ *
+ *  \param  handlerID  WSF handler ID for App.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppHandlerInit(wsfHandlerId_t handlerId)
+{
+  appHandlerId = handlerId;
+
+  AppDbInit();
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  WSF event handler for app framework.
+ *
+ *  \param  event   WSF event mask.
+ *  \param  pMsg    WSF message.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppHandler(wsfEventMask_t event, wsfMsgHdr_t *pMsg)
+{
+  if (pMsg != NULL)
+  {
+    APP_TRACE_INFO1("App got evt %d", pMsg->event);
+
+    if (pMsg->event >= APP_MASTER_MSG_START)
+    {
+      /* pass event to master handler */
+      (*appCb.masterCback)(pMsg);
+    }
+    else if (pMsg->event >= APP_SLAVE_MSG_START)
+    {
+      /* pass event to slave handler */
+      (*appCb.slaveCback)(pMsg);
+    }
+    else
+    {
+      appProcMsg(pMsg);
+    }
+  }
+  else
+  {
+    if (event & APP_BTN_DOWN_EVT)
+    {
+      AppUiBtnPressed();
+    }
+  }
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Handle a passkey request during pairing.  If the passkey is to displayed, a
+ *          random passkey is generated and displayed.  If the passkey is to be entered
+ *          the user is prompted to enter the passkey.
+ *
+ *  \param  pAuthReq  DM authentication requested event structure.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppHandlePasskey(dmSecAuthReqIndEvt_t *pAuthReq)
+{
+  uint32_t passkey;
+  uint8_t  buf[SMP_PIN_LEN];
+
+  if (pAuthReq->display)
+  {
+    /* generate random passkey, limit to 6 digit max */
+    SecRand((uint8_t *) &passkey, sizeof(uint32_t));
+    passkey %= 1000000;
+
+    /* convert to byte buffer */
+    buf[0] = UINT32_TO_BYTE0(passkey);
+    buf[1] = UINT32_TO_BYTE1(passkey);
+    buf[2] = UINT32_TO_BYTE2(passkey);
+
+    /* send authentication response to DM */
+    DmSecAuthRsp((dmConnId_t) pAuthReq->hdr.param, SMP_PIN_LEN, buf);
+
+    /* display passkey */
+    AppUiDisplayPasskey(passkey);
+  }
+  else
+  {
+    /* prompt user to enter passkey */
+    AppUiAction(APP_UI_PASSKEY_PROMPT);
+  }
+}
+
+/*************************************************************************************************/
+/*!
+*  \brief  Handle a numeric comparison indication during pairing.  The confirmation value is
+*          displayed and the user is prompted to verify that the local and peer confirmation
+*          values match.
+*
+*  \param  pCnfInd  DM confirmation indication event structure.
+*
+*  \return None.
+*/
+/*************************************************************************************************/
+void AppHandleNumericComparison(dmSecCnfIndEvt_t *pCnfInd)
+{
+  uint32_t confirm = DmSecGetCompareValue(pCnfInd->confirm);
+
+  /* display confirmation value */
+  AppUiDisplayConfirmValue(confirm);
+
+  /* TODO: Verify that local and peer confirmation values match */
+  DmSecCompareRsp((dmConnId_t)pCnfInd->hdr.param, TRUE);
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Close the connection with the give connection identifier.
+ *
+ *  \param  connId    Connection identifier.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppConnClose(dmConnId_t connId)
+{
+  DmConnClose(DM_CLIENT_ID_APP, connId, HCI_ERR_REMOTE_TERMINATED);
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Get a list of connection identifiers of open connections.
+ *
+ *  \param  pConnIdList    Buffer to hold connection IDs (must be DM_CONN_MAX bytes).
+ *
+ *  \return Number of open connections.
+ *
+ */
+/*************************************************************************************************/
+uint8_t AppConnOpenList(dmConnId_t *pConnIdList)
+{
+  appConnCb_t   *pCcb = appConnCb;
+  uint8_t       i;
+  uint8_t       pos = 0;
+
+  memset(pConnIdList, DM_CONN_ID_NONE, DM_CONN_MAX);
+
+  for (i = DM_CONN_MAX; i > 0; i--, pCcb++)
+  {
+    if (pCcb->connId != DM_CONN_ID_NONE)
+    {
+      pConnIdList[pos++] = pCcb->connId;
+    }
+  }
+
+  return pos;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Check if a connection is open.
+ *
+ *  \return Connection ID of open connection or DM_CONN_ID_NONE if no open connections.
+ */
+/*************************************************************************************************/
+dmConnId_t AppConnIsOpen(void)
+{
+  appConnCb_t   *pCcb = appConnCb;
+  uint8_t       i;
+
+  for (i = DM_CONN_MAX; i > 0; i--, pCcb++)
+  {
+    if (pCcb->connId != DM_CONN_ID_NONE)
+    {
+      return pCcb->connId;
+    }
+  }
+
+  return DM_CONN_ID_NONE;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Get the device database record handle associated with an open connection.
+ *
+ *  \param  connId    Connection identifier.
+ *
+ *  \return Database record handle or APP_DB_HDL_NONE.
+ */
+/*************************************************************************************************/
+appDbHdl_t AppDbGetHdl(dmConnId_t connId)
+{
+  return appConnCb[connId-1].dbHdl;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Add device to resolving list.
+ *
+ *  \param  pMsg    Pointer to DM callback event message.
+ *  \param  connId  Connection identifier.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppAddDevToResList(dmEvt_t *pMsg, dmConnId_t connId)
+{
+  dmSecKey_t *pPeerKey;
+  appDbHdl_t hdl = appConnCb[connId - 1].dbHdl;
+
+  /* if LL Privacy is supported and the peer device has distributed its IRK */
+  if (HciLlPrivacySupported() && ((pPeerKey = AppDbGetKey(hdl, DM_KEY_IRK, NULL))!= NULL))
+  {
+    /* add peer device to resolving list. If all-zero local or peer IRK is used then
+       LL will only use or accept local or peer identity address respectively. */
+    DmPrivAddDevToResList(pPeerKey->irk.addrType, pPeerKey->irk.bdAddr, pPeerKey->irk.key,
+                          DmSecGetLocalIrk(), TRUE, pMsg->hdr.param);
+  }
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Update privacy mode for a given peer device.
+ *
+ *  \param  hdl     Database record handle.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppUpdatePrivacyMode(appDbHdl_t hdl)
+{
+  /* if peer device's been added to resolving list but RPA Only attribute not found on peer device */
+  if ((hdl != APP_DB_HDL_NONE) && AppDbGetPeerAddedToRl(hdl) && !AppDbGetPeerRpao(hdl))
+  {
+    dmSecKey_t *pPeerKey = AppDbGetKey(hdl, DM_KEY_IRK, NULL);
+    if (pPeerKey != NULL)
+    {
+      /* set device privacy mode for this peer device */
+      DmPrivSetPrivacyMode(pPeerKey->irk.addrType, pPeerKey->irk.bdAddr, DM_PRIV_MODE_DEVICE);
+
+      /* make sure resolving list flag cleared */
+      AppDbSetPeerAddedToRl(hdl, FALSE);
+    }
+  }
+}
diff --git a/epicardium/ble/app/common/app_db.c b/epicardium/ble/app/common/app_db.c
new file mode 100644
index 0000000000000000000000000000000000000000..e3033bf5234d9996a19dcc1a5c4fd5e5030a4aca
--- /dev/null
+++ b/epicardium/ble/app/common/app_db.c
@@ -0,0 +1,710 @@
+/*************************************************************************************************/
+/*!
+ *  \file
+ *
+ *  \brief  Application framework device database example, using simple RAM-based storage.
+ *
+ *  Copyright (c) 2011-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.
+ */
+/*************************************************************************************************/
+
+/* card10:
+ * copied from: lib/sdk/Libraries/BTLE/stack/ble-profiles/sources/apps/app/common/app_db.c
+ *
+ * Reason: we need to implement persistent storage for pairings
+ */
+#include <string.h>
+#include "wsf_types.h"
+#include "wsf_assert.h"
+#include "util/bda.h"
+#include "app_api.h"
+#include "app_main.h"
+#include "app_db.h"
+#include "app_cfg.h"
+
+/**************************************************************************************************
+  Data Types
+**************************************************************************************************/
+
+/*! Database record */
+typedef struct
+{
+  /*! Common for all roles */
+  bdAddr_t    peerAddr;                     /*! Peer address */
+  uint8_t     addrType;                     /*! Peer address type */
+  dmSecIrk_t  peerIrk;                      /*! Peer IRK */
+  dmSecCsrk_t peerCsrk;                     /*! Peer CSRK */
+  uint8_t     keyValidMask;                 /*! Valid keys in this record */
+  bool_t      inUse;                        /*! TRUE if record in use */
+  bool_t      valid;                        /*! TRUE if record is valid */
+  bool_t      peerAddedToRl;                /*! TRUE if peer device's been added to resolving list */
+  bool_t      peerRpao;                     /*! TRUE if RPA Only attribute's present on peer device */
+
+  /*! For slave local device */
+  dmSecLtk_t  localLtk;                     /*! Local LTK */
+  uint8_t     localLtkSecLevel;             /*! Local LTK security level */
+  bool_t      peerAddrRes;                  /*! TRUE if address resolution's supported on peer device (master) */
+
+  /*! For master local device */
+  dmSecLtk_t  peerLtk;                      /*! Peer LTK */
+  uint8_t     peerLtkSecLevel;              /*! Peer LTK security level */
+
+  /*! for ATT server local device */
+  uint16_t    cccTbl[APP_DB_NUM_CCCD];      /*! Client characteristic configuration descriptors */
+  uint32_t    peerSignCounter;              /*! Peer Sign Counter */
+
+  /*! for ATT client */
+  uint16_t    hdlList[APP_DB_HDL_LIST_LEN]; /*! Cached handle list */
+  uint8_t     discStatus;                   /*! Service discovery and configuration status */
+} appDbRec_t;
+
+/*! Database type */
+typedef struct
+{
+  appDbRec_t  rec[APP_DB_NUM_RECS];               /*! Device database records */
+  char        devName[ATT_DEFAULT_PAYLOAD_LEN];   /*! Device name */
+  uint8_t     devNameLen;                         /*! Device name length */
+} appDb_t;
+
+/**************************************************************************************************
+  Local Variables
+**************************************************************************************************/
+
+/*! Database */
+static appDb_t appDb;
+
+/*! When all records are allocated use this index to determine which to overwrite */
+static appDbRec_t *pAppDbNewRec = appDb.rec;
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Initialize the device database.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbInit(void)
+{
+  return;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Create a new device database record.
+ *
+ *  \param  addrType  Address type.
+ *  \param  pAddr     Peer device address.
+ *
+ *  \return Database record handle.
+ */
+/*************************************************************************************************/
+appDbHdl_t AppDbNewRecord(uint8_t addrType, uint8_t *pAddr)
+{
+  appDbRec_t  *pRec = appDb.rec;
+  uint8_t     i;
+
+  /* find a free record */
+  for (i = APP_DB_NUM_RECS; i > 0; i--, pRec++)
+  {
+    if (!pRec->inUse)
+    {
+      break;
+    }
+  }
+
+  /* if all records were allocated */
+  if (i == 0)
+  {
+    /* overwrite a record */
+    pRec = pAppDbNewRec;
+
+    /* get next record to overwrite */
+    pAppDbNewRec++;
+    if (pAppDbNewRec == &appDb.rec[APP_DB_NUM_RECS])
+    {
+      pAppDbNewRec = appDb.rec;
+    }
+  }
+
+  /* initialize record */
+  memset(pRec, 0, sizeof(appDbRec_t));
+  pRec->inUse = TRUE;
+  pRec->addrType = addrType;
+  BdaCpy(pRec->peerAddr, pAddr);
+  pRec->peerAddedToRl = FALSE;
+  pRec->peerRpao = FALSE;
+
+  return (appDbHdl_t) pRec;
+}
+
+/*************************************************************************************************/
+/*!
+*  \brief  Get the next database record for a given record. For the first record, the function
+*          should be called with 'hdl' set to 'APP_DB_HDL_NONE'.
+*
+*  \param  hdl  Database record handle.
+*
+*  \return Next record handle found. APP_DB_HDL_NONE, otherwise.
+*/
+/*************************************************************************************************/
+appDbHdl_t AppDbGetNextRecord(appDbHdl_t hdl)
+{
+  appDbRec_t  *pRec;
+
+  /* if first record is requested */
+  if (hdl == APP_DB_HDL_NONE)
+  {
+    pRec = appDb.rec;
+  }
+  /* if valid record passed in */
+  else if (AppDbRecordInUse(hdl))
+  {
+    pRec = (appDbRec_t *)hdl;
+    pRec++;
+  }
+  /* invalid record passed in */
+  else
+  {
+    return APP_DB_HDL_NONE;
+  }
+
+  /* look for next valid record */
+  while (pRec < &appDb.rec[APP_DB_NUM_RECS])
+  {
+    /* if record is in use */
+    if (pRec->inUse && pRec->valid)
+    {
+      /* record found */
+      return (appDbHdl_t)pRec;
+    }
+
+    /* look for next record */
+    pRec++;
+  }
+
+  /* end of records */
+  return APP_DB_HDL_NONE;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Delete a new device database record.
+ *
+ *  \param  hdl       Database record handle.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbDeleteRecord(appDbHdl_t hdl)
+{
+  ((appDbRec_t *) hdl)->inUse = FALSE;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Validate a new device database record.  This function is called when pairing is
+ *          successful and the devices are bonded.
+ *
+ *  \param  hdl       Database record handle.
+ *  \param  keyMask   Bitmask of keys to validate.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbValidateRecord(appDbHdl_t hdl, uint8_t keyMask)
+{
+  ((appDbRec_t *) hdl)->valid = TRUE;
+  ((appDbRec_t *) hdl)->keyValidMask = keyMask;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Check if a record has been validated.  If it has not, delete it.  This function
+ *          is typically called when the connection is closed.
+ *
+ *  \param  hdl       Database record handle.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbCheckValidRecord(appDbHdl_t hdl)
+{
+  if (((appDbRec_t *) hdl)->valid == FALSE)
+  {
+    AppDbDeleteRecord(hdl);
+  }
+}
+
+/*************************************************************************************************/
+/*!
+*  \brief  Check if a database record is in use.
+
+*  \param  hdl       Database record handle.
+*
+*  \return TURE if record in use. FALSE, otherwise.
+*/
+/*************************************************************************************************/
+bool_t AppDbRecordInUse(appDbHdl_t hdl)
+{
+  appDbRec_t  *pRec = appDb.rec;
+  uint8_t     i;
+
+  /* see if record is in database record list */
+  for (i = APP_DB_NUM_RECS; i > 0; i--, pRec++)
+  {
+    if (pRec->inUse && pRec->valid && (pRec == ((appDbRec_t *)hdl)))
+    {
+      return TRUE;
+    }
+  }
+
+  return FALSE;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Check if there is a stored bond with any device.
+ *
+ *  \param  hdl       Database record handle.
+ *
+ *  \return TRUE if a bonded device is found, FALSE otherwise.
+ */
+/*************************************************************************************************/
+bool_t AppDbCheckBonded(void)
+{
+  appDbRec_t  *pRec = appDb.rec;
+  uint8_t     i;
+
+  /* find a record */
+  for (i = APP_DB_NUM_RECS; i > 0; i--, pRec++)
+  {
+    if (pRec->inUse)
+    {
+      return TRUE;
+    }
+  }
+
+  return FALSE;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Delete all database records.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbDeleteAllRecords(void)
+{
+  appDbRec_t  *pRec = appDb.rec;
+  uint8_t     i;
+
+  /* set in use to false for all records */
+  for (i = APP_DB_NUM_RECS; i > 0; i--, pRec++)
+  {
+    pRec->inUse = FALSE;
+  }
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Find a device database record by peer address.
+ *
+ *  \param  addrType  Address type.
+ *  \param  pAddr     Peer device address.
+ *
+ *  \return Database record handle or APP_DB_HDL_NONE if not found.
+ */
+/*************************************************************************************************/
+appDbHdl_t AppDbFindByAddr(uint8_t addrType, uint8_t *pAddr)
+{
+  appDbRec_t  *pRec = appDb.rec;
+  uint8_t     peerAddrType = DmHostAddrType(addrType);
+  uint8_t     i;
+
+  /* find matching record */
+  for (i = APP_DB_NUM_RECS; i > 0; i--, pRec++)
+  {
+    if (pRec->inUse && (pRec->addrType == peerAddrType) && BdaCmp(pRec->peerAddr, pAddr))
+    {
+      return (appDbHdl_t) pRec;
+    }
+  }
+
+  return APP_DB_HDL_NONE;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Find a device database record by data in an LTK request.
+ *
+ *  \param  encDiversifier  Encryption diversifier associated with key.
+ *  \param  pRandNum        Pointer to random number associated with key.
+ *
+ *  \return Database record handle or APP_DB_HDL_NONE if not found.
+ */
+/*************************************************************************************************/
+appDbHdl_t AppDbFindByLtkReq(uint16_t encDiversifier, uint8_t *pRandNum)
+{
+  appDbRec_t  *pRec = appDb.rec;
+  uint8_t     i;
+
+  /* find matching record */
+  for (i = APP_DB_NUM_RECS; i > 0; i--, pRec++)
+  {
+    if (pRec->inUse && (pRec->localLtk.ediv == encDiversifier) &&
+        (memcmp(pRec->localLtk.rand, pRandNum, SMP_RAND8_LEN) == 0))
+    {
+      return (appDbHdl_t) pRec;
+    }
+  }
+
+  return APP_DB_HDL_NONE;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Get a key from a device database record.
+ *
+ *  \param  hdl       Database record handle.
+ *  \param  type      Type of key to get.
+ *  \param  pSecLevel If the key is valid, the security level of the key.
+ *
+ *  \return Pointer to key if key is valid or NULL if not valid.
+ */
+/*************************************************************************************************/
+dmSecKey_t *AppDbGetKey(appDbHdl_t hdl, uint8_t type, uint8_t *pSecLevel)
+{
+  dmSecKey_t *pKey = NULL;
+
+  /* if key valid */
+  if ((type & ((appDbRec_t *) hdl)->keyValidMask) != 0)
+  {
+    switch(type)
+    {
+      case DM_KEY_LOCAL_LTK:
+        *pSecLevel = ((appDbRec_t *) hdl)->localLtkSecLevel;
+        pKey = (dmSecKey_t *) &((appDbRec_t *) hdl)->localLtk;
+        break;
+
+      case DM_KEY_PEER_LTK:
+        *pSecLevel = ((appDbRec_t *) hdl)->peerLtkSecLevel;
+        pKey = (dmSecKey_t *) &((appDbRec_t *) hdl)->peerLtk;
+        break;
+
+      case DM_KEY_IRK:
+        pKey = (dmSecKey_t *)&((appDbRec_t *)hdl)->peerIrk;
+        break;
+
+      case DM_KEY_CSRK:
+        pKey = (dmSecKey_t *)&((appDbRec_t *)hdl)->peerCsrk;
+        break;
+
+      default:
+        break;
+    }
+  }
+
+  return pKey;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Set a key in a device database record.
+ *
+ *  \param  hdl       Database record handle.
+ *  \param  pKey      Key data.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbSetKey(appDbHdl_t hdl, dmSecKeyIndEvt_t *pKey)
+{
+  switch(pKey->type)
+  {
+    case DM_KEY_LOCAL_LTK:
+      ((appDbRec_t *) hdl)->localLtkSecLevel = pKey->secLevel;
+      ((appDbRec_t *) hdl)->localLtk = pKey->keyData.ltk;
+      break;
+
+    case DM_KEY_PEER_LTK:
+      ((appDbRec_t *) hdl)->peerLtkSecLevel = pKey->secLevel;
+      ((appDbRec_t *) hdl)->peerLtk = pKey->keyData.ltk;
+      break;
+
+    case DM_KEY_IRK:
+      ((appDbRec_t *)hdl)->peerIrk = pKey->keyData.irk;
+
+      /* make sure peer record is stored using its identity address */
+      ((appDbRec_t *)hdl)->addrType = pKey->keyData.irk.addrType;
+      BdaCpy(((appDbRec_t *)hdl)->peerAddr, pKey->keyData.irk.bdAddr);
+      break;
+
+    case DM_KEY_CSRK:
+      ((appDbRec_t *)hdl)->peerCsrk = pKey->keyData.csrk;
+
+      /* sign counter must be initialized to zero when CSRK is generated */
+      ((appDbRec_t *)hdl)->peerSignCounter = 0;
+      break;
+
+    default:
+      break;
+  }
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Get the client characteristic configuration descriptor table.
+ *
+ *  \param  hdl       Database record handle.
+ *
+ *  \return Pointer to client characteristic configuration descriptor table.
+ */
+/*************************************************************************************************/
+uint16_t *AppDbGetCccTbl(appDbHdl_t hdl)
+{
+  return ((appDbRec_t *) hdl)->cccTbl;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Set a value in the client characteristic configuration table.
+ *
+ *  \param  hdl       Database record handle.
+ *  \param  idx       Table index.
+ *  \param  value     client characteristic configuration value.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbSetCccTblValue(appDbHdl_t hdl, uint16_t idx, uint16_t value)
+{
+  WSF_ASSERT(idx < APP_DB_NUM_CCCD);
+
+  ((appDbRec_t *) hdl)->cccTbl[idx] = value;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Get the discovery status.
+ *
+ *  \param  hdl       Database record handle.
+ *
+ *  \return Discovery status.
+ */
+/*************************************************************************************************/
+uint8_t AppDbGetDiscStatus(appDbHdl_t hdl)
+{
+  return ((appDbRec_t *) hdl)->discStatus;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Set the discovery status.
+ *
+ *  \param  hdl       Database record handle.
+ *  \param  state     Discovery status.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbSetDiscStatus(appDbHdl_t hdl, uint8_t status)
+{
+  ((appDbRec_t *) hdl)->discStatus = status;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Get the cached handle list.
+ *
+ *  \param  hdl       Database record handle.
+ *
+ *  \return Pointer to handle list.
+ */
+/*************************************************************************************************/
+uint16_t *AppDbGetHdlList(appDbHdl_t hdl)
+{
+  return ((appDbRec_t *) hdl)->hdlList;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Set the cached handle list.
+ *
+ *  \param  hdl       Database record handle.
+ *  \param  pHdlList  Pointer to handle list.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbSetHdlList(appDbHdl_t hdl, uint16_t *pHdlList)
+{
+  memcpy(((appDbRec_t *) hdl)->hdlList, pHdlList, sizeof(((appDbRec_t *) hdl)->hdlList));
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Get the device name.
+ *
+ *  \param  pLen      Returned device name length.
+ *
+ *  \return Pointer to UTF-8 string containing device name or NULL if not set.
+ */
+/*************************************************************************************************/
+char *AppDbGetDevName(uint8_t *pLen)
+{
+  /* if first character of name is NULL assume it is uninitialized */
+  if (appDb.devName[0] == 0)
+  {
+    *pLen = 0;
+    return NULL;
+  }
+  else
+  {
+    *pLen = appDb.devNameLen;
+    return appDb.devName;
+  }
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Set the device name.
+ *
+ *  \param  len       Device name length.
+ *  \param  pStr      UTF-8 string containing device name.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbSetDevName(uint8_t len, char *pStr)
+{
+  /* check for maximum device length */
+  len = (len <= sizeof(appDb.devName)) ? len : sizeof(appDb.devName);
+
+  memcpy(appDb.devName, pStr, len);
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Get address resolution attribute value read from a peer device.
+ *
+ *  \param  hdl        Database record handle.
+ *
+ *  \return TRUE if address resolution is supported in peer device. FALSE, otherwise.
+ */
+/*************************************************************************************************/
+bool_t AppDbGetPeerAddrRes(appDbHdl_t hdl)
+{
+  return ((appDbRec_t *)hdl)->peerAddrRes;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Set address resolution attribute value for a peer device.
+ *
+ *  \param  hdl        Database record handle.
+ *  \param  addrRes    Address resolution attribue value.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbSetPeerAddrRes(appDbHdl_t hdl, uint8_t addrRes)
+{
+  ((appDbRec_t *)hdl)->peerAddrRes = addrRes;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Get sign counter for a peer device.
+ *
+ *  \param  hdl        Database record handle.
+ *
+ *  \return Sign counter for peer device.
+ */
+/*************************************************************************************************/
+uint32_t AppDbGetPeerSignCounter(appDbHdl_t hdl)
+{
+  return ((appDbRec_t *)hdl)->peerSignCounter;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Set sign counter for a peer device.
+ *
+ *  \param  hdl          Database record handle.
+ *  \param  signCounter  Sign counter for peer device.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbSetPeerSignCounter(appDbHdl_t hdl, uint32_t signCounter)
+{
+  ((appDbRec_t *)hdl)->peerSignCounter = signCounter;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Get the peer device added to resolving list flag value.
+ *
+ *  \param  hdl        Database record handle.
+ *
+ *  \return TRUE if peer device's been added to resolving list. FALSE, otherwise.
+ */
+/*************************************************************************************************/
+bool_t AppDbGetPeerAddedToRl(appDbHdl_t hdl)
+{
+  return ((appDbRec_t *)hdl)->peerAddedToRl;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Set the peer device added to resolving list flag to a given value.
+ *
+ *  \param  hdl           Database record handle.
+ *  \param  peerAddedToRl Peer device added to resolving list flag value.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbSetPeerAddedToRl(appDbHdl_t hdl, bool_t peerAddedToRl)
+{
+  ((appDbRec_t *)hdl)->peerAddedToRl = peerAddedToRl;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Get the resolvable private address only attribute flag for a given peer device.
+ *
+ *  \param  hdl        Database record handle.
+ *
+ *  \return TRUE if RPA Only attribute is present on peer device. FALSE, otherwise.
+ */
+/*************************************************************************************************/
+bool_t AppDbGetPeerRpao(appDbHdl_t hdl)
+{
+  return ((appDbRec_t *)hdl)->peerRpao;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Set the resolvable private address only attribute flag for a given peer device.
+ *
+ *  \param  hdl        Database record handle.
+ *  \param  peerRpao   Resolvable private address only attribute flag value.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppDbSetPeerRpao(appDbHdl_t hdl, bool_t peerRpao)
+{
+  ((appDbRec_t *)hdl)->peerRpao = peerRpao;
+}
diff --git a/epicardium/ble/app/common/app_ui.c b/epicardium/ble/app/common/app_ui.c
new file mode 100644
index 0000000000000000000000000000000000000000..d5400fd747957bcf6fa3167990358fc0ac7fd585
--- /dev/null
+++ b/epicardium/ble/app/common/app_ui.c
@@ -0,0 +1,332 @@
+/*************************************************************************************************/
+/*!
+ *  \file
+ *
+ *  \brief  Application framework user interface.
+ *
+ *  Copyright (c) 2011-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 "wsf_types.h"
+#include "wsf_os.h"
+#include "wsf_trace.h"
+#include "app_ui.h"
+
+/* card10:
+ * copied from: lib/sdk/Libraries/BTLE/stack/ble-profiles/sources/apps/app/common/app_ui.c
+ *
+ * Reason: has several user interactions which we likley have to implement
+ */
+
+/**************************************************************************************************
+  Global Variables
+**************************************************************************************************/
+
+/*! \brief Callback struct */
+appUiCback_t appUiCbackTbl;
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Perform a user interface action based on the event value passed to the function.
+ *
+ *  \param  event   User interface event value.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppUiAction(uint8_t event)
+{
+  switch (event)
+  {
+    case APP_UI_NONE:
+      /* no action */
+      break;
+
+    case APP_UI_RESET_CMPL:
+      APP_TRACE_INFO0(">>> Reset complete <<<");
+      break;
+
+    case APP_UI_ADV_START:
+      APP_TRACE_INFO0(">>> Advertising started <<<");
+      break;
+
+    case APP_UI_ADV_STOP:
+      APP_TRACE_INFO0(">>> Advertising stopped <<<");
+      break;
+
+    case APP_UI_SCAN_START:
+      APP_TRACE_INFO0(">>> Scanning started <<<");
+      break;
+
+    case APP_UI_SCAN_STOP:
+      APP_TRACE_INFO0(">>> Scanning stopped <<<");
+      break;
+
+    case APP_UI_SCAN_REPORT:
+      APP_TRACE_INFO0(">>> Scan data received from peer <<<");
+      break;
+
+    case APP_UI_CONN_OPEN:
+      APP_TRACE_INFO0(">>> Connection opened <<<");
+      break;
+
+    case APP_UI_CONN_CLOSE:
+      APP_TRACE_INFO0(">>> Connection closed <<<");
+      break;
+
+    case APP_UI_SEC_PAIR_CMPL:
+      APP_TRACE_INFO0(">>> Pairing completed successfully <<<");
+      break;
+
+    case APP_UI_SEC_PAIR_FAIL:
+      APP_TRACE_INFO0(">>> Pairing failed <<<");
+      break;
+
+    case APP_UI_SEC_ENCRYPT:
+      APP_TRACE_INFO0(">>> Connection encrypted <<<");
+      break;
+
+    case APP_UI_SEC_ENCRYPT_FAIL:
+      APP_TRACE_INFO0(">>> Encryption failed <<<");
+      break;
+
+    case APP_UI_PASSKEY_PROMPT:
+      APP_TRACE_INFO0(">>> Prompt user to enter passkey <<<");
+      break;
+
+    case APP_UI_ALERT_CANCEL:
+      APP_TRACE_INFO0(">>> Cancel a low or high alert <<<");
+      break;
+
+    case APP_UI_ALERT_LOW:
+      APP_TRACE_INFO0(">>> Low alert <<<");
+      break;
+
+    case APP_UI_ALERT_HIGH:
+      APP_TRACE_INFO0(">>> High alert <<<");
+      break;
+
+    case APP_UI_ADV_SET_START_IND:
+      APP_TRACE_INFO0(">>> Advertising set(s) started <<<");
+      break;
+
+    case APP_UI_ADV_SET_STOP_IND:
+      APP_TRACE_INFO0(">>> Advertising set(s) stopped <<<");
+      break;
+
+    case APP_UI_SCAN_REQ_RCVD_IND:
+      APP_TRACE_INFO0(">>> Scan request received <<<");
+      break;
+
+    case APP_UI_EXT_SCAN_START_IND:
+      APP_TRACE_INFO0(">>> Extended scanning started <<<");
+      break;
+
+    case APP_UI_EXT_SCAN_STOP_IND:
+      APP_TRACE_INFO0(">>> Extended scanning stopped <<<");
+      break;
+
+    case APP_UI_PER_ADV_SET_START_IND:
+      APP_TRACE_INFO0(">>> Periodic advertising set started <<<");
+      break;
+
+    case APP_UI_PER_ADV_SET_STOP_IND:
+      APP_TRACE_INFO0(">>> Periodic advertising set stopped <<<");
+      break;
+
+    case APP_UI_PER_ADV_SYNC_EST_IND:
+      APP_TRACE_INFO0(">>> Periodic advertising sync established <<<");
+      break;
+
+    case APP_UI_PER_ADV_SYNC_LOST_IND:
+      APP_TRACE_INFO0(">>> Periodic advertising sync lost <<<");
+      break;
+
+    default:
+      break;
+  }
+
+  if (appUiCbackTbl.actionCback)
+  {
+    (*appUiCbackTbl.actionCback)(event);
+  }
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Display a passkey.
+ *
+ *  \param  passkey   Passkey to display.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppUiDisplayPasskey(uint32_t passkey)
+{
+  APP_TRACE_INFO1(">>> Passkey: %d <<<", passkey);
+}
+
+/*************************************************************************************************/
+/*!
+*  \brief  Display a confirmation value.
+*
+*  \param  confirm    Confirm value to display.
+*
+*  \return None.
+*/
+/*************************************************************************************************/
+void AppUiDisplayConfirmValue(uint32_t confirm)
+{
+  APP_TRACE_INFO1(">>> Confirm Value: %d <<<", confirm);
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Display an RSSI value.
+ *
+ *  \param  rssi   Rssi value to display.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppUiDisplayRssi(int8_t rssi)
+{
+  APP_TRACE_INFO1(">>> RSSI: %d dBm<<<", rssi);
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Handle a UI timer expiration event.
+ *
+ *  \param  pMsg    Pointer to message.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void appUiTimerExpired(wsfMsgHdr_t *pMsg)
+{
+
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Perform button press polling.  This function is called to handle WSF
+ *          message APP_BTN_POLL_IND.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void appUiBtnPoll(void)
+{
+  if (appUiCbackTbl.btnPollCback)
+  {
+    (*appUiCbackTbl.btnPollCback)();
+  }
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Handle a hardware button press.  This function is called to handle WSF
+ *          event APP_BTN_DOWN_EVT.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppUiBtnPressed(void)
+{
+
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Register a callback function to receive application button press events.
+ *
+ *  \return None.
+ *
+ *  \note   Registered by application to receive button events
+ */
+/*************************************************************************************************/
+void AppUiBtnRegister(appUiBtnCback_t btnCback)
+{
+  appUiCbackTbl.btnCback = btnCback;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Register a callback function to receive action events.
+ *
+ *  \return None.
+ *
+ *  \note   Registered by platform
+ */
+/*************************************************************************************************/
+void AppUiActionRegister(appUiActionCback_t actionCback)
+{
+  appUiCbackTbl.actionCback = actionCback;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Register a callback function to receive APP_BTN_POLL_IND events.
+ *
+ *  \return None.
+ *
+ *  \note   Registered by platform
+ */
+/*************************************************************************************************/
+void AppUiBtnPollRegister(appUiBtnPollCback_t btnPollCback)
+{
+  appUiCbackTbl.btnPollCback = btnPollCback;
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Play a sound.
+ *
+ *  \param  pSound   Pointer to sound tone/duration array.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppUiSoundPlay(const appUiSound_t *pSound)
+{
+
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Stop the sound that is currently playing.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppUiSoundStop(void)
+{
+
+}
+
+/*************************************************************************************************/
+/*!
+ *  \brief  Button test function-- for test purposes only.
+ *
+ *  \return None.
+ */
+/*************************************************************************************************/
+void AppUiBtnTest(uint8_t btn)
+{
+  if (appUiCbackTbl.btnCback)
+  {
+    (*appUiCbackTbl.btnCback)(btn);
+  }
+}
+
diff --git a/epicardium/ble/meson.build b/epicardium/ble/meson.build
index ff3fba90a285f8f1b13262fe57508cd9653fd5ac..9e00b125574d16f6c5c2cd62a2c492cca7ee3b45 100644
--- a/epicardium/ble/meson.build
+++ b/epicardium/ble/meson.build
@@ -3,5 +3,8 @@ ble_sources = files(
   'stack.c',
   'ble_main.c',
   'svc_dis.c',
-  'svc_core.c'
+  'svc_core.c',
+  'app/app_main.c',
+  'app/common/app_db.c',
+  'app/common/app_ui.c'
 )
diff --git a/lib/sdk/Libraries/BTLE/meson.build b/lib/sdk/Libraries/BTLE/meson.build
index cf76a8543e272f52c106514562e8263c7c5b5b9f..36d6cda75c8538bc56c88915478c9822abd9915d 100644
--- a/lib/sdk/Libraries/BTLE/meson.build
+++ b/lib/sdk/Libraries/BTLE/meson.build
@@ -59,11 +59,11 @@ sources = files(
 'stack/ble-profiles/sources/apps/app/app_master_ae.c',
 'stack/ble-profiles/sources/apps/app/app_slave_leg.c',
 'stack/ble-profiles/sources/apps/app/app_slave_ae.c',
-'stack/ble-profiles/sources/apps/app/app_main.c',
+#'stack/ble-profiles/sources/apps/app/app_main.c',
 'stack/ble-profiles/sources/apps/app/app_server.c',
 'stack/ble-profiles/sources/apps/app/app_master_leg.c',
-'stack/ble-profiles/sources/apps/app/common/app_db.c',
-'stack/ble-profiles/sources/apps/app/common/app_ui.c',
+#'stack/ble-profiles/sources/apps/app/common/app_db.c',
+#'stack/ble-profiles/sources/apps/app/common/app_ui.c',
 'stack/ble-profiles/sources/apps/app/common/app_hw.c',
 'stack/ble-profiles/sources/apps/app/app_terminal.c',
 'stack/ble-profiles/sources/apps/wdxs/wdxs_oad.c',