annotate libitm/clone.cc @ 131:84e7813d76e9

gcc-8.2
author mir3636
date Thu, 25 Oct 2018 07:37:49 +0900 (2018-10-24)
parents 04ced10e8804
children 1830386684a0
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
1 /* Copyright (C) 2009-2018 Free Software Foundation, Inc.
111
kono
parents:
diff changeset
2 Contributed by Richard Henderson <rth@redhat.com>.
kono
parents:
diff changeset
3
kono
parents:
diff changeset
4 This file is part of the GNU Transactional Memory Library (libitm).
kono
parents:
diff changeset
5
kono
parents:
diff changeset
6 Libitm is free software; you can redistribute it and/or modify it
kono
parents:
diff changeset
7 under the terms of the GNU General Public License as published by
kono
parents:
diff changeset
8 the Free Software Foundation; either version 3 of the License, or
kono
parents:
diff changeset
9 (at your option) any later version.
kono
parents:
diff changeset
10
kono
parents:
diff changeset
11 Libitm is distributed in the hope that it will be useful, but WITHOUT ANY
kono
parents:
diff changeset
12 WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
kono
parents:
diff changeset
13 FOR A PARTICULAR PURPOSE. See the GNU General Public License for
kono
parents:
diff changeset
14 more details.
kono
parents:
diff changeset
15
kono
parents:
diff changeset
16 Under Section 7 of GPL version 3, you are granted additional
kono
parents:
diff changeset
17 permissions described in the GCC Runtime Library Exception, version
kono
parents:
diff changeset
18 3.1, as published by the Free Software Foundation.
kono
parents:
diff changeset
19
kono
parents:
diff changeset
20 You should have received a copy of the GNU General Public License and
kono
parents:
diff changeset
21 a copy of the GCC Runtime Library Exception along with this program;
kono
parents:
diff changeset
22 see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
kono
parents:
diff changeset
23 <http://www.gnu.org/licenses/>. */
kono
parents:
diff changeset
24
kono
parents:
diff changeset
25 #include "libitm_i.h"
kono
parents:
diff changeset
26
kono
parents:
diff changeset
27 using namespace GTM;
kono
parents:
diff changeset
28
kono
parents:
diff changeset
29 struct clone_entry
kono
parents:
diff changeset
30 {
kono
parents:
diff changeset
31 void *orig, *clone;
kono
parents:
diff changeset
32 };
kono
parents:
diff changeset
33
kono
parents:
diff changeset
34 struct clone_table
kono
parents:
diff changeset
35 {
kono
parents:
diff changeset
36 clone_entry *table;
kono
parents:
diff changeset
37 size_t size;
kono
parents:
diff changeset
38 clone_table *next;
kono
parents:
diff changeset
39 };
kono
parents:
diff changeset
40
kono
parents:
diff changeset
41 static clone_table *all_tables;
kono
parents:
diff changeset
42
kono
parents:
diff changeset
43 static void *
kono
parents:
diff changeset
44 find_clone (void *ptr)
kono
parents:
diff changeset
45 {
kono
parents:
diff changeset
46 clone_table *table;
kono
parents:
diff changeset
47
kono
parents:
diff changeset
48 for (table = all_tables; table ; table = table->next)
kono
parents:
diff changeset
49 {
kono
parents:
diff changeset
50 clone_entry *t = table->table;
kono
parents:
diff changeset
51 size_t lo = 0, hi = table->size, i;
kono
parents:
diff changeset
52
kono
parents:
diff changeset
53 /* Quick test for whether PTR is present in this table. */
kono
parents:
diff changeset
54 if (ptr < t[0].orig || ptr > t[hi - 1].orig)
kono
parents:
diff changeset
55 continue;
kono
parents:
diff changeset
56
kono
parents:
diff changeset
57 /* Otherwise binary search. */
kono
parents:
diff changeset
58 while (lo < hi)
kono
parents:
diff changeset
59 {
kono
parents:
diff changeset
60 i = (lo + hi) / 2;
kono
parents:
diff changeset
61 if (ptr < t[i].orig)
kono
parents:
diff changeset
62 hi = i;
kono
parents:
diff changeset
63 else if (ptr > t[i].orig)
kono
parents:
diff changeset
64 lo = i + 1;
kono
parents:
diff changeset
65 else
kono
parents:
diff changeset
66 return t[i].clone;
kono
parents:
diff changeset
67 }
kono
parents:
diff changeset
68
kono
parents:
diff changeset
69 /* Given the quick test above, if we don't find the entry in
kono
parents:
diff changeset
70 this table then it doesn't exist. */
kono
parents:
diff changeset
71 break;
kono
parents:
diff changeset
72 }
kono
parents:
diff changeset
73
kono
parents:
diff changeset
74 return NULL;
kono
parents:
diff changeset
75 }
kono
parents:
diff changeset
76
kono
parents:
diff changeset
77
kono
parents:
diff changeset
78 void * ITM_REGPARM
kono
parents:
diff changeset
79 _ITM_getTMCloneOrIrrevocable (void *ptr)
kono
parents:
diff changeset
80 {
kono
parents:
diff changeset
81 void *ret = find_clone (ptr);
kono
parents:
diff changeset
82 if (ret)
kono
parents:
diff changeset
83 return ret;
kono
parents:
diff changeset
84
kono
parents:
diff changeset
85 gtm_thr()->serialirr_mode ();
kono
parents:
diff changeset
86
kono
parents:
diff changeset
87 return ptr;
kono
parents:
diff changeset
88 }
kono
parents:
diff changeset
89
kono
parents:
diff changeset
90 void * ITM_REGPARM
kono
parents:
diff changeset
91 _ITM_getTMCloneSafe (void *ptr)
kono
parents:
diff changeset
92 {
kono
parents:
diff changeset
93 void *ret = find_clone (ptr);
kono
parents:
diff changeset
94 if (ret == NULL)
kono
parents:
diff changeset
95 abort ();
kono
parents:
diff changeset
96 return ret;
kono
parents:
diff changeset
97 }
kono
parents:
diff changeset
98
kono
parents:
diff changeset
99 static int
kono
parents:
diff changeset
100 clone_entry_compare (const void *a, const void *b)
kono
parents:
diff changeset
101 {
kono
parents:
diff changeset
102 const clone_entry *aa = (const clone_entry *)a;
kono
parents:
diff changeset
103 const clone_entry *bb = (const clone_entry *)b;
kono
parents:
diff changeset
104
kono
parents:
diff changeset
105 if (aa->orig < bb->orig)
kono
parents:
diff changeset
106 return -1;
kono
parents:
diff changeset
107 else if (aa->orig > bb->orig)
kono
parents:
diff changeset
108 return 1;
kono
parents:
diff changeset
109 else
kono
parents:
diff changeset
110 return 0;
kono
parents:
diff changeset
111 }
kono
parents:
diff changeset
112
kono
parents:
diff changeset
113 namespace {
kono
parents:
diff changeset
114
kono
parents:
diff changeset
115 // Within find_clone, we know that we are inside a transaction. Because
kono
parents:
diff changeset
116 // of that, we have already synchronized with serial_lock. By taking the
kono
parents:
diff changeset
117 // serial_lock for write, we exclude all transactions while we make this
kono
parents:
diff changeset
118 // change to the clone tables, without having to synchronize on a separate
kono
parents:
diff changeset
119 // lock. Do be careful not to attempt a recursive write lock.
kono
parents:
diff changeset
120
kono
parents:
diff changeset
121 class ExcludeTransaction
kono
parents:
diff changeset
122 {
kono
parents:
diff changeset
123 bool do_lock;
kono
parents:
diff changeset
124
kono
parents:
diff changeset
125 public:
kono
parents:
diff changeset
126 ExcludeTransaction()
kono
parents:
diff changeset
127 {
kono
parents:
diff changeset
128 gtm_thread *tx = gtm_thr();
kono
parents:
diff changeset
129 do_lock = !(tx && (tx->state & gtm_thread::STATE_SERIAL));
kono
parents:
diff changeset
130
kono
parents:
diff changeset
131 if (do_lock)
kono
parents:
diff changeset
132 gtm_thread::serial_lock.write_lock ();
kono
parents:
diff changeset
133 }
kono
parents:
diff changeset
134
kono
parents:
diff changeset
135 ~ExcludeTransaction()
kono
parents:
diff changeset
136 {
kono
parents:
diff changeset
137 if (do_lock)
kono
parents:
diff changeset
138 gtm_thread::serial_lock.write_unlock ();
kono
parents:
diff changeset
139 }
kono
parents:
diff changeset
140 };
kono
parents:
diff changeset
141
kono
parents:
diff changeset
142 } // end anon namespace
kono
parents:
diff changeset
143
kono
parents:
diff changeset
144
kono
parents:
diff changeset
145 void
kono
parents:
diff changeset
146 _ITM_registerTMCloneTable (void *xent, size_t size)
kono
parents:
diff changeset
147 {
kono
parents:
diff changeset
148 clone_entry *ent = static_cast<clone_entry *>(xent);
kono
parents:
diff changeset
149 clone_table *table;
kono
parents:
diff changeset
150
kono
parents:
diff changeset
151 table = (clone_table *) xmalloc (sizeof (clone_table));
kono
parents:
diff changeset
152 table->table = ent;
kono
parents:
diff changeset
153 table->size = size;
kono
parents:
diff changeset
154
kono
parents:
diff changeset
155 qsort (ent, size, sizeof (clone_entry), clone_entry_compare);
kono
parents:
diff changeset
156
kono
parents:
diff changeset
157 // Hold the serial_lock while we update the ALL_TABLES datastructure.
kono
parents:
diff changeset
158 {
kono
parents:
diff changeset
159 ExcludeTransaction exclude;
kono
parents:
diff changeset
160 table->next = all_tables;
kono
parents:
diff changeset
161 all_tables = table;
kono
parents:
diff changeset
162 }
kono
parents:
diff changeset
163 }
kono
parents:
diff changeset
164
kono
parents:
diff changeset
165 void
kono
parents:
diff changeset
166 _ITM_deregisterTMCloneTable (void *xent)
kono
parents:
diff changeset
167 {
kono
parents:
diff changeset
168 clone_entry *ent = static_cast<clone_entry *>(xent);
kono
parents:
diff changeset
169 clone_table *tab;
kono
parents:
diff changeset
170
kono
parents:
diff changeset
171 // Hold the serial_lock while we update the ALL_TABLES datastructure.
kono
parents:
diff changeset
172 {
kono
parents:
diff changeset
173 ExcludeTransaction exclude;
kono
parents:
diff changeset
174 clone_table **pprev;
kono
parents:
diff changeset
175
kono
parents:
diff changeset
176 for (pprev = &all_tables;
kono
parents:
diff changeset
177 tab = *pprev, tab->table != ent;
kono
parents:
diff changeset
178 pprev = &tab->next)
kono
parents:
diff changeset
179 continue;
kono
parents:
diff changeset
180 *pprev = tab->next;
kono
parents:
diff changeset
181 }
kono
parents:
diff changeset
182
kono
parents:
diff changeset
183 free (tab);
kono
parents:
diff changeset
184 }