4
0
mirror of git://sourceware.org/git/newlib-cygwin.git synced 2025-01-16 03:19:54 +08:00
Corinna Vinschen 2ca9ed527e * fhandler.h (refcnt): Add i interlocked. Explain why.
* winbase.h (ilockadd): New function.
	(InterlockedAdd): Define as ilockadd.
2012-05-23 13:13:56 +00:00

94 lines
2.0 KiB
C

/* winbase.h
Copyright 2002, 2003, 2004, 2008, 2012 Red Hat, Inc.
This software is a copyrighted work licensed under the terms of the
Cygwin license. Please consult the file "CYGWIN_LICENSE" for
details. */
#include_next "winbase.h"
#ifndef _WINBASE2_H
#define _WINBASE2_H
/* For some unknown reason, InterlockedAdd is only supported on Itanium
when using the Windows headers. Fortunately we're not restricted to the
Windows headers :) */
extern __inline__ long
ilockadd (volatile long *m, long value)
{
register int __res;
__asm__ __volatile__ ("\n\
movl %3,%0\n\
lock xadd %0,%1\n\
addl %3,%0\n\
": "=&r" (__res), "=m" (*m): "m" (*m), "r" (value): "cc");
return __res;
}
extern __inline__ long
ilockincr (volatile long *m)
{
register int __res;
__asm__ __volatile__ ("\n\
movl $1,%0\n\
lock xadd %0,%1\n\
inc %0\n\
": "=&r" (__res), "=m" (*m): "m" (*m): "cc");
return __res;
}
extern __inline__ long
ilockdecr (volatile long *m)
{
register int __res;
__asm__ __volatile__ ("\n\
movl $0xffffffff,%0\n\
lock xadd %0,%1\n\
dec %0\n\
": "=&r" (__res), "=m" (*m): "m" (*m): "cc");
return __res;
}
extern __inline__ long
ilockexch (volatile long *t, long v)
{
return
({
register long ret __asm ("%eax");
__asm __volatile ("\n"
"1: lock cmpxchgl %2, %1\n"
" jne 1b\n"
: "=a" (ret), "=m" (*t)
: "r" (v), "m" (*t), "0" (*t)
: "memory");
ret;
});
}
extern __inline__ long
ilockcmpexch (volatile long *t, long v, long c)
{
return
({
register long ret __asm ("%eax");
__asm __volatile ("lock cmpxchgl %2, %1"
: "=a" (ret), "=m" (*t)
: "r" (v), "m" (*t), "0" (c)
: "memory");
ret;
});
}
#undef InterlockedAdd
#define InterlockedAdd ilockadd
#undef InterlockedIncrement
#define InterlockedIncrement ilockincr
#undef InterlockedDecrement
#define InterlockedDecrement ilockdecr
#undef InterlockedExchange
#define InterlockedExchange ilockexch
#undef InterlockedCompareExchange
#define InterlockedCompareExchange ilockcmpexch
#endif /*_WINBASE2_H*/