Fixed incorrect use of mvcle introduced by 2001-07-12 change.
[kopensolaris-gnu/glibc.git] / sysdeps / mips / atomicity.h
1 /* Low-level functions for atomic operations. Mips version.
2    Copyright (C) 2001 Free Software Foundation, Inc.
3    This file is part of the GNU C Library.
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 #ifndef _MIPS_ATOMICITY_H
21 #define _MIPS_ATOMICITY_H    1
22
23 #include <sgidefs.h>
24 #include <inttypes.h>
25
26 #if (_MIPS_ISA >= _MIPS_ISA_MIPS2)
27
28 static inline int
29 __attribute__ ((unused))
30 exchange_and_add (volatile uint32_t *mem, int val)
31 {
32   int result, tmp;
33
34   __asm__ __volatile__
35     ("/* Inline exchange & add */\n\t"
36      "1:\n\t"
37      "ll        %0,%3\n\t"
38      "addu      %1,%4,%0\n\t"
39      "sc        %1,%2\n\t"
40      "beqz      %1,1b\n\t"
41      "/* End exchange & add */"
42      : "=&r"(result), "=&r"(tmp), "=m"(*mem)
43      : "m" (*mem), "r"(val)
44      : "memory");
45
46   return result;
47 }
48
49 static inline void
50 __attribute__ ((unused))
51 atomic_add (volatile uint32_t *mem, int val)
52 {
53   int result;
54
55   __asm__ __volatile__
56     ("/* Inline atomic add */\n\t"
57      "1:\n\t"
58      "ll        %0,%2\n\t"
59      "addu      %0,%3,%0\n\t"
60      "sc        %0,%1\n\t"
61      "beqz      %0,1b\n\t"
62      "/* End atomic add */"
63      : "=&r"(result), "=m"(*mem)
64      : "m" (*mem), "r"(val)
65      : "memory");
66 }
67
68 static inline int
69 __attribute__ ((unused))
70 compare_and_swap (volatile long int *p, long int oldval, long int newval)
71 {
72   long int ret;
73
74   __asm__ __volatile__
75     ("/* Inline compare & swap */\n\t"
76      "1:\n\t"
77      "ll        %0,%4\n\t"
78      ".set      push\n"
79      ".set      noreorder\n\t"
80      "bne       %0,%2,2f\n\t"
81      "move      %0,%3\n\t"
82      ".set      pop\n\t"
83      "sc        %0,%1\n\t"
84      "beqz      %0,1b\n"
85      "2:\n\t"
86      "/* End compare & swap */"
87      : "=&r" (ret), "=m" (*p)
88      : "r" (oldval), "r" (newval), "m" (*p)
89      : "memory");
90
91   return ret;
92 }
93
94 #else /* (_MIPS_ISA >= _MIPS_ISA_MIPS2) */
95
96 #warning MIPS I atomicity functions are not atomic
97
98 static inline int
99 __attribute__ ((unused))
100 exchange_and_add (volatile uint32_t *mem, int val)
101 {
102   int result = *mem;
103   *mem += val;
104   return result;
105 }
106
107 static inline void
108 __attribute__ ((unused))
109 atomic_add (volatile uint32_t *mem, int val)
110 {
111   *mem += val;
112 }
113
114 static inline int
115 __attribute__ ((unused))
116 compare_and_swap (volatile long int *p, long int oldval, long int newval)
117 {
118   if (*p != oldval)
119     return 0;
120
121   *p = newval;
122   return 1;
123 }
124
125 #endif /* !(_MIPS_ISA >= _MIPS_ISA_MIPS2) */
126
127 #endif /* atomicity.h */