forked from Imagelibrary/rtems
malloc: Fix early realloc() allocation
This commit is contained in:
@@ -63,7 +63,7 @@ static void *_Malloc_Get_deferred_free( void )
|
|||||||
return p;
|
return p;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void _Malloc_Process_deferred_frees( void )
|
void _Malloc_Process_deferred_frees( void )
|
||||||
{
|
{
|
||||||
rtems_chain_node *to_be_freed;
|
rtems_chain_node *to_be_freed;
|
||||||
|
|
||||||
|
|||||||
@@ -27,6 +27,8 @@ Malloc_System_state _Malloc_System_state( void );
|
|||||||
|
|
||||||
void _Malloc_Deferred_free( void * );
|
void _Malloc_Deferred_free( void * );
|
||||||
|
|
||||||
|
void _Malloc_Process_deferred_frees( void );
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif /* __cplusplus */
|
#endif /* __cplusplus */
|
||||||
|
|||||||
@@ -19,53 +19,15 @@
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef RTEMS_NEWLIB
|
#ifdef RTEMS_NEWLIB
|
||||||
#include "malloc_p.h"
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
#include <rtems/score/sysstate.h>
|
#include "malloc_p.h"
|
||||||
#include <rtems/score/objectimpl.h>
|
|
||||||
|
|
||||||
void *realloc(
|
static void *new_alloc( void *old_ptr, size_t new_size, size_t old_size )
|
||||||
void *ptr,
|
|
||||||
size_t size
|
|
||||||
)
|
|
||||||
{
|
{
|
||||||
uintptr_t old_size;
|
void *new_ptr;
|
||||||
char *new_area;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Do not attempt to allocate memory if in a critical section or ISR.
|
|
||||||
*/
|
|
||||||
|
|
||||||
if (_System_state_Is_up(_System_state_Get())) {
|
|
||||||
if (!_Thread_Dispatch_is_enabled())
|
|
||||||
return (void *) 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Continue with realloc().
|
|
||||||
*/
|
|
||||||
if ( !ptr )
|
|
||||||
return malloc( size );
|
|
||||||
|
|
||||||
if ( !size ) {
|
|
||||||
free( ptr );
|
|
||||||
return (void *) 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( !_Protected_heap_Get_block_size(RTEMS_Malloc_Heap, ptr, &old_size) ) {
|
|
||||||
errno = EINVAL;
|
|
||||||
return (void *) 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Now resize it.
|
|
||||||
*/
|
|
||||||
if ( _Protected_heap_Resize_block( RTEMS_Malloc_Heap, ptr, size ) ) {
|
|
||||||
return ptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* There used to be a free on this error case but it is wrong to
|
* There used to be a free on this error case but it is wrong to
|
||||||
@@ -73,16 +35,57 @@ void *realloc(
|
|||||||
* and the C Standard.
|
* and the C Standard.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
new_area = malloc( size );
|
new_ptr = malloc( new_size );
|
||||||
|
if ( new_ptr == NULL ) {
|
||||||
if ( !new_area ) {
|
return NULL;
|
||||||
return (void *) 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy( new_area, ptr, (size < old_size) ? size : old_size );
|
memcpy( new_ptr, old_ptr, ( new_size < old_size ) ? new_size : old_size );
|
||||||
free( ptr );
|
free( old_ptr );
|
||||||
|
|
||||||
return new_area;
|
return new_ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void *realloc( void *ptr, size_t size )
|
||||||
|
{
|
||||||
|
Heap_Control *heap;
|
||||||
|
Heap_Resize_status status;
|
||||||
|
uintptr_t old_size;
|
||||||
|
uintptr_t avail_size;
|
||||||
|
|
||||||
|
if ( size == 0 ) {
|
||||||
|
free( ptr );
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( ptr == NULL ) {
|
||||||
|
return malloc( size );
|
||||||
|
}
|
||||||
|
|
||||||
|
heap = RTEMS_Malloc_Heap;
|
||||||
|
|
||||||
|
switch ( _Malloc_System_state() ) {
|
||||||
|
case MALLOC_SYSTEM_STATE_NORMAL:
|
||||||
|
_RTEMS_Lock_allocator();
|
||||||
|
_Malloc_Process_deferred_frees();
|
||||||
|
status = _Heap_Resize_block( heap, ptr, size, &old_size, &avail_size );
|
||||||
|
_RTEMS_Unlock_allocator();
|
||||||
|
break;
|
||||||
|
case MALLOC_SYSTEM_STATE_NO_PROTECTION:
|
||||||
|
status = _Heap_Resize_block( heap, ptr, size, &old_size, &avail_size );
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch ( status ) {
|
||||||
|
case HEAP_RESIZE_SUCCESSFUL:
|
||||||
|
return ptr;
|
||||||
|
case HEAP_RESIZE_UNSATISFIED:
|
||||||
|
return new_alloc( ptr, size, old_size );
|
||||||
|
default:
|
||||||
|
errno = EINVAL;
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -1375,6 +1375,9 @@ static void test_early_malloc( void )
|
|||||||
{
|
{
|
||||||
void *p;
|
void *p;
|
||||||
char *q;
|
char *q;
|
||||||
|
void *r;
|
||||||
|
void *s;
|
||||||
|
void *t;
|
||||||
|
|
||||||
p = malloc( 1 );
|
p = malloc( 1 );
|
||||||
rtems_test_assert( p != NULL );
|
rtems_test_assert( p != NULL );
|
||||||
@@ -1387,6 +1390,20 @@ static void test_early_malloc( void )
|
|||||||
rtems_test_assert( q[0] == 0 );
|
rtems_test_assert( q[0] == 0 );
|
||||||
|
|
||||||
free( q );
|
free( q );
|
||||||
|
|
||||||
|
r = realloc( q, 128 );
|
||||||
|
rtems_test_assert( r == q );
|
||||||
|
|
||||||
|
s = malloc( 1 );
|
||||||
|
rtems_test_assert( s != NULL );
|
||||||
|
|
||||||
|
free( s );
|
||||||
|
|
||||||
|
t = realloc( r, 256 );
|
||||||
|
rtems_test_assert( t != NULL );
|
||||||
|
rtems_test_assert( t != r );
|
||||||
|
|
||||||
|
free( t );
|
||||||
}
|
}
|
||||||
|
|
||||||
RTEMS_SYSINIT_ITEM(
|
RTEMS_SYSINIT_ITEM(
|
||||||
|
|||||||
Reference in New Issue
Block a user