_REENT_ONLY question

Artem B. Bityuckiy abityuckiy@yandex.ru
Fri Nov 28 21:37:00 GMT 2003



J. Johnston wrote:

> Artem B. Bityuckiy wrote:
>
>> Hello.
>>
>> Please, can anybody explain me why _REENT_ONLY macro is needed? For 
>> what? Why some functions have on implementation and another have pair 
>> with _r postfix, e.g. setlocale() and _setlocale_r(), etc. Several 
>> functions have no such pair, e.g. ldiv(). And in all cases (that I've 
>> seen) usual functions call _r pair adding _REENT argument. For 
>> example 'setlocale(category, locale)' just calls '_setlocale_r(p, 
>> category, locale), where p is 'struct _reent'. I can only guess that 
>> this is needed if external (not usual Newlib) "multiprocess support 
>> subsystem" is used...
>>
>> Can anybody explain this and give me some examples when "_r" pairs 
>> are really useful/needed?
>>
>
> I'll start with the _r pairs.  They're used to support 
> threading/reentrancy. Probably the simplest example is errno.  The 
> storage behind errno is actually kept in newlib's reentrancy 
> structure.  Most users don't see this because there is a default 
> reentrancy struct set up on their behalf.  If you look in errno.h you 
> will see that errno is a macro which calls __errno which you will find 
> in libc/errno/errno.c.  The function just references the default 
> reentrancy structure and gives back the address of where errno is stored.
>
> Now, imagine you have multiple threads.  You want an errno per 
> thread.  If you call a library function in a thread, you don't want 
> another thread to overwrite that value or else you won't be able to 
> check the results of your call.
>
> To handle this, we create _r versions of functions where the 
> reentrancy struct is passed in.  In a thread, you call the _r versions 
> of appropriate functions and they set the errno for the reentrancy 
> struct you are passing in.
> There are other such values (e.g. the last token address for strtok).  
> If a function calls another function which has an _r version, it 
> requires an _r version as well.  An example of where this is missing 
> is atol, atoi, and your new atoll.  As soon as this is checked in, I 
> am going to fix them.  They call strtol and strtoll.  These called 
> functions have _r versions because they set errno.  Thus, they should 
> have been modified so an _r version exists as well.
> Notice in errno.h, there is a __errno_r macro which you can use to get 
> the errno for a particular reentrancy struct.
>
> What the _REENT_ONLY flag does is to force the caller to always use 
> the _r routines for appropriate functions.  To do this, the default 
> versions are ifdef'd out and only the _r versions are exposed.  You 
> can't inadvertently call strtok() from a thread because it isn't in 
> the library.  You are forced to call _strtok_r and specify the 
> reentrancy struct.  The reason you want this is that the default 
> versions are not protected.  If two threads call strtok(), it is a 
> race condition for the token address in the reentrancy struct.  This 
> can be difficult to debug as any error might only show up at random.
>
> -- Jeff J.

Thanks Jeff for ** complete ** description!

-- 
Best Regards,
Artem B. Bityuckiy,
St.-Petersburg, Russia.




More information about the Newlib mailing list