mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
a6ceeec29f
There is a corner cases in which the counting of allocated memory previously was wrong: When the allocation of the chunk succeeded but the allocation of the next struct node fails. This was relatively unlikely to happen, as the chunk size was much bigger than the memory required by the struct node. But it happens on the ESP32 boards resulting in failing nightlies. This fixes the issue.
151 lines
3.5 KiB
C
151 lines
3.5 KiB
C
/*
|
|
* Copyright (C) 2013 Benjamin Valentin <benpicco@zedat.fu-berlin.de>
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
|
* Public License v2.1. See the file LICENSE in the top level directory for more
|
|
* details.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup tests
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Simple malloc/free test
|
|
*
|
|
*
|
|
* @author Benjamin Valentin <benpicco@zedat.fu-berlin.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <inttypes.h>
|
|
|
|
#include "test_utils/expect.h"
|
|
|
|
#ifndef CHUNK_SIZE
|
|
#ifdef BOARD_NATIVE
|
|
#define CHUNK_SIZE (1024 * 1024U)
|
|
#else
|
|
#define CHUNK_SIZE (128U)
|
|
#endif
|
|
#endif
|
|
|
|
#ifndef NUMBER_OF_TESTS
|
|
#define NUMBER_OF_TESTS 3
|
|
#endif
|
|
|
|
#ifndef MAX_MEM
|
|
#define MAX_MEM (256 * 1024UL * 1024UL)
|
|
#endif
|
|
|
|
struct node {
|
|
struct node *next;
|
|
void *ptr;
|
|
};
|
|
|
|
static uint32_t total = 0;
|
|
|
|
static uint32_t fill_memory(struct node *head)
|
|
{
|
|
uint32_t allocations = 0;
|
|
|
|
if (head) {
|
|
head->next = NULL;
|
|
}
|
|
|
|
total = 0;
|
|
while (head && (head->ptr = malloc(CHUNK_SIZE)) && total < MAX_MEM) {
|
|
printf("Allocated %"PRIu32" Bytes at 0x%p, total %"PRIu32"\n",
|
|
(uint32_t)CHUNK_SIZE, head->ptr, total += CHUNK_SIZE);
|
|
memset(head->ptr, '@', CHUNK_SIZE);
|
|
head = head->next = malloc(sizeof(struct node));
|
|
if (head) {
|
|
total += sizeof(struct node);
|
|
head->ptr = 0;
|
|
head->next = 0;
|
|
}
|
|
allocations++;
|
|
}
|
|
|
|
printf("Allocations count: %"PRIu32"\n", allocations);
|
|
|
|
return allocations;
|
|
}
|
|
|
|
static void free_memory(struct node *head)
|
|
{
|
|
struct node *old_head;
|
|
|
|
uint32_t freed = 0;
|
|
|
|
while (head) {
|
|
if (head->ptr) {
|
|
if (total >= CHUNK_SIZE) {
|
|
total -= CHUNK_SIZE;
|
|
freed++;
|
|
}
|
|
printf("Free %"PRIu32" Bytes at 0x%p, total %"PRIu32"\n",
|
|
(uint32_t)CHUNK_SIZE, head->ptr, total);
|
|
free(head->ptr);
|
|
}
|
|
|
|
if (head->next) {
|
|
old_head = head;
|
|
head = head->next;
|
|
free(old_head);
|
|
}
|
|
else {
|
|
free(head);
|
|
head = 0;
|
|
}
|
|
|
|
total -= sizeof(struct node);
|
|
}
|
|
|
|
printf("Free count: %"PRIu32"\n", freed);
|
|
}
|
|
|
|
int main(void)
|
|
{
|
|
uint32_t allocations = 0;
|
|
|
|
/* modern compilers warn about nonsense calls to calloc, but this is exactly what we want to
|
|
* test */
|
|
#pragma GCC diagnostic push
|
|
#if !defined(__clang__) && (__GNUC__ > 6)
|
|
#pragma GCC diagnostic ignored "-Walloc-size-larger-than="
|
|
#endif
|
|
/* test if an overflow is correctly detected by calloc(): the size below overflows by 1 byte */
|
|
/* cppcheck-suppress leakReturnValNotUsed; (should return NULL, so nothing to free anyway) */
|
|
expect(NULL == calloc(SIZE_MAX / 16 + 1, 16));
|
|
#pragma GCC diagnostic pop
|
|
|
|
printf("CHUNK_SIZE: %"PRIu32"\n", (uint32_t)CHUNK_SIZE);
|
|
printf("NUMBER_OF_TESTS: %d\n", NUMBER_OF_TESTS);
|
|
|
|
uint8_t test = NUMBER_OF_TESTS;
|
|
while (test--) {
|
|
struct node *head = malloc(sizeof(struct node));
|
|
total += sizeof(struct node);
|
|
|
|
uint32_t new_allocations = fill_memory(head);
|
|
free_memory(head);
|
|
|
|
/* test if all memory was freed/can be allocated again */
|
|
if (allocations == 0) {
|
|
allocations = new_allocations;
|
|
} else if (allocations != new_allocations) {
|
|
puts("[FAILED]");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
puts("[SUCCESS]");
|
|
|
|
return 0;
|
|
}
|