|
|
|
/*
|
|
|
|
* Copyright (c) 2009-2021, Google LLC
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
* * Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* * Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* * Neither the name of Google LLC nor the
|
|
|
|
* names of its contributors may be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
* DISCLAIMED. IN NO EVENT SHALL Google LLC BE LIABLE FOR ANY
|
|
|
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is where we define internal portability macros used across upb.
|
|
|
|
*
|
|
|
|
* All of these macros are undef'd in undef.inc to avoid leaking them to users.
|
|
|
|
*
|
|
|
|
* The correct usage is:
|
|
|
|
*
|
|
|
|
* #include "upb/foobar.h"
|
|
|
|
* #include "upb/baz.h"
|
|
|
|
*
|
|
|
|
* // MUST be last included header.
|
|
|
|
* #include "upb/port/def.inc"
|
|
|
|
*
|
|
|
|
* // Code for this file.
|
|
|
|
* // <...>
|
|
|
|
*
|
|
|
|
* // Can be omitted for .c files, required for .h.
|
|
|
|
* #include "upb/port/undef.inc"
|
|
|
|
*
|
|
|
|
* This file is private and must not be included by users!
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if !((defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901L) || \
|
|
|
|
(defined(__cplusplus) && __cplusplus >= 201402L) || \
|
|
|
|
(defined(_MSC_VER) && _MSC_VER >= 1900))
|
|
|
|
#error upb requires C99 or C++14 or MSVC >= 2015.
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Portable check for GCC minimum version:
|
|
|
|
// https://gcc.gnu.org/onlinedocs/cpp/Common-Predefined-Macros.html
|
|
|
|
#if defined(__GNUC__) && defined(__GNUC_MINOR__) && defined(__GNUC_PATCHLEVEL__)
|
|
|
|
#define UPB_GNUC_MIN(x, y) \
|
|
|
|
(__GNUC__ > (x) || __GNUC__ == (x) && __GNUC_MINOR__ >= (y))
|
|
|
|
#else
|
|
|
|
#define UPB_GNUC_MIN(x, y) 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <setjmp.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#if UINTPTR_MAX == 0xffffffff
|
|
|
|
#define UPB_SIZE(size32, size64) size32
|
|
|
|
#else
|
|
|
|
#define UPB_SIZE(size32, size64) size64
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* If we always read/write as a consistent type to each address, this shouldn't
|
|
|
|
* violate aliasing.
|
|
|
|
*/
|
|
|
|
#define UPB_PTR_AT(msg, ofs, type) ((type*)((char*)(msg) + (ofs)))
|
|
|
|
|
|
|
|
#define UPB_MAPTYPE_STRING 0
|
|
|
|
|
|
|
|
// UPB_EXPORT: always generate a public symbol.
|
|
|
|
#if defined(__GNUC__) || defined(__clang__)
|
|
|
|
#define UPB_EXPORT __attribute__((visibility("default"))) __attribute__((used))
|
|
|
|
#else
|
|
|
|
#define UPB_EXPORT
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// UPB_INLINE: inline if possible, emit standalone code if required.
|
|
|
|
#ifdef __cplusplus
|
|
|
|
#define UPB_INLINE inline
|
|
|
|
#elif defined (__GNUC__) || defined(__clang__)
|
|
|
|
#define UPB_INLINE static __inline__
|
|
|
|
#else
|
|
|
|
#define UPB_INLINE static
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef UPB_BUILD_API
|
|
|
|
#define UPB_API UPB_EXPORT
|
|
|
|
#define UPB_API_INLINE UPB_EXPORT
|
|
|
|
#else
|
|
|
|
#define UPB_API
|
|
|
|
#define UPB_API_INLINE UPB_INLINE
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define UPB_MALLOC_ALIGN 8
|
|
|
|
#define UPB_ALIGN_UP(size, align) (((size) + (align) - 1) / (align) * (align))
|
|
|
|
#define UPB_ALIGN_DOWN(size, align) ((size) / (align) * (align))
|
|
|
|
#define UPB_ALIGN_MALLOC(size) UPB_ALIGN_UP(size, UPB_MALLOC_ALIGN)
|
|
|
|
#define UPB_ALIGN_OF(type) offsetof (struct { char c; type member; }, member)
|
|
|
|
|
|
|
|
// Hints to the compiler about likely/unlikely branches.
|
|
|
|
#if defined (__GNUC__) || defined(__clang__)
|
|
|
|
#define UPB_LIKELY(x) __builtin_expect((bool)(x), 1)
|
|
|
|
#define UPB_UNLIKELY(x) __builtin_expect((bool)(x), 0)
|
|
|
|
#else
|
|
|
|
#define UPB_LIKELY(x) (x)
|
|
|
|
#define UPB_UNLIKELY(x) (x)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Macros for function attributes on compilers that support them.
|
|
|
|
#ifdef __GNUC__
|
|
|
|
#define UPB_FORCEINLINE __inline__ __attribute__((always_inline))
|
|
|
|
#define UPB_NOINLINE __attribute__((noinline))
|
|
|
|
#define UPB_NORETURN __attribute__((__noreturn__))
|
|
|
|
#define UPB_PRINTF(str, first_vararg) __attribute__((format (printf, str, first_vararg)))
|
|
|
|
#elif defined(_MSC_VER)
|
|
|
|
#define UPB_NOINLINE
|
|
|
|
#define UPB_FORCEINLINE
|
|
|
|
#define UPB_NORETURN __declspec(noreturn)
|
|
|
|
#define UPB_PRINTF(str, first_vararg)
|
|
|
|
#else /* !defined(__GNUC__) */
|
|
|
|
#define UPB_FORCEINLINE
|
|
|
|
#define UPB_NOINLINE
|
|
|
|
#define UPB_NORETURN
|
|
|
|
#define UPB_PRINTF(str, first_vararg)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define UPB_MAX(x, y) ((x) > (y) ? (x) : (y))
|
|
|
|
#define UPB_MIN(x, y) ((x) < (y) ? (x) : (y))
|
|
|
|
|
|
|
|
#define UPB_UNUSED(var) (void)var
|
|
|
|
|
|
|
|
// UPB_ASSUME(): in release mode, we tell the compiler to assume this is true.
|
|
|
|
#ifdef NDEBUG
|
|
|
|
#ifdef __GNUC__
|
|
|
|
#define UPB_ASSUME(expr) if (!(expr)) __builtin_unreachable()
|
|
|
|
#elif defined _MSC_VER
|
|
|
|
#define UPB_ASSUME(expr) if (!(expr)) __assume(0)
|
|
|
|
#else
|
|
|
|
#define UPB_ASSUME(expr) do {} while (false && (expr))
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
#define UPB_ASSUME(expr) assert(expr)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* UPB_ASSERT(): in release mode, we use the expression without letting it be
|
|
|
|
* evaluated. This prevents "unused variable" warnings. */
|
|
|
|
#ifdef NDEBUG
|
|
|
|
#define UPB_ASSERT(expr) do {} while (false && (expr))
|
|
|
|
#else
|
|
|
|
#define UPB_ASSERT(expr) assert(expr)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(__GNUC__) || defined(__clang__)
|
|
|
|
#define UPB_UNREACHABLE() do { assert(0); __builtin_unreachable(); } while(0)
|
|
|
|
#elif defined(_MSC_VER)
|
|
|
|
#define UPB_UNREACHABLE() \
|
|
|
|
do { \
|
|
|
|
assert(0); \
|
|
|
|
__assume(0); \
|
|
|
|
} while (0)
|
|
|
|
#else
|
|
|
|
#define UPB_UNREACHABLE() do { assert(0); } while(0)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* UPB_SETJMP() / UPB_LONGJMP(): avoid setting/restoring signal mask. */
|
|
|
|
#ifdef __APPLE__
|
|
|
|
#define UPB_SETJMP(buf) _setjmp(buf)
|
|
|
|
#define UPB_LONGJMP(buf, val) _longjmp(buf, val)
|
|
|
|
#else
|
|
|
|
#define UPB_SETJMP(buf) setjmp(buf)
|
|
|
|
#define UPB_LONGJMP(buf, val) longjmp(buf, val)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* UPB_PTRADD(ptr, ofs): add pointer while avoiding "NULL + 0" UB */
|
|
|
|
#define UPB_PTRADD(ptr, ofs) ((ofs) ? (ptr) + (ofs) : (ptr))
|
|
|
|
|
|
|
|
/* Configure whether fasttable is switched on or not. *************************/
|
|
|
|
|
|
|
|
#ifdef __has_attribute
|
|
|
|
#define UPB_HAS_ATTRIBUTE(x) __has_attribute(x)
|
|
|
|
#else
|
|
|
|
#define UPB_HAS_ATTRIBUTE(x) 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if UPB_HAS_ATTRIBUTE(musttail)
|
|
|
|
#define UPB_MUSTTAIL __attribute__((musttail))
|
|
|
|
#else
|
|
|
|
#define UPB_MUSTTAIL
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#undef UPB_HAS_ATTRIBUTE
|
|
|
|
|
|
|
|
/* This check is not fully robust: it does not require that we have "musttail"
|
|
|
|
* support available. We need tail calls to avoid consuming arbitrary amounts
|
|
|
|
* of stack space.
|
|
|
|
*
|
|
|
|
* GCC/Clang can mostly be trusted to generate tail calls as long as
|
|
|
|
* optimization is enabled, but, debug builds will not generate tail calls
|
|
|
|
* unless "musttail" is available.
|
|
|
|
*
|
|
|
|
* We should probably either:
|
|
|
|
* 1. require that the compiler supports musttail.
|
|
|
|
* 2. add some fallback code for when musttail isn't available (ie. return
|
|
|
|
* instead of tail calling). This is safe and portable, but this comes at
|
|
|
|
* a CPU cost.
|
|
|
|
*/
|
|
|
|
#if (defined(__x86_64__) || defined(__aarch64__)) && defined(__GNUC__)
|
|
|
|
#define UPB_FASTTABLE_SUPPORTED 1
|
|
|
|
#else
|
|
|
|
#define UPB_FASTTABLE_SUPPORTED 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* define UPB_ENABLE_FASTTABLE to force fast table support.
|
|
|
|
* This is useful when we want to ensure we are really getting fasttable,
|
|
|
|
* for example for testing or benchmarking. */
|
|
|
|
#if defined(UPB_ENABLE_FASTTABLE)
|
|
|
|
#if !UPB_FASTTABLE_SUPPORTED
|
|
|
|
#error fasttable is x86-64/ARM64 only and requires GCC or Clang.
|
|
|
|
#endif
|
|
|
|
#define UPB_FASTTABLE 1
|
|
|
|
/* Define UPB_TRY_ENABLE_FASTTABLE to use fasttable if possible.
|
|
|
|
* This is useful for releasing code that might be used on multiple platforms,
|
|
|
|
* for example the PHP or Ruby C extensions. */
|
|
|
|
#elif defined(UPB_TRY_ENABLE_FASTTABLE)
|
|
|
|
#define UPB_FASTTABLE UPB_FASTTABLE_SUPPORTED
|
|
|
|
#else
|
|
|
|
#define UPB_FASTTABLE 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* UPB_FASTTABLE_INIT() allows protos compiled for fasttable to gracefully
|
|
|
|
* degrade to non-fasttable if we are using UPB_TRY_ENABLE_FASTTABLE. */
|
|
|
|
#if !UPB_FASTTABLE && defined(UPB_TRY_ENABLE_FASTTABLE)
|
|
|
|
#define UPB_FASTTABLE_INIT(...)
|
|
|
|
#else
|
|
|
|
#define UPB_FASTTABLE_INIT(...) __VA_ARGS__
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#undef UPB_FASTTABLE_SUPPORTED
|
|
|
|
|
|
|
|
/* ASAN poisoning (for arena).
|
|
|
|
* If using UPB from an interpreted language like Ruby, a build of the
|
|
|
|
* interpreter compiled with ASAN enabled must be used in order to get sane and
|
|
|
|
* expected behavior.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if defined(__SANITIZE_ADDRESS__)
|
|
|
|
#define UPB_ASAN 1
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
void __asan_poison_memory_region(void const volatile *addr, size_t size);
|
|
|
|
void __asan_unpoison_memory_region(void const volatile *addr, size_t size);
|
|
|
|
#ifdef __cplusplus
|
|
|
|
} /* extern "C" */
|
|
|
|
#endif
|
|
|
|
#define UPB_POISON_MEMORY_REGION(addr, size) \
|
|
|
|
__asan_poison_memory_region((addr), (size))
|
|
|
|
#define UPB_UNPOISON_MEMORY_REGION(addr, size) \
|
|
|
|
__asan_unpoison_memory_region((addr), (size))
|
|
|
|
#else
|
|
|
|
#define UPB_ASAN 0
|
|
|
|
#define UPB_POISON_MEMORY_REGION(addr, size) \
|
|
|
|
((void)(addr), (void)(size))
|
|
|
|
#define UPB_UNPOISON_MEMORY_REGION(addr, size) \
|
|
|
|
((void)(addr), (void)(size))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Disable proto2 arena behavior (TEMPORARY) **********************************/
|
|
|
|
|
|
|
|
#ifdef UPB_DISABLE_PROTO2_ENUM_CHECKING
|
|
|
|
#define UPB_TREAT_PROTO2_ENUMS_LIKE_PROTO3 1
|
|
|
|
#else
|
|
|
|
#define UPB_TREAT_PROTO2_ENUMS_LIKE_PROTO3 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(__cplusplus)
|
|
|
|
#if defined(__clang__) || UPB_GNUC_MIN(6, 0)
|
|
|
|
// https://gcc.gnu.org/gcc-6/changes.html
|
|
|
|
#if __cplusplus >= 201402L
|
|
|
|
#define UPB_DEPRECATED [[deprecated]]
|
|
|
|
#else
|
|
|
|
#define UPB_DEPRECATED __attribute__((deprecated))
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
#define UPB_DEPRECATED
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
#define UPB_DEPRECATED
|
|
|
|
#endif
|