Home | History | Annotate | Download | only in unicode
      1 //  2017 and later: Unicode, Inc. and others.
      2 // License & terms of use: http://www.unicode.org/copyright.html
      3 
      4 // char16ptr.h
      5 // created: 2017feb28 Markus W. Scherer
      6 
      7 #ifndef __CHAR16PTR_H__
      8 #define __CHAR16PTR_H__
      9 
     10 #include <cstddef>
     11 #include "unicode/utypes.h"
     12 
     13 /**
     14  * \file
     15  * \brief C++ API: char16_t pointer wrappers with
     16  *        implicit conversion from bit-compatible raw pointer types.
     17  *        Also conversion functions from char16_t * to UChar * and OldUChar *.
     18  */
     19 
     20 U_NAMESPACE_BEGIN
     21 
     22 /**
     23  * \def U_ALIASING_BARRIER
     24  * Barrier for pointer anti-aliasing optimizations even across function boundaries.
     25  * @internal
     26  */
     27 #ifdef U_ALIASING_BARRIER
     28     // Use the predefined value.
     29 #elif (defined(__clang__) || defined(__GNUC__)) && U_PLATFORM != U_PF_BROWSER_NATIVE_CLIENT
     30 #   define U_ALIASING_BARRIER(ptr) asm volatile("" : : "rm"(ptr) : "memory")
     31 #elif defined(U_IN_DOXYGEN)
     32 #   define U_ALIASING_BARRIER(ptr)
     33 #endif
     34 
     35 /**
     36  * char16_t * wrapper with implicit conversion from distinct but bit-compatible pointer types.
     37  * @stable ICU 59
     38  */
     39 class U_COMMON_API Char16Ptr U_FINAL {
     40 public:
     41     /**
     42      * Copies the pointer.
     43      * @param p pointer
     44      * @stable ICU 59
     45      */
     46     inline Char16Ptr(char16_t *p);
     47 #if !U_CHAR16_IS_TYPEDEF
     48     /**
     49      * Converts the pointer to char16_t *.
     50      * @param p pointer to be converted
     51      * @stable ICU 59
     52      */
     53     inline Char16Ptr(uint16_t *p);
     54 #endif
     55 #if U_SIZEOF_WCHAR_T==2 || defined(U_IN_DOXYGEN)
     56     /**
     57      * Converts the pointer to char16_t *.
     58      * (Only defined if U_SIZEOF_WCHAR_T==2.)
     59      * @param p pointer to be converted
     60      * @stable ICU 59
     61      */
     62     inline Char16Ptr(wchar_t *p);
     63 #endif
     64     /**
     65      * nullptr constructor.
     66      * @param p nullptr
     67      * @stable ICU 59
     68      */
     69     inline Char16Ptr(std::nullptr_t p);
     70     /**
     71      * Destructor.
     72      * @stable ICU 59
     73      */
     74     inline ~Char16Ptr();
     75 
     76     /**
     77      * Pointer access.
     78      * @return the wrapped pointer
     79      * @stable ICU 59
     80      */
     81     inline char16_t *get() const;
     82     /**
     83      * char16_t pointer access via type conversion (e.g., static_cast).
     84      * @return the wrapped pointer
     85      * @stable ICU 59
     86      */
     87     inline operator char16_t *() const { return get(); }
     88 
     89 private:
     90     Char16Ptr() = delete;
     91 
     92 #ifdef U_ALIASING_BARRIER
     93     template<typename T> static char16_t *cast(T *t) {
     94         U_ALIASING_BARRIER(t);
     95         return reinterpret_cast<char16_t *>(t);
     96     }
     97 
     98     char16_t *p_;
     99 #else
    100     union {
    101         char16_t *cp;
    102         uint16_t *up;
    103         wchar_t *wp;
    104     } u_;
    105 #endif
    106 };
    107 
    108 /// \cond
    109 #ifdef U_ALIASING_BARRIER
    110 
    111 Char16Ptr::Char16Ptr(char16_t *p) : p_(p) {}
    112 #if !U_CHAR16_IS_TYPEDEF
    113 Char16Ptr::Char16Ptr(uint16_t *p) : p_(cast(p)) {}
    114 #endif
    115 #if U_SIZEOF_WCHAR_T==2
    116 Char16Ptr::Char16Ptr(wchar_t *p) : p_(cast(p)) {}
    117 #endif
    118 Char16Ptr::Char16Ptr(std::nullptr_t p) : p_(p) {}
    119 Char16Ptr::~Char16Ptr() {
    120     U_ALIASING_BARRIER(p_);
    121 }
    122 
    123 char16_t *Char16Ptr::get() const { return p_; }
    124 
    125 #else
    126 
    127 Char16Ptr::Char16Ptr(char16_t *p) { u_.cp = p; }
    128 #if !U_CHAR16_IS_TYPEDEF
    129 Char16Ptr::Char16Ptr(uint16_t *p) { u_.up = p; }
    130 #endif
    131 #if U_SIZEOF_WCHAR_T==2
    132 Char16Ptr::Char16Ptr(wchar_t *p) { u_.wp = p; }
    133 #endif
    134 Char16Ptr::Char16Ptr(std::nullptr_t p) { u_.cp = p; }
    135 Char16Ptr::~Char16Ptr() {}
    136 
    137 char16_t *Char16Ptr::get() const { return u_.cp; }
    138 
    139 #endif
    140 /// \endcond
    141 
    142 /**
    143  * const char16_t * wrapper with implicit conversion from distinct but bit-compatible pointer types.
    144  * @stable ICU 59
    145  */
    146 class U_COMMON_API ConstChar16Ptr U_FINAL {
    147 public:
    148     /**
    149      * Copies the pointer.
    150      * @param p pointer
    151      * @stable ICU 59
    152      */
    153     inline ConstChar16Ptr(const char16_t *p);
    154 #if !U_CHAR16_IS_TYPEDEF
    155     /**
    156      * Converts the pointer to char16_t *.
    157      * @param p pointer to be converted
    158      * @stable ICU 59
    159      */
    160     inline ConstChar16Ptr(const uint16_t *p);
    161 #endif
    162 #if U_SIZEOF_WCHAR_T==2 || defined(U_IN_DOXYGEN)
    163     /**
    164      * Converts the pointer to char16_t *.
    165      * (Only defined if U_SIZEOF_WCHAR_T==2.)
    166      * @param p pointer to be converted
    167      * @stable ICU 59
    168      */
    169     inline ConstChar16Ptr(const wchar_t *p);
    170 #endif
    171     /**
    172      * nullptr constructor.
    173      * @param p nullptr
    174      * @stable ICU 59
    175      */
    176     inline ConstChar16Ptr(const std::nullptr_t p);
    177 
    178     /**
    179      * Destructor.
    180      * @stable ICU 59
    181      */
    182     inline ~ConstChar16Ptr();
    183 
    184     /**
    185      * Pointer access.
    186      * @return the wrapped pointer
    187      * @stable ICU 59
    188      */
    189     inline const char16_t *get() const;
    190     /**
    191      * char16_t pointer access via type conversion (e.g., static_cast).
    192      * @return the wrapped pointer
    193      * @stable ICU 59
    194      */
    195     inline operator const char16_t *() const { return get(); }
    196 
    197 private:
    198     ConstChar16Ptr() = delete;
    199 
    200 #ifdef U_ALIASING_BARRIER
    201     template<typename T> static const char16_t *cast(const T *t) {
    202         U_ALIASING_BARRIER(t);
    203         return reinterpret_cast<const char16_t *>(t);
    204     }
    205 
    206     const char16_t *p_;
    207 #else
    208     union {
    209         const char16_t *cp;
    210         const uint16_t *up;
    211         const wchar_t *wp;
    212     } u_;
    213 #endif
    214 };
    215 
    216 /// \cond
    217 #ifdef U_ALIASING_BARRIER
    218 
    219 ConstChar16Ptr::ConstChar16Ptr(const char16_t *p) : p_(p) {}
    220 #if !U_CHAR16_IS_TYPEDEF
    221 ConstChar16Ptr::ConstChar16Ptr(const uint16_t *p) : p_(cast(p)) {}
    222 #endif
    223 #if U_SIZEOF_WCHAR_T==2
    224 ConstChar16Ptr::ConstChar16Ptr(const wchar_t *p) : p_(cast(p)) {}
    225 #endif
    226 ConstChar16Ptr::ConstChar16Ptr(const std::nullptr_t p) : p_(p) {}
    227 ConstChar16Ptr::~ConstChar16Ptr() {
    228     U_ALIASING_BARRIER(p_);
    229 }
    230 
    231 const char16_t *ConstChar16Ptr::get() const { return p_; }
    232 
    233 #else
    234 
    235 ConstChar16Ptr::ConstChar16Ptr(const char16_t *p) { u_.cp = p; }
    236 #if !U_CHAR16_IS_TYPEDEF
    237 ConstChar16Ptr::ConstChar16Ptr(const uint16_t *p) { u_.up = p; }
    238 #endif
    239 #if U_SIZEOF_WCHAR_T==2
    240 ConstChar16Ptr::ConstChar16Ptr(const wchar_t *p) { u_.wp = p; }
    241 #endif
    242 ConstChar16Ptr::ConstChar16Ptr(const std::nullptr_t p) { u_.cp = p; }
    243 ConstChar16Ptr::~ConstChar16Ptr() {}
    244 
    245 const char16_t *ConstChar16Ptr::get() const { return u_.cp; }
    246 
    247 #endif
    248 /// \endcond
    249 
    250 /**
    251  * Converts from const char16_t * to const UChar *.
    252  * Includes an aliasing barrier if available.
    253  * @param p pointer
    254  * @return p as const UChar *
    255  * @stable ICU 59
    256  */
    257 inline const UChar *toUCharPtr(const char16_t *p) {
    258 #ifdef U_ALIASING_BARRIER
    259     U_ALIASING_BARRIER(p);
    260 #endif
    261     return reinterpret_cast<const UChar *>(p);
    262 }
    263 
    264 /**
    265  * Converts from char16_t * to UChar *.
    266  * Includes an aliasing barrier if available.
    267  * @param p pointer
    268  * @return p as UChar *
    269  * @stable ICU 59
    270  */
    271 inline UChar *toUCharPtr(char16_t *p) {
    272 #ifdef U_ALIASING_BARRIER
    273     U_ALIASING_BARRIER(p);
    274 #endif
    275     return reinterpret_cast<UChar *>(p);
    276 }
    277 
    278 /**
    279  * Converts from const char16_t * to const OldUChar *.
    280  * Includes an aliasing barrier if available.
    281  * @param p pointer
    282  * @return p as const OldUChar *
    283  * @stable ICU 59
    284  */
    285 inline const OldUChar *toOldUCharPtr(const char16_t *p) {
    286 #ifdef U_ALIASING_BARRIER
    287     U_ALIASING_BARRIER(p);
    288 #endif
    289     return reinterpret_cast<const OldUChar *>(p);
    290 }
    291 
    292 /**
    293  * Converts from char16_t * to OldUChar *.
    294  * Includes an aliasing barrier if available.
    295  * @param p pointer
    296  * @return p as OldUChar *
    297  * @stable ICU 59
    298  */
    299 inline OldUChar *toOldUCharPtr(char16_t *p) {
    300 #ifdef U_ALIASING_BARRIER
    301     U_ALIASING_BARRIER(p);
    302 #endif
    303     return reinterpret_cast<OldUChar *>(p);
    304 }
    305 
    306 U_NAMESPACE_END
    307 
    308 #endif  // __CHAR16PTR_H__
    309