]> git.ipfire.org Git - thirdparty/glibc.git/blame - sysdeps/unix/sysv/linux/x86/elision-lock.c
Update copyright dates with scripts/update-copyrights.
[thirdparty/glibc.git] / sysdeps / unix / sysv / linux / x86 / elision-lock.c
CommitLineData
1cdbe579 1/* elision-lock.c: Elided pthread mutex lock.
04277e02 2 Copyright (C) 2011-2019 Free Software Foundation, Inc.
1cdbe579
AK
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, see
075b9322 17 <http://www.gnu.org/licenses/>. */
1cdbe579
AK
18
19#include <pthread.h>
20#include "pthreadP.h"
21#include "lowlevellock.h"
22#include "hle.h"
23#include <elision-conf.h>
24
25#if !defined(LLL_LOCK) && !defined(EXTRAARG)
26/* Make sure the configuration code is always linked in for static
27 libraries. */
28#include "elision-conf.c"
29#endif
30
31#ifndef EXTRAARG
32#define EXTRAARG
33#endif
34#ifndef LLL_LOCK
35#define LLL_LOCK(a,b) lll_lock(a,b), 0
36#endif
37
38#define aconf __elision_aconf
39
40/* Adaptive lock using transactions.
41 By default the lock region is run as a transaction, and when it
42 aborts or the lock is busy the lock adapts itself. */
43
44int
45__lll_lock_elision (int *futex, short *adapt_count, EXTRAARG int private)
46{
ca6e601a
TR
47 /* adapt_count can be accessed concurrently; these accesses can be both
48 inside of transactions (if critical sections are nested and the outer
49 critical section uses lock elision) and outside of transactions. Thus,
50 we need to use atomic accesses to avoid data races. However, the
51 value of adapt_count is just a hint, so relaxed MO accesses are
52 sufficient. */
53 if (atomic_load_relaxed (adapt_count) <= 0)
1cdbe579
AK
54 {
55 unsigned status;
56 int try_xbegin;
57
58 for (try_xbegin = aconf.retry_try_xbegin;
59 try_xbegin > 0;
60 try_xbegin--)
61 {
62 if ((status = _xbegin()) == _XBEGIN_STARTED)
63 {
64 if (*futex == 0)
65 return 0;
66
075b9322 67 /* Lock was busy. Fall back to normal locking.
1cdbe579
AK
68 Could also _xend here but xabort with 0xff code
69 is more visible in the profiler. */
70 _xabort (_ABORT_LOCK_BUSY);
71 }
72
73 if (!(status & _XABORT_RETRY))
74 {
75 if ((status & _XABORT_EXPLICIT)
76 && _XABORT_CODE (status) == _ABORT_LOCK_BUSY)
77 {
075b9322 78 /* Right now we skip here. Better would be to wait a bit
ca6e601a
TR
79 and retry. This likely needs some spinning. See
80 above for why relaxed MO is sufficient. */
81 if (atomic_load_relaxed (adapt_count)
82 != aconf.skip_lock_busy)
83 atomic_store_relaxed (adapt_count, aconf.skip_lock_busy);
1cdbe579 84 }
075b9322 85 /* Internal abort. There is no chance for retry.
1cdbe579 86 Use the normal locking and next time use lock.
ca6e601a
TR
87 Be careful to avoid writing to the lock. See above for why
88 relaxed MO is sufficient. */
89 else if (atomic_load_relaxed (adapt_count)
90 != aconf.skip_lock_internal_abort)
91 atomic_store_relaxed (adapt_count,
92 aconf.skip_lock_internal_abort);
1cdbe579
AK
93 break;
94 }
95 }
96 }
97 else
98 {
99 /* Use a normal lock until the threshold counter runs out.
100 Lost updates possible. */
ca6e601a
TR
101 atomic_store_relaxed (adapt_count,
102 atomic_load_relaxed (adapt_count) - 1);
1cdbe579
AK
103 }
104
105 /* Use a normal lock here. */
106 return LLL_LOCK ((*futex), private);
107}