You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
179 lines
6.0 KiB
C
179 lines
6.0 KiB
C
/*
|
|
* FreeRTOS Kernel <DEVELOPMENT BRANCH>
|
|
* Copyright (C) 2021 Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
*
|
|
* SPDX-License-Identifier: MIT
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of
|
|
* this software and associated documentation files (the "Software"), to deal in
|
|
* the Software without restriction, including without limitation the rights to
|
|
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
|
* the Software, and to permit persons to whom the Software is furnished to do so,
|
|
* subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in all
|
|
* copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
|
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
|
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
|
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*
|
|
* https://www.FreeRTOS.org
|
|
* https://github.com/FreeRTOS
|
|
*
|
|
*/
|
|
|
|
|
|
/*
|
|
* The simplest possible implementation of pvPortMalloc(). Note that this
|
|
* implementation does NOT allow allocated memory to be freed again.
|
|
*
|
|
* See heap_2.c, heap_3.c and heap_4.c for alternative implementations, and the
|
|
* memory management pages of https://www.FreeRTOS.org for more information.
|
|
*/
|
|
#include <stdlib.h>
|
|
|
|
/* Defining MPU_WRAPPERS_INCLUDED_FROM_API_FILE prevents task.h from redefining
|
|
* all the API functions to use the MPU wrappers. That should only be done when
|
|
* task.h is included from an application file. */
|
|
#define MPU_WRAPPERS_INCLUDED_FROM_API_FILE
|
|
|
|
#include "FreeRTOS.h"
|
|
#include "task.h"
|
|
|
|
#undef MPU_WRAPPERS_INCLUDED_FROM_API_FILE
|
|
|
|
#if ( configSUPPORT_DYNAMIC_ALLOCATION == 0 )
|
|
#error This file must not be used if configSUPPORT_DYNAMIC_ALLOCATION is 0
|
|
#endif
|
|
|
|
/* A few bytes might be lost to byte aligning the heap start address. */
|
|
#define configADJUSTED_HEAP_SIZE ( configTOTAL_HEAP_SIZE - portBYTE_ALIGNMENT )
|
|
|
|
/* Max value that fits in a size_t type. */
|
|
#define heapSIZE_MAX ( ~( ( size_t ) 0 ) )
|
|
|
|
/* Check if adding a and b will result in overflow. */
|
|
#define heapADD_WILL_OVERFLOW( a, b ) ( ( a ) > ( heapSIZE_MAX - ( b ) ) )
|
|
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/* Allocate the memory for the heap. */
|
|
#if ( configAPPLICATION_ALLOCATED_HEAP == 1 )
|
|
|
|
/* The application writer has already defined the array used for the RTOS
|
|
* heap - probably so it can be placed in a special segment or address. */
|
|
extern uint8_t ucHeap[ configTOTAL_HEAP_SIZE ];
|
|
#else
|
|
static uint8_t ucHeap[ configTOTAL_HEAP_SIZE ];
|
|
#endif /* configAPPLICATION_ALLOCATED_HEAP */
|
|
|
|
/* Index into the ucHeap array. */
|
|
static size_t xNextFreeByte = ( size_t ) 0U;
|
|
|
|
/*-----------------------------------------------------------*/
|
|
|
|
void * pvPortMalloc( size_t xWantedSize )
|
|
{
|
|
void * pvReturn = NULL;
|
|
static uint8_t * pucAlignedHeap = NULL;
|
|
|
|
/* Ensure that blocks are always aligned. */
|
|
#if ( portBYTE_ALIGNMENT != 1 )
|
|
{
|
|
size_t xAdditionalRequiredSize;
|
|
|
|
if( ( xWantedSize & portBYTE_ALIGNMENT_MASK ) != 0x00 )
|
|
{
|
|
/* Byte alignment required. */
|
|
xAdditionalRequiredSize = portBYTE_ALIGNMENT - ( xWantedSize & portBYTE_ALIGNMENT_MASK );
|
|
|
|
if( heapADD_WILL_OVERFLOW( xWantedSize, xAdditionalRequiredSize ) == 0 )
|
|
{
|
|
xWantedSize += xAdditionalRequiredSize;
|
|
}
|
|
else
|
|
{
|
|
xWantedSize = 0;
|
|
}
|
|
}
|
|
}
|
|
#endif /* if ( portBYTE_ALIGNMENT != 1 ) */
|
|
|
|
vTaskSuspendAll();
|
|
{
|
|
if( pucAlignedHeap == NULL )
|
|
{
|
|
/* Ensure the heap starts on a correctly aligned boundary. */
|
|
pucAlignedHeap = ( uint8_t * ) ( ( ( portPOINTER_SIZE_TYPE ) &( ucHeap[ portBYTE_ALIGNMENT - 1 ] ) ) &
|
|
( ~( ( portPOINTER_SIZE_TYPE ) portBYTE_ALIGNMENT_MASK ) ) );
|
|
}
|
|
|
|
/* Check there is enough room left for the allocation. */
|
|
if( ( xWantedSize > 0 ) &&
|
|
( heapADD_WILL_OVERFLOW( xNextFreeByte, xWantedSize ) == 0 ) &&
|
|
( ( xNextFreeByte + xWantedSize ) < configADJUSTED_HEAP_SIZE ) )
|
|
{
|
|
/* Return the next free byte then increment the index past this
|
|
* block. */
|
|
pvReturn = pucAlignedHeap + xNextFreeByte;
|
|
xNextFreeByte += xWantedSize;
|
|
}
|
|
|
|
traceMALLOC( pvReturn, xWantedSize );
|
|
}
|
|
( void ) xTaskResumeAll();
|
|
|
|
#if ( configUSE_MALLOC_FAILED_HOOK == 1 )
|
|
{
|
|
if( pvReturn == NULL )
|
|
{
|
|
vApplicationMallocFailedHook();
|
|
}
|
|
}
|
|
#endif
|
|
|
|
return pvReturn;
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
void vPortFree( void * pv )
|
|
{
|
|
/* Memory cannot be freed using this scheme. See heap_2.c, heap_3.c and
|
|
* heap_4.c for alternative implementations, and the memory management pages of
|
|
* https://www.FreeRTOS.org for more information. */
|
|
( void ) pv;
|
|
|
|
/* Force an assert as it is invalid to call this function. */
|
|
configASSERT( pv == NULL );
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
void vPortInitialiseBlocks( void )
|
|
{
|
|
/* Only required when static memory is not cleared. */
|
|
xNextFreeByte = ( size_t ) 0;
|
|
}
|
|
/*-----------------------------------------------------------*/
|
|
|
|
size_t xPortGetFreeHeapSize( void )
|
|
{
|
|
return( configADJUSTED_HEAP_SIZE - xNextFreeByte );
|
|
}
|
|
|
|
/*-----------------------------------------------------------*/
|
|
|
|
/*
|
|
* Reset the state in this file. This state is normally initialized at start up.
|
|
* This function must be called by the application before restarting the
|
|
* scheduler.
|
|
*/
|
|
void vPortHeapResetState( void )
|
|
{
|
|
xNextFreeByte = ( size_t ) 0U;
|
|
}
|
|
/*-----------------------------------------------------------*/
|