Graphite Version 3
An experimental 3D geometry processing program
Loading...
Searching...
No Matches
delaunay_sync.h
Go to the documentation of this file.
1/*
2 * Copyright (c) 2000-2022 Inria
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 *
8 * * Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation
12 * and/or other materials provided with the distribution.
13 * * Neither the name of the ALICE Project-Team nor the names of its
14 * contributors may be used to endorse or promote products derived from this
15 * software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
18 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
21 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27 * POSSIBILITY OF SUCH DAMAGE.
28 *
29 * Contact: Bruno Levy
30 *
31 * https://www.inria.fr/fr/bruno-levy
32 *
33 * Inria,
34 * Domaine de Voluceau,
35 * 78150 Le Chesnay - Rocquencourt
36 * FRANCE
37 *
38 */
39
40#ifndef DELAUNAY_SYNC_H
41#define DELAUNAY_SYNC_H
42
45#include <atomic>
46
47// In GARGANTUA mode (64-bit indices), we also enable
48// more than 127 concurrent threads (16-bits cell status,
49// that contain owner thread id).
50#ifdef GARGANTUA
51#define GEO_CONNECTION_MACHINE
52#endif
53
59namespace GEO {
60
70 public:
71#ifdef GEO_CONNECTION_MACHINE
72 // For machines that can run more than 127 concurrent threads
73 typedef uint16_t thread_index_t;
74 typedef uint16_t cell_status_t;
75 static constexpr cell_status_t FREE_CELL = 32767;
76 static constexpr cell_status_t THREAD_MASK = 32767;
77 static constexpr cell_status_t CONFLICT_MASK = 32768;
78#else
79 typedef uint8_t thread_index_t;
80 typedef uint8_t cell_status_t;
81 static constexpr cell_status_t FREE_CELL = 127;
82 static constexpr cell_status_t THREAD_MASK = 127;
83 static constexpr cell_status_t CONFLICT_MASK = 128;
84#endif
85
86 static constexpr index_t MAX_THREADS = index_t(THREAD_MASK)-1;
87
91 CellStatusArray() : cell_status_(nullptr), size_(0), capacity_(0) {
92 }
93
99 cell_status_(nullptr), size_(0), capacity_(0) {
100 resize(size_in,size_in);
101 }
102
110 clear();
111 }
112
116 CellStatusArray(const CellStatusArray& rhs) = delete;
117
122
132 cell_status_t acquire_cell(index_t cell, cell_status_t status) {
133 geo_debug_assert(cell < size_);
134 cell_status_t expected = FREE_CELL;
135 // strong: acquire_cell is not used in a spinlock-like
136 // spinning loop (so we do not want to have "false negatives")
137 cell_status_[cell].compare_exchange_strong(
138 expected,status,
139 std::memory_order_acquire,std::memory_order_acquire
140 ); // this one could probably be relaxed ----^
141 // if compare_exchange was not sucessful, expected contains
142 // the current stored value.
143 return (expected & THREAD_MASK);
144 }
145
152 geo_debug_assert(cell < size_);
153 cell_status_[cell].store(FREE_CELL, std::memory_order_release);
154 }
155
162 cell_status_t cell_thread(index_t cell) const {
163 geo_debug_assert(cell < size_);
164 return (
165 cell_status_[cell].load(std::memory_order_relaxed) &
166 THREAD_MASK
167 );
168 }
169
178 geo_debug_assert(cell < size_);
179 return(
180 (
181 cell_status_[cell].load(std::memory_order_relaxed) &
182 CONFLICT_MASK
183 ) != 0
184 );
185 }
186
193 // memory_order_relaxed because this function is always called from
194 // a thread that previously acquired the cell.
195 cell_status_[cell].fetch_or(
196 CONFLICT_MASK, std::memory_order_relaxed
197 );
198 }
199
206 void set_cell_status(index_t cell, cell_status_t status) {
207 geo_debug_assert(cell < size_);
208 cell_status_[cell].store(status, std::memory_order_relaxed);
209 }
210
218 void resize(index_t size_in, index_t capacity_in) {
219 geo_debug_assert(capacity_in >= size_in);
221 if(capacity_in > capacity_) {
222 capacity_ = capacity_in;
223 std::atomic<cell_status_t>* old_cell_status = cell_status_;
224 cell_status_ = new std::atomic<cell_status_t>[capacity_];
225 for(index_t i=0; i<capacity_; ++i) {
226 cell_status_t val = (i < size_) ?
227 old_cell_status[i].load(std::memory_order_relaxed) :
228 FREE_CELL;
229 std::atomic_init(&cell_status_[i],val);
230 }
231 delete[] old_cell_status;
232 }
233 size_ = size_in;
234#ifdef __cpp_lib_atomic_is_always_lock_free
235 static_assert(std::atomic<cell_status_t>::is_always_lock_free);
236#else
237 geo_debug_assert(size_ == 0 || cell_status_[0].is_lock_free());
238#endif
239 }
240
246 void resize(index_t size_in) {
247 resize(size_in, size_in);
248 }
249
257 void reserve(index_t new_capacity) {
258 if(new_capacity > capacity_) {
259 resize(size_, new_capacity);
260 }
261 }
262
267 void grow() {
268 if(size_+1 >= capacity_) {
269 resize(size_+1, std::max(capacity_*2,size_+1));
270 } else {
271 resize(size_+1, capacity_);
272 }
273 }
274
279 index_t size() const {
280 return size_;
281 }
282
288 void clear() {
290#ifdef GEO_DEBUG
291 for(index_t i=0; i<size_; ++i) {
292 geo_debug_assert(cell_thread(i) == FREE_CELL);
293 }
294#endif
295 delete[] cell_status_;
296 cell_status_ = nullptr;
297 size_ = 0;
298 capacity_ = 0;
299 }
300
301 private:
302 std::atomic<cell_status_t>* cell_status_;
303 index_t size_;
304 index_t capacity_;
305 };
306}
307
308#endif
Assertion checking mechanism.
#define geo_debug_assert(x)
Verifies that a condition is met.
Definition assert.h:196
An array of cell status codes associates to each tetrahedron in a Delaunay tetrahedralization.
void resize(index_t size_in)
Resizes this CellStatusArray.
void reserve(index_t new_capacity)
Reserves additional space.
void grow()
Increases the size of the array for one additional element.
CellStatusArray(index_t size_in)
Creates a CellStatusArray.
CellStatusArray(const CellStatusArray &rhs)=delete
Forbids copy.
void mark_cell_as_conflict(index_t cell)
Marks a cell as conflict.
CellStatusArray()
Creates an empty CellStatusArray.
CellStatusArray & operator=(const CellStatusArray &rhs)=delete
Forbids copy.
void release_cell(index_t cell)
Releases a cell.
void resize(index_t size_in, index_t capacity_in)
Resizes this CellStatusArray.
void clear()
Clears this CellStatusArray.
bool cell_is_marked_as_conflict(index_t cell) const
Tests whether a cell is marked as conflict.
~CellStatusArray()
CellStatusArray destructor.
void set_cell_status(index_t cell, cell_status_t status)
Sets the status of a cell.
index_t size() const
Gets the size of this CellStatusArray.
cell_status_t cell_thread(index_t cell) const
Gets the thread that acquired a cell.
cell_status_t acquire_cell(index_t cell, cell_status_t status)
Tentatively acquires a cell.
Common include file, providing basic definitions. Should be included before anything else by all head...
bool is_running_threads()
Checks whether threads are running.
Global Vorpaline namespace.
geo_index_t index_t
The type for storing and manipulating indices.
Definition numeric.h:329