00001
00002
00003
00004
00005
00006
00007 #ifndef COH_COMPATIBILITY_HPP
00008 #define COH_COMPATIBILITY_HPP
00009
00010
00011
00012 #include <typeinfo>
00013
00014
00015
00016 #if defined(_WIN32)
00017 # define COH_OS_WINDOWS
00018 # define COH_LIB_PREFIX ""
00019 # define COH_LIB_SUFFIX ".dll"
00020 # if defined(_WIN64)
00021 # define COH_OS_WIN64
00022 # define COH_PLATFORM Microsoft Windows x64
00023 # else
00024 # define COH_OS_WIN32
00025 # define COH_PLATFORM Microsoft Windows x86
00026 # endif
00027 #elif defined(__sun) || defined(sun)
00028 # define COH_OS_SOLARIS // Oracle Solaris
00029 # define COH_OS_UNIX
00030 # define COH_LIB_PREFIX "lib"
00031 # define COH_LIB_SUFFIX ".so"
00032 # include <sys/types.h>
00033 # if defined(__sparc)
00034 # if defined(_LP64)
00035 # define COH_OS_SOLARIS64
00036 # define COH_PLATFORM Oracle Solaris SPARC 64b
00037 # else
00038 # define COH_OS_SOLARIS32
00039 # define COH_PLATFORM Oracle Solaris SPARC 32b
00040 # endif
00041 # elif defined(__x86)
00042 # if defined(_LP64)
00043 # define COH_OS_SOLARIS64
00044 # define COH_PLATFORM Oracle Solaris x64
00045 # else
00046 # define COH_OS_SOLARIS32
00047 # define COH_PLATFORM Oracle Solaris x86
00048 # endif
00049 # endif
00050 #elif defined(__linux__)
00051 # define COH_OS_LINUX // Linux
00052 # define COH_LIB_PREFIX "lib"
00053 # define COH_LIB_SUFFIX ".so"
00054 # define COH_OS_UNIX
00055 # if defined(__x86_64__) || defined(__amd64__)
00056 # define COH_OS_LINUX64
00057 # define COH_PLATFORM Linux x64
00058 # elif defined(__x86_32__) || defined(__i386__)
00059 # define COH_OS_LINUX32
00060 # define COH_PLATFORM Linux x86
00061 # elif defined (__aarch64__)
00062 # define COH_OS_LINUXAARCH64
00063 # define COH_PLATFORM Linux aarch64
00064 # endif
00065 #elif defined(__APPLE__)
00066 # define COH_OS_DARWIN // OS X
00067 # define COH_OS_UNIX
00068 # define COH_LIB_PREFIX "lib"
00069 # define COH_LIB_SUFFIX ".dylib"
00070 # if defined(__x86_64__) || defined(__amd64__)
00071 # define COH_OS_DARWIN64
00072 # define COH_PLATFORM Apple macOS x64
00073 # elif defined (__aarch64__)
00074 # define COH_OS_DARWINAARCH64
00075 # define COH_PLATFORM Apple macOS arm64
00076 # endif
00077 #endif
00078
00079 #ifndef COH_PLATFORM
00080 # error "Coherence for C++ does not support this platform."
00081 #endif
00082
00083
00084
00085
00086 #if defined(_MSC_VER) && _MSC_VER >= 1600
00087 # define COH_CC_MSVC // Microsoft Visual C/C++
00088 # define COH_CC MSVC
00089 #elif defined(__SUNPRO_CC) && __SUNPRO_CC >= 0x590
00090 # define COH_CC_SUN // Oracle Solaris Studio
00091 # if defined (STLPORT)
00092 # define COH_CC SunPro STLport
00093 # else
00094 # define COH_CC SunPro
00095 # endif
00096 #elif defined(__GNUG__) && (__GNUC__ >= 4 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 4))
00097 # define COH_CC_GNU // GNU C++
00098 # if defined (__clang__)
00099 # define COH_CC LLVM (clang)
00100 # else
00101 # define COH_CC GCC
00102 # endif
00103 #else
00104 # error "Coherence for C++ does not support this compiler or compiler version."
00105 #endif
00106
00107
00108
00109
00110
00111
00112
00113
00114
00115
00116 #if defined(COH_CC_MSVC)
00117 #define COH_PRAGMA_PUSH \
00118 __pragma(warning(push)) \
00119 \
00120 __pragma(warning(disable : 4250)) \
00121 \
00122 __pragma(warning(disable : 4251)) \
00123 \
00124 __pragma(warning(disable : 4275)) \
00125 \
00126 __pragma(warning(disable : 4521)) \
00127 \
00128 __pragma(warning(disable : 4522)) \
00129
00130 \
00131 __pragma(warning(disable : 4715; disable : 4716))
00132
00133 #define COH_PRAGMA_POP __pragma(warning(pop))
00134 #else
00135 #define COH_PRAGMA_PUSH
00136 #define COH_PRAGMA_POP
00137 #endif
00138
00139
00140
00141
00142
00143
00144 #define COH_NO_WARN(STMT) COH_PRAGMA_PUSH STMT COH_PRAGMA_POP
00145
00146
00147
00148
00149
00150
00151
00152
00153
00154
00155
00156
00157 #if defined(COH_CC_MSVC)
00158 #define COH_NO_RETURN_PRE __declspec(noreturn)
00159 #define COH_NO_RETURN_POST
00160 #define COH_NO_RETURN_STMT(expr) expr
00161 #elif defined(COH_CC_GNU)
00162 #define COH_NO_RETURN_PRE
00163 #define COH_NO_RETURN_POST __attribute__((noreturn))
00164 #define COH_NO_RETURN_STMT(expr) expr
00165 #elif defined(COH_CC_SUN)
00166 #define COH_NO_RETURN_PRE
00167 #define COH_NO_RETURN_POST
00168 #define COH_NO_RETURN_STMT(expr)\
00169 do { expr; throw std::exception(); } while (0)
00170 #else
00171 #define COH_NO_RETURN_PRE
00172 #define COH_NO_RETURN_POST
00173 #define COH_NO_RETURN_STMT(expr) expr
00174 #endif
00175
00176
00177
00178
00179
00180
00181
00182 namespace coherence { namespace lang {}}
00183
00184 #define COH_OPEN_NAMESPACE(ns) namespace ns { \
00185 COH_PRAGMA_PUSH \
00186 using namespace coherence::lang;
00187
00188 #define COH_INNER_NAMESPACE(ns) namespace ns {
00189
00190 #define COH_OPEN_NAMESPACE2(ns1, ns2)\
00191 COH_OPEN_NAMESPACE (ns1) COH_INNER_NAMESPACE (ns2)
00192
00193 #define COH_OPEN_NAMESPACE3(ns1, ns2, ns3)\
00194 COH_OPEN_NAMESPACE2 (ns1, ns2) COH_INNER_NAMESPACE (ns3)
00195
00196 #define COH_OPEN_NAMESPACE4(ns1, ns2, ns3, ns4)\
00197 COH_OPEN_NAMESPACE3 (ns1, ns2, ns3) COH_INNER_NAMESPACE (ns4)
00198
00199 #define COH_OPEN_NAMESPACE5(ns1, ns2, ns3, ns4, ns5)\
00200 COH_OPEN_NAMESPACE4 (ns1, ns2, ns3, ns4) COH_INNER_NAMESPACE (ns5)
00201
00202 #define COH_OPEN_NAMESPACE6(ns1, ns2, ns3, ns4, ns5, ns6)\
00203 COH_OPEN_NAMESPACE5 (ns1, ns2, ns3, ns4, ns5) COH_INNER_NAMESPACE (ns6)
00204
00205 #define COH_OPEN_NAMESPACE7(ns1, ns2, ns3, ns4, ns5, ns6, ns7)\
00206 COH_OPEN_NAMESPACE6 (ns1, ns2, ns3, ns4, ns5, ns6) COH_INNER_NAMESPACE (ns7)
00207
00208 #define COH_CLOSE_NAMESPACE COH_PRAGMA_POP }
00209
00210 #define COH_CLOSE_NAMESPACE2 COH_PRAGMA_POP } }
00211
00212 #define COH_CLOSE_NAMESPACE3 COH_PRAGMA_POP } } }
00213
00214 #define COH_CLOSE_NAMESPACE4 COH_PRAGMA_POP } } } }
00215
00216 #define COH_CLOSE_NAMESPACE5 COH_PRAGMA_POP } } } } }
00217
00218 #define COH_CLOSE_NAMESPACE6 COH_PRAGMA_POP } } } } } }
00219
00220 #define COH_CLOSE_NAMESPACE7 COH_PRAGMA_POP } } } } } } }
00221
00222
00223
00224
00225
00226
00227
00228
00229
00230
00231
00232
00233
00234 #if defined(COH_CC_MSVC) && _MSC_VER >= 1900
00235 #define COH_OPEN_NAMESPACE_ANON(NAME)\
00236 namespace COH_JOIN(_anon_, NAME) {}\
00237 using namespace COH_JOIN(_anon_, NAME);\
00238 namespace COH_JOIN(_anon_, NAME) {
00239 #else
00240 #define COH_OPEN_NAMESPACE_ANON(NAME) namespace {
00241 #endif
00242
00243 #define COH_CLOSE_NAMESPACE_ANON }
00244
00245
00246
00247
00248
00249
00250
00251
00252
00253
00254
00255
00256 #define COH_JOIN(X, Y) COH_DO_JOIN(X, Y)
00257 #define COH_DO_JOIN(X, Y) COH_DO_JOIN2(X, Y)
00258 #define COH_DO_JOIN2(X, Y) X##Y
00259 #ifdef COH_CC_MSVC_NET // MSVC 2002 and later
00260 # define COH_UNIQUE_IDENTIFIER(Name) COH_JOIN(Name, __COUNTER__)
00261 #else
00262 # define COH_UNIQUE_IDENTIFIER(Name) COH_JOIN(Name, __LINE__)
00263 #endif
00264
00265
00266
00267
00268
00269
00270
00271
00272
00273
00274
00275
00276
00277
00278
00279
00280 #ifdef COH_EAGER_INIT
00281 # define COH_STATIC_INIT_EX(N, FUNC) \
00282 static const bool COH_UNIQUE_IDENTIFIER(coh_static_init_func##N##_) = (FUNC, true)
00283 #else
00284
00285
00286
00287
00288
00289 # define COH_STATIC_INIT_EX(N, FUNC) \
00290 static void COH_UNIQUE_IDENTIFIER(coh_static_init_func##N##_)() {FUNC;} \
00291 static coherence::lang::coh_initializer \
00292 COH_UNIQUE_IDENTIFIER(coh_static_init_reg##N##_) \
00293 (&COH_UNIQUE_IDENTIFIER(coh_static_init_func##N##_))
00294 #endif
00295
00296 #define COH_STATIC_INIT(FUNC) COH_STATIC_INIT_EX(0, FUNC)
00297
00298 COH_OPEN_NAMESPACE2(coherence,lang)
00299 template <bool x> struct STATIC_ASSERTION_FAILURE;
00300 template<> struct STATIC_ASSERTION_FAILURE<true> { enum { value = 1 }; };
00301 COH_CLOSE_NAMESPACE2
00302
00303
00304
00305
00306
00307
00308
00309
00310
00311
00312
00313
00314
00315
00316
00317
00318
00319 #define COH_STATIC_ASSERT(B) \
00320 enum { COH_UNIQUE_IDENTIFIER(coh_static_assert_enum_) =\
00321 sizeof(coherence::lang::STATIC_ASSERTION_FAILURE<(bool)(B)>) }
00322
00323
00324
00325
00326 #if defined(COH_CC_MSVC)
00327 #ifdef COH_BUILD
00328 #define COH_EXPORT __declspec(dllexport)
00329 #else
00330 #define COH_EXPORT __declspec(dllimport)
00331 #endif
00332 #define COH_EXPORT_SPEC __declspec(dllexport)
00333 #if _MSC_VER >= 1900
00334 #define COH_EXPORT_SPEC_MEMBER(DECL) DECL;
00335 #else
00336 #define COH_EXPORT_SPEC_MEMBER(DECL)
00337 #endif
00338 #else
00339 #define COH_EXPORT
00340 #define COH_EXPORT_SPEC
00341 #define COH_EXPORT_SPEC_MEMBER(DECL) DECL;
00342 #endif
00343
00344
00345
00346
00347 #if defined(COH_CC_MSVC)
00348 #define COH_INLINE __forceinline
00349 #elif defined(COH_CC_GNU)
00350 #define COH_INLINE __attribute__((always_inline)) inline
00351 #else
00352 #define COH_INLINE __attribute__((always_inline)) inline
00353 #endif
00354
00355
00356
00357
00358 #if defined(__GNUC__)
00359 #define COH_CURRENT_FUNCTION __PRETTY_FUNCTION__
00360 #elif defined(__FUNCSIG__)
00361 #define COH_CURRENT_FUNCTION __FUNCSIG__
00362 #elif defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901)
00363 #define COH_CURRENT_FUNCTION __func__
00364 #else
00365 #define COH_CURRENT_FUNCTION "(unknown function)"
00366 #endif
00367
00368
00369
00370
00371 #if defined(COH_CC_GNU)
00372 #define COH_ALIGN(B, TYPE, NAME) TYPE NAME __attribute__ ((aligned (B)))
00373 #elif defined(COH_CC_SUN)
00374 #define COH_SUNPRO_PRAGMA(S) _Pragma(S)
00375 #define COH_ALIGN(B, TYPE, NAME) COH_SUNPRO_PRAGMA(COH_SYMB_STRING(align B(NAME))) TYPE NAME
00376 #elif defined(COH_CC_MSVC)
00377 #define COH_ALIGN(B, TYPE, NAME) __declspec(align(B)) TYPE NAME
00378 #else
00379 #error "No Coherence alignment macro for this compiler"
00380 #endif
00381
00382
00383
00384
00385 #if __cplusplus > 199711L ||\
00386 defined(_MSC_VER) && _MSC_VER >= 1900
00387 # define COH_NOEXCEPT(expr) noexcept(expr)
00388 #else
00389 # define COH_NOEXCEPT(expr)
00390 #endif
00391
00392
00393
00394
00395
00396
00397 #if defined(__GXX_EXPERIMENTAL_CXX0X__) || __cplusplus >= 201103L
00398 # define COH_AUTO_PTR std::unique_ptr
00399 #else
00400 # define COH_AUTO_PTR std::auto_ptr
00401 #endif
00402
00403
00404
00405
00406
00407
00408 #include <utility>
00409 #if defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901 ||\
00410 defined(_POSIX_VERSION) && _POSIX_VERSION >= 200100 ||\
00411 defined(COH_OS_LINUX) &&\
00412 defined(__GLIBC__) &&\
00413 (__GLIBC__ > 2 || __GLIBC__ == 2 && __GLIBC_MINOR__ >= 1) &&\
00414 defined(__GNUC__) ||\
00415 defined(COH_OS_DARWIN) && __MACH__ && !defined(_MSL_USING_MSL_C)
00416 # define COH_HAS_STDINT_H
00417 #endif
00418
00419 #if defined(__GCCXML__) ||\
00420 defined(__GNUC__) ||\
00421 defined(_MSC_EXTENSIONS)
00422 # define COH_HAS_LONG_LONG
00423 #endif
00424
00425 #if (defined(__GLIBCPP__) || defined(__GLIBCXX__)) &&\
00426 !defined(_GLIBCPP_USE_LONG_LONG) && \
00427 !defined(_GLIBCXX_USE_LONG_LONG) && \
00428 defined(COH_HAS_LONG_LONG)
00429
00430
00431
00432
00433 # include <ostream>
00434 namespace std
00435 {
00436 template<class E, class T>
00437 inline basic_ostream<E, T>& operator<<
00438 (basic_ostream<E, T>& out, long long l)
00439 {
00440 return out << static_cast<long>(l);
00441 }
00442 template<class E, class T>
00443 inline basic_ostream<E, T>& operator<<
00444 (basic_ostream<E, T>& out, unsigned long long l)
00445 {
00446 return out << static_cast<unsigned long>(l);
00447 }
00448 }
00449 # undef COH_HAS_LONG_LONG
00450 #endif
00451
00452 #if !defined(COH_HAS_LONG_LONG) && !defined(COH_CC_MSVC)
00453 # include <limits.h>
00454 # if defined(ULLONG_MAX) || defined(ULONG_LONG_MAX) ||\
00455 defined(ULONGLONG_MAX)
00456 # define COH_HAS_LONG_LONG
00457 # endif
00458 #endif
00459
00460 #if defined(_MSC_VER)
00461 # define COH_HAS_MS_INT64
00462 #endif
00463
00464
00465
00466
00467 #if defined(COH_HAS_STDINT_H)
00468 # include <stdint.h>
00469 #elif defined(COH_CC_SUN)
00470 # include <inttypes.h>
00471 #else
00472
00473
00474
00475
00476 # include <climits>
00477 # ifdef COH_NAMESPACED_FIXED_INTS
00478 COH_OPEN_NAMESPACE2(coherence,lang)
00479 # endif
00480 COH_STATIC_ASSERT(UCHAR_MAX == 0xFF);
00481 typedef signed char int8_t;
00482 typedef unsigned char uint8_t;
00483 COH_STATIC_ASSERT(USHRT_MAX == 0xFFFF);
00484 typedef short int16_t;
00485 typedef unsigned short uint16_t;
00486 # if UINT_MAX == 0xFFFFFFFF
00487 typedef int int32_t;
00488 typedef unsigned int uint32_t;
00489 # elif ULONG_MAX == 0xFFFFFFFF
00490 typedef long int32_t;
00491 typedef unsigned long uint32_t;
00492 # else
00493 # error int size not correct
00494 # endif
00495 # if defined(COH_HAS_LONG_LONG) &&\
00496 !defined(COH_CC_MSVC) &&\
00497 (!defined(__GLIBCPP__) || defined(_GLIBCPP_USE_LONG_LONG)) && \
00498 (defined(ULLONG_MAX) ||\
00499 defined(ULONG_LONG_MAX) ||\
00500 defined(ULONGLONG_MAX))
00501 # if defined(ULLONG_MAX)
00502 COH_STATIC_ASSERT(ULLONG_MAX == 0xFFFFFFFFFFFFFFFFULL);
00503 # elif defined(ULONG_LONG_MAX)
00504 COH_STATIC_ASSERT
00505 (ULONG_LONG_MAX == 0xFFFFFFFFFFFFFFFFULL);
00506 # elif defined(ULONGLONG_MAX)
00507 COH_STATIC_ASSERT
00508 (ULONGLONG_MAX == 0xFFFFFFFFFFFFFFFFULL));
00509 # else
00510 # error long long size not correct
00511 # endif
00512 typedef long long int64_t;
00513 typedef unsigned long long uint64_t;
00514 # elif ULONG_MAX != 0xFFFFFFFF
00515 COH_STATIC_ASSERT(ULONG_MAX == 0xFFFFFFFFFFFFFFFFULL);
00516 typedef long int64_t;
00517 typedef unsigned long uint64_t;
00518 # elif defined(__GNUC__) && defined(COH_HAS_LONG_LONG)
00519 __extension__ typedef long long int64_t;
00520 __extension__ typedef unsigned long long uint64_t;
00521 # elif defined(COH_HAS_MS_INT64)
00522 typedef __int64 int64_t;
00523 typedef unsigned __int64 uint64_t;
00524 # else
00525 # error no 64-bit integer support
00526 # endif
00527 # ifdef COH_NAMESPACED_FIXED_INTS
00528 COH_CLOSE_NAMESPACE2
00529 # endif
00530 #endif
00531
00532
00533
00534
00535 COH_OPEN_NAMESPACE2(coherence,lang)
00536 typedef unsigned char octet_t;
00537 typedef uint16_t wchar16_t;
00538 typedef uint32_t size32_t;
00539 typedef uint64_t size64_t;
00540 typedef float float32_t; COH_STATIC_ASSERT(sizeof(float32_t) >= sizeof(int32_t));
00541 typedef double float64_t; COH_STATIC_ASSERT(sizeof(float64_t) >= sizeof(int64_t));
00542 COH_CLOSE_NAMESPACE2
00543
00544
00545
00546
00547
00548
00549
00550
00551 #define COH_INT64(HIGH, LOW) int64_t(uint64_t(HIGH) << 32 | uint64_t(LOW))
00552
00553
00554 #define COH_SYMB_STRING(SYMB) #SYMB
00555 #define COH_SYMB_TO_STRING(SYMB) COH_SYMB_STRING(SYMB)
00556
00557
00558
00559
00560 #include <typeinfo>
00561 COH_OPEN_NAMESPACE2(coherence,lang)
00562 class Class;
00563
00564 typedef void (*coh_static_initializer)();
00565
00566
00567
00568
00569 extern COH_EXPORT void coh_register_initializer(
00570 coh_static_initializer pInit, bool fAdd);
00571
00572
00573
00574
00575 class coh_initializer
00576 {
00577 public:
00578
00579 coh_initializer(coh_static_initializer pInit)
00580 : m_pInit(pInit)
00581 {
00582 coh_register_initializer(pInit, true);
00583 }
00584
00585 ~coh_initializer()
00586 {
00587 coh_register_initializer(m_pInit, false);
00588 }
00589
00590 private:
00591 coh_static_initializer m_pInit;
00592 };
00593
00594
00595
00596
00597 extern COH_EXPORT void coh_throw_npe(const std::type_info&);
00598 extern COH_EXPORT void coh_throw_class_cast(const std::type_info&,
00599 const std::type_info&);
00600 extern COH_EXPORT void coh_throw_const_cast(const std::type_info&,
00601 const std::type_info&);
00602 extern COH_EXPORT void coh_throw_illegal_state(const char* achMsg);
00603 extern COH_EXPORT void coh_throw_illegal_argument(const char* achMsg);
00604 extern COH_EXPORT void coh_throw_unsupported_operation(const char* achMsg);
00605 extern COH_EXPORT const Class* coh_loadClassByType(const std::type_info& ti);
00606
00607
00608
00609
00610
00611
00612
00613
00614 template<class A, class B>
00615 class assignment
00616 {
00617 protected:
00618 typedef char PathA;
00619 class Other {char unused[2];};
00620 static PathA route(A*);
00621 static Other route(...);
00622 static B* test();
00623
00624 public:
00625
00626
00627
00628 static A* safe(A* a)
00629 {
00630 return a;
00631 }
00632
00633
00634
00635
00636 static A* safe(...)
00637 {
00638 coh_throw_illegal_state("unsafe cast");
00639 return 0;
00640 }
00641
00642 public:
00643
00644
00645
00646 enum {allowed = (sizeof(route(test())) == sizeof(PathA))};
00647 };
00648
00649
00650
00651
00652 template<class A>
00653 class constness
00654 {
00655 public:
00656 enum {applied = assignment<A, const A>::allowed};
00657 };
00658
00659
00660
00661 #if defined(COH_CC_MSVC)
00662
00663
00664
00665
00666
00667
00668
00669
00670 typedef const std::type_info& coh_class_id;
00671
00672
00673
00674
00675 #define COH_CLASS_ID(CLASS) CLASS::_classId()
00676
00677
00678
00679
00680
00681
00682 #define COH_GENERATE_CLASS_ID(T) \
00683 static coh_class_id _classId() \
00684 { \
00685 return typeid(T); \
00686 }
00687 #else
00688
00689
00690
00691
00692
00693
00694
00695 typedef void (*coh_class_id)();
00696
00697
00698
00699
00700 #define COH_CLASS_ID(CLASS) &CLASS::_classId
00701
00702
00703
00704
00705
00706
00707 #define COH_GENERATE_CLASS_ID(T) static void _classId() {}
00708 #endif
00709
00710 COH_CLOSE_NAMESPACE2
00711
00712
00713
00714 #endif // COH_COMPATIBILITY_HPP