314 lines
7.2 KiB
C++
314 lines
7.2 KiB
C++
// © 2017 and later: Unicode, Inc. and others.
|
|
// License & terms of use: http://www.unicode.org/copyright.html
|
|
|
|
// char16ptr.h
|
|
// created: 2017feb28 Markus W. Scherer
|
|
|
|
#ifndef __CHAR16PTR_H__
|
|
#define __CHAR16PTR_H__
|
|
|
|
#include "unicode/utypes.h"
|
|
|
|
#if U_SHOW_CPLUSPLUS_API
|
|
|
|
#include <cstddef>
|
|
|
|
/**
|
|
* \file
|
|
* \brief C++ API: char16_t pointer wrappers with
|
|
* implicit conversion from bit-compatible raw pointer types.
|
|
* Also conversion functions from char16_t * to UChar * and OldUChar *.
|
|
*/
|
|
|
|
U_NAMESPACE_BEGIN
|
|
|
|
/**
|
|
* \def U_ALIASING_BARRIER
|
|
* Barrier for pointer anti-aliasing optimizations even across function boundaries.
|
|
* @internal
|
|
*/
|
|
#ifdef U_ALIASING_BARRIER
|
|
// Use the predefined value.
|
|
#elif (defined(__clang__) || defined(__GNUC__)) && U_PLATFORM != U_PF_BROWSER_NATIVE_CLIENT
|
|
# define U_ALIASING_BARRIER(ptr) asm volatile("" : : "rm"(ptr) : "memory")
|
|
#elif defined(U_IN_DOXYGEN)
|
|
# define U_ALIASING_BARRIER(ptr)
|
|
#endif
|
|
|
|
/**
|
|
* char16_t * wrapper with implicit conversion from distinct but bit-compatible pointer types.
|
|
* @stable ICU 59
|
|
*/
|
|
class U_COMMON_API Char16Ptr final {
|
|
public:
|
|
/**
|
|
* Copies the pointer.
|
|
* @param p pointer
|
|
* @stable ICU 59
|
|
*/
|
|
inline Char16Ptr(char16_t *p);
|
|
#if !U_CHAR16_IS_TYPEDEF
|
|
/**
|
|
* Converts the pointer to char16_t *.
|
|
* @param p pointer to be converted
|
|
* @stable ICU 59
|
|
*/
|
|
inline Char16Ptr(uint16_t *p);
|
|
#endif
|
|
#if U_SIZEOF_WCHAR_T==2 || defined(U_IN_DOXYGEN)
|
|
/**
|
|
* Converts the pointer to char16_t *.
|
|
* (Only defined if U_SIZEOF_WCHAR_T==2.)
|
|
* @param p pointer to be converted
|
|
* @stable ICU 59
|
|
*/
|
|
inline Char16Ptr(wchar_t *p);
|
|
#endif
|
|
/**
|
|
* nullptr constructor.
|
|
* @param p nullptr
|
|
* @stable ICU 59
|
|
*/
|
|
inline Char16Ptr(std::nullptr_t p);
|
|
/**
|
|
* Destructor.
|
|
* @stable ICU 59
|
|
*/
|
|
inline ~Char16Ptr();
|
|
|
|
/**
|
|
* Pointer access.
|
|
* @return the wrapped pointer
|
|
* @stable ICU 59
|
|
*/
|
|
inline char16_t *get() const;
|
|
/**
|
|
* char16_t pointer access via type conversion (e.g., static_cast).
|
|
* @return the wrapped pointer
|
|
* @stable ICU 59
|
|
*/
|
|
inline operator char16_t *() const { return get(); }
|
|
|
|
private:
|
|
Char16Ptr() = delete;
|
|
|
|
#ifdef U_ALIASING_BARRIER
|
|
template<typename T> static char16_t *cast(T *t) {
|
|
U_ALIASING_BARRIER(t);
|
|
return reinterpret_cast<char16_t *>(t);
|
|
}
|
|
|
|
char16_t *p_;
|
|
#else
|
|
union {
|
|
char16_t *cp;
|
|
uint16_t *up;
|
|
wchar_t *wp;
|
|
} u_;
|
|
#endif
|
|
};
|
|
|
|
/// \cond
|
|
#ifdef U_ALIASING_BARRIER
|
|
|
|
Char16Ptr::Char16Ptr(char16_t *p) : p_(p) {}
|
|
#if !U_CHAR16_IS_TYPEDEF
|
|
Char16Ptr::Char16Ptr(uint16_t *p) : p_(cast(p)) {}
|
|
#endif
|
|
#if U_SIZEOF_WCHAR_T==2
|
|
Char16Ptr::Char16Ptr(wchar_t *p) : p_(cast(p)) {}
|
|
#endif
|
|
Char16Ptr::Char16Ptr(std::nullptr_t p) : p_(p) {}
|
|
Char16Ptr::~Char16Ptr() {
|
|
U_ALIASING_BARRIER(p_);
|
|
}
|
|
|
|
char16_t *Char16Ptr::get() const { return p_; }
|
|
|
|
#else
|
|
|
|
Char16Ptr::Char16Ptr(char16_t *p) { u_.cp = p; }
|
|
#if !U_CHAR16_IS_TYPEDEF
|
|
Char16Ptr::Char16Ptr(uint16_t *p) { u_.up = p; }
|
|
#endif
|
|
#if U_SIZEOF_WCHAR_T==2
|
|
Char16Ptr::Char16Ptr(wchar_t *p) { u_.wp = p; }
|
|
#endif
|
|
Char16Ptr::Char16Ptr(std::nullptr_t p) { u_.cp = p; }
|
|
Char16Ptr::~Char16Ptr() {}
|
|
|
|
char16_t *Char16Ptr::get() const { return u_.cp; }
|
|
|
|
#endif
|
|
/// \endcond
|
|
|
|
/**
|
|
* const char16_t * wrapper with implicit conversion from distinct but bit-compatible pointer types.
|
|
* @stable ICU 59
|
|
*/
|
|
class U_COMMON_API ConstChar16Ptr final {
|
|
public:
|
|
/**
|
|
* Copies the pointer.
|
|
* @param p pointer
|
|
* @stable ICU 59
|
|
*/
|
|
inline ConstChar16Ptr(const char16_t *p);
|
|
#if !U_CHAR16_IS_TYPEDEF
|
|
/**
|
|
* Converts the pointer to char16_t *.
|
|
* @param p pointer to be converted
|
|
* @stable ICU 59
|
|
*/
|
|
inline ConstChar16Ptr(const uint16_t *p);
|
|
#endif
|
|
#if U_SIZEOF_WCHAR_T==2 || defined(U_IN_DOXYGEN)
|
|
/**
|
|
* Converts the pointer to char16_t *.
|
|
* (Only defined if U_SIZEOF_WCHAR_T==2.)
|
|
* @param p pointer to be converted
|
|
* @stable ICU 59
|
|
*/
|
|
inline ConstChar16Ptr(const wchar_t *p);
|
|
#endif
|
|
/**
|
|
* nullptr constructor.
|
|
* @param p nullptr
|
|
* @stable ICU 59
|
|
*/
|
|
inline ConstChar16Ptr(const std::nullptr_t p);
|
|
|
|
/**
|
|
* Destructor.
|
|
* @stable ICU 59
|
|
*/
|
|
inline ~ConstChar16Ptr();
|
|
|
|
/**
|
|
* Pointer access.
|
|
* @return the wrapped pointer
|
|
* @stable ICU 59
|
|
*/
|
|
inline const char16_t *get() const;
|
|
/**
|
|
* char16_t pointer access via type conversion (e.g., static_cast).
|
|
* @return the wrapped pointer
|
|
* @stable ICU 59
|
|
*/
|
|
inline operator const char16_t *() const { return get(); }
|
|
|
|
private:
|
|
ConstChar16Ptr() = delete;
|
|
|
|
#ifdef U_ALIASING_BARRIER
|
|
template<typename T> static const char16_t *cast(const T *t) {
|
|
U_ALIASING_BARRIER(t);
|
|
return reinterpret_cast<const char16_t *>(t);
|
|
}
|
|
|
|
const char16_t *p_;
|
|
#else
|
|
union {
|
|
const char16_t *cp;
|
|
const uint16_t *up;
|
|
const wchar_t *wp;
|
|
} u_;
|
|
#endif
|
|
};
|
|
|
|
/// \cond
|
|
#ifdef U_ALIASING_BARRIER
|
|
|
|
ConstChar16Ptr::ConstChar16Ptr(const char16_t *p) : p_(p) {}
|
|
#if !U_CHAR16_IS_TYPEDEF
|
|
ConstChar16Ptr::ConstChar16Ptr(const uint16_t *p) : p_(cast(p)) {}
|
|
#endif
|
|
#if U_SIZEOF_WCHAR_T==2
|
|
ConstChar16Ptr::ConstChar16Ptr(const wchar_t *p) : p_(cast(p)) {}
|
|
#endif
|
|
ConstChar16Ptr::ConstChar16Ptr(const std::nullptr_t p) : p_(p) {}
|
|
ConstChar16Ptr::~ConstChar16Ptr() {
|
|
U_ALIASING_BARRIER(p_);
|
|
}
|
|
|
|
const char16_t *ConstChar16Ptr::get() const { return p_; }
|
|
|
|
#else
|
|
|
|
ConstChar16Ptr::ConstChar16Ptr(const char16_t *p) { u_.cp = p; }
|
|
#if !U_CHAR16_IS_TYPEDEF
|
|
ConstChar16Ptr::ConstChar16Ptr(const uint16_t *p) { u_.up = p; }
|
|
#endif
|
|
#if U_SIZEOF_WCHAR_T==2
|
|
ConstChar16Ptr::ConstChar16Ptr(const wchar_t *p) { u_.wp = p; }
|
|
#endif
|
|
ConstChar16Ptr::ConstChar16Ptr(const std::nullptr_t p) { u_.cp = p; }
|
|
ConstChar16Ptr::~ConstChar16Ptr() {}
|
|
|
|
const char16_t *ConstChar16Ptr::get() const { return u_.cp; }
|
|
|
|
#endif
|
|
/// \endcond
|
|
|
|
/**
|
|
* Converts from const char16_t * to const UChar *.
|
|
* Includes an aliasing barrier if available.
|
|
* @param p pointer
|
|
* @return p as const UChar *
|
|
* @stable ICU 59
|
|
*/
|
|
inline const UChar *toUCharPtr(const char16_t *p) {
|
|
#ifdef U_ALIASING_BARRIER
|
|
U_ALIASING_BARRIER(p);
|
|
#endif
|
|
return reinterpret_cast<const UChar *>(p);
|
|
}
|
|
|
|
/**
|
|
* Converts from char16_t * to UChar *.
|
|
* Includes an aliasing barrier if available.
|
|
* @param p pointer
|
|
* @return p as UChar *
|
|
* @stable ICU 59
|
|
*/
|
|
inline UChar *toUCharPtr(char16_t *p) {
|
|
#ifdef U_ALIASING_BARRIER
|
|
U_ALIASING_BARRIER(p);
|
|
#endif
|
|
return reinterpret_cast<UChar *>(p);
|
|
}
|
|
|
|
/**
|
|
* Converts from const char16_t * to const OldUChar *.
|
|
* Includes an aliasing barrier if available.
|
|
* @param p pointer
|
|
* @return p as const OldUChar *
|
|
* @stable ICU 59
|
|
*/
|
|
inline const OldUChar *toOldUCharPtr(const char16_t *p) {
|
|
#ifdef U_ALIASING_BARRIER
|
|
U_ALIASING_BARRIER(p);
|
|
#endif
|
|
return reinterpret_cast<const OldUChar *>(p);
|
|
}
|
|
|
|
/**
|
|
* Converts from char16_t * to OldUChar *.
|
|
* Includes an aliasing barrier if available.
|
|
* @param p pointer
|
|
* @return p as OldUChar *
|
|
* @stable ICU 59
|
|
*/
|
|
inline OldUChar *toOldUCharPtr(char16_t *p) {
|
|
#ifdef U_ALIASING_BARRIER
|
|
U_ALIASING_BARRIER(p);
|
|
#endif
|
|
return reinterpret_cast<OldUChar *>(p);
|
|
}
|
|
|
|
U_NAMESPACE_END
|
|
|
|
#endif /* U_SHOW_CPLUSPLUS_API */
|
|
|
|
#endif // __CHAR16PTR_H__
|