/* * Copyright (c) 2015 Andrew Kelley * * This file is part of zig, which is MIT licensed. * See http://opensource.org/licenses/MIT */ #ifndef ZIG_UTIL_HPP #define ZIG_UTIL_HPP #include #include #include #include #if defined(_MSC_VER) #include #define ATTRIBUTE_COLD __declspec(noinline) #define ATTRIBUTE_PRINTF(a, b) #define ATTRIBUTE_RETURNS_NOALIAS __declspec(restrict) #define ATTRIBUTE_NORETURN __declspec(noreturn) #define ATTRIBUTE_MUST_USE #else #define ATTRIBUTE_COLD __attribute__((cold)) #define ATTRIBUTE_PRINTF(a, b) __attribute__((format(printf, a, b))) #define ATTRIBUTE_RETURNS_NOALIAS __attribute__((__malloc__)) #define ATTRIBUTE_NORETURN __attribute__((noreturn)) #define ATTRIBUTE_MUST_USE __attribute__((warn_unused_result)) #endif #include "softfloat.hpp" #define BREAKPOINT __asm("int $0x03") ATTRIBUTE_COLD ATTRIBUTE_NORETURN ATTRIBUTE_PRINTF(1, 2) void zig_panic(const char *format, ...); #ifdef WIN32 #define __func__ __FUNCTION__ #endif #define zig_unreachable() zig_panic("unreachable: %s:%s:%d", __FILE__, __func__, __LINE__) // Assertions in stage1 are always on, and they call zig @panic. #undef assert void assert(bool ok); #if defined(_MSC_VER) static inline int clzll(unsigned long long mask) { unsigned long lz; #if defined(_WIN64) if (_BitScanReverse64(&lz, mask)) return static_cast(63 - lz); zig_unreachable(); #else if (_BitScanReverse(&lz, mask >> 32)) lz += 32; else _BitScanReverse(&lz, mask & 0xffffffff); return 63 - lz; #endif } static inline int ctzll(unsigned long long mask) { unsigned long result; #if defined(_WIN64) if (_BitScanForward64(&result, mask)) return result; zig_unreachable(); #else if (_BitScanForward(&result, mask & 0xffffffff)) return result; } if (_BitScanForward(&result, mask >> 32)) return 32 + result; zig_unreachable(); #endif } #else #define clzll(x) __builtin_clzll(x) #define ctzll(x) __builtin_ctzll(x) #endif template ATTRIBUTE_RETURNS_NOALIAS static inline T *allocate_nonzero(size_t count) { #ifndef NDEBUG // make behavior when size == 0 portable if (count == 0) return nullptr; #endif T *ptr = reinterpret_cast(malloc(count * sizeof(T))); if (!ptr) zig_panic("allocation failed"); return ptr; } template ATTRIBUTE_RETURNS_NOALIAS static inline T *allocate(size_t count) { #ifndef NDEBUG // make behavior when size == 0 portable if (count == 0) return nullptr; #endif T *ptr = reinterpret_cast(calloc(count, sizeof(T))); if (!ptr) zig_panic("allocation failed"); return ptr; } template static inline T *reallocate(T *old, size_t old_count, size_t new_count) { T *ptr = reallocate_nonzero(old, old_count, new_count); if (new_count > old_count) { memset(&ptr[old_count], 0, (new_count - old_count) * sizeof(T)); } return ptr; } template static inline T *reallocate_nonzero(T *old, size_t old_count, size_t new_count) { #ifndef NDEBUG // make behavior when size == 0 portable if (new_count == 0 && old == nullptr) return nullptr; #endif T *ptr = reinterpret_cast(realloc(old, new_count * sizeof(T))); if (!ptr) zig_panic("allocation failed"); return ptr; } template constexpr size_t array_length(const T (&)[n]) { return n; } template static inline T max(T a, T b) { return (a >= b) ? a : b; } template static inline T min(T a, T b) { return (a <= b) ? a : b; } template static inline T clamp(T min_value, T value, T max_value) { return max(min(value, max_value), min_value); } static inline bool mem_eql_mem(const char *a_ptr, size_t a_len, const char *b_ptr, size_t b_len) { if (a_len != b_len) return false; return memcmp(a_ptr, b_ptr, a_len) == 0; } static inline bool mem_eql_str(const char *mem, size_t mem_len, const char *str) { return mem_eql_mem(mem, mem_len, str, strlen(str)); } static inline bool is_power_of_2(uint64_t x) { return x != 0 && ((x & (~x + 1)) == x); } static inline uint64_t round_to_next_power_of_2(uint64_t x) { --x; x |= x >> 1; x |= x >> 2; x |= x >> 4; x |= x >> 8; x |= x >> 16; x |= x >> 32; return x + 1; } uint32_t int_hash(int i); bool int_eq(int a, int b); uint32_t uint64_hash(uint64_t i); bool uint64_eq(uint64_t a, uint64_t b); uint32_t ptr_hash(const void *ptr); bool ptr_eq(const void *a, const void *b); static inline uint8_t log2_u64(uint64_t x) { return (63 - clzll(x)); } static inline float16_t zig_double_to_f16(double x) { float64_t y; static_assert(sizeof(x) == sizeof(y), ""); memcpy(&y, &x, sizeof(x)); return f64_to_f16(y); } // Return value is safe to coerce to float even when |x| is NaN or Infinity. static inline double zig_f16_to_double(float16_t x) { float64_t y = f16_to_f64(x); double z; static_assert(sizeof(y) == sizeof(z), ""); memcpy(&z, &y, sizeof(y)); return z; } template struct Optional { T value; bool is_some; static inline Optional some(T x) { return {x, true}; } static inline Optional none() { return {{}, false}; } inline bool unwrap(T *res) { *res = value; return is_some; } }; template struct Slice { T *ptr; size_t len; inline T &at(size_t i) { assert(i < len); return &ptr[i]; } inline Slice slice(size_t start, size_t end) { assert(end <= len); assert(end >= start); return { ptr + start, end - start, }; } inline Slice sliceFrom(size_t start) { assert(start <= len); return { ptr + start, len - start, }; } static inline Slice alloc(size_t n) { return {allocate_nonzero(n), n}; } }; template struct Array { static const size_t len = n; T items[n]; inline Slice slice() { return { &items[0], len, }; } }; static inline Slice str(const char *literal) { return {(uint8_t*)(literal), strlen(literal)}; } // Ported from std/mem.zig template static inline bool memEql(Slice a, Slice b) { if (a.len != b.len) return false; for (size_t i = 0; i < a.len; i += 1) { if (a.ptr[i] != b.ptr[i]) return false; } return true; } // Ported from std/mem.zig template static inline bool memStartsWith(Slice haystack, Slice needle) { if (needle.len > haystack.len) return false; return memEql(haystack.slice(0, needle.len), needle); } // Ported from std/mem.zig template static inline void memCopy(Slice dest, Slice src) { assert(dest.len >= src.len); memcpy(dest.ptr, src.ptr, src.len * sizeof(T)); } // Ported from std/mem.zig. // Coordinate struct fields with memSplit function struct SplitIterator { size_t index; Slice buffer; Slice split_bytes; }; bool SplitIterator_isSplitByte(SplitIterator *self, uint8_t byte); Optional< Slice > SplitIterator_next(SplitIterator *self); Slice SplitIterator_rest(SplitIterator *self); SplitIterator memSplit(Slice buffer, Slice split_bytes); #endif