xref: /nrf52832-nimble/packages/NimBLE-latest/porting/npl/dummy/src/hci_dummy.c (revision 042d53a763ad75cb1465103098bb88c245d95138)
1*042d53a7SEvalZero /*
2*042d53a7SEvalZero  * Licensed to the Apache Software Foundation (ASF) under one
3*042d53a7SEvalZero  * or more contributor license agreements.  See the NOTICE file
4*042d53a7SEvalZero  * distributed with this work for additional information
5*042d53a7SEvalZero  * regarding copyright ownership.  The ASF licenses this file
6*042d53a7SEvalZero  * to you under the Apache License, Version 2.0 (the
7*042d53a7SEvalZero  * "License"); you may not use this file except in compliance
8*042d53a7SEvalZero  * with the License.  You may obtain a copy of the License at
9*042d53a7SEvalZero  *
10*042d53a7SEvalZero  *  http://www.apache.org/licenses/LICENSE-2.0
11*042d53a7SEvalZero  *
12*042d53a7SEvalZero  * Unless required by applicable law or agreed to in writing,
13*042d53a7SEvalZero  * software distributed under the License is distributed on an
14*042d53a7SEvalZero  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15*042d53a7SEvalZero  * KIND, either express or implied.  See the License for the
16*042d53a7SEvalZero  * specific language governing permissions and limitations
17*042d53a7SEvalZero  * under the License.
18*042d53a7SEvalZero  */
19*042d53a7SEvalZero 
20*042d53a7SEvalZero #include <assert.h>
21*042d53a7SEvalZero #include <stdio.h>
22*042d53a7SEvalZero #include <stdint.h>
23*042d53a7SEvalZero #include "syscfg/syscfg.h"
24*042d53a7SEvalZero #include "sysinit/sysinit.h"
25*042d53a7SEvalZero #include "os/os_mempool.h"
26*042d53a7SEvalZero #include "nimble/ble.h"
27*042d53a7SEvalZero #include "nimble/ble_hci_trans.h"
28*042d53a7SEvalZero #include "nimble/hci_common.h"
29*042d53a7SEvalZero 
30*042d53a7SEvalZero /* HCI packet types */
31*042d53a7SEvalZero #define HCI_PKT_CMD     0x01
32*042d53a7SEvalZero #define HCI_PKT_ACL     0x02
33*042d53a7SEvalZero #define HCI_PKT_EVT     0x04
34*042d53a7SEvalZero #define HCI_PKT_GTL     0x05
35*042d53a7SEvalZero 
36*042d53a7SEvalZero /* Buffers for HCI commands data */
37*042d53a7SEvalZero static uint8_t trans_buf_cmd[BLE_HCI_TRANS_CMD_SZ];
38*042d53a7SEvalZero static uint8_t trans_buf_cmd_allocd;
39*042d53a7SEvalZero 
40*042d53a7SEvalZero /* Buffers for HCI events data */
41*042d53a7SEvalZero static uint8_t trans_buf_evt_hi_pool_buf[ OS_MEMPOOL_BYTES(
42*042d53a7SEvalZero                                             MYNEWT_VAL(BLE_HCI_EVT_HI_BUF_COUNT),
43*042d53a7SEvalZero                                             MYNEWT_VAL(BLE_HCI_EVT_BUF_SIZE)) ];
44*042d53a7SEvalZero static struct os_mempool trans_buf_evt_hi_pool;
45*042d53a7SEvalZero static uint8_t trans_buf_evt_lo_pool_buf[ OS_MEMPOOL_BYTES(
46*042d53a7SEvalZero                                             MYNEWT_VAL(BLE_HCI_EVT_LO_BUF_COUNT),
47*042d53a7SEvalZero                                             MYNEWT_VAL(BLE_HCI_EVT_BUF_SIZE)) ];
48*042d53a7SEvalZero static struct os_mempool trans_buf_evt_lo_pool;
49*042d53a7SEvalZero 
50*042d53a7SEvalZero /* Buffers for HCI ACL data */
51*042d53a7SEvalZero #define ACL_POOL_BLOCK_SIZE OS_ALIGN(MYNEWT_VAL(BLE_ACL_BUF_SIZE) + \
52*042d53a7SEvalZero                                             BLE_MBUF_MEMBLOCK_OVERHEAD + \
53*042d53a7SEvalZero                                             BLE_HCI_DATA_HDR_SZ, OS_ALIGNMENT)
54*042d53a7SEvalZero static uint8_t trans_buf_acl_pool_buf[ OS_MEMPOOL_BYTES(
55*042d53a7SEvalZero                                             MYNEWT_VAL(BLE_ACL_BUF_COUNT),
56*042d53a7SEvalZero                                             ACL_POOL_BLOCK_SIZE) ];
57*042d53a7SEvalZero static struct os_mempool trans_buf_acl_pool;
58*042d53a7SEvalZero static struct os_mbuf_pool trans_buf_acl_mbuf_pool;
59*042d53a7SEvalZero 
60*042d53a7SEvalZero /* Host interface */
61*042d53a7SEvalZero static ble_hci_trans_rx_cmd_fn *trans_rx_cmd_cb;
62*042d53a7SEvalZero static void *trans_rx_cmd_arg;
63*042d53a7SEvalZero static ble_hci_trans_rx_acl_fn *trans_rx_acl_cb;
64*042d53a7SEvalZero static void *trans_rx_acl_arg;
65*042d53a7SEvalZero 
66*042d53a7SEvalZero /* Called by NimBLE host to reset HCI transport state (i.e. on host reset) */
67*042d53a7SEvalZero int
ble_hci_trans_reset(void)68*042d53a7SEvalZero ble_hci_trans_reset(void)
69*042d53a7SEvalZero {
70*042d53a7SEvalZero     return 0;
71*042d53a7SEvalZero }
72*042d53a7SEvalZero 
73*042d53a7SEvalZero /* Called by NimBLE host to setup callbacks from HCI transport */
74*042d53a7SEvalZero void
ble_hci_trans_cfg_hs(ble_hci_trans_rx_cmd_fn * cmd_cb,void * cmd_arg,ble_hci_trans_rx_acl_fn * acl_cb,void * acl_arg)75*042d53a7SEvalZero ble_hci_trans_cfg_hs(ble_hci_trans_rx_cmd_fn *cmd_cb, void *cmd_arg,
76*042d53a7SEvalZero                      ble_hci_trans_rx_acl_fn *acl_cb, void *acl_arg)
77*042d53a7SEvalZero {
78*042d53a7SEvalZero     trans_rx_cmd_cb = cmd_cb;
79*042d53a7SEvalZero     trans_rx_cmd_arg = cmd_arg;
80*042d53a7SEvalZero     trans_rx_acl_cb = acl_cb;
81*042d53a7SEvalZero     trans_rx_acl_arg = acl_arg;
82*042d53a7SEvalZero }
83*042d53a7SEvalZero 
84*042d53a7SEvalZero /*
85*042d53a7SEvalZero  * Called by NimBLE host to allocate buffer for HCI Command packet.
86*042d53a7SEvalZero  * Called by HCI transport to allocate buffer for HCI Event packet.
87*042d53a7SEvalZero  */
88*042d53a7SEvalZero uint8_t *
ble_hci_trans_buf_alloc(int type)89*042d53a7SEvalZero ble_hci_trans_buf_alloc(int type)
90*042d53a7SEvalZero {
91*042d53a7SEvalZero     uint8_t *buf;
92*042d53a7SEvalZero 
93*042d53a7SEvalZero     switch (type) {
94*042d53a7SEvalZero     case BLE_HCI_TRANS_BUF_CMD:
95*042d53a7SEvalZero         assert(!trans_buf_cmd_allocd);
96*042d53a7SEvalZero         trans_buf_cmd_allocd = 1;
97*042d53a7SEvalZero         buf = trans_buf_cmd;
98*042d53a7SEvalZero         break;
99*042d53a7SEvalZero     case BLE_HCI_TRANS_BUF_EVT_HI:
100*042d53a7SEvalZero         buf = os_memblock_get(&trans_buf_evt_hi_pool);
101*042d53a7SEvalZero         if (buf) {
102*042d53a7SEvalZero             break;
103*042d53a7SEvalZero         }
104*042d53a7SEvalZero         /* no break */
105*042d53a7SEvalZero     case BLE_HCI_TRANS_BUF_EVT_LO:
106*042d53a7SEvalZero         buf = os_memblock_get(&trans_buf_evt_lo_pool);
107*042d53a7SEvalZero         break;
108*042d53a7SEvalZero     default:
109*042d53a7SEvalZero         assert(0);
110*042d53a7SEvalZero         buf = NULL;
111*042d53a7SEvalZero     }
112*042d53a7SEvalZero 
113*042d53a7SEvalZero     return buf;
114*042d53a7SEvalZero }
115*042d53a7SEvalZero 
116*042d53a7SEvalZero /*
117*042d53a7SEvalZero  * Called by NimBLE host to free buffer allocated for HCI Event packet.
118*042d53a7SEvalZero  * Called by HCI transport to free buffer allocated for HCI Command packet.
119*042d53a7SEvalZero  */
120*042d53a7SEvalZero void
ble_hci_trans_buf_free(uint8_t * buf)121*042d53a7SEvalZero ble_hci_trans_buf_free(uint8_t *buf)
122*042d53a7SEvalZero {
123*042d53a7SEvalZero     int rc;
124*042d53a7SEvalZero 
125*042d53a7SEvalZero     if (buf == trans_buf_cmd) {
126*042d53a7SEvalZero         assert(trans_buf_cmd_allocd);
127*042d53a7SEvalZero         trans_buf_cmd_allocd = 0;
128*042d53a7SEvalZero     } else if (os_memblock_from(&trans_buf_evt_hi_pool, buf)) {
129*042d53a7SEvalZero         rc = os_memblock_put(&trans_buf_evt_hi_pool, buf);
130*042d53a7SEvalZero         assert(rc == 0);
131*042d53a7SEvalZero     } else {
132*042d53a7SEvalZero         assert(os_memblock_from(&trans_buf_evt_lo_pool, buf));
133*042d53a7SEvalZero         rc = os_memblock_put(&trans_buf_evt_lo_pool, buf);
134*042d53a7SEvalZero         assert(rc == 0);
135*042d53a7SEvalZero     }
136*042d53a7SEvalZero }
137*042d53a7SEvalZero 
138*042d53a7SEvalZero /* Called by NimBLE host to send HCI Command packet over HCI transport */
139*042d53a7SEvalZero int
ble_hci_trans_hs_cmd_tx(uint8_t * cmd)140*042d53a7SEvalZero ble_hci_trans_hs_cmd_tx(uint8_t *cmd)
141*042d53a7SEvalZero {
142*042d53a7SEvalZero     uint8_t *buf = cmd;
143*042d53a7SEvalZero 
144*042d53a7SEvalZero     /*
145*042d53a7SEvalZero      * TODO Send HCI Command packet somewhere.
146*042d53a7SEvalZero      * Buffer pointed by 'cmd' contains complete HCI Command packet as defined
147*042d53a7SEvalZero      * by Core spec.
148*042d53a7SEvalZero      */
149*042d53a7SEvalZero 
150*042d53a7SEvalZero     ble_hci_trans_buf_free(buf);
151*042d53a7SEvalZero 
152*042d53a7SEvalZero     return 0;
153*042d53a7SEvalZero }
154*042d53a7SEvalZero 
155*042d53a7SEvalZero /* Called by NimBLE host to send HCI ACL Data packet over HCI transport */
156*042d53a7SEvalZero int
ble_hci_trans_hs_acl_tx(struct os_mbuf * om)157*042d53a7SEvalZero ble_hci_trans_hs_acl_tx(struct os_mbuf *om)
158*042d53a7SEvalZero {
159*042d53a7SEvalZero     uint8_t *buf = om->om_data;
160*042d53a7SEvalZero 
161*042d53a7SEvalZero     /*
162*042d53a7SEvalZero      * TODO Send HCI ACL Data packet somewhere.
163*042d53a7SEvalZero      * mbuf pointed by 'om' contains complete HCI ACL Data packet as defined
164*042d53a7SEvalZero      * by Core spec.
165*042d53a7SEvalZero      */
166*042d53a7SEvalZero     (void)buf;
167*042d53a7SEvalZero 
168*042d53a7SEvalZero     os_mbuf_free_chain(om);
169*042d53a7SEvalZero 
170*042d53a7SEvalZero     return 0;
171*042d53a7SEvalZero }
172*042d53a7SEvalZero 
173*042d53a7SEvalZero /* Called by application to send HCI ACL Data packet to host */
174*042d53a7SEvalZero int
hci_transport_send_acl_to_host(uint8_t * buf,uint16_t size)175*042d53a7SEvalZero hci_transport_send_acl_to_host(uint8_t *buf, uint16_t size)
176*042d53a7SEvalZero {
177*042d53a7SEvalZero     struct os_mbuf *trans_mbuf;
178*042d53a7SEvalZero     int rc;
179*042d53a7SEvalZero 
180*042d53a7SEvalZero     trans_mbuf = os_mbuf_get_pkthdr(&trans_buf_acl_mbuf_pool,
181*042d53a7SEvalZero                                     sizeof(struct ble_mbuf_hdr));
182*042d53a7SEvalZero     os_mbuf_append(trans_mbuf, buf, size);
183*042d53a7SEvalZero     rc = trans_rx_acl_cb(trans_mbuf, trans_rx_acl_arg);
184*042d53a7SEvalZero 
185*042d53a7SEvalZero     return rc;
186*042d53a7SEvalZero }
187*042d53a7SEvalZero 
188*042d53a7SEvalZero /* Called by application to send HCI Event packet to host */
189*042d53a7SEvalZero int
hci_transport_send_evt_to_host(uint8_t * buf,uint8_t size)190*042d53a7SEvalZero hci_transport_send_evt_to_host(uint8_t *buf, uint8_t size)
191*042d53a7SEvalZero {
192*042d53a7SEvalZero     uint8_t *trans_buf;
193*042d53a7SEvalZero     int rc;
194*042d53a7SEvalZero 
195*042d53a7SEvalZero     /* Allocate LE Advertising Report Event from lo pool only */
196*042d53a7SEvalZero     if ((buf[0] == BLE_HCI_EVCODE_LE_META) &&
197*042d53a7SEvalZero         (buf[2] == BLE_HCI_LE_SUBEV_ADV_RPT)) {
198*042d53a7SEvalZero         trans_buf = ble_hci_trans_buf_alloc(BLE_HCI_TRANS_BUF_EVT_LO);
199*042d53a7SEvalZero         if (!trans_buf) {
200*042d53a7SEvalZero             /* Skip advertising report if we're out of memory */
201*042d53a7SEvalZero             return 0;
202*042d53a7SEvalZero         }
203*042d53a7SEvalZero     } else {
204*042d53a7SEvalZero         trans_buf = ble_hci_trans_buf_alloc(BLE_HCI_TRANS_BUF_EVT_HI);
205*042d53a7SEvalZero     }
206*042d53a7SEvalZero 
207*042d53a7SEvalZero     memcpy(trans_buf, buf, size);
208*042d53a7SEvalZero 
209*042d53a7SEvalZero     rc = trans_rx_cmd_cb(trans_buf, trans_rx_cmd_arg);
210*042d53a7SEvalZero     if (rc != 0) {
211*042d53a7SEvalZero         ble_hci_trans_buf_free(trans_buf);
212*042d53a7SEvalZero     }
213*042d53a7SEvalZero 
214*042d53a7SEvalZero     return rc;
215*042d53a7SEvalZero }
216*042d53a7SEvalZero 
217*042d53a7SEvalZero /* Called by application to initialize transport structures */
218*042d53a7SEvalZero int
hci_transport_init(void)219*042d53a7SEvalZero hci_transport_init(void)
220*042d53a7SEvalZero {
221*042d53a7SEvalZero     int rc;
222*042d53a7SEvalZero 
223*042d53a7SEvalZero     trans_buf_cmd_allocd = 0;
224*042d53a7SEvalZero 
225*042d53a7SEvalZero     rc = os_mempool_init(&trans_buf_acl_pool, MYNEWT_VAL(BLE_ACL_BUF_COUNT),
226*042d53a7SEvalZero                                 ACL_POOL_BLOCK_SIZE, trans_buf_acl_pool_buf,
227*042d53a7SEvalZero                                 "dummy_hci_acl_pool");
228*042d53a7SEvalZero     SYSINIT_PANIC_ASSERT(rc == 0);
229*042d53a7SEvalZero 
230*042d53a7SEvalZero     rc = os_mbuf_pool_init(&trans_buf_acl_mbuf_pool, &trans_buf_acl_pool,
231*042d53a7SEvalZero                                 ACL_POOL_BLOCK_SIZE,
232*042d53a7SEvalZero                                 MYNEWT_VAL(BLE_ACL_BUF_COUNT));
233*042d53a7SEvalZero     SYSINIT_PANIC_ASSERT(rc == 0);
234*042d53a7SEvalZero 
235*042d53a7SEvalZero     rc = os_mempool_init(&trans_buf_evt_hi_pool,
236*042d53a7SEvalZero                                 MYNEWT_VAL(BLE_HCI_EVT_HI_BUF_COUNT),
237*042d53a7SEvalZero                                 MYNEWT_VAL(BLE_HCI_EVT_BUF_SIZE),
238*042d53a7SEvalZero                                 trans_buf_evt_hi_pool_buf,
239*042d53a7SEvalZero                                 "dummy_hci_hci_evt_hi_pool");
240*042d53a7SEvalZero     SYSINIT_PANIC_ASSERT(rc == 0);
241*042d53a7SEvalZero 
242*042d53a7SEvalZero     rc = os_mempool_init(&trans_buf_evt_lo_pool,
243*042d53a7SEvalZero                                 MYNEWT_VAL(BLE_HCI_EVT_LO_BUF_COUNT),
244*042d53a7SEvalZero                                 MYNEWT_VAL(BLE_HCI_EVT_BUF_SIZE),
245*042d53a7SEvalZero                                 trans_buf_evt_lo_pool_buf,
246*042d53a7SEvalZero                                 "dummy_hci_hci_evt_lo_pool");
247*042d53a7SEvalZero     SYSINIT_PANIC_ASSERT(rc == 0);
248*042d53a7SEvalZero 
249*042d53a7SEvalZero     return 0;
250*042d53a7SEvalZero }
251