xref: /btstack/tool/btstack_memory_generator.py (revision 6bdecec7ba8f55d805af8d13b9f45c3a7f6d5810)
15c544019SMatthias Ringwald#!/usr/bin/env python3
2c0a711d9SMatthias Ringwaldimport os
3c0a711d9SMatthias Ringwaldimport sys
41ca3442bSMatthias Ringwald
5a2673d88SMatthias Ringwaldimport os
6a2673d88SMatthias Ringwaldimport sys
7a2673d88SMatthias Ringwald
81ca3442bSMatthias Ringwaldcopyright = """/*
91ca3442bSMatthias Ringwald * Copyright (C) 2014 BlueKitchen GmbH
101ca3442bSMatthias Ringwald *
111ca3442bSMatthias Ringwald * Redistribution and use in source and binary forms, with or without
121ca3442bSMatthias Ringwald * modification, are permitted provided that the following conditions
131ca3442bSMatthias Ringwald * are met:
141ca3442bSMatthias Ringwald *
151ca3442bSMatthias Ringwald * 1. Redistributions of source code must retain the above copyright
161ca3442bSMatthias Ringwald *    notice, this list of conditions and the following disclaimer.
171ca3442bSMatthias Ringwald * 2. Redistributions in binary form must reproduce the above copyright
181ca3442bSMatthias Ringwald *    notice, this list of conditions and the following disclaimer in the
191ca3442bSMatthias Ringwald *    documentation and/or other materials provided with the distribution.
201ca3442bSMatthias Ringwald * 3. Neither the name of the copyright holders nor the names of
211ca3442bSMatthias Ringwald *    contributors may be used to endorse or promote products derived
221ca3442bSMatthias Ringwald *    from this software without specific prior written permission.
231ca3442bSMatthias Ringwald * 4. Any redistribution, use, or modification is done solely for
241ca3442bSMatthias Ringwald *    personal benefit and not for any commercial purpose or for
251ca3442bSMatthias Ringwald *    monetary gain.
261ca3442bSMatthias Ringwald *
271ca3442bSMatthias Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS
281ca3442bSMatthias Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
291ca3442bSMatthias Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
301ca3442bSMatthias Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS
311ca3442bSMatthias Ringwald * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
321ca3442bSMatthias Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
331ca3442bSMatthias Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
341ca3442bSMatthias Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
351ca3442bSMatthias Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
361ca3442bSMatthias Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
371ca3442bSMatthias Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
381ca3442bSMatthias Ringwald * SUCH DAMAGE.
391ca3442bSMatthias Ringwald *
401ca3442bSMatthias Ringwald * Please inquire about commercial licensing options at
411ca3442bSMatthias Ringwald * [email protected]
421ca3442bSMatthias Ringwald *
431ca3442bSMatthias Ringwald */
441ca3442bSMatthias Ringwald"""
451ca3442bSMatthias Ringwald
461ca3442bSMatthias Ringwaldhfile_header_begin = """
471ca3442bSMatthias Ringwald
481ca3442bSMatthias Ringwald/*
491ca3442bSMatthias Ringwald *  btstack_memory.h
501ca3442bSMatthias Ringwald *
511ca3442bSMatthias Ringwald *  @brief BTstack memory management via configurable memory pools
521ca3442bSMatthias Ringwald *
531ca3442bSMatthias Ringwald */
541ca3442bSMatthias Ringwald
5580e33422SMatthias Ringwald#ifndef BTSTACK_MEMORY_H
5680e33422SMatthias Ringwald#define BTSTACK_MEMORY_H
571ca3442bSMatthias Ringwald
581ca3442bSMatthias Ringwald#if defined __cplusplus
591ca3442bSMatthias Ringwaldextern "C" {
601ca3442bSMatthias Ringwald#endif
611ca3442bSMatthias Ringwald
627907f069SMatthias Ringwald#include "btstack_config.h"
631ca3442bSMatthias Ringwald
643edc84c5SMatthias Ringwald// Core
651ca3442bSMatthias Ringwald#include "hci.h"
661ca3442bSMatthias Ringwald#include "l2cap.h"
671ca3442bSMatthias Ringwald
683edc84c5SMatthias Ringwald// Classic
69208d3378SMilanka Ringwald#include "classic/avdtp_sink.h"
70208d3378SMilanka Ringwald#include "classic/avdtp_source.h"
71be32e7f1SMilanka Ringwald#include "classic/avrcp.h"
72f399f7fbSMilanka Ringwald#include "classic/bnep.h"
73f399f7fbSMilanka Ringwald#include "classic/btstack_link_key_db.h"
74f399f7fbSMilanka Ringwald#include "classic/btstack_link_key_db_memory.h"
75f399f7fbSMilanka Ringwald#include "classic/hfp.h"
76f399f7fbSMilanka Ringwald#include "classic/hid_host.h"
77f399f7fbSMilanka Ringwald#include "classic/rfcomm.h"
78f399f7fbSMilanka Ringwald#include "classic/sdp_server.h"
793edc84c5SMatthias Ringwald
803edc84c5SMatthias Ringwald// BLE
81a9a4c409SMatthias Ringwald#ifdef ENABLE_BLE
823edc84c5SMatthias Ringwald#include "ble/gatt_client.h"
83*6bdecec7SMatthias Ringwald#include "ble/gatt-service/battery_service_client.h"
843edc84c5SMatthias Ringwald#include "ble/sm.h"
8544c5d856SMatthias Ringwald#endif
8644c5d856SMatthias Ringwald
8744c5d856SMatthias Ringwald#ifdef ENABLE_MESH
8877ba3d3fSMatthias Ringwald#include "mesh/mesh_network.h"
89a5a7b6daSMatthias Ringwald#include "mesh/mesh_keys.h"
90a5a7b6daSMatthias Ringwald#include "mesh/mesh_virtual_addresses.h"
911ca3442bSMatthias Ringwald#endif
921ca3442bSMatthias Ringwald
931ca3442bSMatthias Ringwald/* API_START */
941ca3442bSMatthias Ringwald
951ca3442bSMatthias Ringwald/**
961ca3442bSMatthias Ringwald * @brief Initializes BTstack memory pools.
971ca3442bSMatthias Ringwald */
981ca3442bSMatthias Ringwaldvoid btstack_memory_init(void);
991ca3442bSMatthias Ringwald
100b6269742SMatthias Ringwald/**
101b6269742SMatthias Ringwald * @brief Deinitialize BTstack memory pools
102b6269742SMatthias Ringwald * @note if HAVE_MALLOC is defined, all previously allocated buffers are free'd
103b6269742SMatthias Ringwald */
104b6269742SMatthias Ringwaldvoid btstack_memory_deinit(void);
105b6269742SMatthias Ringwald
1061ca3442bSMatthias Ringwald/* API_END */
1071ca3442bSMatthias Ringwald"""
1081ca3442bSMatthias Ringwald
1091ca3442bSMatthias Ringwaldhfile_header_end = """
1101ca3442bSMatthias Ringwald#if defined __cplusplus
1111ca3442bSMatthias Ringwald}
1121ca3442bSMatthias Ringwald#endif
1131ca3442bSMatthias Ringwald
11480e33422SMatthias Ringwald#endif // BTSTACK_MEMORY_H
1151ca3442bSMatthias Ringwald"""
1161ca3442bSMatthias Ringwald
1171ca3442bSMatthias Ringwaldcfile_header_begin = """
1185c544019SMatthias Ringwald#define BTSTACK_FILE__ "btstack_memory.c"
1195c544019SMatthias Ringwald
1205c544019SMatthias Ringwald
1211ca3442bSMatthias Ringwald/*
1225c544019SMatthias Ringwald *  btstack_memory.c
1231ca3442bSMatthias Ringwald *
1241ca3442bSMatthias Ringwald *  @brief BTstack memory management via configurable memory pools
1251ca3442bSMatthias Ringwald *
126a98592bcSMatthias Ringwald *  @note code generated by tool/btstack_memory_generator.py
127a2673d88SMatthias Ringwald *  @note returnes buffers are initialized with 0
1281ca3442bSMatthias Ringwald *
1291ca3442bSMatthias Ringwald */
1301ca3442bSMatthias Ringwald
1311ca3442bSMatthias Ringwald#include "btstack_memory.h"
132d2e6c4b7SMatthias Ringwald#include "btstack_memory_pool.h"
133b6269742SMatthias Ringwald#include "btstack_debug.h"
1341ca3442bSMatthias Ringwald
1351ca3442bSMatthias Ringwald#include <stdlib.h>
1361ca3442bSMatthias Ringwald
137b6269742SMatthias Ringwald#ifdef HAVE_MALLOC
138b6269742SMatthias Ringwaldtypedef struct btstack_memory_buffer {
139b6269742SMatthias Ringwald    struct btstack_memory_buffer * next;
140b6269742SMatthias Ringwald    struct btstack_memory_buffer * prev;
141b6269742SMatthias Ringwald} btstack_memory_buffer_t;
142b6269742SMatthias Ringwald
143798bd46fSMatthias Ringwaldtypedef struct {
144798bd46fSMatthias Ringwald    btstack_memory_buffer_t tracking;
145798bd46fSMatthias Ringwald    void * pointer;
146798bd46fSMatthias Ringwald} test_buffer_t;
147798bd46fSMatthias Ringwald
148b6269742SMatthias Ringwaldstatic btstack_memory_buffer_t * btstack_memory_malloc_buffers;
14919ef97d2SMatthias Ringwaldstatic uint32_t btstack_memory_malloc_counter;
150b6269742SMatthias Ringwald
1512a95308bSMatthias Ringwaldstatic void btstack_memory_tracking_add(btstack_memory_buffer_t * buffer){
1522a95308bSMatthias Ringwald    btstack_assert(buffer != NULL);
15319ef97d2SMatthias Ringwald    if (btstack_memory_malloc_buffers != NULL) {
15419ef97d2SMatthias Ringwald        // let current first item prev point to new first item
15519ef97d2SMatthias Ringwald        btstack_memory_malloc_buffers->prev = buffer;
15619ef97d2SMatthias Ringwald    }
157b6269742SMatthias Ringwald    buffer->prev = NULL;
158b6269742SMatthias Ringwald    buffer->next = btstack_memory_malloc_buffers;
159b6269742SMatthias Ringwald    btstack_memory_malloc_buffers = buffer;
16019ef97d2SMatthias Ringwald
16119ef97d2SMatthias Ringwald    btstack_memory_malloc_counter++;
162b6269742SMatthias Ringwald}
163b6269742SMatthias Ringwald
1642a95308bSMatthias Ringwaldstatic void btstack_memory_tracking_remove(btstack_memory_buffer_t * buffer){
1652a95308bSMatthias Ringwald    btstack_assert(buffer != NULL);
166b6269742SMatthias Ringwald    if (buffer->prev == NULL){
167b6269742SMatthias Ringwald        // first item
168b6269742SMatthias Ringwald        btstack_memory_malloc_buffers = buffer->next;
169b6269742SMatthias Ringwald    } else {
170b6269742SMatthias Ringwald        buffer->prev->next = buffer->next;
171b6269742SMatthias Ringwald    }
172b6269742SMatthias Ringwald    if (buffer->next != NULL){
173b6269742SMatthias Ringwald        buffer->next->prev = buffer->prev;
174b6269742SMatthias Ringwald    }
17519ef97d2SMatthias Ringwald
17619ef97d2SMatthias Ringwald    btstack_memory_malloc_counter--;
177b6269742SMatthias Ringwald}
178b6269742SMatthias Ringwald#endif
179b6269742SMatthias Ringwald
180b6269742SMatthias Ringwaldvoid btstack_memory_deinit(void){
181b6269742SMatthias Ringwald#ifdef HAVE_MALLOC
182b6269742SMatthias Ringwald    while (btstack_memory_malloc_buffers != NULL){
183b6269742SMatthias Ringwald        btstack_memory_buffer_t * buffer = btstack_memory_malloc_buffers;
184b6269742SMatthias Ringwald        btstack_memory_malloc_buffers = buffer->next;
185b6269742SMatthias Ringwald        free(buffer);
186b6269742SMatthias Ringwald    }
18719ef97d2SMatthias Ringwald    btstack_assert(btstack_memory_malloc_counter == 0);
188b6269742SMatthias Ringwald#endif
189b6269742SMatthias Ringwald}
1901ca3442bSMatthias Ringwald"""
1911ca3442bSMatthias Ringwald
1921ca3442bSMatthias Ringwaldheader_template = """STRUCT_NAME_t * btstack_memory_STRUCT_NAME_get(void);
1931ca3442bSMatthias Ringwaldvoid   btstack_memory_STRUCT_NAME_free(STRUCT_NAME_t *STRUCT_NAME);"""
1941ca3442bSMatthias Ringwald
1951ca3442bSMatthias Ringwaldcode_template = """
1961ca3442bSMatthias Ringwald// MARK: STRUCT_TYPE
197a265b909SMatthias Ringwald#if !defined(HAVE_MALLOC) && !defined(POOL_COUNT)
198a265b909SMatthias Ringwald    #if defined(POOL_COUNT_OLD_NO)
199a265b909SMatthias Ringwald        #error "Deprecated POOL_COUNT_OLD_NO defined instead of POOL_COUNT. Please update your btstack_config.h to use POOL_COUNT."
200a265b909SMatthias Ringwald    #else
201a265b909SMatthias Ringwald        #define POOL_COUNT 0
202a265b909SMatthias Ringwald    #endif
203a265b909SMatthias Ringwald#endif
204a265b909SMatthias Ringwald
2051ca3442bSMatthias Ringwald#ifdef POOL_COUNT
2061ca3442bSMatthias Ringwald#if POOL_COUNT > 0
2071ca3442bSMatthias Ringwaldstatic STRUCT_TYPE STRUCT_NAME_storage[POOL_COUNT];
20829d0c4f7SMatthias Ringwaldstatic btstack_memory_pool_t STRUCT_NAME_pool;
2091ca3442bSMatthias RingwaldSTRUCT_NAME_t * btstack_memory_STRUCT_NAME_get(void){
210a2673d88SMatthias Ringwald    void * buffer = btstack_memory_pool_get(&STRUCT_NAME_pool);
211a2673d88SMatthias Ringwald    if (buffer){
212a2673d88SMatthias Ringwald        memset(buffer, 0, sizeof(STRUCT_TYPE));
213a2673d88SMatthias Ringwald    }
214a2673d88SMatthias Ringwald    return (STRUCT_NAME_t *) buffer;
2151ca3442bSMatthias Ringwald}
2161ca3442bSMatthias Ringwaldvoid btstack_memory_STRUCT_NAME_free(STRUCT_NAME_t *STRUCT_NAME){
21729d0c4f7SMatthias Ringwald    btstack_memory_pool_free(&STRUCT_NAME_pool, STRUCT_NAME);
2181ca3442bSMatthias Ringwald}
2191ca3442bSMatthias Ringwald#else
2201ca3442bSMatthias RingwaldSTRUCT_NAME_t * btstack_memory_STRUCT_NAME_get(void){
2211ca3442bSMatthias Ringwald    return NULL;
2221ca3442bSMatthias Ringwald}
2231ca3442bSMatthias Ringwaldvoid btstack_memory_STRUCT_NAME_free(STRUCT_NAME_t *STRUCT_NAME){
224b6269742SMatthias Ringwald    UNUSED(STRUCT_NAME);
2251ca3442bSMatthias Ringwald};
2261ca3442bSMatthias Ringwald#endif
2271ca3442bSMatthias Ringwald#elif defined(HAVE_MALLOC)
2282a95308bSMatthias Ringwald
2292a95308bSMatthias Ringwaldtypedef struct {
2302a95308bSMatthias Ringwald    btstack_memory_buffer_t tracking;
231798bd46fSMatthias Ringwald    STRUCT_NAME_t data;
2322a95308bSMatthias Ringwald} btstack_memory_STRUCT_NAME_t;
2332a95308bSMatthias Ringwald
2341ca3442bSMatthias RingwaldSTRUCT_NAME_t * btstack_memory_STRUCT_NAME_get(void){
2352a95308bSMatthias Ringwald    btstack_memory_STRUCT_NAME_t * buffer = (btstack_memory_STRUCT_NAME_t *) malloc(sizeof(btstack_memory_STRUCT_NAME_t));
236a2673d88SMatthias Ringwald    if (buffer){
237798bd46fSMatthias Ringwald        memset(buffer, 0, sizeof(btstack_memory_STRUCT_NAME_t));
2382a95308bSMatthias Ringwald        btstack_memory_tracking_add(&buffer->tracking);
2392a95308bSMatthias Ringwald        return &buffer->data;
240b6269742SMatthias Ringwald    } else {
241b6269742SMatthias Ringwald        return NULL;
242a2673d88SMatthias Ringwald    }
2431ca3442bSMatthias Ringwald}
2441ca3442bSMatthias Ringwaldvoid btstack_memory_STRUCT_NAME_free(STRUCT_NAME_t *STRUCT_NAME){
245798bd46fSMatthias Ringwald    // reconstruct buffer start
246798bd46fSMatthias Ringwald    btstack_memory_buffer_t * buffer = &((btstack_memory_buffer_t *) STRUCT_NAME)[-1];
247798bd46fSMatthias Ringwald    btstack_memory_tracking_remove(buffer);
248b6269742SMatthias Ringwald    free(buffer);
2491ca3442bSMatthias Ringwald}
2501ca3442bSMatthias Ringwald#endif
2511ca3442bSMatthias Ringwald"""
252798bd46fSMatthias Ringwaldinit_header = '''
253798bd46fSMatthias Ringwald// init
254798bd46fSMatthias Ringwaldvoid btstack_memory_init(void){
255798bd46fSMatthias Ringwald#ifdef HAVE_MALLOC
256798bd46fSMatthias Ringwald    // assert that there is no unexpected padding for combined buffer
257798bd46fSMatthias Ringwald    btstack_assert(sizeof(test_buffer_t) == sizeof(btstack_memory_buffer_t) + sizeof(void *));
258798bd46fSMatthias Ringwald#endif
259798bd46fSMatthias Ringwald
260798bd46fSMatthias Ringwald'''
2611ca3442bSMatthias Ringwald
2621ca3442bSMatthias Ringwaldinit_template = """#if POOL_COUNT > 0
26329d0c4f7SMatthias Ringwald    btstack_memory_pool_create(&STRUCT_NAME_pool, STRUCT_NAME_storage, POOL_COUNT, sizeof(STRUCT_TYPE));
2641ca3442bSMatthias Ringwald#endif"""
2651ca3442bSMatthias Ringwald
2661ca3442bSMatthias Ringwalddef writeln(f, data):
2671ca3442bSMatthias Ringwald    f.write(data + "\n")
2681ca3442bSMatthias Ringwald
2691ca3442bSMatthias Ringwalddef replacePlaceholder(template, struct_name):
2701ca3442bSMatthias Ringwald    struct_type = struct_name + '_t'
2711ca3442bSMatthias Ringwald    if struct_name.endswith('try'):
272a265b909SMatthias Ringwald        pool_count = "MAX_NR_" + struct_name.upper()[:-3] + "TRIES"
2731ca3442bSMatthias Ringwald    else:
274a265b909SMatthias Ringwald        pool_count = "MAX_NR_" + struct_name.upper() + "S"
275a265b909SMatthias Ringwald    pool_count_old_no = pool_count.replace("MAX_NR_", "MAX_NO_")
276a265b909SMatthias Ringwald    snippet = template.replace("STRUCT_TYPE", struct_type).replace("STRUCT_NAME", struct_name).replace("POOL_COUNT_OLD_NO", pool_count_old_no).replace("POOL_COUNT", pool_count)
2771ca3442bSMatthias Ringwald    return snippet
2781ca3442bSMatthias Ringwald
279b3401248SMatthias Ringwaldlist_of_structs = [
280b3401248SMatthias Ringwald    ["hci_connection"],
281b3401248SMatthias Ringwald    ["l2cap_service", "l2cap_channel"],
28244c5d856SMatthias Ringwald]
28344c5d856SMatthias Ringwaldlist_of_classic_structs = [
284b3401248SMatthias Ringwald    ["rfcomm_multiplexer", "rfcomm_service", "rfcomm_channel"],
2852c455dadSMatthias Ringwald    ["btstack_link_key_db_memory_entry"],
286b3401248SMatthias Ringwald    ["bnep_service", "bnep_channel"],
287b3401248SMatthias Ringwald    ["hfp_connection"],
288f399f7fbSMilanka Ringwald    ["hid_host_connection"],
28927faf85aSMilanka Ringwald    ["service_record_item"],
29012e7f38cSMilanka Ringwald    ["avdtp_stream_endpoint"],
29191451a2bSMilanka Ringwald    ["avdtp_connection"],
292f12a3722SMilanka Ringwald    ["avrcp_connection"],
293ebb73e1fSMatthias Ringwald    ["avrcp_browsing_connection"],
294b3401248SMatthias Ringwald]
295ebb73e1fSMatthias Ringwaldlist_of_le_structs = [
296174a0c1cSMilanka Ringwald    ["battery_service_client", "gatt_client", "sm_lookup_entry", "whitelist_entry"],
29744c5d856SMatthias Ringwald]
29844c5d856SMatthias Ringwaldlist_of_mesh_structs = [
299039cbf1dSMatthias Ringwald    ['mesh_network_pdu', 'mesh_segmented_pdu', 'mesh_upper_transport_pdu', 'mesh_network_key', 'mesh_transport_key', 'mesh_virtual_address', 'mesh_subnet']
300ebb73e1fSMatthias Ringwald]
3011ca3442bSMatthias Ringwald
302a2673d88SMatthias Ringwaldbtstack_root = os.path.abspath(os.path.dirname(sys.argv[0]) + '/..')
303a2673d88SMatthias Ringwaldfile_name = btstack_root + "/src/btstack_memory"
304a2673d88SMatthias Ringwaldprint ('Generating %s.[h|c]' % file_name)
3051ca3442bSMatthias Ringwald
3061ca3442bSMatthias Ringwaldf = open(file_name+".h", "w")
3071ca3442bSMatthias Ringwaldwriteln(f, copyright)
3081ca3442bSMatthias Ringwaldwriteln(f, hfile_header_begin)
3091ca3442bSMatthias Ringwaldfor struct_names in list_of_structs:
3101ca3442bSMatthias Ringwald    writeln(f, "// "+ ", ".join(struct_names))
3111ca3442bSMatthias Ringwald    for struct_name in struct_names:
3121ca3442bSMatthias Ringwald        writeln(f, replacePlaceholder(header_template, struct_name))
3131ca3442bSMatthias Ringwald    writeln(f, "")
31444c5d856SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_CLASSIC")
31544c5d856SMatthias Ringwaldfor struct_names in list_of_classic_structs:
31644c5d856SMatthias Ringwald    writeln(f, "// "+ ", ".join(struct_names))
31744c5d856SMatthias Ringwald    for struct_name in struct_names:
31844c5d856SMatthias Ringwald        writeln(f, replacePlaceholder(header_template, struct_name))
31944c5d856SMatthias Ringwald    writeln(f, "")
32044c5d856SMatthias Ringwaldwriteln(f, "#endif")
321a9a4c409SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_BLE")
3221ca3442bSMatthias Ringwaldfor struct_names in list_of_le_structs:
3231ca3442bSMatthias Ringwald    writeln(f, "// "+ ", ".join(struct_names))
3241ca3442bSMatthias Ringwald    for struct_name in struct_names:
3251ca3442bSMatthias Ringwald        writeln(f, replacePlaceholder(header_template, struct_name))
3261ca3442bSMatthias Ringwaldwriteln(f, "#endif")
32744c5d856SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_MESH")
32844c5d856SMatthias Ringwaldfor struct_names in list_of_mesh_structs:
32944c5d856SMatthias Ringwald    writeln(f, "// "+ ", ".join(struct_names))
33044c5d856SMatthias Ringwald    for struct_name in struct_names:
33144c5d856SMatthias Ringwald        writeln(f, replacePlaceholder(header_template, struct_name))
33244c5d856SMatthias Ringwaldwriteln(f, "#endif")
3331ca3442bSMatthias Ringwaldwriteln(f, hfile_header_end)
3341ca3442bSMatthias Ringwaldf.close();
3351ca3442bSMatthias Ringwald
3361ca3442bSMatthias Ringwald
3371ca3442bSMatthias Ringwaldf = open(file_name+".c", "w")
3381ca3442bSMatthias Ringwaldwriteln(f, copyright)
3391ca3442bSMatthias Ringwaldwriteln(f, cfile_header_begin)
3401ca3442bSMatthias Ringwaldfor struct_names in list_of_structs:
3411ca3442bSMatthias Ringwald    for struct_name in struct_names:
3421ca3442bSMatthias Ringwald        writeln(f, replacePlaceholder(code_template, struct_name))
3431ca3442bSMatthias Ringwald    writeln(f, "")
34444c5d856SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_CLASSIC")
34544c5d856SMatthias Ringwaldfor struct_names in list_of_classic_structs:
34644c5d856SMatthias Ringwald    for struct_name in struct_names:
34744c5d856SMatthias Ringwald        writeln(f, replacePlaceholder(code_template, struct_name))
34844c5d856SMatthias Ringwald    writeln(f, "")
34944c5d856SMatthias Ringwaldwriteln(f, "#endif")
350a9a4c409SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_BLE")
3511ca3442bSMatthias Ringwaldfor struct_names in list_of_le_structs:
3521ca3442bSMatthias Ringwald    for struct_name in struct_names:
3531ca3442bSMatthias Ringwald        writeln(f, replacePlaceholder(code_template, struct_name))
3541ca3442bSMatthias Ringwald    writeln(f, "")
3551ca3442bSMatthias Ringwaldwriteln(f, "#endif")
35644c5d856SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_MESH")
35744c5d856SMatthias Ringwaldfor struct_names in list_of_mesh_structs:
35844c5d856SMatthias Ringwald    for struct_name in struct_names:
35944c5d856SMatthias Ringwald        writeln(f, replacePlaceholder(code_template, struct_name))
36044c5d856SMatthias Ringwald    writeln(f, "")
36144c5d856SMatthias Ringwaldwriteln(f, "#endif")
3621ca3442bSMatthias Ringwald
363798bd46fSMatthias Ringwaldf.write(init_header)
3641ca3442bSMatthias Ringwaldfor struct_names in list_of_structs:
3651ca3442bSMatthias Ringwald    for struct_name in struct_names:
3661ca3442bSMatthias Ringwald        writeln(f, replacePlaceholder(init_template, struct_name))
36744c5d856SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_CLASSIC")
36850dc57fcSMatthias Ringwaldfor struct_names in list_of_classic_structs:
36944c5d856SMatthias Ringwald    for struct_name in struct_names:
37044c5d856SMatthias Ringwald        writeln(f, replacePlaceholder(init_template, struct_name))
37144c5d856SMatthias Ringwaldwriteln(f, "#endif")
372a9a4c409SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_BLE")
3731ca3442bSMatthias Ringwaldfor struct_names in list_of_le_structs:
3741ca3442bSMatthias Ringwald    for struct_name in struct_names:
3751ca3442bSMatthias Ringwald        writeln(f, replacePlaceholder(init_template, struct_name))
3761ca3442bSMatthias Ringwaldwriteln(f, "#endif")
37744c5d856SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_MESH")
37850dc57fcSMatthias Ringwaldfor struct_names in list_of_mesh_structs:
37944c5d856SMatthias Ringwald    for struct_name in struct_names:
38044c5d856SMatthias Ringwald        writeln(f, replacePlaceholder(init_template, struct_name))
38144c5d856SMatthias Ringwaldwriteln(f, "#endif")
3821ca3442bSMatthias Ringwaldwriteln(f, "}")
3831ca3442bSMatthias Ringwaldf.close();
3841ca3442bSMatthias Ringwald
3856d34f98eSMatthias Ringwald# also generate test code
3866d34f98eSMatthias Ringwaldtest_header = """
3876d34f98eSMatthias Ringwald#include <stdint.h>
3886d34f98eSMatthias Ringwald#include <stdio.h>
3896d34f98eSMatthias Ringwald#include <stdlib.h>
3906d34f98eSMatthias Ringwald#include <string.h>
3916d34f98eSMatthias Ringwald
3926d34f98eSMatthias Ringwald// malloc hook
3936d34f98eSMatthias Ringwaldstatic int simulate_no_memory;
3946d34f98eSMatthias Ringwaldextern "C" void * test_malloc(size_t size);
3956d34f98eSMatthias Ringwaldvoid * test_malloc(size_t size){
3966d34f98eSMatthias Ringwald    if (simulate_no_memory) return NULL;
3976d34f98eSMatthias Ringwald    return malloc(size);
3986d34f98eSMatthias Ringwald}
3996d34f98eSMatthias Ringwald
4006d34f98eSMatthias Ringwald#include "btstack_config.h"
4016d34f98eSMatthias Ringwald
4026d34f98eSMatthias Ringwald#include "CppUTest/TestHarness.h"
4036d34f98eSMatthias Ringwald#include "CppUTest/CommandLineTestRunner.h"
4046d34f98eSMatthias Ringwald
4056d34f98eSMatthias Ringwald#include "bluetooth_data_types.h"
4066d34f98eSMatthias Ringwald#include "btstack_util.h"
4076d34f98eSMatthias Ringwald#include "btstack_memory.h"
4086d34f98eSMatthias Ringwald
4096d34f98eSMatthias Ringwald
4106d34f98eSMatthias RingwaldTEST_GROUP(btstack_memory){
4116d34f98eSMatthias Ringwald    void setup(void){
4126d34f98eSMatthias Ringwald        btstack_memory_init();
4136d34f98eSMatthias Ringwald        simulate_no_memory = 0;
4146d34f98eSMatthias Ringwald    }
4156d34f98eSMatthias Ringwald};
4166d34f98eSMatthias Ringwald
4176d34f98eSMatthias Ringwald#ifdef HAVE_MALLOC
4186d34f98eSMatthias RingwaldTEST(btstack_memory, deinit){
4196d34f98eSMatthias Ringwald    // alloc buffers 1,2,3
4206d34f98eSMatthias Ringwald    hci_connection_t * buffer_1 = btstack_memory_hci_connection_get();
4216d34f98eSMatthias Ringwald    hci_connection_t * buffer_2 = btstack_memory_hci_connection_get();
4226d34f98eSMatthias Ringwald    hci_connection_t * buffer_3 = btstack_memory_hci_connection_get();
4236d34f98eSMatthias Ringwald    // free first one in list
4246d34f98eSMatthias Ringwald    btstack_memory_hci_connection_free(buffer_3);
4256d34f98eSMatthias Ringwald    // free second one in list
4266d34f98eSMatthias Ringwald    btstack_memory_hci_connection_free(buffer_1);
4276d34f98eSMatthias Ringwald    // leave buffer in list
4286d34f98eSMatthias Ringwald    (void) buffer_2;
4296d34f98eSMatthias Ringwald    btstack_memory_deinit();
4306d34f98eSMatthias Ringwald}
4316d34f98eSMatthias Ringwald#endif
4326d34f98eSMatthias Ringwald
4336d34f98eSMatthias Ringwald"""
4346d34f98eSMatthias Ringwald
4356d34f98eSMatthias Ringwaldtest_template = """
4366d34f98eSMatthias Ringwald
4376d34f98eSMatthias RingwaldTEST(btstack_memory, STRUCT_NAME_GetAndFree){
4386d34f98eSMatthias Ringwald    STRUCT_NAME_t * context;
4396d34f98eSMatthias Ringwald#ifdef HAVE_MALLOC
4406d34f98eSMatthias Ringwald    context = btstack_memory_STRUCT_NAME_get();
4416d34f98eSMatthias Ringwald    CHECK(context != NULL);
4426d34f98eSMatthias Ringwald    btstack_memory_STRUCT_NAME_free(context);
4436d34f98eSMatthias Ringwald#else
4446d34f98eSMatthias Ringwald#ifdef POOL_COUNT
4456d34f98eSMatthias Ringwald    // single
4466d34f98eSMatthias Ringwald    context = btstack_memory_STRUCT_NAME_get();
4476d34f98eSMatthias Ringwald    CHECK(context != NULL);
4486d34f98eSMatthias Ringwald    btstack_memory_STRUCT_NAME_free(context);
4496d34f98eSMatthias Ringwald#else
4506d34f98eSMatthias Ringwald    // none
4516d34f98eSMatthias Ringwald    context = btstack_memory_STRUCT_NAME_get();
4526d34f98eSMatthias Ringwald    CHECK(context == NULL);
4536d34f98eSMatthias Ringwald    btstack_memory_STRUCT_NAME_free(context);
4546d34f98eSMatthias Ringwald#endif
4556d34f98eSMatthias Ringwald#endif
4566d34f98eSMatthias Ringwald}
4576d34f98eSMatthias Ringwald
4586d34f98eSMatthias RingwaldTEST(btstack_memory, STRUCT_NAME_NotEnoughBuffers){
4596d34f98eSMatthias Ringwald    STRUCT_NAME_t * context;
4606d34f98eSMatthias Ringwald#ifdef HAVE_MALLOC
4616d34f98eSMatthias Ringwald    simulate_no_memory = 1;
4626d34f98eSMatthias Ringwald#else
4636d34f98eSMatthias Ringwald#ifdef POOL_COUNT
4646d34f98eSMatthias Ringwald    int i;
4656d34f98eSMatthias Ringwald    // alloc all static buffers
4666d34f98eSMatthias Ringwald    for (i = 0; i < POOL_COUNT; i++){
4676d34f98eSMatthias Ringwald        context = btstack_memory_STRUCT_NAME_get();
4686d34f98eSMatthias Ringwald        CHECK(context != NULL);
4696d34f98eSMatthias Ringwald    }
4706d34f98eSMatthias Ringwald#endif
4716d34f98eSMatthias Ringwald#endif
4726d34f98eSMatthias Ringwald    // get one more
4736d34f98eSMatthias Ringwald    context = btstack_memory_STRUCT_NAME_get();
4746d34f98eSMatthias Ringwald    CHECK(context == NULL);
4756d34f98eSMatthias Ringwald}
4766d34f98eSMatthias Ringwald"""
4776d34f98eSMatthias Ringwald
4786d34f98eSMatthias Ringwaldtest_footer = """
4796d34f98eSMatthias Ringwaldint main (int argc, const char * argv[]){
4806d34f98eSMatthias Ringwald    return CommandLineTestRunner::RunAllTests(argc, argv);
4816d34f98eSMatthias Ringwald}
4826d34f98eSMatthias Ringwald"""
4836d34f98eSMatthias Ringwald
4846d34f98eSMatthias Ringwaldfile_name = btstack_root + "/test/btstack_memory/btstack_memory_test.c"
4856d34f98eSMatthias Ringwaldprint ('Generating %s' % file_name)
4866d34f98eSMatthias Ringwald
4876d34f98eSMatthias Ringwaldf = open(file_name, "w")
4886d34f98eSMatthias Ringwaldwriteln(f, copyright)
4896d34f98eSMatthias Ringwaldwriteln(f, test_header)
4906d34f98eSMatthias Ringwaldfor struct_names in list_of_structs:
4916d34f98eSMatthias Ringwald    for struct_name in struct_names:
4926d34f98eSMatthias Ringwald        writeln(f, replacePlaceholder(test_template, struct_name))
4936d34f98eSMatthias Ringwaldwriteln(f, "#ifdef ENABLE_CLASSIC")
4946d34f98eSMatthias Ringwaldfor struct_names in list_of_classic_structs:
4956d34f98eSMatthias Ringwald    for struct_name in struct_names:
4966d34f98eSMatthias Ringwald        writeln(f, replacePlaceholder(test_template, struct_name))
4976d34f98eSMatthias Ringwaldwriteln(f, "#endif")
4986d34f98eSMatthias Ringwaldwriteln(f, "#ifdef ENABLE_BLE")
4996d34f98eSMatthias Ringwaldfor struct_names in list_of_le_structs:
5006d34f98eSMatthias Ringwald    for struct_name in struct_names:
5016d34f98eSMatthias Ringwald        writeln(f, replacePlaceholder(test_template, struct_name))
5026d34f98eSMatthias Ringwaldwriteln(f, "#endif")
5036d34f98eSMatthias Ringwaldwriteln(f, "#ifdef ENABLE_MESH")
5046d34f98eSMatthias Ringwaldfor struct_names in list_of_mesh_structs:
5056d34f98eSMatthias Ringwald    for struct_name in struct_names:
5066d34f98eSMatthias Ringwald        writeln(f, replacePlaceholder(test_template, struct_name))
5076d34f98eSMatthias Ringwaldwriteln(f, "#endif")
5086d34f98eSMatthias Ringwaldwriteln(f, test_footer)