Wake only when there are waiters.
[kopensolaris-gnu/glibc.git] / nptl / sysdeps / unix / sysv / linux / i386 / i486 / sem_post.S
1 /* Copyright (C) 2002, 2003, 2005, 2007 Free Software Foundation, Inc.
2    This file is part of the GNU C Library.
3    Contributed by Ulrich Drepper <drepper@redhat.com>, 2002.
4
5    The GNU C Library is free software; you can redistribute it and/or
6    modify it under the terms of the GNU Lesser General Public
7    License as published by the Free Software Foundation; either
8    version 2.1 of the License, or (at your option) any later version.
9
10    The GNU C Library is distributed in the hope that it will be useful,
11    but WITHOUT ANY WARRANTY; without even the implied warranty of
12    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13    Lesser General Public License for more details.
14
15    You should have received a copy of the GNU Lesser General Public
16    License along with the GNU C Library; if not, write to the Free
17    Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
18    02111-1307 USA.  */
19
20 #include <sysdep.h>
21 #include <shlib-compat.h>
22 #include <pthread-errnos.h>
23 #include <structsem.h>
24
25 #ifndef UP
26 # define LOCK lock
27 #else
28 # define
29 #endif
30
31 #define SYS_futex               240
32 #define FUTEX_WAKE              1
33
34
35         .text
36
37         .globl  __new_sem_post
38         .type   __new_sem_post,@function
39         .align  16
40 __new_sem_post:
41         pushl   %ebx
42
43         movl    8(%esp), %ebx
44
45         LOCK
46 #if VALUE == 0
47         addl    $1, (%ebx)
48 #else
49         addl    $1, VALUE(%ebx)
50 #endif
51
52         cmpl    $0, NWAITERS(%ebx)
53         je      2f
54
55         movl    $SYS_futex, %eax
56         movl    $FUTEX_WAKE, %ecx
57         movl    $1, %edx
58         ENTER_KERNEL
59
60         testl   %eax, %eax
61         js      1f
62
63 2:      xorl    %eax, %eax
64         popl    %ebx
65         ret
66
67 1:
68 #ifdef PIC
69         call    __i686.get_pc_thunk.bx
70 #else
71         movl    $4f, %ebx
72 4:
73 #endif
74         addl    $_GLOBAL_OFFSET_TABLE_, %ebx
75 #if USE___THREAD
76 # ifdef NO_TLS_DIRECT_SEG_REFS
77         movl    errno@gotntpoff(%ebx), %edx
78         addl    %gs:0, %edx
79         movl    $EINVAL, (%edx)
80 # else
81         movl    errno@gotntpoff(%ebx), %edx
82         movl    $EINVAL, %gs:(%edx)
83 # endif
84 #else
85         call    __errno_location@plt
86         movl    $EINVAL, (%eax)
87 #endif
88
89         orl     $-1, %eax
90         popl    %ebx
91         ret
92         .size   __new_sem_post,.-__new_sem_post
93         versioned_symbol(libpthread, __new_sem_post, sem_post, GLIBC_2_1)
94 #if SHLIB_COMPAT(libpthread, GLIBC_2_0, GLIBC_2_1)
95         .global __old_sem_post
96 __old_sem_post = __new_sem_post
97         compat_symbol(libpthread, __old_sem_post, sem_post, GLIBC_2_0)
98 #endif
99
100
101 #ifdef PIC
102         .section .gnu.linkonce.t.__i686.get_pc_thunk.bx,"ax",@progbits
103         .globl  __i686.get_pc_thunk.bx
104         .hidden __i686.get_pc_thunk.bx
105         .type   __i686.get_pc_thunk.bx,@function
106 __i686.get_pc_thunk.bx:
107         movl (%esp), %ebx;
108         ret
109         .size   __i686.get_pc_thunk.bx,.-__i686.get_pc_thunk.bx
110 #endif