Weird behavior of mallocr() function
Jeff Johnston
jjohnstn@redhat.com
Wed Jan 6 20:09:00 GMT 2010
On 05/01/10 03:39 AM, Denis Micolani wrote:
>> For starters, your linker script is restricting the heap to be 0x400 bytes
>> (1024). That is less that 1344 so your return code makes sense. Malloc
>> creates bins to reuse for small allocations so if you are really this
>> constrained for space, you won't be able to use the malloc logic in newlib
>> and will have to override it.
>
> It seems kinda weird to me that malloc is asking an allocation for
> about 1.34Kbytes while I'm just asking for 1 byte. Anyway if that's
> the way it works, I got nothing to complain about, I'll just expand
> the heap area in the linker script to be a bit larger.
>
The algorithm is trying to balance speed and storage usage as well as
minimize fragmentation. The point of the fixed-size bins is that when
you ask for small amounts of storage, it doesn't go to the external
pool. Instead, it has a reusable set of slots that can be quickly
accessed/reused and it doesn't have to waste bytes on overhead (slot is
free or taken which can be tracked with a single bit). Imagine if you
instead always called sbrk for storage and asked for 1000 bytes then 1
byte, then another 1000. If you free both of the 1000 byte blocks, you
can't reclaim the 2000+ bytes together until the 1 byte block is freed.
The one byte block here is actually taking up more space due to the
overhead of keeping track of it. If you free n 1 byte blocks in a row,
combining them together still doesn't net you much. If the really small
stuff is kept off in the bins, everything left is worth recombining
after freeing to get larger areas.
>
>> As well, your current code isn't going to work. You are adding incr to
>> heap_start and comparing this to heap_end (i.e. neither changes). I'm
>> pretty sure you meant to use heap_current_end in your if statement.
>> As well, if you are going to have a reentrant version, you might as well set
>> errno properly using the reentrant struct that is input.
>> See newlib/libc/include/sys/errno.h for the __errno_r macro.
>
> Thanks for pointing that out!
>
>
>> Most sbrk implementations use the logic found in libgloss/libnosys/sbrk.c
>> which has the heap and stack going in opposite directions and allocates a
>> lump sum for the two in the linker script. This allows peaks for either the
>> stack or heap as long as the total is within limits. You might want to
>> check out this design before implementing your own.
>
> I did not realize there were such a directory with all those useful
> functions within, thanks for pointing this out too!
>
>
> Just one more question: AFAIK, in order to get malloc to be reentrant,
> I should provide my own stubs to lock and unlock the memory, these
> should be: __malloc_unlock() and __malloc_lock().
> Anyway if I define them just like other stubs in the global scope I
> get the compiler to fail, saying that there are multiple definitions
> of those. Specifically, saying that they were already defined in
> newlib-1.18.0/newlib/libc/stdlib/mlock.c .So, how could I manage to
> get it to work? Should I place my own stub directly in newlib's source
> code and recompile it (it doesn't seem to me like a good idea) ?
Normally, a platform has its own libc/machine subdirectory. Files there
are used to override the ones in the shared directories of newlib. For
example, if you have a libc/machine/XXX directory and put your own
mlock.c file there, it will override the one in stdlib (where XXX is the
name of your platform). There are plenty of examples in the newlib
repository under libc/machine. You have to specify this in
configure.host. There are instructions on the newlib web-site in the
FAQ section.
-- Jeff J.
More information about the Newlib
mailing list