2006-10-17  Jakub Jelinek  <jakub@redhat.com>
	* sunrpc/xdr_mem.c (xdrmem_setpos): Don't compare addresses
	as signed longs, check for x_base + pos overflow.
	* sunrpc/Makefile (tests): Add tst-xdrmem2.
	* sunrpc/tst-xdrmem2.c: New test.

2006-10-18  Ulrich Drepper  <drepper@redhat.com>

	* elf/dl-lookup.c (_dl_lookup_symbol_x): Add warning to
	_dl_lookup_symbol_x code.

2006-10-17  Jakub Jelinek  <jakub@redhat.com>

	* elf/dl-runtime.c: Include sysdep-cancel.h.
	(_dl_fixup, _dl_profile_fixup): Use __rtld_mrlock_* and
	scoperec->nusers only if !SINGLE_THREAD_P.  Use atomic_*
	instead of catomic_* macros.
	* elf/dl-sym.c: Include sysdep-cancel.h.
	(do_sym): Use __rtld_mrlock_* and scoperec->nusers only
	if !SINGLE_THREAD_P.  Use atomic_* instead of catomic_* macros.
	* elf/dl-close.c: Include sysdep-cancel.h.
	(_dl_close): Use __rtld_mrlock_* and scoperec->nusers only
	if !SINGLE_THREAD_P.  Use atomic_* instead of catomic_* macros.
	* elf/dl-open.c: Include sysdep-cancel.h.
	(dl_open_worker): Use __rtld_mrlock_* and scoperec->nusers only
	if !SINGLE_THREAD_P.  Use atomic_* instead of catomic_* macros.

