Graphite Version 3
An experimental 3D geometry processing program
Loading...
Searching...
No Matches
memory.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 GEOGRAM_BASIC_MEMORY
41#define GEOGRAM_BASIC_MEMORY
42
47#include <vector>
48#include <string.h>
49#include <stdlib.h>
50
51#ifdef GEO_OS_WINDOWS
52
53#include <windows.h>
54#ifdef min
55#undef min
56#endif
57#ifdef max
58#undef max
59#endif
60
61#else
62
63#include <unistd.h>
64
65#endif
66
67// Stack size depending on OS:
68// Linux: 10 Mb
69// Windows: 1 Mb
70// Mac OSX: 512 Kb
71// GEO_HAS_BIG_STACK is defined under Linux
72// and lets some of the functions that
73// manipulate exact precision numbers
74// allocate temporaries on the stack.
75
76#ifdef GEO_OS_LINUX
77#define GEO_HAS_BIG_STACK
78#endif
79
85namespace GEO {
86
90 namespace Memory {
92 typedef unsigned char byte;
93
95 typedef unsigned char word8;
96
98 typedef unsigned short word16;
99
101 typedef unsigned int word32;
102
104 typedef byte* pointer;
105
107 typedef void (*function_pointer)();
108
116 inline void clear(void* addr, size_t size) {
117 ::memset(addr, 0, size);
118 }
119
129 inline void copy(void* to, const void* from, size_t size) {
130 ::memcpy(to, from, size);
131 }
132
144 ) {
145 // I know this is ugly, but I did not find a simpler warning-free
146 // way that is portable between all compilers.
147 pointer result = nullptr;
148 ::memcpy(&result, &fptr, sizeof(pointer));
149 return result;
150 }
151
162 pointer ptr
163 ) {
164 // I know this is ugly, but I did not find a simpler warning-free
165 // way that is portable between all compilers.
166 function_pointer result = nullptr;
167 ::memcpy(&result, &ptr, sizeof(pointer));
168 return result;
169 }
170
181 // I know this is ugly, but I did not find a simpler warning-free
182 // way that is portable between all compilers.
183 function_pointer result = nullptr;
184 ::memcpy(&result, &ptr, sizeof(pointer));
185 return result;
186 }
187
196#define GEO_MEMORY_ALIGNMENT 64
197
208 template <int DIM>
214 static const size_t value = 1;
215 };
216
221 template <>
222 struct PointAlignment<2> {
223 static const size_t value = 16;
224 };
225
230 template <>
231 struct PointAlignment<3> {
232 static const size_t value = 8;
233 };
234
239 template <>
240 struct PointAlignment<4> {
241 static const size_t value = 32;
242 };
243
248 template <>
249 struct PointAlignment<6> {
250 static const size_t value = 16;
251 };
252
257 template <>
258 struct PointAlignment<8> {
259 static const size_t value = 64;
260 };
261
270#define geo_dim_alignment(dim) GEO::Memory::PointAlignment<dim>::value
271
281 inline void* aligned_malloc(
282 size_t size, size_t alignment = GEO_MEMORY_ALIGNMENT
283 ) {
284#if defined(GEO_OS_ANDROID)
285 // Alignment not supported under Android.
286 geo_argused(alignment);
287 return malloc(size);
288#elif defined(GEO_COMPILER_INTEL)
289 return _mm_malloc(size, alignment);
290#elif defined(GEO_COMPILER_GCC) || defined(GEO_COMPILER_CLANG)
291 void* result;
292 return posix_memalign(&result, alignment, size) == 0
293 ? result : nullptr;
294#elif defined(GEO_COMPILER_MSVC)
295 return _aligned_malloc(size, alignment);
296#else
297 geo_argused(alignment);
298 return malloc(size);
299#endif
300 }
301
309 inline void aligned_free(void* p) {
310#if defined(GEO_OS_ANDROID)
311 // Alignment not supported under Android.
312 free(p);
313#elif defined(GEO_COMPILER_INTEL)
314 _mm_free(p);
315#elif defined(GEO_COMPILER_GCC_FAMILY)
316 free(p);
317#elif defined(GEO_COMPILER_MSVC)
318 _aligned_free(p);
319#else
320 free(p);
321#endif
322 }
323
337#if defined(GEO_OS_ANDROID)
338#define geo_decl_aligned(var) var
339#elif defined(GEO_COMPILER_INTEL)
340#define geo_decl_aligned(var) __declspec(aligned(GEO_MEMORY_ALIGNMENT)) var
341#elif defined(GEO_COMPILER_GCC_FAMILY)
342#define geo_decl_aligned(var) var __attribute__((aligned(GEO_MEMORY_ALIGNMENT)))
343#elif defined(GEO_COMPILER_MSVC)
344#define geo_decl_aligned(var) __declspec(align(GEO_MEMORY_ALIGNMENT)) var
345#elif defined(GEO_COMPILER_EMSCRIPTEN)
346#define geo_decl_aligned(var) var
347#endif
348
365#if defined(GEO_OS_ANDROID)
366#define geo_assume_aligned(var, alignment)
367#elif defined(GEO_COMPILER_INTEL)
368#define geo_assume_aligned(var, alignment) \
369 __assume_aligned(var, alignment)
370#elif defined(GEO_COMPILER_CLANG)
371#define geo_assume_aligned(var, alignment)
372 // GCC __builtin_assume_aligned is not yet supported by clang-3.3
373#elif defined(GEO_COMPILER_GCC)
374#if __GNUC__ >= 4 && __GNUC_MINOR__ >= 7
375#define geo_assume_aligned(var, alignment) \
376 *(void**) (&var) = __builtin_assume_aligned(var, alignment)
377 // the GCC way of specifying that a pointer is aligned returns
378 // the aligned pointer (I can't figure out why). It needs to be
379 // affected otherwise it is not taken into account (verified by
380 // looking at the output of gcc -S)
381#else
382#define geo_assume_aligned(var, alignment)
383#endif
384#elif defined(GEO_COMPILER_MSVC)
385#define geo_assume_aligned(var, alignment)
386 // TODO: I do not know how to do that with MSVC
387#elif defined(GEO_COMPILER_EMSCRIPTEN)
388#define geo_assume_aligned(var, alignment)
389#elif defined(GEO_COMPILER_MINGW)
390#define geo_assume_aligned(var, alignment)
391#endif
392
403#if defined(GEO_COMPILER_INTEL)
404#define geo_restrict __restrict
405#elif defined(GEO_COMPILER_GCC_FAMILY)
406#define geo_restrict __restrict__
407#elif defined(GEO_COMPILER_MSVC)
408#define geo_restrict __restrict
409#elif defined(GEO_COMPILER_EMSCRIPTEN)
410#define geo_restrict
411#endif
412
420 inline bool is_aligned(
421 void* p, size_t alignment = GEO_MEMORY_ALIGNMENT
422 ) {
423 return (reinterpret_cast<size_t>(p) & (alignment - 1)) == 0;
424 }
425
429 inline void* align(void* p) {
430 size_t offset = (
432 (reinterpret_cast<size_t>(p) & (GEO_MEMORY_ALIGNMENT - 1))
433 ) & (GEO_MEMORY_ALIGNMENT - 1);
434 return reinterpret_cast<char*>(p) + offset;
435 }
436
446#define geo_aligned_alloca(size) \
447 GEO::Memory::align(alloca(size + GEO_MEMORY_ALIGNMENT - 1))
448
456 template <class T, int ALIGN = GEO_MEMORY_ALIGNMENT>
458 public:
460 typedef T value_type;
461
463 typedef T* pointer;
464
466 typedef T& reference;
467
469 typedef const T* const_pointer;
470
472 typedef const T& const_reference;
473
475 typedef ::std::size_t size_type;
476
478 typedef ::std::ptrdiff_t difference_type;
479
481 static constexpr int ALIGNMENT = ALIGN;
482
487 template <class U>
488 struct rebind {
491 };
492
499 return &x;
500 }
501
508 return &x;
509 }
510
529 size_type nb_elt, const void* hint = nullptr
530 ) {
531 nb_elt = std::max(nb_elt,size_type(1));
532 geo_argused(hint);
533 while(true) {
534 pointer result = static_cast<pointer>(
535 aligned_malloc(sizeof(T) * nb_elt, ALIGNMENT)
536 );
537 if(result != nullptr) {
538 return result;
539 }
540 // under Linux, a process requesting more mem than available
541 // is killed (and there is nothing we can capture). Under
542 // other OSes, the standard mechanism to let the runtime
543 // know is as follows:
544 // see: https://stackoverflow.com/questions/7194127/
545 // how-should-i-write-iso-c-standard-conformant-custom-
546 // new-and-delete-operators
547 // (if there is a handler, call it repeatedly until
548 // allocation succeeds, else throw a bad_alloc exception)
549 std::new_handler handler = std::get_new_handler();
550 if(handler != nullptr) {
551 (*handler)();
552 } else {
553 throw std::bad_alloc();
554 }
555 }
556 }
557
569 void deallocate(pointer p, size_type nb_elt) {
570 geo_argused(nb_elt);
571 aligned_free(p);
572 }
573
581 ::std::allocator<char> a;
582 return std::allocator_traits<decltype(a)>::max_size(a) /
583 sizeof(T);
584 }
585
599 new (static_cast<void*>(p))value_type(val);
600 }
601
610 void destroy(pointer p) {
611 p->~value_type();
612#ifdef GEO_COMPILER_MSVC
613 (void) p; // to avoid a "unreferenced variable" warning
614#endif
615 }
616
621 template <class T2, int A2> operator aligned_allocator<T2, A2>() {
623 }
624 };
625
630 template <typename T1, int A1, typename T2, int A2>
631 inline bool operator== (
633 ) {
634 return true;
635 }
636
641 template <typename T1, int A1, typename T2, int A2>
642 inline bool operator!= (
644 ) {
645 return false;
646 }
647 }
648
649 /************************************************************************/
650
659 template <class T>
660 class vector : public ::std::vector<T, Memory::aligned_allocator<T> > {
665
669 typedef ::std::vector<T, Memory::aligned_allocator<T> > baseclass;
670
671
672
673 public:
678 baseclass() {
679 }
680
687 explicit vector(index_t size) :
688 baseclass(size) {
689 }
690
698 explicit vector(index_t size, const T& val) :
699 baseclass(size, val) {
700 }
701
706 index_t size() const {
707 // casts baseclass::size() from size_t (64 bits)
708 // to index_t (32 bits), because all
709 // indices in Vorpaline are supposed to fit in 32 bits (index_t).
710 // TODO: geo_debug_assert(baseclass::size() < max index_t)
711 return index_t(baseclass::size());
712 }
713
720 geo_debug_assert(i < size());
721 return baseclass::operator[] (i);
722 }
723
730 const T& operator[] (index_t i) const {
731 geo_debug_assert(i < size());
732 return baseclass::operator[] (i);
733 }
734
741 geo_debug_assert(i >= 0 && index_t(i) < size());
742 return baseclass::operator[] (index_t(i));
743 }
744
751 const T& operator[] (signed_index_t i) const {
752 geo_debug_assert(i >= 0 && index_t(i) < size());
753 return baseclass::operator[] (index_t(i));
754 }
755
756
757#ifdef GARGANTUA // If compiled with 64 bits index_t
758
764 T& operator[] (int i) {
765 geo_debug_assert(i >= 0 && index_t(i) < size());
766 return baseclass::operator[] (index_t(i));
767 }
768
775 const T& operator[] (int i) const {
776 geo_debug_assert(i >= 0 && index_t(i) < size());
777 return baseclass::operator[] (index_t(i));
778 }
779
785 T& operator[] (unsigned int i) {
787 return baseclass::operator[] (index_t(i));
788 }
789
796 const T& operator[] (unsigned int i) const {
798 return baseclass::operator[] (index_t(i));
799 }
800#endif
801
806 T* data() {
807 T* result = baseclass::data();
808 // Tell the compiler that the pointer is aligned, to enable AVX
809 // vectorization, can be useful when using vector<double>
810 // with blas-like operations. I hope the hint will propagate to
811 // the caller (not sure...)
812 geo_assume_aligned(result, allocator::ALIGNMENT);
813 return result;
814 }
815
820 const T* data() const {
821 const T* result = baseclass::data();
822 // Tell the compiler that the pointer is aligned, to enable AVX
823 // vectorization, can be useful when using vector<double>
824 // with blas-like operations. I hope the hint will propagate to
825 // the caller (not sure...)
826 geo_assume_aligned(result, allocator::ALIGNMENT);
827 return result;
828 }
829
830
837 vector<T> other;
838 this->swap(other);
839 }
840 };
841
848 template <>
849 class vector<bool> : public ::std::vector<bool> {
850 typedef ::std::vector<bool> baseclass;
851
852 public:
855 baseclass() {
856 }
857
859 explicit vector(index_t size) :
860 baseclass(size) {
861 }
862
864 explicit vector(index_t size, bool val) :
865 baseclass(size, val) {
866 }
867
869 index_t size() const {
870 // casts baseclass::size() from size_t (64 bits)
871 // to index_t (32 bits), because all
872 // indices in Vorpaline are supposed to fit in 32 bits (index_t).
873 // TODO: geo_debug_assert(baseclass::size() < max index_t)
874 return index_t(baseclass::size());
875 }
876
877 // TODO: operator[] with bounds checking (more complicated
878 // than just returning bool&, check implementation in STL).
879 };
880}
881
882#endif
A function to suppress unused parameters compilation warnings.
Assertion checking mechanism.
#define geo_debug_assert(x)
Verifies that a condition is met.
Definition assert.h:196
An allocator that performs aligned memory allocations.
Definition memory.h:457
size_type max_size() const
Gets the maximum size possible to allocate.
Definition memory.h:580
T * pointer
Pointer to element.
Definition memory.h:463
const T & const_reference
Reference to constant element.
Definition memory.h:472
void destroy(pointer p)
Destroys an object.
Definition memory.h:610
const T * const_pointer
Pointer to constant element.
Definition memory.h:469
pointer address(reference x)
Gets the address of an object.
Definition memory.h:498
pointer allocate(size_type nb_elt, const void *hint=nullptr)
Allocates a block of storage.
Definition memory.h:528
T value_type
Element type.
Definition memory.h:460
static constexpr int ALIGNMENT
Alignment in bytes.
Definition memory.h:481
const_pointer address(const_reference x)
Gets the address of a object.
Definition memory.h:507
T & reference
Reference to element.
Definition memory.h:466
void construct(pointer p, const_reference val)
Constructs an object.
Definition memory.h:598
::std::size_t size_type
Quantities of elements.
Definition memory.h:475
void deallocate(pointer p, size_type nb_elt)
Releases a block of storage.
Definition memory.h:569
::std::ptrdiff_t difference_type
Difference between two pointers.
Definition memory.h:478
index_t size() const
Gets the number of elements.
Definition memory.h:869
vector(index_t size)
Creates a pre-allocated vector.
Definition memory.h:859
vector(index_t size, bool val)
Creates a pre-initialized vector.
Definition memory.h:864
vector()
Creates an empty vector.
Definition memory.h:854
Vector with aligned memory allocation.
Definition memory.h:660
void clear_and_deallocate()
Resizes this vector to zero and deallocated all the memory.
Definition memory.h:836
vector()
Creates an empty vector.
Definition memory.h:677
const T * data() const
Gets a pointer to the array of elements.
Definition memory.h:820
vector(index_t size, const T &val)
Creates a pre-initialized vector.
Definition memory.h:698
vector(index_t size)
Creates a pre-allocated vector.
Definition memory.h:687
T * data()
Gets a pointer to the array of elements.
Definition memory.h:806
T & operator[](index_t i)
Gets a vector element.
Definition memory.h:719
index_t size() const
Gets the number of elements.
Definition memory.h:706
Common include file, providing basic definitions. Should be included before anything else by all head...
#define GEO_MEMORY_ALIGNMENT
Default memory alignment for efficient vector operations.
Definition memory.h:196
void * aligned_malloc(size_t size, size_t alignment=GEO_MEMORY_ALIGNMENT)
Allocates aligned memory.
Definition memory.h:281
unsigned short word16
Unsigned 16 bits integer.
Definition memory.h:98
unsigned char byte
Unsigned byte type.
Definition memory.h:92
void(* function_pointer)()
Generic function pointer.
Definition memory.h:107
void * align(void *p)
Returns the smallest aligned memory address from p.
Definition memory.h:429
pointer function_pointer_to_generic_pointer(function_pointer fptr)
Converts a function pointer to a generic pointer.
Definition memory.h:142
void aligned_free(void *p)
Deallocates aligned memory.
Definition memory.h:309
unsigned int word32
Unsigned 32 bits integer.
Definition memory.h:101
byte * pointer
Pointer to unsigned byte(s)
Definition memory.h:104
unsigned char word8
Unsigned 8 bits integer.
Definition memory.h:95
void clear(void *addr, size_t size)
Clears a memory block.
Definition memory.h:116
bool is_aligned(void *p, size_t alignment=GEO_MEMORY_ALIGNMENT)
Checks whether a pointer is aligned.
Definition memory.h:420
bool operator!=(const aligned_allocator< T1, A1 > &, const aligned_allocator< T2, A2 > &)
Tests whether two aligned_allocators are different.
Definition memory.h:642
function_pointer generic_pointer_to_function_pointer(pointer ptr)
Converts a generic pointer to a function pointer.
Definition memory.h:161
bool operator==(const aligned_allocator< T1, A1 > &, const aligned_allocator< T2, A2 > &)
Tests whether two aligned_allocators are equal.
Definition memory.h:631
void copy(void *to, const void *from, size_t size)
Copies a memory block.
Definition memory.h:129
Global Vorpaline namespace.
void geo_argused(const T &)
Suppresses compiler warnings about unused parameters.
Definition argused.h:60
geo_signed_index_t signed_index_t
The type for storing and manipulating indices differences.
Definition numeric.h:343
geo_index_t index_t
The type for storing and manipulating indices.
Definition numeric.h:329
Types and functions for numbers manipulation.
Functions for string manipulation.
Defines the memory alignment of points in a vector.
Definition memory.h:209
static const size_t value
Alignment value in bytes.
Definition memory.h:214
Defines the same allocator for other types.
Definition memory.h:488
aligned_allocator< U > other
Definition memory.h:490