xref: /btstack/tool/btstack_memory_generator.py (revision dbca66ff5b7d5d6c4b8a34e88f8c8497a15a22f5)
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
30154c56ceSMatthias Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BLUEKITCHEN
31154c56ceSMatthias Ringwald * GMBH 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 *
51b45b7749SMilanka Ringwald * @brief BTstack memory management using 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
69ef83f5c4SMatthias Ringwald#ifdef ENABLE_CLASSIC
70208d3378SMilanka Ringwald#include "classic/avdtp_sink.h"
71208d3378SMilanka Ringwald#include "classic/avdtp_source.h"
72be32e7f1SMilanka Ringwald#include "classic/avrcp.h"
73f399f7fbSMilanka Ringwald#include "classic/bnep.h"
74f399f7fbSMilanka Ringwald#include "classic/btstack_link_key_db.h"
75f399f7fbSMilanka Ringwald#include "classic/btstack_link_key_db_memory.h"
76f399f7fbSMilanka Ringwald#include "classic/hfp.h"
77f399f7fbSMilanka Ringwald#include "classic/hid_host.h"
78f399f7fbSMilanka Ringwald#include "classic/rfcomm.h"
79f399f7fbSMilanka Ringwald#include "classic/sdp_server.h"
80ef83f5c4SMatthias Ringwald#endif
813edc84c5SMatthias Ringwald
823edc84c5SMatthias Ringwald// BLE
83a9a4c409SMatthias Ringwald#ifdef ENABLE_BLE
846bdecec7SMatthias Ringwald#include "ble/gatt-service/battery_service_client.h"
85cf26c8fbSMilanka Ringwald#include "ble/gatt-service/hids_client.h"
8678ae886bSMilanka Ringwald#include "ble/gatt-service/scan_parameters_service_client.h"
87cf26c8fbSMilanka Ringwald#include "ble/gatt_client.h"
88cf26c8fbSMilanka Ringwald#include "ble/sm.h"
8944c5d856SMatthias Ringwald#endif
9044c5d856SMatthias Ringwald
9144c5d856SMatthias Ringwald#ifdef ENABLE_MESH
9277ba3d3fSMatthias Ringwald#include "mesh/mesh_network.h"
93a5a7b6daSMatthias Ringwald#include "mesh/mesh_keys.h"
94a5a7b6daSMatthias Ringwald#include "mesh/mesh_virtual_addresses.h"
951ca3442bSMatthias Ringwald#endif
961ca3442bSMatthias Ringwald
971ca3442bSMatthias Ringwald/* API_START */
981ca3442bSMatthias Ringwald
991ca3442bSMatthias Ringwald/**
1001ca3442bSMatthias Ringwald * @brief Initializes BTstack memory pools.
1011ca3442bSMatthias Ringwald */
1021ca3442bSMatthias Ringwaldvoid btstack_memory_init(void);
1031ca3442bSMatthias Ringwald
104b6269742SMatthias Ringwald/**
105b6269742SMatthias Ringwald * @brief Deinitialize BTstack memory pools
106b6269742SMatthias Ringwald * @note if HAVE_MALLOC is defined, all previously allocated buffers are free'd
107b6269742SMatthias Ringwald */
108b6269742SMatthias Ringwaldvoid btstack_memory_deinit(void);
109b6269742SMatthias Ringwald
1101ca3442bSMatthias Ringwald/* API_END */
1111ca3442bSMatthias Ringwald"""
1121ca3442bSMatthias Ringwald
1131ca3442bSMatthias Ringwaldhfile_header_end = """
1141ca3442bSMatthias Ringwald#if defined __cplusplus
1151ca3442bSMatthias Ringwald}
1161ca3442bSMatthias Ringwald#endif
1171ca3442bSMatthias Ringwald
11880e33422SMatthias Ringwald#endif // BTSTACK_MEMORY_H
1191ca3442bSMatthias Ringwald"""
1201ca3442bSMatthias Ringwald
1211ca3442bSMatthias Ringwaldcfile_header_begin = """
1225c544019SMatthias Ringwald#define BTSTACK_FILE__ "btstack_memory.c"
1235c544019SMatthias Ringwald
1245c544019SMatthias Ringwald
1251ca3442bSMatthias Ringwald/*
1265c544019SMatthias Ringwald *  btstack_memory.c
1271ca3442bSMatthias Ringwald *
1281ca3442bSMatthias Ringwald *  @brief BTstack memory management via configurable memory pools
1291ca3442bSMatthias Ringwald *
130a98592bcSMatthias Ringwald *  @note code generated by tool/btstack_memory_generator.py
131a2673d88SMatthias Ringwald *  @note returnes buffers are initialized with 0
1321ca3442bSMatthias Ringwald *
1331ca3442bSMatthias Ringwald */
1341ca3442bSMatthias Ringwald
1351ca3442bSMatthias Ringwald#include "btstack_memory.h"
136d2e6c4b7SMatthias Ringwald#include "btstack_memory_pool.h"
137b6269742SMatthias Ringwald#include "btstack_debug.h"
1381ca3442bSMatthias Ringwald
1391ca3442bSMatthias Ringwald#include <stdlib.h>
1401ca3442bSMatthias Ringwald
141e3c90686SMatthias Ringwald#ifdef ENABLE_MALLOC_TEST
1424490dd1dSMatthias Ringwaldvoid * test_malloc(size_t size);
143e3c90686SMatthias Ringwald#define malloc test_malloc
144e3c90686SMatthias Ringwald#endif
145e3c90686SMatthias Ringwald
146b6269742SMatthias Ringwald#ifdef HAVE_MALLOC
147b6269742SMatthias Ringwaldtypedef struct btstack_memory_buffer {
148b6269742SMatthias Ringwald    struct btstack_memory_buffer * next;
149b6269742SMatthias Ringwald    struct btstack_memory_buffer * prev;
150b6269742SMatthias Ringwald} btstack_memory_buffer_t;
151b6269742SMatthias Ringwald
152798bd46fSMatthias Ringwaldtypedef struct {
153798bd46fSMatthias Ringwald    btstack_memory_buffer_t tracking;
154798bd46fSMatthias Ringwald    void * pointer;
155798bd46fSMatthias Ringwald} test_buffer_t;
156798bd46fSMatthias Ringwald
157b6269742SMatthias Ringwaldstatic btstack_memory_buffer_t * btstack_memory_malloc_buffers;
15819ef97d2SMatthias Ringwaldstatic uint32_t btstack_memory_malloc_counter;
159b6269742SMatthias Ringwald
1602a95308bSMatthias Ringwaldstatic void btstack_memory_tracking_add(btstack_memory_buffer_t * buffer){
1612a95308bSMatthias Ringwald    btstack_assert(buffer != NULL);
16219ef97d2SMatthias Ringwald    if (btstack_memory_malloc_buffers != NULL) {
16319ef97d2SMatthias Ringwald        // let current first item prev point to new first item
16419ef97d2SMatthias Ringwald        btstack_memory_malloc_buffers->prev = buffer;
16519ef97d2SMatthias Ringwald    }
166b6269742SMatthias Ringwald    buffer->prev = NULL;
167b6269742SMatthias Ringwald    buffer->next = btstack_memory_malloc_buffers;
168b6269742SMatthias Ringwald    btstack_memory_malloc_buffers = buffer;
16919ef97d2SMatthias Ringwald
17019ef97d2SMatthias Ringwald    btstack_memory_malloc_counter++;
171b6269742SMatthias Ringwald}
172b6269742SMatthias Ringwald
1732a95308bSMatthias Ringwaldstatic void btstack_memory_tracking_remove(btstack_memory_buffer_t * buffer){
1742a95308bSMatthias Ringwald    btstack_assert(buffer != NULL);
175b6269742SMatthias Ringwald    if (buffer->prev == NULL){
176b6269742SMatthias Ringwald        // first item
177b6269742SMatthias Ringwald        btstack_memory_malloc_buffers = buffer->next;
178b6269742SMatthias Ringwald    } else {
179b6269742SMatthias Ringwald        buffer->prev->next = buffer->next;
180b6269742SMatthias Ringwald    }
181b6269742SMatthias Ringwald    if (buffer->next != NULL){
182b6269742SMatthias Ringwald        buffer->next->prev = buffer->prev;
183b6269742SMatthias Ringwald    }
18419ef97d2SMatthias Ringwald
18519ef97d2SMatthias Ringwald    btstack_memory_malloc_counter--;
186b6269742SMatthias Ringwald}
187b6269742SMatthias Ringwald#endif
188b6269742SMatthias Ringwald
189b6269742SMatthias Ringwaldvoid btstack_memory_deinit(void){
190b6269742SMatthias Ringwald#ifdef HAVE_MALLOC
191b6269742SMatthias Ringwald    while (btstack_memory_malloc_buffers != NULL){
192b6269742SMatthias Ringwald        btstack_memory_buffer_t * buffer = btstack_memory_malloc_buffers;
193b6269742SMatthias Ringwald        btstack_memory_malloc_buffers = buffer->next;
194b6269742SMatthias Ringwald        free(buffer);
195154c56ceSMatthias Ringwald        btstack_memory_malloc_counter--;
196b6269742SMatthias Ringwald    }
19719ef97d2SMatthias Ringwald    btstack_assert(btstack_memory_malloc_counter == 0);
198b6269742SMatthias Ringwald#endif
199b6269742SMatthias Ringwald}
2001ca3442bSMatthias Ringwald"""
2011ca3442bSMatthias Ringwald
2021ca3442bSMatthias Ringwaldheader_template = """STRUCT_NAME_t * btstack_memory_STRUCT_NAME_get(void);
2031ca3442bSMatthias Ringwaldvoid   btstack_memory_STRUCT_NAME_free(STRUCT_NAME_t *STRUCT_NAME);"""
2041ca3442bSMatthias Ringwald
2051ca3442bSMatthias Ringwaldcode_template = """
2061ca3442bSMatthias Ringwald// MARK: STRUCT_TYPE
207a265b909SMatthias Ringwald#if !defined(HAVE_MALLOC) && !defined(POOL_COUNT)
208a265b909SMatthias Ringwald    #if defined(POOL_COUNT_OLD_NO)
209a265b909SMatthias Ringwald        #error "Deprecated POOL_COUNT_OLD_NO defined instead of POOL_COUNT. Please update your btstack_config.h to use POOL_COUNT."
210a265b909SMatthias Ringwald    #else
211a265b909SMatthias Ringwald        #define POOL_COUNT 0
212a265b909SMatthias Ringwald    #endif
213a265b909SMatthias Ringwald#endif
214a265b909SMatthias Ringwald
2151ca3442bSMatthias Ringwald#ifdef POOL_COUNT
2161ca3442bSMatthias Ringwald#if POOL_COUNT > 0
2171ca3442bSMatthias Ringwaldstatic STRUCT_TYPE STRUCT_NAME_storage[POOL_COUNT];
21829d0c4f7SMatthias Ringwaldstatic btstack_memory_pool_t STRUCT_NAME_pool;
2191ca3442bSMatthias RingwaldSTRUCT_NAME_t * btstack_memory_STRUCT_NAME_get(void){
220a2673d88SMatthias Ringwald    void * buffer = btstack_memory_pool_get(&STRUCT_NAME_pool);
221a2673d88SMatthias Ringwald    if (buffer){
222a2673d88SMatthias Ringwald        memset(buffer, 0, sizeof(STRUCT_TYPE));
223a2673d88SMatthias Ringwald    }
224a2673d88SMatthias Ringwald    return (STRUCT_NAME_t *) buffer;
2251ca3442bSMatthias Ringwald}
2261ca3442bSMatthias Ringwaldvoid btstack_memory_STRUCT_NAME_free(STRUCT_NAME_t *STRUCT_NAME){
22729d0c4f7SMatthias Ringwald    btstack_memory_pool_free(&STRUCT_NAME_pool, STRUCT_NAME);
2281ca3442bSMatthias Ringwald}
2291ca3442bSMatthias Ringwald#else
2301ca3442bSMatthias RingwaldSTRUCT_NAME_t * btstack_memory_STRUCT_NAME_get(void){
2311ca3442bSMatthias Ringwald    return NULL;
2321ca3442bSMatthias Ringwald}
2331ca3442bSMatthias Ringwaldvoid btstack_memory_STRUCT_NAME_free(STRUCT_NAME_t *STRUCT_NAME){
234b6269742SMatthias Ringwald    UNUSED(STRUCT_NAME);
2351ca3442bSMatthias Ringwald};
2361ca3442bSMatthias Ringwald#endif
2371ca3442bSMatthias Ringwald#elif defined(HAVE_MALLOC)
2382a95308bSMatthias Ringwald
2392a95308bSMatthias Ringwaldtypedef struct {
2402a95308bSMatthias Ringwald    btstack_memory_buffer_t tracking;
241798bd46fSMatthias Ringwald    STRUCT_NAME_t data;
2422a95308bSMatthias Ringwald} btstack_memory_STRUCT_NAME_t;
2432a95308bSMatthias Ringwald
2441ca3442bSMatthias RingwaldSTRUCT_NAME_t * btstack_memory_STRUCT_NAME_get(void){
2452a95308bSMatthias Ringwald    btstack_memory_STRUCT_NAME_t * buffer = (btstack_memory_STRUCT_NAME_t *) malloc(sizeof(btstack_memory_STRUCT_NAME_t));
246a2673d88SMatthias Ringwald    if (buffer){
247798bd46fSMatthias Ringwald        memset(buffer, 0, sizeof(btstack_memory_STRUCT_NAME_t));
2482a95308bSMatthias Ringwald        btstack_memory_tracking_add(&buffer->tracking);
2492a95308bSMatthias Ringwald        return &buffer->data;
250b6269742SMatthias Ringwald    } else {
251b6269742SMatthias Ringwald        return NULL;
252a2673d88SMatthias Ringwald    }
2531ca3442bSMatthias Ringwald}
2541ca3442bSMatthias Ringwaldvoid btstack_memory_STRUCT_NAME_free(STRUCT_NAME_t *STRUCT_NAME){
255798bd46fSMatthias Ringwald    // reconstruct buffer start
256798bd46fSMatthias Ringwald    btstack_memory_buffer_t * buffer = &((btstack_memory_buffer_t *) STRUCT_NAME)[-1];
257798bd46fSMatthias Ringwald    btstack_memory_tracking_remove(buffer);
258b6269742SMatthias Ringwald    free(buffer);
2591ca3442bSMatthias Ringwald}
2601ca3442bSMatthias Ringwald#endif
2611ca3442bSMatthias Ringwald"""
262798bd46fSMatthias Ringwaldinit_header = '''
263798bd46fSMatthias Ringwald// init
264798bd46fSMatthias Ringwaldvoid btstack_memory_init(void){
265798bd46fSMatthias Ringwald#ifdef HAVE_MALLOC
266798bd46fSMatthias Ringwald    // assert that there is no unexpected padding for combined buffer
267798bd46fSMatthias Ringwald    btstack_assert(sizeof(test_buffer_t) == sizeof(btstack_memory_buffer_t) + sizeof(void *));
268798bd46fSMatthias Ringwald#endif
269798bd46fSMatthias Ringwald
270798bd46fSMatthias Ringwald'''
2711ca3442bSMatthias Ringwald
2721ca3442bSMatthias Ringwaldinit_template = """#if POOL_COUNT > 0
27329d0c4f7SMatthias Ringwald    btstack_memory_pool_create(&STRUCT_NAME_pool, STRUCT_NAME_storage, POOL_COUNT, sizeof(STRUCT_TYPE));
2741ca3442bSMatthias Ringwald#endif"""
2751ca3442bSMatthias Ringwald
2761ca3442bSMatthias Ringwalddef writeln(f, data):
2771ca3442bSMatthias Ringwald    f.write(data + "\n")
2781ca3442bSMatthias Ringwald
2791ca3442bSMatthias Ringwalddef replacePlaceholder(template, struct_name):
2801ca3442bSMatthias Ringwald    struct_type = struct_name + '_t'
2811ca3442bSMatthias Ringwald    if struct_name.endswith('try'):
282a265b909SMatthias Ringwald        pool_count = "MAX_NR_" + struct_name.upper()[:-3] + "TRIES"
2831ca3442bSMatthias Ringwald    else:
284a265b909SMatthias Ringwald        pool_count = "MAX_NR_" + struct_name.upper() + "S"
285a265b909SMatthias Ringwald    pool_count_old_no = pool_count.replace("MAX_NR_", "MAX_NO_")
286a265b909SMatthias 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)
2871ca3442bSMatthias Ringwald    return snippet
2881ca3442bSMatthias Ringwald
289b3401248SMatthias Ringwaldlist_of_structs = [
290b3401248SMatthias Ringwald    ["hci_connection"],
291b3401248SMatthias Ringwald    ["l2cap_service", "l2cap_channel"],
29244c5d856SMatthias Ringwald]
29344c5d856SMatthias Ringwaldlist_of_classic_structs = [
294b3401248SMatthias Ringwald    ["rfcomm_multiplexer", "rfcomm_service", "rfcomm_channel"],
2952c455dadSMatthias Ringwald    ["btstack_link_key_db_memory_entry"],
296b3401248SMatthias Ringwald    ["bnep_service", "bnep_channel"],
297b3401248SMatthias Ringwald    ["hfp_connection"],
298f399f7fbSMilanka Ringwald    ["hid_host_connection"],
29927faf85aSMilanka Ringwald    ["service_record_item"],
30012e7f38cSMilanka Ringwald    ["avdtp_stream_endpoint"],
30191451a2bSMilanka Ringwald    ["avdtp_connection"],
302f12a3722SMilanka Ringwald    ["avrcp_connection"],
303ebb73e1fSMatthias Ringwald    ["avrcp_browsing_connection"],
304b3401248SMatthias Ringwald]
305ebb73e1fSMatthias Ringwaldlist_of_le_structs = [
306*dbca66ffSMatthias Ringwald    ["battery_service_client", "gatt_client", "hids_client", "scan_parameters_service_client", "sm_lookup_entry", "whitelist_entry", "periodic_advertiser_list_entry"],
30744c5d856SMatthias Ringwald]
30844c5d856SMatthias Ringwaldlist_of_mesh_structs = [
309039cbf1dSMatthias Ringwald    ['mesh_network_pdu', 'mesh_segmented_pdu', 'mesh_upper_transport_pdu', 'mesh_network_key', 'mesh_transport_key', 'mesh_virtual_address', 'mesh_subnet']
310ebb73e1fSMatthias Ringwald]
3111ca3442bSMatthias Ringwald
312a2673d88SMatthias Ringwaldbtstack_root = os.path.abspath(os.path.dirname(sys.argv[0]) + '/..')
313a2673d88SMatthias Ringwaldfile_name = btstack_root + "/src/btstack_memory"
314a2673d88SMatthias Ringwaldprint ('Generating %s.[h|c]' % file_name)
3151ca3442bSMatthias Ringwald
3161ca3442bSMatthias Ringwaldf = open(file_name+".h", "w")
3171ca3442bSMatthias Ringwaldwriteln(f, copyright)
3181ca3442bSMatthias Ringwaldwriteln(f, hfile_header_begin)
3191ca3442bSMatthias Ringwaldfor struct_names in list_of_structs:
3201ca3442bSMatthias Ringwald    writeln(f, "// "+ ", ".join(struct_names))
3211ca3442bSMatthias Ringwald    for struct_name in struct_names:
3221ca3442bSMatthias Ringwald        writeln(f, replacePlaceholder(header_template, struct_name))
3231ca3442bSMatthias Ringwald    writeln(f, "")
32444c5d856SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_CLASSIC")
32544c5d856SMatthias Ringwaldfor struct_names in list_of_classic_structs:
32644c5d856SMatthias Ringwald    writeln(f, "// "+ ", ".join(struct_names))
32744c5d856SMatthias Ringwald    for struct_name in struct_names:
32844c5d856SMatthias Ringwald        writeln(f, replacePlaceholder(header_template, struct_name))
32944c5d856SMatthias Ringwald    writeln(f, "")
33044c5d856SMatthias Ringwaldwriteln(f, "#endif")
331a9a4c409SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_BLE")
3321ca3442bSMatthias Ringwaldfor struct_names in list_of_le_structs:
3331ca3442bSMatthias Ringwald    writeln(f, "// "+ ", ".join(struct_names))
3341ca3442bSMatthias Ringwald    for struct_name in struct_names:
3351ca3442bSMatthias Ringwald        writeln(f, replacePlaceholder(header_template, struct_name))
3361ca3442bSMatthias Ringwaldwriteln(f, "#endif")
33744c5d856SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_MESH")
33844c5d856SMatthias Ringwaldfor struct_names in list_of_mesh_structs:
33944c5d856SMatthias Ringwald    writeln(f, "// "+ ", ".join(struct_names))
34044c5d856SMatthias Ringwald    for struct_name in struct_names:
34144c5d856SMatthias Ringwald        writeln(f, replacePlaceholder(header_template, struct_name))
34244c5d856SMatthias Ringwaldwriteln(f, "#endif")
3431ca3442bSMatthias Ringwaldwriteln(f, hfile_header_end)
3441ca3442bSMatthias Ringwaldf.close();
3451ca3442bSMatthias Ringwald
3461ca3442bSMatthias Ringwald
3471ca3442bSMatthias Ringwaldf = open(file_name+".c", "w")
3481ca3442bSMatthias Ringwaldwriteln(f, copyright)
3491ca3442bSMatthias Ringwaldwriteln(f, cfile_header_begin)
3501ca3442bSMatthias Ringwaldfor struct_names in list_of_structs:
3511ca3442bSMatthias Ringwald    for struct_name in struct_names:
3521ca3442bSMatthias Ringwald        writeln(f, replacePlaceholder(code_template, struct_name))
3531ca3442bSMatthias Ringwald    writeln(f, "")
35444c5d856SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_CLASSIC")
35544c5d856SMatthias Ringwaldfor struct_names in list_of_classic_structs:
35644c5d856SMatthias Ringwald    for struct_name in struct_names:
35744c5d856SMatthias Ringwald        writeln(f, replacePlaceholder(code_template, struct_name))
35844c5d856SMatthias Ringwald    writeln(f, "")
35944c5d856SMatthias Ringwaldwriteln(f, "#endif")
360a9a4c409SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_BLE")
3611ca3442bSMatthias Ringwaldfor struct_names in list_of_le_structs:
3621ca3442bSMatthias Ringwald    for struct_name in struct_names:
3631ca3442bSMatthias Ringwald        writeln(f, replacePlaceholder(code_template, struct_name))
3641ca3442bSMatthias Ringwald    writeln(f, "")
3651ca3442bSMatthias Ringwaldwriteln(f, "#endif")
36644c5d856SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_MESH")
36744c5d856SMatthias Ringwaldfor struct_names in list_of_mesh_structs:
36844c5d856SMatthias Ringwald    for struct_name in struct_names:
36944c5d856SMatthias Ringwald        writeln(f, replacePlaceholder(code_template, struct_name))
37044c5d856SMatthias Ringwald    writeln(f, "")
37144c5d856SMatthias Ringwaldwriteln(f, "#endif")
3721ca3442bSMatthias Ringwald
373798bd46fSMatthias Ringwaldf.write(init_header)
3741ca3442bSMatthias Ringwaldfor struct_names in list_of_structs:
3751ca3442bSMatthias Ringwald    for struct_name in struct_names:
3761ca3442bSMatthias Ringwald        writeln(f, replacePlaceholder(init_template, struct_name))
37744c5d856SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_CLASSIC")
37850dc57fcSMatthias Ringwaldfor struct_names in list_of_classic_structs:
37944c5d856SMatthias Ringwald    for struct_name in struct_names:
38044c5d856SMatthias Ringwald        writeln(f, replacePlaceholder(init_template, struct_name))
38144c5d856SMatthias Ringwaldwriteln(f, "#endif")
382a9a4c409SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_BLE")
3831ca3442bSMatthias Ringwaldfor struct_names in list_of_le_structs:
3841ca3442bSMatthias Ringwald    for struct_name in struct_names:
3851ca3442bSMatthias Ringwald        writeln(f, replacePlaceholder(init_template, struct_name))
3861ca3442bSMatthias Ringwaldwriteln(f, "#endif")
38744c5d856SMatthias Ringwaldwriteln(f, "#ifdef ENABLE_MESH")
38850dc57fcSMatthias Ringwaldfor struct_names in list_of_mesh_structs:
38944c5d856SMatthias Ringwald    for struct_name in struct_names:
39044c5d856SMatthias Ringwald        writeln(f, replacePlaceholder(init_template, struct_name))
39144c5d856SMatthias Ringwaldwriteln(f, "#endif")
3921ca3442bSMatthias Ringwaldwriteln(f, "}")
3931ca3442bSMatthias Ringwaldf.close();
3941ca3442bSMatthias Ringwald
3956d34f98eSMatthias Ringwald# also generate test code
3966d34f98eSMatthias Ringwaldtest_header = """
3976d34f98eSMatthias Ringwald#include <stdint.h>
3986d34f98eSMatthias Ringwald#include <stdio.h>
3996d34f98eSMatthias Ringwald#include <stdlib.h>
4006d34f98eSMatthias Ringwald#include <string.h>
4016d34f98eSMatthias Ringwald
4026d34f98eSMatthias Ringwald// malloc hook
4036d34f98eSMatthias Ringwaldstatic int simulate_no_memory;
4046d34f98eSMatthias Ringwaldextern "C" void * test_malloc(size_t size);
4056d34f98eSMatthias Ringwaldvoid * test_malloc(size_t size){
4066d34f98eSMatthias Ringwald    if (simulate_no_memory) return NULL;
4076d34f98eSMatthias Ringwald    return malloc(size);
4086d34f98eSMatthias Ringwald}
4096d34f98eSMatthias Ringwald
4106d34f98eSMatthias Ringwald#include "btstack_config.h"
4116d34f98eSMatthias Ringwald
4126d34f98eSMatthias Ringwald#include "CppUTest/TestHarness.h"
4136d34f98eSMatthias Ringwald#include "CppUTest/CommandLineTestRunner.h"
4146d34f98eSMatthias Ringwald
4156d34f98eSMatthias Ringwald#include "bluetooth_data_types.h"
4166d34f98eSMatthias Ringwald#include "btstack_util.h"
4176d34f98eSMatthias Ringwald#include "btstack_memory.h"
4186d34f98eSMatthias Ringwald
4196d34f98eSMatthias Ringwald
4206d34f98eSMatthias RingwaldTEST_GROUP(btstack_memory){
4216d34f98eSMatthias Ringwald    void setup(void){
4226d34f98eSMatthias Ringwald        btstack_memory_init();
4236d34f98eSMatthias Ringwald        simulate_no_memory = 0;
4246d34f98eSMatthias Ringwald    }
4256d34f98eSMatthias Ringwald};
4266d34f98eSMatthias Ringwald
4276d34f98eSMatthias Ringwald#ifdef HAVE_MALLOC
4286d34f98eSMatthias RingwaldTEST(btstack_memory, deinit){
4296d34f98eSMatthias Ringwald    // alloc buffers 1,2,3
4306d34f98eSMatthias Ringwald    hci_connection_t * buffer_1 = btstack_memory_hci_connection_get();
4316d34f98eSMatthias Ringwald    hci_connection_t * buffer_2 = btstack_memory_hci_connection_get();
4326d34f98eSMatthias Ringwald    hci_connection_t * buffer_3 = btstack_memory_hci_connection_get();
4336d34f98eSMatthias Ringwald    // free first one in list
4346d34f98eSMatthias Ringwald    btstack_memory_hci_connection_free(buffer_3);
4356d34f98eSMatthias Ringwald    // free second one in list
4366d34f98eSMatthias Ringwald    btstack_memory_hci_connection_free(buffer_1);
4376d34f98eSMatthias Ringwald    // leave buffer in list
4386d34f98eSMatthias Ringwald    (void) buffer_2;
4396d34f98eSMatthias Ringwald    btstack_memory_deinit();
4406d34f98eSMatthias Ringwald}
4416d34f98eSMatthias Ringwald#endif
4426d34f98eSMatthias Ringwald
4436d34f98eSMatthias Ringwald"""
4446d34f98eSMatthias Ringwald
4456d34f98eSMatthias Ringwaldtest_template = """
4466d34f98eSMatthias Ringwald
4476d34f98eSMatthias RingwaldTEST(btstack_memory, STRUCT_NAME_GetAndFree){
4486d34f98eSMatthias Ringwald    STRUCT_NAME_t * context;
4496d34f98eSMatthias Ringwald#ifdef HAVE_MALLOC
4506d34f98eSMatthias Ringwald    context = btstack_memory_STRUCT_NAME_get();
4516d34f98eSMatthias Ringwald    CHECK(context != NULL);
4526d34f98eSMatthias Ringwald    btstack_memory_STRUCT_NAME_free(context);
4536d34f98eSMatthias Ringwald#else
4546d34f98eSMatthias Ringwald#ifdef POOL_COUNT
4556d34f98eSMatthias Ringwald    // single
4566d34f98eSMatthias Ringwald    context = btstack_memory_STRUCT_NAME_get();
4576d34f98eSMatthias Ringwald    CHECK(context != NULL);
4586d34f98eSMatthias Ringwald    btstack_memory_STRUCT_NAME_free(context);
4596d34f98eSMatthias Ringwald#else
4606d34f98eSMatthias Ringwald    // none
4616d34f98eSMatthias Ringwald    context = btstack_memory_STRUCT_NAME_get();
4626d34f98eSMatthias Ringwald    CHECK(context == NULL);
4636d34f98eSMatthias Ringwald    btstack_memory_STRUCT_NAME_free(context);
4646d34f98eSMatthias Ringwald#endif
4656d34f98eSMatthias Ringwald#endif
4666d34f98eSMatthias Ringwald}
4676d34f98eSMatthias Ringwald
4686d34f98eSMatthias RingwaldTEST(btstack_memory, STRUCT_NAME_NotEnoughBuffers){
4696d34f98eSMatthias Ringwald    STRUCT_NAME_t * context;
4706d34f98eSMatthias Ringwald#ifdef HAVE_MALLOC
4716d34f98eSMatthias Ringwald    simulate_no_memory = 1;
4726d34f98eSMatthias Ringwald#else
4736d34f98eSMatthias Ringwald#ifdef POOL_COUNT
4746d34f98eSMatthias Ringwald    int i;
4756d34f98eSMatthias Ringwald    // alloc all static buffers
4766d34f98eSMatthias Ringwald    for (i = 0; i < POOL_COUNT; i++){
4776d34f98eSMatthias Ringwald        context = btstack_memory_STRUCT_NAME_get();
4786d34f98eSMatthias Ringwald        CHECK(context != NULL);
4796d34f98eSMatthias Ringwald    }
4806d34f98eSMatthias Ringwald#endif
4816d34f98eSMatthias Ringwald#endif
4826d34f98eSMatthias Ringwald    // get one more
4836d34f98eSMatthias Ringwald    context = btstack_memory_STRUCT_NAME_get();
4846d34f98eSMatthias Ringwald    CHECK(context == NULL);
4856d34f98eSMatthias Ringwald}
4866d34f98eSMatthias Ringwald"""
4876d34f98eSMatthias Ringwald
4886d34f98eSMatthias Ringwaldtest_footer = """
4896d34f98eSMatthias Ringwaldint main (int argc, const char * argv[]){
4906d34f98eSMatthias Ringwald    return CommandLineTestRunner::RunAllTests(argc, argv);
4916d34f98eSMatthias Ringwald}
4926d34f98eSMatthias Ringwald"""
4936d34f98eSMatthias Ringwald
4946d34f98eSMatthias Ringwaldfile_name = btstack_root + "/test/btstack_memory/btstack_memory_test.c"
4956d34f98eSMatthias Ringwaldprint ('Generating %s' % file_name)
4966d34f98eSMatthias Ringwald
4976d34f98eSMatthias Ringwaldf = open(file_name, "w")
4986d34f98eSMatthias Ringwaldwriteln(f, copyright)
4996d34f98eSMatthias Ringwaldwriteln(f, test_header)
5006d34f98eSMatthias Ringwaldfor struct_names in list_of_structs:
5016d34f98eSMatthias Ringwald    for struct_name in struct_names:
5026d34f98eSMatthias Ringwald        writeln(f, replacePlaceholder(test_template, struct_name))
5036d34f98eSMatthias Ringwaldwriteln(f, "#ifdef ENABLE_CLASSIC")
5046d34f98eSMatthias Ringwaldfor struct_names in list_of_classic_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, "#ifdef ENABLE_BLE")
5096d34f98eSMatthias Ringwaldfor struct_names in list_of_le_structs:
5106d34f98eSMatthias Ringwald    for struct_name in struct_names:
5116d34f98eSMatthias Ringwald        writeln(f, replacePlaceholder(test_template, struct_name))
5126d34f98eSMatthias Ringwaldwriteln(f, "#endif")
5136d34f98eSMatthias Ringwaldwriteln(f, "#ifdef ENABLE_MESH")
5146d34f98eSMatthias Ringwaldfor struct_names in list_of_mesh_structs:
5156d34f98eSMatthias Ringwald    for struct_name in struct_names:
5166d34f98eSMatthias Ringwald        writeln(f, replacePlaceholder(test_template, struct_name))
5176d34f98eSMatthias Ringwaldwriteln(f, "#endif")
5186d34f98eSMatthias Ringwaldwriteln(f, test_footer)