mirror of
git://sourceware.org/git/newlib-cygwin.git
synced 2025-01-20 05:19:21 +08:00
166 lines
5.3 KiB
C
166 lines
5.3 KiB
C
|
/* EPIPHANY implementation of _sbrk ()
|
||
|
|
||
|
Copyright (c) 2011, Adapteva, Inc.
|
||
|
All rights reserved.
|
||
|
|
||
|
Contributor Jeremy Bennett <jeremy.bennett@embecosm.com> for Adapteva Inc
|
||
|
|
||
|
Redistribution and use in source and binary forms, with or without
|
||
|
modification, are permitted provided that the following conditions are met:
|
||
|
* Redistributions of source code must retain the above copyright notice,
|
||
|
this list of conditions and the following disclaimer.
|
||
|
* Redistributions in binary form must reproduce the above copyright
|
||
|
notice, this list of conditions and the following disclaimer in the
|
||
|
documentation and/or other materials provided with the distribution.
|
||
|
* Neither the name of Adapteva nor the names of its contributors may be
|
||
|
used to endorse or promote products derived from this software without
|
||
|
specific prior written permission.
|
||
|
|
||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||
|
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||
|
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
|
||
|
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
||
|
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
||
|
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
||
|
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
||
|
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||
|
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||
|
POSSIBILITY OF SUCH DAMAGE. */
|
||
|
/* ------------------------------------------------------------------------- */
|
||
|
/* This implementation is suitable for use with the Verilator simulation of
|
||
|
the EPIPHANY.
|
||
|
|
||
|
Commenting suitable for processing by Doxygen is provided throughout. */
|
||
|
/* ------------------------------------------------------------------------- */
|
||
|
#include <sys/types.h>
|
||
|
#include <errno.h>
|
||
|
#include <stddef.h>
|
||
|
|
||
|
/* ------------------------------------------------------------------------- */
|
||
|
/*!Increase program data space
|
||
|
|
||
|
This is a minimal implementation. Assuming the heap is growing upwards
|
||
|
from __heap_start towards __heap_end.
|
||
|
See linker file for definition.
|
||
|
|
||
|
@param[in] incr The number of bytes to increment the stack by.
|
||
|
|
||
|
@return A pointer to the start of the new block of memory */
|
||
|
/* ------------------------------------------------------------------------- */
|
||
|
void *
|
||
|
_sbrk (int incr)
|
||
|
{
|
||
|
extern char __heap_start;//set by linker
|
||
|
extern char __heap_end;//set by linker
|
||
|
|
||
|
static char *heap_end; /* Previous end of heap or 0 if none */
|
||
|
char *prev_heap_end;
|
||
|
|
||
|
if (0 == heap_end) {
|
||
|
heap_end = &__heap_start; /* Initialize first time round */
|
||
|
}
|
||
|
|
||
|
prev_heap_end = heap_end;
|
||
|
heap_end += incr;
|
||
|
//check
|
||
|
if( heap_end < (&__heap_end)) {
|
||
|
|
||
|
} else {
|
||
|
errno = ENOMEM;
|
||
|
return (char*)-1;
|
||
|
}
|
||
|
return (void *) prev_heap_end;
|
||
|
|
||
|
} /* _sbrk () */
|
||
|
|
||
|
|
||
|
|
||
|
///* sbrk support */
|
||
|
//
|
||
|
///* The current plan is to have one sbrk handler for all cpus.
|
||
|
// Hence use `asm' for each global variable here to avoid the cpu prefix.
|
||
|
// We can't intrude on the user's namespace (another reason to use asm). */
|
||
|
//
|
||
|
//#include <sys/types.h>
|
||
|
///*#include <sys/syscall.h>*/
|
||
|
//#include <errno.h>
|
||
|
//#include <stddef.h>
|
||
|
//
|
||
|
///* These variables are publicly accessible for debugging purposes.
|
||
|
// The user is also free to set sbrk_size to something different.
|
||
|
// See mem-layout.c. */
|
||
|
//
|
||
|
//extern int sbrk_size asm ("sbrk_size");
|
||
|
//
|
||
|
//caddr_t sbrk_start asm ("sbrk_start");
|
||
|
//caddr_t sbrk_loc asm ("sbrk_loc");
|
||
|
//extern char end; /* Defined by linker. */
|
||
|
//
|
||
|
///*caddr_t _sbrk_r (struct _reent *, size_t) asm ("__sbrk_r");*/
|
||
|
//
|
||
|
//
|
||
|
///* This symbol can be defined with the --defsym linker option. */
|
||
|
//extern char __heap_end __attribute__((weak));
|
||
|
//
|
||
|
//
|
||
|
///* FIXME: We need a semaphore here. */
|
||
|
//
|
||
|
//caddr_t
|
||
|
//_sbrk_r (struct _reent *r, size_t nbytes)
|
||
|
//{
|
||
|
// extern char end; /* Defined by linker. */
|
||
|
// static char *heap_end; /* Previous end of heap or 0 if none */
|
||
|
// char *prev_heap_end;
|
||
|
// register char* stack asm("sp");
|
||
|
//
|
||
|
// if (0 == heap_end)
|
||
|
// {
|
||
|
// heap_end = &end; /* Initialize first time round */
|
||
|
// }
|
||
|
//
|
||
|
// prev_heap_end = heap_end;
|
||
|
//
|
||
|
// if (stack < (prev_heap_end+nbytes)) {
|
||
|
// /* heap would overlap the current stack depth.
|
||
|
// * Future: use sbrk() system call to make simulator grow memory beyond
|
||
|
// * the stack and allocate that
|
||
|
// */
|
||
|
// errno = ENOMEM;
|
||
|
// return (char*)-1;
|
||
|
// }
|
||
|
//
|
||
|
// heap_end += nbytes;
|
||
|
//
|
||
|
// return (caddr_t) prev_heap_end;
|
||
|
//
|
||
|
//} /* _sbrk () */
|
||
|
|
||
|
|
||
|
//caddr_t
|
||
|
//_sbrk_r (struct _reent *r, size_t nbytes)
|
||
|
//{
|
||
|
// caddr_t result;
|
||
|
//
|
||
|
// void *heap_end;
|
||
|
//
|
||
|
// heap_end = &__heap_end;
|
||
|
// if (!heap_end)
|
||
|
// heap_end = sbrk_start + sbrk_size;
|
||
|
// if (
|
||
|
// /* Ensure we don't underflow. */
|
||
|
// sbrk_loc + nbytes < sbrk_start
|
||
|
// /* Ensure we don't overflow. */
|
||
|
// || sbrk_loc + nbytes > (caddr_t)heap_end)
|
||
|
// {
|
||
|
// errno = ENOMEM;
|
||
|
// return ((caddr_t) -1);
|
||
|
// }
|
||
|
//
|
||
|
// if (0 == sbrk_loc)
|
||
|
// sbrk_loc = &end; /* Initialize first time round */
|
||
|
// result = sbrk_loc;
|
||
|
// sbrk_loc += nbytes;
|
||
|
// return result;
|
||
|
//}
|