2006-10-17  Jakub Jelinek  <jakub@redhat.com>

	[BZ #3313]
	* malloc/malloc.c (malloc_consolidate): Set maxfb to address of last
	fastbin rather than end of fastbin array.

2006-10-18  Ulrich Drepper  <drepper@redhat.com>

	* sysdeps/i386/i486/bits/atomic.h (catomic_decrement): Use correct
	body macro.
	* sysdeps/x86_64/bits/atomic.h
	(__arch_c_compare_and_exchange_val_64_acq): Add missing casts.
	(catomic_decrement): Use correct body macro.

2006-10-17  Jakub Jelinek  <jakub@redhat.com>

	* include/atomic.h: Add a unique prefix to all local variables
	in macros.
	* csu/tst-atomic.c (do_test): Test also catomic_* macros.

	* include/link.h: Include <rtld-lowlevel.h>.  Define struct
This commit is contained in:
Ulrich Drepper 2006-10-18 19:26:30 +00:00
parent 4a85a8ee31
commit 609cf61467
16 changed files with 491 additions and 193 deletions

View File

@ -1,3 +1,51 @@
2006-10-17 Jakub Jelinek <jakub@redhat.com>
* sunrpc/xdr_mem.c (xdrmem_setpos): Don't compare addresses
as signed longs, check for x_base + pos overflow.
* sunrpc/Makefile (tests): Add tst-xdrmem2.
* sunrpc/tst-xdrmem2.c: New test.
2006-10-18 Ulrich Drepper <drepper@redhat.com>
* elf/dl-lookup.c (_dl_lookup_symbol_x): Add warning to
_dl_lookup_symbol_x code.
2006-10-17 Jakub Jelinek <jakub@redhat.com>
* elf/dl-runtime.c: Include sysdep-cancel.h.
(_dl_fixup, _dl_profile_fixup): Use __rtld_mrlock_* and
scoperec->nusers only if !SINGLE_THREAD_P. Use atomic_*
instead of catomic_* macros.
* elf/dl-sym.c: Include sysdep-cancel.h.
(do_sym): Use __rtld_mrlock_* and scoperec->nusers only
if !SINGLE_THREAD_P. Use atomic_* instead of catomic_* macros.
* elf/dl-close.c: Include sysdep-cancel.h.
(_dl_close): Use __rtld_mrlock_* and scoperec->nusers only
if !SINGLE_THREAD_P. Use atomic_* instead of catomic_* macros.
* elf/dl-open.c: Include sysdep-cancel.h.
(dl_open_worker): Use __rtld_mrlock_* and scoperec->nusers only
if !SINGLE_THREAD_P. Use atomic_* instead of catomic_* macros.
2006-10-17 Jakub Jelinek <jakub@redhat.com>
[BZ #3313]
* malloc/malloc.c (malloc_consolidate): Set maxfb to address of last
fastbin rather than end of fastbin array.
2006-10-18 Ulrich Drepper <drepper@redhat.com>
* sysdeps/i386/i486/bits/atomic.h (catomic_decrement): Use correct
body macro.
* sysdeps/x86_64/bits/atomic.h
(__arch_c_compare_and_exchange_val_64_acq): Add missing casts.
(catomic_decrement): Use correct body macro.
2006-10-17 Jakub Jelinek <jakub@redhat.com>
* include/atomic.h: Add a unique prefix to all local variables
in macros.
* csu/tst-atomic.c (do_test): Test also catomic_* macros.
2006-10-16 Ulrich Drepper <drepper@redhat.com> 2006-10-16 Ulrich Drepper <drepper@redhat.com>
[BZ #3369] [BZ #3369]
@ -108,7 +156,7 @@
* elf/dl-load.c: Likewise. * elf/dl-load.c: Likewise.
* elf/dl-object.c: Likewise. * elf/dl-object.c: Likewise.
* elf/rtld.c: Likewise. * elf/rtld.c: Likewise.
* include/link.h: Inlcude <rtld-lowlevel.h>. Define struct * include/link.h: Include <rtld-lowlevel.h>. Define struct
r_scoperec. Replace r_scope with pointer to r_scoperec structure. r_scoperec. Replace r_scope with pointer to r_scoperec structure.
Add l_scoperec_lock. Add l_scoperec_lock.
* sysdeps/generic/ldsodefs.h: Include <rtld-lowlevel.h>. * sysdeps/generic/ldsodefs.h: Include <rtld-lowlevel.h>.

View File

@ -1,5 +1,5 @@
/* Tests for atomic.h macros. /* Tests for atomic.h macros.
Copyright (C) 2003, 2004 Free Software Foundation, Inc. Copyright (C) 2003, 2004, 2006 Free Software Foundation, Inc.
This file is part of the GNU C Library. This file is part of the GNU C Library.
Contributed by Jakub Jelinek <jakub@redhat.com>, 2003. Contributed by Jakub Jelinek <jakub@redhat.com>, 2003.
@ -379,6 +379,117 @@ do_test (void)
} }
#endif #endif
#ifdef catomic_compare_and_exchange_val_acq
mem = 24;
if (catomic_compare_and_exchange_val_acq (&mem, 35, 24) != 24
|| mem != 35)
{
puts ("catomic_compare_and_exchange_val_acq test 1 failed");
ret = 1;
}
mem = 12;
if (catomic_compare_and_exchange_val_acq (&mem, 10, 15) != 12
|| mem != 12)
{
puts ("catomic_compare_and_exchange_val_acq test 2 failed");
ret = 1;
}
mem = -15;
if (catomic_compare_and_exchange_val_acq (&mem, -56, -15) != -15
|| mem != -56)
{
puts ("catomic_compare_and_exchange_val_acq test 3 failed");
ret = 1;
}
mem = -1;
if (catomic_compare_and_exchange_val_acq (&mem, 17, 0) != -1
|| mem != -1)
{
puts ("catomic_compare_and_exchange_val_acq test 4 failed");
ret = 1;
}
#endif
mem = 24;
if (catomic_compare_and_exchange_bool_acq (&mem, 35, 24)
|| mem != 35)
{
puts ("catomic_compare_and_exchange_bool_acq test 1 failed");
ret = 1;
}
mem = 12;
if (! catomic_compare_and_exchange_bool_acq (&mem, 10, 15)
|| mem != 12)
{
puts ("catomic_compare_and_exchange_bool_acq test 2 failed");
ret = 1;
}
mem = -15;
if (catomic_compare_and_exchange_bool_acq (&mem, -56, -15)
|| mem != -56)
{
puts ("catomic_compare_and_exchange_bool_acq test 3 failed");
ret = 1;
}
mem = -1;
if (! catomic_compare_and_exchange_bool_acq (&mem, 17, 0)
|| mem != -1)
{
puts ("catomic_compare_and_exchange_bool_acq test 4 failed");
ret = 1;
}
mem = 2;
if (catomic_exchange_and_add (&mem, 11) != 2
|| mem != 13)
{
puts ("catomic_exchange_and_add test failed");
ret = 1;
}
mem = -21;
catomic_add (&mem, 22);
if (mem != 1)
{
puts ("catomic_add test failed");
ret = 1;
}
mem = -1;
catomic_increment (&mem);
if (mem != 0)
{
puts ("catomic_increment test failed");
ret = 1;
}
mem = 2;
if (catomic_increment_val (&mem) != 3)
{
puts ("catomic_increment_val test failed");
ret = 1;
}
mem = 17;
catomic_decrement (&mem);
if (mem != 16)
{
puts ("catomic_decrement test failed");
ret = 1;
}
if (catomic_decrement_val (&mem) != 15)
{
puts ("catomic_decrement_val test failed");
ret = 1;
}
return ret; return ret;
} }

View File

@ -30,6 +30,7 @@
#include <ldsodefs.h> #include <ldsodefs.h>
#include <sys/types.h> #include <sys/types.h>
#include <sys/mman.h> #include <sys/mman.h>
#include <sysdep-cancel.h>
/* Type of the constructor functions. */ /* Type of the constructor functions. */
@ -419,16 +420,21 @@ _dl_close (void *_map)
struct r_scoperec *old = imap->l_scoperec; struct r_scoperec *old = imap->l_scoperec;
__rtld_mrlock_change (imap->l_scoperec_lock); if (SINGLE_THREAD_P)
imap->l_scoperec = newp; imap->l_scoperec = newp;
__rtld_mrlock_done (imap->l_scoperec_lock); else
if (catomic_increment_val (&old->nusers) != 1)
{ {
old->remove_after_use = true; __rtld_mrlock_change (imap->l_scoperec_lock);
old->notify = true; imap->l_scoperec = newp;
if (catomic_decrement_val (&old->nusers) != 0) __rtld_mrlock_done (imap->l_scoperec_lock);
__rtld_waitzero (old->nusers);
if (atomic_increment_val (&old->nusers) != 1)
{
old->remove_after_use = true;
old->notify = true;
if (atomic_decrement_val (&old->nusers) != 0)
__rtld_waitzero (old->nusers);
}
} }
/* No user anymore, we can free it now. */ /* No user anymore, we can free it now. */

View File

@ -207,7 +207,11 @@ _dl_debug_bindings (const char *undef_name, struct link_map *undef_map,
/* Search loaded objects' symbol tables for a definition of the symbol /* Search loaded objects' symbol tables for a definition of the symbol
UNDEF_NAME, perhaps with a requested version for the symbol. */ UNDEF_NAME, perhaps with a requested version for the symbol.
We must never have calls to the audit functions inside this function
or in any function which gets called. If this would happen the audit
code might create a thread which can throw off all the scope locking. */
lookup_t lookup_t
internal_function internal_function
_dl_lookup_symbol_x (const char *undef_name, struct link_map *undef_map, _dl_lookup_symbol_x (const char *undef_name, struct link_map *undef_map,

View File

@ -31,6 +31,7 @@
#include <ldsodefs.h> #include <ldsodefs.h>
#include <bp-sym.h> #include <bp-sym.h>
#include <caller.h> #include <caller.h>
#include <sysdep-cancel.h>
#include <dl-dst.h> #include <dl-dst.h>
@ -423,15 +424,20 @@ dl_open_worker (void *a)
if (old == &imap->l_scoperec_mem) if (old == &imap->l_scoperec_mem)
imap->l_scoperec = newp; imap->l_scoperec = newp;
else if (SINGLE_THREAD_P)
{
imap->l_scoperec = newp;
free (old);
}
else else
{ {
__rtld_mrlock_change (imap->l_scoperec_lock); __rtld_mrlock_change (imap->l_scoperec_lock);
imap->l_scoperec = newp; imap->l_scoperec = newp;
__rtld_mrlock_done (imap->l_scoperec_lock); __rtld_mrlock_done (imap->l_scoperec_lock);
catomic_increment (&old->nusers); atomic_increment (&old->nusers);
old->remove_after_use = true; old->remove_after_use = true;
if (catomic_decrement_val (&old->nusers) == 0) if (atomic_decrement_val (&old->nusers) == 0)
/* No user, we can free it here and now. */ /* No user, we can free it here and now. */
free (old); free (old);
} }

View File

@ -24,6 +24,7 @@
#include <unistd.h> #include <unistd.h>
#include <sys/param.h> #include <sys/param.h>
#include <ldsodefs.h> #include <ldsodefs.h>
#include <sysdep-cancel.h>
#include "dynamic-link.h" #include "dynamic-link.h"
#if (!defined ELF_MACHINE_NO_RELA && !defined ELF_MACHINE_PLT_REL) \ #if (!defined ELF_MACHINE_NO_RELA && !defined ELF_MACHINE_PLT_REL) \
@ -93,11 +94,11 @@ _dl_fixup (
} }
struct r_scoperec *scoperec = l->l_scoperec; struct r_scoperec *scoperec = l->l_scoperec;
if (l->l_type == lt_loaded) if (l->l_type == lt_loaded && !SINGLE_THREAD_P)
{ {
__rtld_mrlock_lock (l->l_scoperec_lock); __rtld_mrlock_lock (l->l_scoperec_lock);
scoperec = l->l_scoperec; scoperec = l->l_scoperec;
catomic_increment (&scoperec->nusers); atomic_increment (&scoperec->nusers);
__rtld_mrlock_unlock (l->l_scoperec_lock); __rtld_mrlock_unlock (l->l_scoperec_lock);
} }
@ -106,8 +107,8 @@ _dl_fixup (
ELF_RTYPE_CLASS_PLT, ELF_RTYPE_CLASS_PLT,
DL_LOOKUP_ADD_DEPENDENCY, NULL); DL_LOOKUP_ADD_DEPENDENCY, NULL);
if (l->l_type == lt_loaded if (l->l_type == lt_loaded && !SINGLE_THREAD_P
&& catomic_decrement_val (&scoperec->nusers) == 0 && atomic_decrement_val (&scoperec->nusers) == 0
&& __builtin_expect (scoperec->remove_after_use, 0)) && __builtin_expect (scoperec->remove_after_use, 0))
{ {
if (scoperec->notify) if (scoperec->notify)
@ -195,11 +196,11 @@ _dl_profile_fixup (
} }
struct r_scoperec *scoperec = l->l_scoperec; struct r_scoperec *scoperec = l->l_scoperec;
if (l->l_type == lt_loaded) if (l->l_type == lt_loaded && !SINGLE_THREAD_P)
{ {
__rtld_mrlock_lock (l->l_scoperec_lock); __rtld_mrlock_lock (l->l_scoperec_lock);
scoperec = l->l_scoperec; scoperec = l->l_scoperec;
catomic_increment (&scoperec->nusers); atomic_increment (&scoperec->nusers);
__rtld_mrlock_unlock (l->l_scoperec_lock); __rtld_mrlock_unlock (l->l_scoperec_lock);
} }
@ -208,8 +209,8 @@ _dl_profile_fixup (
ELF_RTYPE_CLASS_PLT, ELF_RTYPE_CLASS_PLT,
DL_LOOKUP_ADD_DEPENDENCY, NULL); DL_LOOKUP_ADD_DEPENDENCY, NULL);
if (l->l_type == lt_loaded if (l->l_type == lt_loaded && !SINGLE_THREAD_P
&& catomic_decrement_val (&scoperec->nusers) == 0 && atomic_decrement_val (&scoperec->nusers) == 0
&& __builtin_expect (scoperec->remove_after_use, 0)) && __builtin_expect (scoperec->remove_after_use, 0))
{ {
if (scoperec->notify) if (scoperec->notify)

View File

@ -25,6 +25,7 @@
#include <dlfcn.h> #include <dlfcn.h>
#include <ldsodefs.h> #include <ldsodefs.h>
#include <dl-hash.h> #include <dl-hash.h>
#include <sysdep-cancel.h>
#ifdef USE_TLS #ifdef USE_TLS
# include <dl-tls.h> # include <dl-tls.h>
#endif #endif
@ -115,7 +116,7 @@ do_sym (void *handle, const char *name, void *who,
the initial binary. And then the more complex part the initial binary. And then the more complex part
where the object is dynamically loaded and the scope where the object is dynamically loaded and the scope
array can change. */ array can change. */
if (match->l_type != lt_loaded) if (match->l_type != lt_loaded || SINGLE_THREAD_P)
result = GLRO(dl_lookup_symbol_x) (name, match, &ref, result = GLRO(dl_lookup_symbol_x) (name, match, &ref,
match->l_scoperec->scope, vers, 0, match->l_scoperec->scope, vers, 0,
flags | DL_LOOKUP_ADD_DEPENDENCY, flags | DL_LOOKUP_ADD_DEPENDENCY,
@ -124,7 +125,7 @@ do_sym (void *handle, const char *name, void *who,
{ {
__rtld_mrlock_lock (match->l_scoperec_lock); __rtld_mrlock_lock (match->l_scoperec_lock);
struct r_scoperec *scoperec = match->l_scoperec; struct r_scoperec *scoperec = match->l_scoperec;
catomic_increment (&scoperec->nusers); atomic_increment (&scoperec->nusers);
__rtld_mrlock_unlock (match->l_scoperec_lock); __rtld_mrlock_unlock (match->l_scoperec_lock);
struct call_dl_lookup_args args; struct call_dl_lookup_args args;
@ -141,7 +142,7 @@ do_sym (void *handle, const char *name, void *who,
int err = GLRO(dl_catch_error) (&objname, &errstring, &malloced, int err = GLRO(dl_catch_error) (&objname, &errstring, &malloced,
call_dl_lookup, &args); call_dl_lookup, &args);
if (catomic_decrement_val (&scoperec->nusers) == 0 if (atomic_decrement_val (&scoperec->nusers) == 0
&& __builtin_expect (scoperec->remove_after_use, 0)) && __builtin_expect (scoperec->remove_after_use, 0))
{ {
if (scoperec->notify) if (scoperec->notify)

View File

@ -39,7 +39,12 @@
Architectures must provide a few lowlevel macros (the compare Architectures must provide a few lowlevel macros (the compare
and exchange definitions). All others are optional. They and exchange definitions). All others are optional. They
should only be provided if the architecture has specific should only be provided if the architecture has specific
support for the operation. */ support for the operation.
As <atomic.h> macros are usually heavily nested and often use local
variables to make sure side-effects are evaluated properly, use for
macro local variables a per-macro unique prefix. This file uses
__atgN_ prefix where N is different in each macro. */
#include <stdlib.h> #include <stdlib.h>
@ -50,33 +55,33 @@
and following args. */ and following args. */
#define __atomic_val_bysize(pre, post, mem, ...) \ #define __atomic_val_bysize(pre, post, mem, ...) \
({ \ ({ \
__typeof (*mem) __result; \ __typeof (*mem) __atg1_result; \
if (sizeof (*mem) == 1) \ if (sizeof (*mem) == 1) \
__result = pre##_8_##post (mem, __VA_ARGS__); \ __atg1_result = pre##_8_##post (mem, __VA_ARGS__); \
else if (sizeof (*mem) == 2) \ else if (sizeof (*mem) == 2) \
__result = pre##_16_##post (mem, __VA_ARGS__); \ __atg1_result = pre##_16_##post (mem, __VA_ARGS__); \
else if (sizeof (*mem) == 4) \ else if (sizeof (*mem) == 4) \
__result = pre##_32_##post (mem, __VA_ARGS__); \ __atg1_result = pre##_32_##post (mem, __VA_ARGS__); \
else if (sizeof (*mem) == 8) \ else if (sizeof (*mem) == 8) \
__result = pre##_64_##post (mem, __VA_ARGS__); \ __atg1_result = pre##_64_##post (mem, __VA_ARGS__); \
else \ else \
abort (); \ abort (); \
__result; \ __atg1_result; \
}) })
#define __atomic_bool_bysize(pre, post, mem, ...) \ #define __atomic_bool_bysize(pre, post, mem, ...) \
({ \ ({ \
int __result; \ int __atg2_result; \
if (sizeof (*mem) == 1) \ if (sizeof (*mem) == 1) \
__result = pre##_8_##post (mem, __VA_ARGS__); \ __atg2_result = pre##_8_##post (mem, __VA_ARGS__); \
else if (sizeof (*mem) == 2) \ else if (sizeof (*mem) == 2) \
__result = pre##_16_##post (mem, __VA_ARGS__); \ __atg2_result = pre##_16_##post (mem, __VA_ARGS__); \
else if (sizeof (*mem) == 4) \ else if (sizeof (*mem) == 4) \
__result = pre##_32_##post (mem, __VA_ARGS__); \ __atg2_result = pre##_32_##post (mem, __VA_ARGS__); \
else if (sizeof (*mem) == 8) \ else if (sizeof (*mem) == 8) \
__result = pre##_64_##post (mem, __VA_ARGS__); \ __atg2_result = pre##_64_##post (mem, __VA_ARGS__); \
else \ else \
abort (); \ abort (); \
__result; \ __atg2_result; \
}) })
@ -124,8 +129,9 @@
# define atomic_compare_and_exchange_bool_acq(mem, newval, oldval) \ # define atomic_compare_and_exchange_bool_acq(mem, newval, oldval) \
({ /* Cannot use __oldval here, because macros later in this file might \ ({ /* Cannot use __oldval here, because macros later in this file might \
call this macro with __oldval argument. */ \ call this macro with __oldval argument. */ \
__typeof (oldval) __old = (oldval); \ __typeof (oldval) __atg3_old = (oldval); \
atomic_compare_and_exchange_val_acq (mem, newval, __old) != __old; \ atomic_compare_and_exchange_val_acq (mem, newval, __atg3_old) \
!= __atg3_old; \
}) })
# endif # endif
#endif #endif
@ -140,8 +146,9 @@
# define catomic_compare_and_exchange_bool_acq(mem, newval, oldval) \ # define catomic_compare_and_exchange_bool_acq(mem, newval, oldval) \
({ /* Cannot use __oldval here, because macros later in this file might \ ({ /* Cannot use __oldval here, because macros later in this file might \
call this macro with __oldval argument. */ \ call this macro with __oldval argument. */ \
__typeof (oldval) __old = (oldval); \ __typeof (oldval) __atg4_old = (oldval); \
catomic_compare_and_exchange_val_acq (mem, newval, __old) != __old; \ catomic_compare_and_exchange_val_acq (mem, newval, __atg4_old) \
!= __atg4_old; \
}) })
# endif # endif
#endif #endif
@ -162,18 +169,17 @@
/* Store NEWVALUE in *MEM and return the old value. */ /* Store NEWVALUE in *MEM and return the old value. */
#ifndef atomic_exchange_acq #ifndef atomic_exchange_acq
# define atomic_exchange_acq(mem, newvalue) \ # define atomic_exchange_acq(mem, newvalue) \
({ __typeof (*(mem)) __oldval; \ ({ __typeof (*(mem)) __atg5_oldval; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg5_memp = (mem); \
__typeof (*(mem)) __value = (newvalue); \ __typeof (*(mem)) __atg5_value = (newvalue); \
\ \
do \ do \
__oldval = *__memp; \ __atg5_oldval = *__atg5_memp; \
while (__builtin_expect (atomic_compare_and_exchange_bool_acq (__memp, \ while (__builtin_expect \
__value, \ (atomic_compare_and_exchange_bool_acq (__atg5_memp, __atg5_value, \
__oldval),\ __atg5_oldval), 0)); \
0)); \
\ \
__oldval; }) __atg5_oldval; })
#endif #endif
#ifndef atomic_exchange_rel #ifndef atomic_exchange_rel
@ -184,54 +190,53 @@
/* Add VALUE to *MEM and return the old value of *MEM. */ /* Add VALUE to *MEM and return the old value of *MEM. */
#ifndef atomic_exchange_and_add #ifndef atomic_exchange_and_add
# define atomic_exchange_and_add(mem, value) \ # define atomic_exchange_and_add(mem, value) \
({ __typeof (*(mem)) __oldval; \ ({ __typeof (*(mem)) __atg6_oldval; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg6_memp = (mem); \
__typeof (*(mem)) __value = (value); \ __typeof (*(mem)) __atg6_value = (value); \
\ \
do \ do \
__oldval = *__memp; \ __atg6_oldval = *__atg6_memp; \
while (__builtin_expect (atomic_compare_and_exchange_bool_acq (__memp, \ while (__builtin_expect \
__oldval \ (atomic_compare_and_exchange_bool_acq (__atg6_memp, \
+ __value,\ __atg6_oldval \
__oldval),\ + __atg6_value, \
0)); \ __atg6_oldval), 0)); \
\ \
__oldval; }) __atg6_oldval; })
#endif #endif
#ifndef catomic_exchange_and_add #ifndef catomic_exchange_and_add
# define catomic_exchange_and_add(mem, value) \ # define catomic_exchange_and_add(mem, value) \
({ __typeof (*(mem)) __oldv; \ ({ __typeof (*(mem)) __atg7_oldv; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg7_memp = (mem); \
__typeof (*(mem)) __value = (value); \ __typeof (*(mem)) __atg7_value = (value); \
\ \
do \ do \
__oldv = *__memp; \ __atg7_oldv = *__atg7_memp; \
while (__builtin_expect (catomic_compare_and_exchange_bool_acq (__memp, \ while (__builtin_expect \
__oldv \ (catomic_compare_and_exchange_bool_acq (__atg7_memp, \
+ __value,\ __atg7_oldv \
__oldv), \ + __atg7_value, \
0)); \ __atg7_oldv), 0)); \
\ \
__oldv; }) __atg7_oldv; })
#endif #endif
#ifndef atomic_max #ifndef atomic_max
# define atomic_max(mem, value) \ # define atomic_max(mem, value) \
do { \ do { \
__typeof (*(mem)) __oldval; \ __typeof (*(mem)) __atg8_oldval; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg8_memp = (mem); \
__typeof (*(mem)) __value = (value); \ __typeof (*(mem)) __atg8_value = (value); \
do { \ do { \
__oldval = *__memp; \ __atg8_oldval = *__atg8_memp; \
if (__oldval >= __value) \ if (__atg8_oldval >= __atg8_value) \
break; \ break; \
} while (__builtin_expect (atomic_compare_and_exchange_bool_acq (__memp, \ } while (__builtin_expect \
__value, \ (atomic_compare_and_exchange_bool_acq (__atg8_memp, __atg8_value,\
__oldval),\ __atg8_oldval), 0)); \
0)); \
} while (0) } while (0)
#endif #endif
@ -239,17 +244,17 @@
#ifndef catomic_max #ifndef catomic_max
# define catomic_max(mem, value) \ # define catomic_max(mem, value) \
do { \ do { \
__typeof (*(mem)) __oldv; \ __typeof (*(mem)) __atg9_oldv; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg9_memp = (mem); \
__typeof (*(mem)) __value = (value); \ __typeof (*(mem)) __atg9_value = (value); \
do { \ do { \
__oldv = *__memp; \ __atg9_oldv = *__atg9_memp; \
if (__oldv >= __value) \ if (__atg9_oldv >= __atg9_value) \
break; \ break; \
} while (__builtin_expect (catomic_compare_and_exchange_bool_acq (__memp, \ } while (__builtin_expect \
__value,\ (catomic_compare_and_exchange_bool_acq (__atg9_memp, \
__oldv),\ __atg9_value, \
0)); \ __atg9_oldv), 0)); \
} while (0) } while (0)
#endif #endif
@ -257,17 +262,17 @@
#ifndef atomic_min #ifndef atomic_min
# define atomic_min(mem, value) \ # define atomic_min(mem, value) \
do { \ do { \
__typeof (*(mem)) __oldval; \ __typeof (*(mem)) __atg10_oldval; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg10_memp = (mem); \
__typeof (*(mem)) __value = (value); \ __typeof (*(mem)) __atg10_value = (value); \
do { \ do { \
__oldval = *__memp; \ __atg10_oldval = *__atg10_memp; \
if (__oldval <= __value) \ if (__atg10_oldval <= __atg10_value) \
break; \ break; \
} while (__builtin_expect (atomic_compare_and_exchange_bool_acq (__memp, \ } while (__builtin_expect \
__value, \ (atomic_compare_and_exchange_bool_acq (__atg10_memp, \
__oldval),\ __atg10_value, \
0)); \ __atg10_oldval), 0)); \
} while (0) } while (0)
#endif #endif
@ -340,35 +345,34 @@
/* Decrement *MEM if it is > 0, and return the old value. */ /* Decrement *MEM if it is > 0, and return the old value. */
#ifndef atomic_decrement_if_positive #ifndef atomic_decrement_if_positive
# define atomic_decrement_if_positive(mem) \ # define atomic_decrement_if_positive(mem) \
({ __typeof (*(mem)) __oldval; \ ({ __typeof (*(mem)) __atg11_oldval; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg11_memp = (mem); \
\ \
do \ do \
{ \ { \
__oldval = *__memp; \ __atg11_oldval = *__atg11_memp; \
if (__builtin_expect (__oldval <= 0, 0)) \ if (__builtin_expect (__atg11_oldval <= 0, 0)) \
break; \ break; \
} \ } \
while (__builtin_expect (atomic_compare_and_exchange_bool_acq (__memp, \ while (__builtin_expect \
__oldval \ (atomic_compare_and_exchange_bool_acq (__atg11_memp, \
- 1, \ __atg11_oldval - 1, \
__oldval),\ __atg11_oldval), 0)); \
0));\ __atg11_oldval; })
__oldval; })
#endif #endif
#ifndef atomic_add_negative #ifndef atomic_add_negative
# define atomic_add_negative(mem, value) \ # define atomic_add_negative(mem, value) \
({ __typeof (value) __aan_value = (value); \ ({ __typeof (value) __atg12_value = (value); \
atomic_exchange_and_add (mem, __aan_value) < -__aan_value; }) atomic_exchange_and_add (mem, __atg12_value) < -__atg12_value; })
#endif #endif
#ifndef atomic_add_zero #ifndef atomic_add_zero
# define atomic_add_zero(mem, value) \ # define atomic_add_zero(mem, value) \
({ __typeof (value) __aaz_value = (value); \ ({ __typeof (value) __atg13_value = (value); \
atomic_exchange_and_add (mem, __aaz_value) == -__aaz_value; }) atomic_exchange_and_add (mem, __atg13_value) == -__atg13_value; })
#endif #endif
@ -380,108 +384,102 @@
#ifndef atomic_bit_test_set #ifndef atomic_bit_test_set
# define atomic_bit_test_set(mem, bit) \ # define atomic_bit_test_set(mem, bit) \
({ __typeof (*(mem)) __oldval; \ ({ __typeof (*(mem)) __atg14_old; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg14_memp = (mem); \
__typeof (*(mem)) __mask = ((__typeof (*(mem))) 1 << (bit)); \ __typeof (*(mem)) __atg14_mask = ((__typeof (*(mem))) 1 << (bit)); \
\ \
do \ do \
__oldval = (*__memp); \ __atg14_old = (*__atg14_memp); \
while (__builtin_expect (atomic_compare_and_exchange_bool_acq (__memp, \ while (__builtin_expect \
__oldval \ (atomic_compare_and_exchange_bool_acq (__atg14_memp, \
| __mask, \ __atg14_old | __atg14_mask,\
__oldval),\ __atg14_old), 0)); \
0)); \
\ \
__oldval & __mask; }) __atg14_old & __atg14_mask; })
#endif #endif
/* Atomically *mem &= mask. */ /* Atomically *mem &= mask. */
#ifndef atomic_and #ifndef atomic_and
# define atomic_and(mem, mask) \ # define atomic_and(mem, mask) \
do { \ do { \
__typeof (*(mem)) __oldval; \ __typeof (*(mem)) __atg15_old; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg15_memp = (mem); \
__typeof (*(mem)) __mask = (mask); \ __typeof (*(mem)) __atg15_mask = (mask); \
\ \
do \ do \
__oldval = (*__memp); \ __atg15_old = (*__atg15_memp); \
while (__builtin_expect (atomic_compare_and_exchange_bool_acq (__memp, \ while (__builtin_expect \
__oldval \ (atomic_compare_and_exchange_bool_acq (__atg15_memp, \
& __mask, \ __atg15_old & __atg15_mask, \
__oldval), \ __atg15_old), 0)); \
0)); \
} while (0) } while (0)
#endif #endif
/* Atomically *mem &= mask and return the old value of *mem. */ /* Atomically *mem &= mask and return the old value of *mem. */
#ifndef atomic_and_val #ifndef atomic_and_val
# define atomic_and_val(mem, mask) \ # define atomic_and_val(mem, mask) \
({ __typeof (*(mem)) __oldval; \ ({ __typeof (*(mem)) __atg16_old; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg16_memp = (mem); \
__typeof (*(mem)) __mask = (mask); \ __typeof (*(mem)) __atg16_mask = (mask); \
\ \
do \ do \
__oldval = (*__memp); \ __atg16_old = (*__atg16_memp); \
while (__builtin_expect (atomic_compare_and_exchange_bool_acq (__memp, \ while (__builtin_expect \
__oldval \ (atomic_compare_and_exchange_bool_acq (__atg16_memp, \
& __mask, \ __atg16_old & __atg16_mask,\
__oldval),\ __atg16_old), 0)); \
0)); \
\ \
__oldval; }) __atg16_old; })
#endif #endif
/* Atomically *mem |= mask and return the old value of *mem. */ /* Atomically *mem |= mask and return the old value of *mem. */
#ifndef atomic_or #ifndef atomic_or
# define atomic_or(mem, mask) \ # define atomic_or(mem, mask) \
do { \ do { \
__typeof (*(mem)) __oldval; \ __typeof (*(mem)) __atg17_old; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg17_memp = (mem); \
__typeof (*(mem)) __mask = (mask); \ __typeof (*(mem)) __atg17_mask = (mask); \
\ \
do \ do \
__oldval = (*__memp); \ __atg17_old = (*__atg17_memp); \
while (__builtin_expect (atomic_compare_and_exchange_bool_acq (__memp, \ while (__builtin_expect \
__oldval \ (atomic_compare_and_exchange_bool_acq (__atg17_memp, \
| __mask, \ __atg17_old | __atg17_mask, \
__oldval), \ __atg17_old), 0)); \
0)); \
} while (0) } while (0)
#endif #endif
#ifndef catomic_or #ifndef catomic_or
# define catomic_or(mem, mask) \ # define catomic_or(mem, mask) \
do { \ do { \
__typeof (*(mem)) __oldval; \ __typeof (*(mem)) __atg18_old; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg18_memp = (mem); \
__typeof (*(mem)) __mask = (mask); \ __typeof (*(mem)) __atg18_mask = (mask); \
\ \
do \ do \
__oldval = (*__memp); \ __atg18_old = (*__atg18_memp); \
while (__builtin_expect (catomic_compare_and_exchange_bool_acq (__memp, \ while (__builtin_expect \
__oldval \ (catomic_compare_and_exchange_bool_acq (__atg18_memp, \
| __mask, \ __atg18_old | __atg18_mask,\
__oldval),\ __atg18_old), 0)); \
0)); \
} while (0) } while (0)
#endif #endif
/* Atomically *mem |= mask and return the old value of *mem. */ /* Atomically *mem |= mask and return the old value of *mem. */
#ifndef atomic_or_val #ifndef atomic_or_val
# define atomic_or_val(mem, mask) \ # define atomic_or_val(mem, mask) \
({ __typeof (*(mem)) __oldval; \ ({ __typeof (*(mem)) __atg19_old; \
__typeof (mem) __memp = (mem); \ __typeof (mem) __atg19_memp = (mem); \
__typeof (*(mem)) __mask = (mask); \ __typeof (*(mem)) __atg19_mask = (mask); \
\ \
do \ do \
__oldval = (*__memp); \ __atg19_old = (*__atg19_memp); \
while (__builtin_expect (atomic_compare_and_exchange_bool_acq (__memp, \ while (__builtin_expect \
__oldval \ (atomic_compare_and_exchange_bool_acq (__atg19_memp, \
| __mask, \ __atg19_old | __atg19_mask,\
__oldval),\ __atg19_old), 0)); \
0)); \
\ \
__oldval; }) __atg19_old; })
#endif #endif
#ifndef atomic_full_barrier #ifndef atomic_full_barrier

View File

@ -4699,7 +4699,7 @@ static void malloc_consolidate(av) mstate av;
search all bins all the time. */ search all bins all the time. */
maxfb = &(av->fastbins[fastbin_index(get_max_fast ())]); maxfb = &(av->fastbins[fastbin_index(get_max_fast ())]);
#else #else
maxfb = &(av->fastbins[NFASTBINS]); maxfb = &(av->fastbins[NFASTBINS - 1]);
#endif #endif
fb = &(av->fastbins[0]); fb = &(av->fastbins[0]);
do { do {

View File

@ -1,3 +1,9 @@
2006-10-17 Jakub Jelinek <jakub@redhat.com>
* sysdeps/unix/sysv/linux/rtld-lowlevel.h (__rtld_mrlock_lock,
__rtld_mrlock_unlock, __rtld_mrlock_change, __rtld_mrlock_done): Use
atomic_* instead of catomic_* macros.
2006-10-12 Ulrich Drepper <drepper@redhat.com> 2006-10-12 Ulrich Drepper <drepper@redhat.com>
[BZ #3285] [BZ #3285]

View File

@ -62,9 +62,9 @@ typedef int __rtld_mrlock_t;
{ \ { \
int newval = ((oldval & __RTLD_MRLOCK_RBITS) \ int newval = ((oldval & __RTLD_MRLOCK_RBITS) \
+ __RTLD_MRLOCK_INC); \ + __RTLD_MRLOCK_INC); \
int ret = catomic_compare_and_exchange_val_acq (&(lock), \ int ret = atomic_compare_and_exchange_val_acq (&(lock), \
newval, \ newval, \
oldval); \ oldval); \
if (__builtin_expect (ret == oldval, 1)) \ if (__builtin_expect (ret == oldval, 1)) \
goto out; \ goto out; \
} \ } \
@ -72,7 +72,7 @@ typedef int __rtld_mrlock_t;
} \ } \
if ((oldval & __RTLD_MRLOCK_RWAIT) == 0) \ if ((oldval & __RTLD_MRLOCK_RWAIT) == 0) \
{ \ { \
catomic_or (&(lock), __RTLD_MRLOCK_RWAIT); \ atomic_or (&(lock), __RTLD_MRLOCK_RWAIT); \
oldval |= __RTLD_MRLOCK_RWAIT; \ oldval |= __RTLD_MRLOCK_RWAIT; \
} \ } \
lll_futex_wait (lock, oldval); \ lll_futex_wait (lock, oldval); \
@ -83,7 +83,7 @@ typedef int __rtld_mrlock_t;
#define __rtld_mrlock_unlock(lock) \ #define __rtld_mrlock_unlock(lock) \
do { \ do { \
int oldval = catomic_exchange_and_add (&(lock), -__RTLD_MRLOCK_INC); \ int oldval = atomic_exchange_and_add (&(lock), -__RTLD_MRLOCK_INC); \
if (__builtin_expect ((oldval \ if (__builtin_expect ((oldval \
& (__RTLD_MRLOCK_RBITS | __RTLD_MRLOCK_WWAIT)) \ & (__RTLD_MRLOCK_RBITS | __RTLD_MRLOCK_WWAIT)) \
== (__RTLD_MRLOCK_INC | __RTLD_MRLOCK_WWAIT), 0)) \ == (__RTLD_MRLOCK_INC | __RTLD_MRLOCK_WWAIT), 0)) \
@ -103,11 +103,11 @@ typedef int __rtld_mrlock_t;
for (int tries = 0; tries < __RTLD_MRLOCK_TRIES; ++tries) \ for (int tries = 0; tries < __RTLD_MRLOCK_TRIES; ++tries) \
{ \ { \
oldval = lock; \ oldval = lock; \
while (__builtin_expect ((oldval & __RTLD_MRLOCK_RBITS) == 0, 1))\ while (__builtin_expect ((oldval & __RTLD_MRLOCK_RBITS) == 0, 1)) \
{ \ { \
int newval = ((oldval & __RTLD_MRLOCK_RWAIT) \ int newval = ((oldval & __RTLD_MRLOCK_RWAIT) \
+ __RTLD_MRLOCK_WRITER); \ + __RTLD_MRLOCK_WRITER); \
int ret = catomic_compare_and_exchange_val_acq (&(lock), \ int ret = atomic_compare_and_exchange_val_acq (&(lock), \
newval, \ newval, \
oldval); \ oldval); \
if (__builtin_expect (ret == oldval, 1)) \ if (__builtin_expect (ret == oldval, 1)) \
@ -115,7 +115,7 @@ typedef int __rtld_mrlock_t;
} \ } \
atomic_delay (); \ atomic_delay (); \
} \ } \
catomic_or (&(lock), __RTLD_MRLOCK_WWAIT); \ atomic_or (&(lock), __RTLD_MRLOCK_WWAIT); \
oldval |= __RTLD_MRLOCK_WWAIT; \ oldval |= __RTLD_MRLOCK_WWAIT; \
lll_futex_wait (lock, oldval); \ lll_futex_wait (lock, oldval); \
} \ } \
@ -125,7 +125,7 @@ typedef int __rtld_mrlock_t;
#define __rtld_mrlock_done(lock) \ #define __rtld_mrlock_done(lock) \
do { \ do { \
int oldval = catomic_exchange_and_add (&(lock), -__RTLD_MRLOCK_WRITER); \ int oldval = atomic_exchange_and_add (&(lock), -__RTLD_MRLOCK_WRITER); \
if (__builtin_expect ((oldval & __RTLD_MRLOCK_RWAIT) != 0, 0)) \ if (__builtin_expect ((oldval & __RTLD_MRLOCK_RWAIT) != 0, 0)) \
lll_futex_wake (&(lock), 0x7fffffff); \ lll_futex_wake (&(lock), 0x7fffffff); \
} while (0) } while (0)

View File

@ -1,4 +1,4 @@
# Copyright (C) 1994-2004, 2005 Free Software Foundation, Inc. # Copyright (C) 1994-2004, 2005, 2006 Free Software Foundation, Inc.
# This file is part of the GNU C Library. # This file is part of the GNU C Library.
# The GNU C Library is free software; you can redistribute it and/or # The GNU C Library is free software; you can redistribute it and/or
@ -85,7 +85,7 @@ all: # Make this the default target; it will be defined in Rules.
include ../Makeconfig include ../Makeconfig
tests = tst-xdrmem tests = tst-xdrmem tst-xdrmem2
xtests := tst-getmyaddr xtests := tst-getmyaddr
ifeq ($(have-thread-library),yes) ifeq ($(have-thread-library),yes)

114
sunrpc/tst-xdrmem2.c Normal file
View File

@ -0,0 +1,114 @@
/* Copyright (C) 2006 Free Software Foundation, Inc.
This file is part of the GNU C Library.
Contributed by Jakub Jelinek <jakub@redhat.com>, 2006.
The GNU C Library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
The GNU C Library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with the GNU C Library; if not, write to the Free
Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
02111-1307 USA. */
#include <limits.h>
#include <stdio.h>
#include <string.h>
#include <rpc/rpc.h>
#include <sys/mman.h>
#include <unistd.h>
static int
do_test (void)
{
XDR xdrs;
void *buf;
size_t ps = sysconf (_SC_PAGESIZE);
uintptr_t half = -1;
int v_int;
u_short v_u_short;
half = (half >> 1) & ~(uintptr_t) (ps - 1);
buf = mmap ((void *) half, 2 * ps, PROT_READ | PROT_WRITE,
MAP_PRIVATE | MAP_ANON, -1, 0);
if (buf == MAP_FAILED || buf != (void *) half)
{
puts ("Couldn't mmap 2 pages in the middle of address space");
return 0;
}
xdrmem_create (&xdrs, (char *) buf, 2 * ps, XDR_ENCODE);
#define T(type, val) \
v_##type = val; \
if (! xdr_##type (&xdrs, &v_##type)) \
{ \
puts ("encoding of " #type \
" " #val " failed"); \
return 1; \
}
T(int, 127)
u_int pos = xdr_getpos (&xdrs);
T(u_short, 31)
if (! xdr_setpos (&xdrs, pos))
{
puts ("xdr_setpos during encoding failed");
return 1;
}
T(u_short, 36)
#undef T
xdr_destroy (&xdrs);
xdrmem_create (&xdrs, (char *) buf, 2 * ps, XDR_DECODE);
#define T(type, val) \
v_##type = 0x15; \
if (! xdr_##type (&xdrs, &v_##type)) \
{ \
puts ("decoding of " #type \
" " #val " failed"); \
return 1; \
} \
if (v_##type != val) \
{ \
puts ("decoded value differs, " \
"type " #type " " #val); \
return 1; \
}
T(int, 127)
pos = xdr_getpos (&xdrs);
T(u_short, 36)
if (! xdr_setpos (&xdrs, pos))
{
puts ("xdr_setpos during encoding failed");
return 1;
}
T(u_short, 36)
#undef T
xdr_destroy (&xdrs);
return 0;
}
#define TEST_FUNCTION do_test ()
#include "../test-skeleton.c"

View File

@ -177,13 +177,15 @@ xdrmem_setpos (xdrs, pos)
{ {
caddr_t newaddr = xdrs->x_base + pos; caddr_t newaddr = xdrs->x_base + pos;
caddr_t lastaddr = xdrs->x_private + xdrs->x_handy; caddr_t lastaddr = xdrs->x_private + xdrs->x_handy;
size_t handy = lastaddr - newaddr;
if ((long) newaddr > (long) lastaddr if (newaddr > lastaddr
|| (UINT_MAX < LONG_MAX || newaddr < xdrs->x_base
&& (long) UINT_MAX < (long) lastaddr - (long) newaddr)) || handy != (u_int) handy)
return FALSE; return FALSE;
xdrs->x_private = newaddr; xdrs->x_private = newaddr;
xdrs->x_handy = (long) lastaddr - (long) newaddr; xdrs->x_handy = (u_int) handy;
return TRUE; return TRUE;
} }

View File

@ -425,7 +425,7 @@ typedef uintmax_t uatomic_max_t;
"cmpl $0, %%gs:%P2\n\tje 0f\n\tlock\n0:\t" "cmpl $0, %%gs:%P2\n\tje 0f\n\tlock\n0:\t"
#define catomic_decrement(mem) \ #define catomic_decrement(mem) \
__arch_increment_body (__arch_decrement_cprefix, __arch_c, mem) __arch_decrement_body (__arch_decrement_cprefix, __arch_c, mem)
#define atomic_decrement_and_test(mem) \ #define atomic_decrement_and_test(mem) \

View File

@ -81,8 +81,8 @@ typedef uintmax_t uatomic_max_t;
({ __typeof (*mem) ret; \ ({ __typeof (*mem) ret; \
__asm __volatile (LOCK_PREFIX "cmpxchgq %q2, %1" \ __asm __volatile (LOCK_PREFIX "cmpxchgq %q2, %1" \
: "=a" (ret), "=m" (*mem) \ : "=a" (ret), "=m" (*mem) \
: "r" ((long) (newval)), "m" (*mem), \ : "r" ((long int) (newval)), "m" (*mem), \
"0" ((long) (oldval))); \ "0" ((long int) (oldval))); \
ret; }) ret; })
@ -121,12 +121,13 @@ typedef uintmax_t uatomic_max_t;
#define __arch_c_compare_and_exchange_val_64_acq(mem, newval, oldval) \ #define __arch_c_compare_and_exchange_val_64_acq(mem, newval, oldval) \
({ __typeof (*mem) ret; \ ({ __typeof (*mem) ret; \
__asm __volatile ("cmpl $0, %%fs:%P5\n\t" \ __asm __volatile ("cmpl $0, %%fs:%P5\n\t" \
"je 0f\n\t" \ "je 0f\n\t" \
"lock\n" \ "lock\n" \
"0:\tcmpxchgq %q2, %1" \ "0:\tcmpxchgq %q2, %1" \
: "=a" (ret), "=m" (*mem) \ : "=a" (ret), "=m" (*mem) \
: "q" (newval), "m" (*mem), "0" (oldval), \ : "q" ((long int) (newval)), "m" (*mem), \
"0" ((long int)oldval), \
"i" (offsetof (tcbhead_t, multiple_threads))); \ "i" (offsetof (tcbhead_t, multiple_threads))); \
ret; }) ret; })
@ -351,7 +352,7 @@ typedef uintmax_t uatomic_max_t;
"cmpl $0, %%fs:%P2\n\tje 0f\n\tlock\n0:\t" "cmpl $0, %%fs:%P2\n\tje 0f\n\tlock\n0:\t"
#define catomic_decrement(mem) \ #define catomic_decrement(mem) \
__arch_increment_body (__arch_decrement_cprefix, mem) __arch_decrement_body (__arch_decrement_cprefix, mem)
#define atomic_decrement_and_test(mem) \ #define atomic_decrement_and_test(mem) \