mirror of
git://sourceware.org/git/newlib-cygwin.git
synced 2025-01-25 08:37:33 +08:00
9b51beeb2a
Previously, __internal_syscall() compiled into asm-code that unconditionally sets the syscall argument registers a0 to a5. For example, the instruction sequence for a exit syscall looked like this: li a0, 1 # in ther caller of exit() # ... # in newlib: li a1, 0 # unused arguments li a2, 0 li a3, 0 li a4, 0 li a5, 0 li a7, 93 # exit syscall number (i.e. the binary contains then 5 superfluous instructions for this one argument syscall) This commit changes the RISC-V syscall code such that only the required syscall argument registers are set. GCC detects that argc is known at compile time and thus evaluates all the if-statements where argc is used at compile time (tested with -O2 and -Os).
57 lines
1.3 KiB
C
57 lines
1.3 KiB
C
#ifdef USING_SIM_SPECS
|
|
|
|
// Gdb simulator requires that sbrk be implemented without a syscall.
|
|
extern char _end[]; /* _end is set in the linker command file */
|
|
char *heap_ptr;
|
|
|
|
/*
|
|
* sbrk -- changes heap size size. Get nbytes more
|
|
* RAM. We just increment a pointer in what's
|
|
* left of memory on the board.
|
|
*/
|
|
char *
|
|
_sbrk (nbytes)
|
|
int nbytes;
|
|
{
|
|
char *base;
|
|
|
|
if (!heap_ptr)
|
|
heap_ptr = (char *)&_end;
|
|
base = heap_ptr;
|
|
heap_ptr += nbytes;
|
|
|
|
return base;
|
|
}
|
|
|
|
#else
|
|
|
|
// QEMU uses a syscall.
|
|
#include <machine/syscall.h>
|
|
#include <sys/types.h>
|
|
#include "internal_syscall.h"
|
|
|
|
/* Increase program data space. As malloc and related functions depend
|
|
on this, it is useful to have a working implementation. The following
|
|
is suggested by the newlib docs and suffices for a standalone
|
|
system. */
|
|
void *
|
|
_sbrk(ptrdiff_t incr)
|
|
{
|
|
static unsigned long heap_end;
|
|
|
|
if (heap_end == 0)
|
|
{
|
|
long brk = __internal_syscall (SYS_brk, 1, 0, 0, 0, 0, 0, 0);
|
|
if (brk == -1)
|
|
return (void *)__syscall_error (-ENOMEM);
|
|
heap_end = brk;
|
|
}
|
|
|
|
if (__internal_syscall (SYS_brk, 1, heap_end + incr, 0, 0, 0, 0, 0) != heap_end + incr)
|
|
return (void *)__syscall_error (-ENOMEM);
|
|
|
|
heap_end += incr;
|
|
return (void *)(heap_end - incr);
|
|
}
|
|
#endif
|