gb/gb.hpp

4155 lines
98 KiB
C++
Raw Normal View History

2015-12-17 07:26:24 -05:00
// gb.hpp - v0.32 - public domain C++11 helper library - no warranty implied; use at your own risk
2015-09-27 14:03:47 -04:00
// (Experimental) A C++11 helper library without STL geared towards game development
2015-10-05 16:30:55 -04:00
/*
LICENSE
This software is in the public domain. Where that dedication is not
recognized, you are granted a perpetual, irrevocable license to copy,
distribute, and modify this file as you see fit.
WARNING
- This library is _highly_ experimental and features may not work as expected.
- This also means that many functions are not documented.
- This library is not compatible with STL at all! (By design)
2015-11-18 04:53:00 -05:00
CONTENTS:
2015-10-05 16:30:55 -04:00
- Common Macros
- Assert
- Types
2015-10-18 07:18:54 -04:00
- Type Traits
2015-10-05 16:30:55 -04:00
- C++11 Move Semantics
- Defer
2015-10-18 07:18:54 -04:00
- Casts
- bit_cast
2015-11-19 17:27:57 -05:00
- pseudo_cast
2015-10-05 16:30:55 -04:00
- Memory
- Mutex
- Atomics
2015-10-28 14:32:42 -04:00
- Semaphore
- Thread
2015-10-05 16:30:55 -04:00
- Allocator
- Heap Allocator
- Arena Allocator
2015-10-28 14:32:42 -04:00
- Functions
2015-10-05 16:30:55 -04:00
- String
- Array
- Hash Table
- Hash Functions
*/
2015-12-02 10:38:18 -05:00
/*
Version History:
2015-12-17 07:26:24 -05:00
0.32 - Change const position convention
0.31a - Minor fixes
0.31 - Remove `_Allocator` suffix for allocator types
2015-12-02 15:34:35 -05:00
0.30 - sort::quick
2015-12-02 10:38:18 -05:00
0.29 - GB_ASSERT prints call stack
0.28 - Pool Allocator
0.27 - Dealloc to Free & More Hashing Functions
0.26a - Heap_Allocator Fix
0.26 - Better Allocation system
0.25a - Array bug fix
0.25 - Faster Heap_Allocator for Windows using HeapAlloc
0.24b - Even More Hash_Table Bug Fixes
0.24a - Hash_Table Bug Fixes
0.24 - More documentation and bug fixes
0.23 - Move Semantics for Array and Hash_Table
0.22 - Code rearrangment into namespaces
0.21d - Fix array::free
0.21c - Fix Another Typo causing unresolved external symbol
0.21b - Typo fixes
0.21a - Better `static` keywords
0.21 - Separate Math Library
0.20a - #ifndef for many macros
0.20 - Angle
0.19 - Cache friendly Transform and String fixes
0.18 - Hash_Table bug fixes
0.17 - Death to OOP
0.16 - All References are const convention
0.15 - Namespaced Types
0.14 - Casts and Quaternion Look At
0.13a - Fix Todos
0.13 - Basic Type Traits
0.12 - Random
0.11 - Complex
0.10 - Atomics
0.09 - Bug Fixes
0.08 - Matrix(2,3)
0.07 - Bug Fixes
0.06 - Os spec ideas
0.05 - Transform Type and Quaternion Functions
0.04 - String
0.03 - Hash Functions
0.02 - Hash Table
0.01 - Initial Version
*/
2015-09-27 14:03:47 -04:00
#ifndef GB_INCLUDE_GB_HPP
#define GB_INCLUDE_GB_HPP
#if !defined(__cplusplus) && __cplusplus >= 201103L
2015-10-18 07:18:54 -04:00
#error This library is only for C++11 and above
2015-09-27 14:03:47 -04:00
#endif
// NOTE(bill): Because static means three different things in C/C++
// Great Design(!)
2015-11-17 18:45:30 -05:00
#ifndef global_variable
#define global_variable static
#define internal_linkage static
#define local_persist static
2015-11-17 05:45:07 -05:00
#endif
2015-09-27 14:03:47 -04:00
2015-12-02 10:06:36 -05:00
/*
Example for static defines
global_variable const f32 TAU = 6.283185f;
2015-12-02 15:34:35 -05:00
global_variable void* g_memory;
2015-12-02 10:06:36 -05:00
internal_linkage void
some_function(...)
{
local_persist u32 count = 0;
...
count++;
...
}
*/
// `global_variable` : variable is in the global scope
// `internal_linkage` : function is only visible for this linkage
// `local_persist` : variable persists for the particular function scope
2015-09-27 14:03:47 -04:00
#if defined(_MSC_VER)
2015-10-18 07:18:54 -04:00
#define _ALLOW_KEYWORD_MACROS
2015-09-27 14:03:47 -04:00
2015-11-17 05:45:07 -05:00
#ifndef alignof // Needed for MSVC 2013 'cause Microsoft "loves" standards
2015-09-27 14:03:47 -04:00
#define alignof(x) __alignof(x)
2015-11-17 05:45:07 -05:00
#endif
2015-09-27 14:03:47 -04:00
#endif
2015-11-17 05:45:07 -05:00
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// System OS //
// //
2015-09-27 14:03:47 -04:00
////////////////////////////////
#if defined(_WIN32) || defined(_WIN64)
2015-11-17 05:45:07 -05:00
#ifndef GB_SYSTEM_WINDOWS
2015-10-18 07:18:54 -04:00
#define GB_SYSTEM_WINDOWS 1
2015-11-17 05:45:07 -05:00
#endif
2015-09-27 14:03:47 -04:00
#elif defined(__APPLE__) && defined(__MACH__)
2015-11-17 05:45:07 -05:00
#ifndef GB_SYSTEM_OSX
2015-10-18 07:18:54 -04:00
#define GB_SYSTEM_OSX 1
2015-11-17 05:45:07 -05:00
#endif
2015-09-27 14:03:47 -04:00
#elif defined(__unix__)
2015-11-17 05:45:07 -05:00
#ifndef GB_SYSTEM_UNIX
2015-10-18 07:18:54 -04:00
#define GB_SYSTEM_UNIX 1
2015-11-17 05:45:07 -05:00
#endif
2015-09-27 14:03:47 -04:00
#if defined(__linux__)
2015-11-17 05:45:07 -05:00
#ifndef GB_SYSTEM_LINUX
2015-10-18 07:18:54 -04:00
#define GB_SYSTEM_LINUX 1
2015-11-17 05:45:07 -05:00
#endif
2015-09-27 14:03:47 -04:00
#elif defined(__FreeBSD__) || defined(__FreeBSD_kernel__)
2015-11-17 05:45:07 -05:00
#ifndef GB_SYSTEM_FREEBSD
2015-10-18 07:18:54 -04:00
#define GB_SYSTEM_FREEBSD 1
2015-11-17 05:45:07 -05:00
#endif
2015-09-27 14:03:47 -04:00
#else
2015-10-18 07:18:54 -04:00
#error This UNIX operating system is not supported by gb.hpp
2015-09-27 14:03:47 -04:00
#endif
#else
2015-10-18 07:18:54 -04:00
#error This operating system is not supported by gb.hpp
2015-09-27 14:03:47 -04:00
#endif
2015-11-18 19:35:56 -05:00
#if defined(_MSC_VER)
// Microsoft Visual Studio
#define GB_COMPILER_MSVC 1
#elif defined(__clang__)
// Clang
#define GB_COMPILER_CLANG 1
#elif defined(__GNUC__) || defined(__GNUG__) && !(defined(__clang__) || defined(__INTEL_COMPILER))
// GNU GCC/G++ Compiler
#define GB_COMPILER_GNU_GCC 1
#elif defined(__INTEL_COMPILER)
// Intel C++ Compiler
#define GB_COMPILER_INTEL 1
#endif
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Environment Bit Size //
// //
2015-09-27 14:03:47 -04:00
////////////////////////////////
#if defined(_WIN32) || defined(_WIN64)
#if defined(_WIN64)
2015-11-17 05:45:07 -05:00
#ifndef GB_ARCH_64_BIT
2015-10-18 07:18:54 -04:00
#define GB_ARCH_64_BIT 1
2015-11-17 05:45:07 -05:00
#endif
2015-09-27 14:03:47 -04:00
#else
2015-11-17 05:45:07 -05:00
#ifndef GB_ARCH_32_BIT
2015-10-18 07:18:54 -04:00
#define GB_ARCH_32_BIT 1
2015-11-17 05:45:07 -05:00
#endif
2015-09-27 14:03:47 -04:00
#endif
#endif
2015-12-02 10:38:18 -05:00
// TODO(bill): Check if this works on clang
2015-09-27 14:03:47 -04:00
#if defined(__GNUC__)
#if defined(__x86_64__) || defined(__ppc64__)
2015-11-17 05:45:07 -05:00
#ifndef GB_ARCH_64_BIT
2015-10-18 07:18:54 -04:00
#define GB_ARCH_64_BIT 1
2015-11-17 05:45:07 -05:00
#endif
2015-09-27 14:03:47 -04:00
#else
2015-11-17 05:45:07 -05:00
#ifndef GB_ARCH_32_BIT
2015-10-18 07:18:54 -04:00
#define GB_ARCH_32_BIT 1
2015-11-17 05:45:07 -05:00
#endif
2015-09-27 14:03:47 -04:00
#endif
#endif
2015-10-18 07:18:54 -04:00
2015-12-02 10:06:36 -05:00
#ifndef GB_EDIAN_ORDER
#define GB_EDIAN_ORDER
#define GB_IS_BIG_EDIAN (!*(unsigned char*)&(unsigned short){1})
#define GB_IS_LITTLE_EDIAN (!GB_IS_BIG_EDIAN)
#endif
2015-10-18 07:18:54 -04:00
2015-09-27 14:03:47 -04:00
#define GB_IS_POWER_OF_TWO(x) ((x) != 0) && !((x) & ((x) - 1))
2015-10-18 07:18:54 -04:00
2015-10-05 16:30:55 -04:00
#if !defined(GB_HAS_NO_CONSTEXPR)
#if defined(_GNUC_VER) && _GNUC_VER < 406 // Less than gcc 4.06
2015-10-28 14:32:42 -04:00
#define GB_HAS_NO_CONSTEXPR 1
2015-10-05 16:30:55 -04:00
#elif defined(_MSC_VER) && _MSC_VER < 1900 // Less than Visual Studio 2015/MSVC++ 14.0
2015-10-28 14:32:42 -04:00
#define GB_HAS_NO_CONSTEXPR 1
2015-10-05 16:30:55 -04:00
#elif !defined(__GXX_EXPERIMENTAL_CXX0X__) && __cplusplus < 201103L
2015-10-28 14:32:42 -04:00
#define GB_HAS_NO_CONSTEXPR 1
2015-10-05 16:30:55 -04:00
#endif
#endif
#if defined(GB_HAS_NO_CONSTEXPR)
2015-10-18 07:18:54 -04:00
#define GB_CONSTEXPR
2015-10-05 16:30:55 -04:00
#else
2015-10-18 07:18:54 -04:00
#define GB_CONSTEXPR constexpr
2015-10-05 16:30:55 -04:00
#endif
#ifndef GB_FORCE_INLINE
2015-10-05 13:17:30 -04:00
#if defined(_MSC_VER)
2015-10-18 07:18:54 -04:00
#define GB_FORCE_INLINE __forceinline
2015-10-05 13:17:30 -04:00
#else
2015-10-28 14:32:42 -04:00
#define GB_FORCE_INLINE __attribute__ ((__always_inline__))
2015-10-05 13:17:30 -04:00
#endif
#endif
2015-12-02 10:06:36 -05:00
////////////////////////////////
// //
// Includes //
// //
////////////////////////////////
#include <math.h>
2015-12-17 07:26:24 -05:00
#include <stdarg.h>
#if !defined(GB_NO_STDIO)
2015-12-02 10:06:36 -05:00
#include <stdio.h>
2015-12-17 07:26:24 -05:00
#endif
2015-12-02 10:06:36 -05:00
2015-10-04 15:59:40 -04:00
#if defined(GB_SYSTEM_WINDOWS)
2015-10-18 07:18:54 -04:00
#define NOMINMAX 1
#define VC_EXTRALEAN 1
#define WIN32_EXTRA_LEAN 1
#define WIN32_LEAN_AND_MEAN 1
2015-10-04 15:59:40 -04:00
2015-12-02 15:34:35 -05:00
#include <windows.h> // TODO(bill): Should we include only the needed headers?
2015-10-18 07:18:54 -04:00
#include <mmsystem.h> // Time functions
2015-10-04 15:59:40 -04:00
2015-10-18 07:18:54 -04:00
#undef NOMINMAX
#undef VC_EXTRALEAN
#undef WIN32_EXTRA_LEAN
#undef WIN32_LEAN_AND_MEAN
2015-10-04 15:59:40 -04:00
2015-10-18 07:18:54 -04:00
#include <intrin.h>
2015-09-27 14:03:47 -04:00
#else
2015-10-18 07:18:54 -04:00
#include <pthread.h>
#include <sys/time.h>
2015-09-27 14:03:47 -04:00
#endif
2015-11-29 10:53:59 -05:00
2015-12-17 07:26:24 -05:00
#ifndef GB_UNUSED
#define GB_UNUSED(x) ((void)sizeof(x))
#endif
2015-12-02 10:38:18 -05:00
2015-11-17 05:45:07 -05:00
#if !defined(GB_ASSERT)
#if !defined(NDEBUG)
2015-12-02 15:34:35 -05:00
#define GB_ASSERT(x, ...) ((void)(gb__assert_handler((x), #x, __FILE__, __LINE__, ##__VA_ARGS__)))
2015-11-17 05:45:07 -05:00
2015-12-02 10:06:36 -05:00
// Helper function used as a better alternative to assert which allows for
// optional printf style error messages
2015-12-02 10:38:18 -05:00
extern "C" void
2015-12-17 07:26:24 -05:00
gb__assert_handler(bool condition, char const* condition_str,
char const* filename, size_t line,
char const* error_text = nullptr, ...);
2015-11-17 05:45:07 -05:00
#else
#define GB_ASSERT(x, ...) ((void)sizeof(x))
#endif
#endif
2015-09-28 16:58:33 -04:00
2015-10-18 07:18:54 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// snprintf_msvc //
// //
2015-10-18 07:18:54 -04:00
////////////////////////////////
2015-12-17 07:26:24 -05:00
#if !defined(GB_NO_STDIO) && defined(_MSC_VER)
2015-11-17 05:45:07 -05:00
extern "C" inline int
2015-12-17 07:26:24 -05:00
gb__vsnprintf_compatible(char* buffer, size_t size, char const* format, va_list args)
2015-11-17 05:45:07 -05:00
{
int result = -1;
if (size > 0)
result = _vsnprintf_s(buffer, size, _TRUNCATE, format, args);
if (result == -1)
return _vscprintf(format, args);
2015-09-28 16:58:33 -04:00
2015-11-17 05:45:07 -05:00
return result;
}
2015-10-18 07:18:54 -04:00
2015-11-17 05:45:07 -05:00
extern "C" inline int
2015-12-17 07:26:24 -05:00
gb__snprintf_compatible(char* buffer, size_t size, char const* format, ...)
2015-11-17 05:45:07 -05:00
{
va_list args;
va_start(args, format);
int result = gb__vsnprintf_compatible(buffer, size, format, args);
va_end(args);
return result;
}
2015-10-18 07:18:54 -04:00
2015-11-17 05:45:07 -05:00
#if !defined(GB_DO_NOT_USE_MSVC_SPRINTF_FIX)
#define snprintf gb__snprintf_compatible
#define vsnprintf gb__vsnprintf_compatible
#endif // GB_DO_NOT_USE_MSVC_SPRINTF_FIX
2015-10-18 07:18:54 -04:00
#endif
#if defined(GB_NO_GB_NAMESPACE)
#define __GB_NAMESPACE_START
#define __GB_NAMESPACE_END
2015-11-18 19:35:56 -05:00
#define __GB_NAMESPACE_PREFIX
2015-10-18 07:18:54 -04:00
#else
2015-11-18 19:35:56 -05:00
#ifndef __GB_NAMESPACE_PREFIX
#define __GB_NAMESPACE_PREFIX gb
#endif
// NOTE(bill): __GB_NAMESPACE_PREFIX cannot be blank
// This is why macros that not meant to be touched have `__` prefix
// You can change them if you know what you are doing
2015-10-18 07:18:54 -04:00
#define __GB_NAMESPACE_START namespace __GB_NAMESPACE_PREFIX {
#define __GB_NAMESPACE_END } // namespace __GB_NAMESPACE_PREFIX
#endif
#if !defined(GB_BASIC_WITHOUT_NAMESPACE)
__GB_NAMESPACE_START
2015-10-05 16:30:55 -04:00
#endif // GB_BASIC_WITHOUT_NAMESPACE
2015-10-18 07:18:54 -04:00
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Types //
// //
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-11-17 05:45:07 -05:00
#ifndef GB_BASIC_TYPES
#define GB_BASIC_TYPES
#if defined(_MSC_VER)
using u8 = unsigned __int8;
using s8 = signed __int8;
using u16 = unsigned __int16;
using s16 = signed __int16;
using u32 = unsigned __int32;
using s32 = signed __int32;
using u64 = unsigned __int64;
using s64 = signed __int64;
#else
2015-12-02 15:34:35 -05:00
// NOTE(bill): Of the platforms that I build for, these will be correct
2015-11-17 05:45:07 -05:00
using u8 = unsigned char;
using s8 = signed char;
using u16 = unsigned short;
using s16 = signed short;
using u32 = unsigned int;
using s32 = signed int;
using u64 = unsigned long long;
using s64 = signed long long;
#endif
2015-09-27 14:03:47 -04:00
2015-11-17 05:45:07 -05:00
static_assert( sizeof(u8) == 1, "u8 is not 8 bits");
static_assert(sizeof(u16) == 2, "u16 is not 16 bits");
static_assert(sizeof(u32) == 4, "u32 is not 32 bits");
static_assert(sizeof(u64) == 8, "u64 is not 64 bits");
2015-09-27 14:03:47 -04:00
2015-11-17 05:45:07 -05:00
using f32 = float;
using f64 = double;
2015-09-27 14:03:47 -04:00
2015-11-17 05:45:07 -05:00
#if defined(GB_B8_AS_BOOL)
2015-11-18 19:35:56 -05:00
using bool8 = bool;
2015-11-17 05:45:07 -05:00
#else
2015-11-18 19:35:56 -05:00
using bool8 = s8;
2015-11-17 05:45:07 -05:00
#endif
2015-11-18 19:35:56 -05:00
using bool32 = s32;
2015-11-17 05:45:07 -05:00
// NOTE(bill): (std::)size_t is not used not because it's a bad concept but on
// the platforms that I will be using:
2015-11-18 04:53:00 -05:00
// sizeof(size_t) == sizeof(usize) == sizeof(ssize)
2015-11-17 05:45:07 -05:00
// NOTE(bill): This also allows for a signed version of size_t which is similar
// to ptrdiff_t
// NOTE(bill): If (u)intptr is a better fit, please use that.
// NOTE(bill): Also, I hate the `_t` suffix
#if defined(GB_ARCH_64_BIT)
using ssize = s64;
using usize = u64;
#elif defined(GB_ARCH_32_BIT)
using usize = s32;
using usize = u32;
#else
#error Unknown architecture bit size
#endif
2015-09-27 14:03:47 -04:00
2015-11-17 05:45:07 -05:00
static_assert(sizeof(usize) == sizeof(size_t),
"`usize` is not the same size as `size_t`");
static_assert(sizeof(ssize) == sizeof(usize),
"`ssize` is not the same size as `usize`");
2015-09-27 14:03:47 -04:00
2015-11-17 05:45:07 -05:00
using intptr = intptr_t;
using uintptr = uintptr_t;
2015-10-04 15:59:40 -04:00
2015-11-17 05:45:07 -05:00
using ptrdiff = ptrdiff_t;
2015-10-04 15:59:40 -04:00
2015-11-17 05:45:07 -05:00
#endif
2015-10-04 15:59:40 -04:00
2015-11-17 05:45:07 -05:00
#if !defined(GB_U8_MIN)
#define GB_U8_MIN 0u
#define GB_U8_MAX 0xffu
#define GB_S8_MIN (-0x7f - 1)
#define GB_S8_MAX 0x7f
#define GB_U16_MIN 0u
#define GB_U16_MAX 0xffffu
#define GB_S16_MIN (-0x7fff - 1)
#define GB_S16_MAX 0x7fff
#define GB_U32_MIN 0u
#define GB_U32_MAX 0xffffffffu
#define GB_S32_MIN (-0x7fffffff - 1)
#define GB_S32_MAX 0x7fffffff
#define GB_U64_MIN 0ull
#define GB_U64_MAX 0xffffffffffffffffull
#define GB_S64_MIN (-0x7fffffffffffffffll - 1)
#define GB_S64_MAX 0x7fffffffffffffffll
#endif
2015-10-04 15:59:40 -04:00
2015-11-17 05:45:07 -05:00
#if defined(GB_ARCH_64_BIT) && !defined(GB_USIZE_MIX)
2015-10-28 14:32:42 -04:00
#define GB_USIZE_MIX GB_U64_MIN
#define GB_USIZE_MAX GB_U64_MAX
2015-10-04 15:59:40 -04:00
2015-10-28 14:32:42 -04:00
#define GB_SSIZE_MIX GB_S64_MIN
#define GB_SSIZE_MAX GB_S64_MAX
2015-11-17 05:45:07 -05:00
#elif defined(GB_ARCH_32_BIT) && !defined(GB_USIZE_MIX)
2015-10-28 14:32:42 -04:00
#define GB_USIZE_MIX GB_U32_MIN
#define GB_USIZE_MAX GB_U32_MAX
2015-10-04 15:59:40 -04:00
2015-10-28 14:32:42 -04:00
#define GB_SSIZE_MIX GB_S32_MIN
#define GB_SSIZE_MAX GB_S32_MAX
2015-10-04 15:59:40 -04:00
#endif
2015-11-17 05:45:07 -05:00
#if defined(GB_BASIC_WITHOUT_NAMESPACE) && !defined(U8_MIN)
2015-10-18 07:18:54 -04:00
#define U8_MIN 0u
#define U8_MAX 0xffu
#define S8_MIN (-0x7f - 1)
#define S8_MAX 0x7f
#define U16_MIN 0u
#define U16_MAX 0xffffu
#define S16_MIN (-0x7fff - 1)
#define S16_MAX 0x7fff
#define U32_MIN 0u
#define U32_MAX 0xffffffffu
#define S32_MIN (-0x7fffffff - 1)
#define S32_MAX 0x7fffffff
#define U64_MIN 0ull
#define U64_MAX 0xffffffffffffffffull
#define S64_MIN (-0x7fffffffffffffffll - 1)
#define S64_MAX 0x7fffffffffffffffll
2015-11-17 05:45:07 -05:00
#if defined(GB_ARCH_64_BIT) && !defined(GB_USIZE_MIX)
2015-10-18 07:18:54 -04:00
#define USIZE_MIX U64_MIN
#define USIZE_MAX U64_MAX
#define SSIZE_MIX S64_MIN
#define SSIZE_MAX S64_MAX
2015-11-17 05:45:07 -05:00
#elif defined(GB_ARCH_32_BIT) && !defined(GB_USIZE_MIX)
2015-10-18 07:18:54 -04:00
#define USIZE_MIX U32_MIN
#define USIZE_MAX U32_MAX
#define SSIZE_MIX S32_MIN
#define SSIZE_MAX S32_MAX
#endif
2015-10-03 10:26:29 -04:00
#endif
2015-10-04 15:59:40 -04:00
2015-10-05 16:30:55 -04:00
#if !defined(GB_BASIC_WITHOUT_NAMESPACE)
2015-10-18 07:18:54 -04:00
__GB_NAMESPACE_END
2015-10-05 16:30:55 -04:00
#endif // GB_BASIC_WITHOUT_NAMESPACE
2015-09-30 17:12:16 -04:00
2015-12-02 10:06:36 -05:00
2015-10-18 07:18:54 -04:00
__GB_NAMESPACE_START
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// C++11 Types Traits //
// //
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-10-05 16:30:55 -04:00
template <typename T> struct Add_Const_Def { using Type = const T; };
template <typename T> using Add_Const = typename Add_Const_Def<T>::Type;
template <typename T> struct Add_Volatile_Def { using Type = volatile T; };
template <typename T> using Add_Volatile = typename Add_Volatile_Def<T>::Type;
template <typename T> using Add_Const_Volatile = Add_Const<Add_Volatile<T>>;
2015-11-19 17:27:57 -05:00
template <typename T> struct Add_Lvalue_Reference_Def { using Type = T&; };
template <typename T> struct Add_Lvalue_Reference_Def<T&> { using Type = T&; };
template <typename T> struct Add_Lvalue_Reference_Def<T&&> { using Type = T&; };
template <> struct Add_Lvalue_Reference_Def<void> { using Type = void; };
2015-12-17 07:26:24 -05:00
template <> struct Add_Lvalue_Reference_Def<void const> { using Type = void const; };
template <> struct Add_Lvalue_Reference_Def<void volatile> { using Type = void volatile; };
template <> struct Add_Lvalue_Reference_Def<void const volatile> { using Type = void const volatile; };
2015-11-19 17:27:57 -05:00
template <typename T> using Add_Lvalue_Reference = typename Add_Lvalue_Reference_Def<T>::Type;
template <typename T> struct Add_Rvalue_Reference_Def { using Type = T&&; };
template <typename T> struct Add_Rvalue_Reference_Def<T&> { using Type = T&; };
template <typename T> struct Add_Rvalue_Reference_Def<T&&> { using Type = T&&; };
template <> struct Add_Rvalue_Reference_Def<void> { using Type = void; };
2015-12-17 07:26:24 -05:00
template <> struct Add_Rvalue_Reference_Def<void const> { using Type = void const; };
template <> struct Add_Rvalue_Reference_Def<void volatile> { using Type = void volatile; };
template <> struct Add_Rvalue_Reference_Def<void const volatile> { using Type = void const volatile; };
2015-10-05 16:30:55 -04:00
template <typename T> using Add_Rvalue_Reference = typename Add_Rvalue_Reference_Def<T>::Type;
template <typename T> struct Remove_Pointer_Def { using Type = T; };
template <typename T> struct Remove_Pointer_Def<T*> { using Type = T; };
template <typename T> struct Remove_Pointer_Def<T* const> { using Type = T; };
template <typename T> struct Remove_Pointer_Def<T* volatile> { using Type = T; };
template <typename T> struct Remove_Pointer_Def<T* const volatile> { using Type = T; };
template <typename T> using Remove_Pointer = typename Remove_Pointer_Def<T>::Type;
template <typename T> struct Add_Pointer_Def { using Type = T*; };
template <typename T> using Add_Pointer = typename Add_Pointer_Def<T>::Type;
template <typename T> struct Remove_Const_Def { using Type = T; };
template <typename T> struct Remove_Const_Def<const T> { using Type = T; };
template <typename T> using Remove_Const = typename Remove_Const_Def<T>::Type;
template <typename T> struct Remove_Volatile_Def { using Type = T; };
template <typename T> struct Remove_Volatile_Def<volatile T> { using Type = T; };
template <typename T> using Remove_Volatile = typename Remove_Const_Def<T>::Type;
template <typename T> using Remove_Const_Volatile = Remove_Const<Remove_Volatile<T>>;
template <typename T> struct Remove_Reference_Def { using Type = T; };
template <typename T> struct Remove_Reference_Def<T&> { using Type = T; };
template <typename T> struct Remove_Reference_Def<T&&> { using Type = T; };
template <typename T> using Remove_Reference = typename Remove_Reference_Def<T>::Type;
2015-11-17 18:45:30 -05:00
template <typename T, T v> struct Integral_Constant { global_variable const T VALUE = v; using Value_Type = T; using Type = Integral_Constant; };
template <typename T, usize N = 0> struct Extent : Integral_Constant<usize, 0> {};
2015-11-19 17:27:57 -05:00
template <typename T> struct Extent<T[], 0> : Integral_Constant<usize, 0> {};
template <typename T, usize N> struct Extent<T[], N> : Integral_Constant<usize, Extent<T, N-1>::VALUE> {};
template <typename T, usize N> struct Extent<T[N], 0> : Integral_Constant<usize, N> {};
template <typename T, usize I, usize N> struct Extent<T[I], N> : Integral_Constant<usize, Extent<T, N-1>::VALUE> {};
2015-11-18 19:35:56 -05:00
template <typename T> struct Remove_Extent_Def { using Type = T; };
template <typename T> struct Remove_Extent_Def<T[]> { using Type = T; };
template <typename T, usize N> struct Remove_Extent_Def<T[N]> { using Type = T; };
2015-12-02 15:34:35 -05:00
// TODO(bill): Do I "need" all of these template traits?
2015-10-05 16:30:55 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// C++11 Move Semantics //
// //
2015-10-05 16:30:55 -04:00
////////////////////////////////
2015-11-18 19:35:56 -05:00
2015-12-02 10:06:36 -05:00
// TODO(bill): Are these decent names? Are `forward` and `move` clear enough?
2015-09-27 14:03:47 -04:00
template <typename T>
inline T&&
2015-10-05 16:30:55 -04:00
forward(Remove_Reference<T>& t)
2015-09-27 14:03:47 -04:00
{
2015-09-28 15:31:26 -04:00
return static_cast<T&&>(t);
2015-09-27 14:03:47 -04:00
}
template <typename T>
inline T&&
2015-10-05 16:30:55 -04:00
forward(Remove_Reference<T>&& t)
2015-09-27 14:03:47 -04:00
{
2015-09-28 15:31:26 -04:00
return static_cast<T&&>(t);
2015-09-27 14:03:47 -04:00
}
template <typename T>
2015-10-05 16:30:55 -04:00
inline Remove_Reference<T>&&
2015-09-27 14:03:47 -04:00
move(T&& t)
{
2015-10-05 16:30:55 -04:00
return static_cast<Remove_Reference<T>&&>(t);
2015-09-27 14:03:47 -04:00
}
2015-11-17 05:45:07 -05:00
__GB_NAMESPACE_END
2015-09-27 14:03:47 -04:00
2015-12-02 10:06:36 -05:00
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Defer //
// //
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-11-18 19:35:56 -05:00
2015-11-17 05:45:07 -05:00
#ifndef GB_DEFER
#define GB_DEFER
__GB_NAMESPACE_START
namespace impl
{
template <typename Func>
struct Defer
{
Func f;
Defer(Func&& f) : f{forward<Func>(f)} {}
~Defer() { f(); };
};
template <typename Func>
inline Defer<Func>
defer_func(Func&& f) { return Defer<Func>(forward<Func>(f)); }
} // namespace impl
__GB_NAMESPACE_END
// NOTE(bill): These macros are in the global namespace thus, defer can be treated without a __GB_NAMESPACE_PREFIX:: prefix
#define GB_DEFER_1(x, y) x##y
#define GB_DEFER_2(x, y) GB_DEFER_1(x, y)
#define GB_DEFER_3(x) GB_DEFER_2(GB_DEFER_2(GB_DEFER_2(x, __COUNTER__), _), __LINE__)
#define defer(code) auto GB_DEFER_3(_defer_) = __GB_NAMESPACE_PREFIX::impl::defer_func([&](){code;})
/* EXAMPLES
// `defer (...)` will defer a statement till the end of scope
FILE* file = fopen("test.txt", "rb");
if (file == nullptr)
{
// Handle Error
}
2015-11-18 19:35:56 -05:00
defer (fclose(file)); // Will always be called at the end of scope
//
auto m = mutex::make();
defer (mutex::destroy(&m)); // Mutex will be destroyed at the end of scope
{
mutex::lock(&m);
defer (mutex::unlock(&m)); // Mutex will unlock at end of scope
// Do whatever
}
// You can scope multiple statements together if needed with {...}
defer ({
func1();
func2();
func3();
});
*/
2015-11-17 05:45:07 -05:00
#endif
2015-09-27 14:03:47 -04:00
2015-12-02 10:06:36 -05:00
2015-10-05 16:30:55 -04:00
#if !defined(GB_CASTS_WITHOUT_NAMESPACE)
2015-10-18 07:18:54 -04:00
__GB_NAMESPACE_START
2015-10-05 16:30:55 -04:00
#endif // GB_CASTS_WITHOUT_NAMESPACE
2015-11-17 05:45:07 -05:00
#ifndef GB_SPECIAL_CASTS
#define GB_SPECIAL_CASTS
// NOTE(bill): Very similar to doing `*(T*)(&u)`
template <typename Dest, typename Source>
inline Dest
2015-12-17 07:26:24 -05:00
bit_cast(Source const& source)
2015-11-17 05:45:07 -05:00
{
static_assert(sizeof(Dest) <= sizeof(Source),
2015-12-17 07:26:24 -05:00
"bit_cast<Dest>(Source const&) - sizeof(Dest) <= sizeof(Source)");
2015-11-17 05:45:07 -05:00
Dest dest;
::memcpy(&dest, &source, sizeof(Dest));
return dest;
}
// IMPORTANT NOTE(bill): Very similar to doing `*(T*)(&u)` but easier/clearer to write
// however, it can be dangerous if sizeof(T) > sizeof(U) e.g. unintialized memory, undefined behavior
// *(T*)(&u) ~~ pseudo_cast<T>(u)
template <typename T, typename U>
inline T
2015-12-17 07:26:24 -05:00
pseudo_cast(U const& u)
{
2015-12-17 07:26:24 -05:00
return reinterpret_cast<T const&>(u);
}
2015-11-18 19:35:56 -05:00
/*
EXAMPLES:
// bit_cast
u8 arr[4] = {0x78, 0x56, 0x34, 0x12};
u32 var = bit_cast<u32>(arr); // Little edian => 0x12345678
2015-11-19 17:27:57 -05:00
// pseudo_cast - except from gb_math.hpp
Sphere
2015-12-17 07:26:24 -05:00
calculate_min_bounding(void const* vertices, usize num_vertices, usize stride, usize offset, f32 step)
2015-11-19 17:27:57 -05:00
{
auto gen = random::make(0);
2015-12-17 07:26:24 -05:00
u8 const* vertex = reinterpret_cast<u8 const*>(vertices);
2015-11-19 17:27:57 -05:00
vertex += offset;
Vector3 position = pseudo_cast<Vector3>(vertex[0]);
Vector3 center = position;
center += pseudo_cast<Vector3>(vertex[1 * stride]);
center *= 0.5f;
Vector3 d = position - center;
f32 max_dist_sq = math::dot(d, d);
f32 radius_step = step * 0.37f;
bool done;
do
{
done = true;
for (u32 i = 0, index = random::uniform_u32(&gen, 0, num_vertices-1);
i < num_vertices;
i++, index = (index + 1)%num_vertices)
{
Vector3 position = pseudo_cast<Vector3>(vertex[index * stride]);
d = position - center;
f32 dist_sq = math::dot(d, d);
if (dist_sq > max_dist_sq)
{
done = false;
center = d * radius_step;
max_dist_sq = math::lerp(max_dist_sq, dist_sq, step);
break;
}
}
}
while (!done);
Sphere result;
result.center = center;
result.radius = math::sqrt(max_dist_sq);
return result;
}
2015-11-18 19:35:56 -05:00
*/
2015-11-17 05:45:07 -05:00
#endif
2015-10-18 07:18:54 -04:00
// FORENOTE(bill): There used to be a magic_cast that was equivalent to
// a C-style cast but I removed it as I could not get it work as intented
2015-12-02 10:06:36 -05:00
// for everything using only C++ style casts (it needed to a c-style cast)
2015-10-05 16:58:47 -04:00
2015-10-05 16:30:55 -04:00
#if !defined(GB_CASTS_WITHOUT_NAMESPACE)
2015-10-18 07:18:54 -04:00
__GB_NAMESPACE_END
2015-10-05 16:30:55 -04:00
#endif // GB_CASTS_WITHOUT_NAMESPACE
2015-12-02 10:06:36 -05:00
2015-10-18 07:18:54 -04:00
__GB_NAMESPACE_START
2015-11-18 19:35:56 -05:00
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Memory //
// //
2015-09-27 14:03:47 -04:00
////////////////////////////////
template <typename T, usize N>
2015-12-17 07:26:24 -05:00
inline usize array_count(T const(& )[N]) { return N; }
2015-11-29 10:53:59 -05:00
inline s64 kilobytes(s64 x) { return (x) * 1024ll; }
inline s64 megabytes(s64 x) { return kilobytes(x) * 1024ll; }
inline s64 gigabytes(s64 x) { return megabytes(x) * 1024ll; }
inline s64 terabytes(s64 x) { return gigabytes(x) * 1024ll; }
2015-12-02 10:06:36 -05:00
2015-09-27 14:03:47 -04:00
struct Mutex
{
2015-10-04 15:59:40 -04:00
#if defined(GB_SYSTEM_WINDOWS)
2015-09-27 14:03:47 -04:00
HANDLE win32_mutex;
#else
pthread_mutex_t posix_mutex;
#endif
};
2015-10-18 07:18:54 -04:00
namespace mutex
{
2015-10-28 14:32:42 -04:00
Mutex make();
void destroy(Mutex* mutex);
void lock(Mutex* mutex);
bool try_lock(Mutex* mutex);
void unlock(Mutex* mutex);
2015-10-18 07:18:54 -04:00
} // namespace mutex
2015-09-27 14:03:47 -04:00
2015-12-02 10:06:36 -05:00
// Atomic Types
2015-10-18 07:18:54 -04:00
struct Atomic32 { u32 nonatomic; };
struct Atomic64 { u64 nonatomic; };
2015-10-03 10:26:29 -04:00
namespace atomic
{
2015-12-17 07:26:24 -05:00
u32 load(Atomic32 const volatile* object);
void store(Atomic32 volatile* object, u32 value);
u32 compare_exchange_strong(Atomic32 volatile* object, u32 expected, u32 desired);
u32 exchanged(Atomic32 volatile* object, u32 desired);
u32 fetch_add(Atomic32 volatile* object, s32 operand);
u32 fetch_and(Atomic32 volatile* object, u32 operand);
u32 fetch_or(Atomic32 volatile* object, u32 operand);
u64 load(Atomic64 const volatile* object);
void store(Atomic64 volatile* object, u64 value);
u64 compare_exchange_strong(Atomic64 volatile* object, u64 expected, u64 desired);
u64 exchanged(Atomic64 volatile* object, u64 desired);
u64 fetch_add(Atomic64 volatile* object, s64 operand);
u64 fetch_and(Atomic64 volatile* object, u64 operand);
u64 fetch_or(Atomic64 volatile* object, u64 operand);
2015-10-03 10:26:29 -04:00
} // namespace atomic
2015-12-02 10:06:36 -05:00
2015-10-28 14:32:42 -04:00
struct Semaphore
{
#if defined(GB_SYSTEM_WINDOWS)
HANDLE win32_handle;
#else
Mutex mutex;
pthread_cond_t cond;
s32 count;
#endif
};
namespace semaphore
{
Semaphore make();
void destroy(Semaphore* semaphore);
void post(Semaphore* semaphore, u32 count = 1);
void wait(Semaphore* semaphore);
} // namespace semaphore
2015-12-02 10:06:36 -05:00
2015-12-02 15:34:35 -05:00
// TODO(bill): Is this thread procedure definition good enough?
using Thread_Procedure = void(void*);
2015-10-28 14:32:42 -04:00
struct Thread
{
#if defined(GB_SYSTEM_WINDOWS)
HANDLE win32_handle;
#else
pthread_t posix_handle;
#endif
2015-12-02 15:34:35 -05:00
Thread_Procedure* function;
2015-10-28 14:32:42 -04:00
void* data;
Semaphore semaphore;
usize stack_size;
2015-11-18 19:35:56 -05:00
bool32 is_running;
2015-10-28 14:32:42 -04:00
};
namespace thread
{
Thread make();
2015-10-31 20:45:54 -04:00
void destroy(Thread* t);
2015-12-02 15:34:35 -05:00
void start(Thread* t, Thread_Procedure* func, void* data = nullptr, usize stack_size = 0);
void join(Thread* t);
2015-12-17 07:26:24 -05:00
bool is_running(Thread const& t);
2015-11-19 17:27:57 -05:00
u32 current_id();
2015-10-28 14:32:42 -04:00
} // namespace thread
2015-12-02 10:06:36 -05:00
// Default alignment for memory allocations
2015-09-28 15:31:26 -04:00
#ifndef GB_DEFAULT_ALIGNMENT
2015-11-29 10:53:59 -05:00
#define GB_DEFAULT_ALIGNMENT 8
#endif
2015-12-02 10:06:36 -05:00
// Base class for memory allocators - Pretty much a vtable
2015-09-27 14:03:47 -04:00
struct Allocator
{
2015-12-02 10:06:36 -05:00
// Allocates the specified amount of memory aligned to the specified alignment
2015-11-29 10:53:59 -05:00
void* (*alloc)(Allocator* a, usize size, usize align);
2015-12-02 10:06:36 -05:00
// Frees an allocation made with alloc()
void (*free)(Allocator* a, void* ptr);
2015-12-02 10:06:36 -05:00
// Returns the amount of usuable memory allocated at `ptr`.
2015-10-18 07:18:54 -04:00
///
2015-12-02 10:06:36 -05:00
// If the allocator does not support tracking of the allocation size,
// the function will return -1
2015-12-17 07:26:24 -05:00
s64 (*allocated_size)(Allocator* a, void const* ptr);
2015-12-02 10:06:36 -05:00
// Returns the total amount of memory allocated by this allocator
2015-10-18 07:18:54 -04:00
///
2015-12-02 10:06:36 -05:00
// If the allocator does not track memory, the function will return -1
2015-11-29 10:53:59 -05:00
s64 (*total_allocated)(Allocator* a);
2015-09-27 14:03:47 -04:00
};
2015-12-02 10:06:36 -05:00
struct Heap : Allocator
2015-09-27 14:03:47 -04:00
{
struct Header
{
usize size;
};
2015-11-29 13:04:33 -05:00
Mutex mutex;
bool32 use_mutex;
s64 total_allocated_count;
s64 allocation_count;
2015-09-27 14:03:47 -04:00
#if defined(GB_SYSTEM_WINDOWS)
2015-11-29 13:04:33 -05:00
HANDLE win32_heap_handle;
#endif
2015-09-27 14:03:47 -04:00
};
namespace heap
2015-11-19 17:27:57 -05:00
{
Heap make(bool use_mutex = true);
void destroy(Heap* heap);
} // namespace heap
2015-11-19 17:27:57 -05:00
2015-12-02 10:06:36 -05:00
struct Arena : Allocator
2015-09-27 14:03:47 -04:00
{
2015-10-03 10:26:29 -04:00
Allocator* backing;
void* physical_start;
s64 total_size;
s64 total_allocated_count;
s64 temp_count;
2015-09-27 14:03:47 -04:00
};
namespace arena
2015-11-29 10:53:59 -05:00
{
Arena make(Allocator* backing, usize size);
Arena make(void* start, usize size);
void destroy(Arena* arena);
void clear(Arena* arena);
} // namespace arena
2015-11-29 10:53:59 -05:00
2015-12-02 10:06:36 -05:00
2015-10-18 07:18:54 -04:00
struct Temporary_Arena_Memory
{
Arena* arena;
s64 original_count;
2015-10-18 07:18:54 -04:00
};
namespace temporary_arena_memory
{
Temporary_Arena_Memory make(Arena* arena);
void free(Temporary_Arena_Memory tmp);
} // namespace temporary_arena_memory
2015-12-02 10:06:36 -05:00
struct Pool : Allocator
2015-12-02 10:06:36 -05:00
{
Allocator* backing;
void* physical_start;
void* free_list;
usize block_size;
usize block_align;
s64 total_size;
};
namespace pool
2015-12-02 10:06:36 -05:00
{
Pool make(Allocator* backing, usize num_blocks, usize block_size,
usize block_align = GB_DEFAULT_ALIGNMENT);
void destroy(Pool* pool);
} // namespace pool
2015-12-02 10:06:36 -05:00
2015-10-18 07:18:54 -04:00
namespace memory
{
void* align_forward(void* ptr, usize align);
2015-10-28 14:32:42 -04:00
void* pointer_add(void* ptr, usize bytes);
void* pointer_sub(void* ptr, usize bytes);
2015-12-17 07:26:24 -05:00
void const* pointer_add(void const* ptr, usize bytes);
void const* pointer_sub(void const* ptr, usize bytes);
2015-10-18 07:18:54 -04:00
2015-11-29 13:04:33 -05:00
void* set(void* ptr, usize bytes, u8 value);
2015-11-19 17:27:57 -05:00
2015-10-18 07:18:54 -04:00
void* zero(void* ptr, usize bytes);
2015-12-17 07:26:24 -05:00
void* copy(void const* src, usize bytes, void* dest);
void* move(void const* src, usize bytes, void* dest);
bool equals(void const* a, void const* b, usize bytes);
2015-11-19 17:27:57 -05:00
// TODO(bill): Should this be just zero(T*) ???
2015-11-19 17:27:57 -05:00
template <typename T>
T* zero_struct(T* ptr);
2015-11-29 13:04:33 -05:00
template <typename T>
T* zero_array(T* ptr, usize count);
2015-11-19 17:27:57 -05:00
template <typename T>
2015-12-17 07:26:24 -05:00
T* copy_array(T const* src_array, usize count, T* dest_array);
2015-11-19 17:27:57 -05:00
// TODO(bill): Should I implement something like std::copy, std::fill, std::fill_n ???
template <typename T>
void swap(T* a, T* b);
template <typename T, usize N>
void swap(T (& a)[N], T (& b)[N]);
2015-10-18 07:18:54 -04:00
} // namespace memory
2015-12-02 15:34:35 -05:00
// Allocator Functions
2015-11-19 17:27:57 -05:00
void* alloc(Allocator* a, usize size, usize align = GB_DEFAULT_ALIGNMENT);
void free(Allocator* a, void* ptr);
2015-12-17 07:26:24 -05:00
s64 allocated_size(Allocator* a, void const* ptr);
2015-11-19 18:22:21 -05:00
s64 total_allocated(Allocator* a);
2015-10-18 07:18:54 -04:00
template <typename T>
inline T* alloc_struct(Allocator* a) { return static_cast<T*>(alloc(a, sizeof(T), alignof(T))); }
2015-10-18 07:18:54 -04:00
template <typename T>
inline T* alloc_array(Allocator* a, usize count) { return static_cast<T*>(alloc(a, count * sizeof(T), alignof(T))); }
2015-10-18 07:18:54 -04:00
2015-12-02 10:06:36 -05:00
2015-09-28 16:58:33 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// String //
// //
// C compatible string //
// //
2015-09-28 16:58:33 -04:00
////////////////////////////////
2015-10-18 07:18:54 -04:00
2015-12-02 10:06:36 -05:00
// A "better" string type that is compatible with C style read-only functions
2015-09-28 16:58:33 -04:00
using String = char*;
2015-10-18 07:18:54 -04:00
namespace string
{
using Size = u32;
struct Header
2015-09-28 16:58:33 -04:00
{
Allocator* allocator;
Size length;
Size capacity;
2015-09-28 16:58:33 -04:00
};
2015-12-02 15:34:35 -05:00
inline Header* header(String str) { return reinterpret_cast<Header*>(str) - 1; }
2015-09-28 16:58:33 -04:00
2015-12-17 07:26:24 -05:00
String make(Allocator* a, char const* str = "");
String make(Allocator* a, void const* str, Size num_bytes);
2015-10-28 14:32:42 -04:00
void free(String str);
2015-09-28 16:58:33 -04:00
2015-12-17 07:26:24 -05:00
String duplicate(Allocator* a, String const str);
2015-09-28 16:58:33 -04:00
2015-12-17 07:26:24 -05:00
Size length(String const str);
Size capacity(String const str);
Size available_space(String const str);
2015-09-28 16:58:33 -04:00
2015-10-18 07:18:54 -04:00
void clear(String str);
2015-09-28 16:58:33 -04:00
void append(String* str, char c);
2015-12-17 07:26:24 -05:00
void append(String* str, String const other);
void append_cstring(String* str, char const* other);
void append(String* str, void const* other, Size num_bytes);
2015-09-28 16:58:33 -04:00
void make_space_for(String* str, Size add_len);
2015-12-17 07:26:24 -05:00
usize allocation_size(String const str);
2015-09-28 16:58:33 -04:00
2015-12-17 07:26:24 -05:00
bool equals(String const lhs, String const rhs);
int compare(String const lhs, String const rhs); // NOTE(bill): three-way comparison
2015-09-28 16:58:33 -04:00
2015-12-17 07:26:24 -05:00
void trim(String* str, char const* cut_set);
void trim_space(String* str);
2015-10-18 07:18:54 -04:00
} // namespace string
2015-12-02 10:06:36 -05:00
2015-10-03 10:26:29 -04:00
// TODO(bill): string libraries
2015-09-28 16:58:33 -04:00
2015-12-02 10:06:36 -05:00
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Array //
// //
2015-09-27 14:03:47 -04:00
////////////////////////////////
#ifndef GB_ARRAY_BOUND_CHECKING
#define GB_ARRAY_BOUND_CHECKING 1
#endif
2015-12-02 10:06:36 -05:00
// Dynamic resizable array for POD types only
2015-09-27 14:03:47 -04:00
template <typename T>
struct Array
{
2015-11-29 10:53:59 -05:00
using Type = T;
2015-09-27 14:03:47 -04:00
Allocator* allocator;
2015-09-27 15:43:22 -04:00
s64 count;
2015-10-18 07:18:54 -04:00
s64 capacity;
2015-09-27 15:43:22 -04:00
T* data;
2015-09-27 14:03:47 -04:00
2015-09-27 15:43:22 -04:00
Array() = default;
explicit Array(Allocator* a, usize count = 0);
2015-10-18 07:18:54 -04:00
~Array();
2015-12-17 07:26:24 -05:00
Array(Array const& array);
Array(Array&& array);
2015-09-27 14:03:47 -04:00
2015-12-17 07:26:24 -05:00
Array& operator=(Array const& array);
Array& operator=(Array&& array);
2015-12-17 07:26:24 -05:00
T const& operator[](usize index) const;
T& operator[](usize index);
2015-09-27 14:03:47 -04:00
};
2015-12-02 15:34:35 -05:00
// TODO(bill): Should I even have ctor, dtor, copy/move overloads for Array<T>?
// Should these be explicit functions e.g.
/*
auto old_array = array::make(...);
auto new_array = array::copy(old_array);
array::free(&old_array);
array::free(&new_array);
*/
2015-10-18 07:18:54 -04:00
namespace array
{
2015-12-02 10:06:36 -05:00
// Helper functions to make and free an array
template <typename T> Array<T> make(Allocator* allocator, usize count = 0);
template <typename T> void free(Array<T>* array);
2015-10-18 07:18:54 -04:00
2015-12-02 10:06:36 -05:00
// Appends `item` to the end of the array
2015-12-17 07:26:24 -05:00
template <typename T> void append(Array<T>* a, T const& item);
template <typename T> void append(Array<T>* a, T&& item);
2015-12-02 10:06:36 -05:00
// Appends `items[count]` to the end of the array
2015-12-17 07:26:24 -05:00
template <typename T> void append(Array<T>* a, T const* items, usize count);
2015-10-18 07:18:54 -04:00
2015-12-02 10:06:36 -05:00
// Pops the last item form the array. The array cannot be empty.
2015-11-18 19:35:56 -05:00
template <typename T> void pop(Array<T>* a);
2015-10-18 07:18:54 -04:00
2015-12-02 10:06:36 -05:00
// Removes all items from the array - does not free memory
template <typename T> void clear(Array<T>* a);
2015-12-02 10:06:36 -05:00
// Modify the size of a array - only reallocates when necessary
template <typename T> void resize(Array<T>* a, usize count);
2015-12-02 10:06:36 -05:00
// Makes sure that the array has at least the specified capacity - or the array the grows
template <typename T> void reserve(Array<T>* a, usize capacity);
2015-12-02 10:06:36 -05:00
// Reallocates the array to the specific capacity
template <typename T> void set_capacity(Array<T>* a, usize capacity);
2015-12-02 10:06:36 -05:00
// Grows the array to keep append() to be O(1)
template <typename T> void grow(Array<T>* a, usize min_capacity = 0);
2015-10-18 07:18:54 -04:00
} // namespace array
2015-12-02 10:06:36 -05:00
// Used to iterate over the array with a C++11 for loop
2015-12-17 07:26:24 -05:00
template <typename T> inline T* begin(Array<T>& a) { return a.data; }
template <typename T> inline T const* begin(Array<T> const& a) { return a.data; }
template <typename T> inline T* begin(Array<T>&& a) { return a.data; }
template <typename T> inline T* end(Array<T>& a) { return a.data + a.count; }
template <typename T> inline T const* end(Array<T> const& a) { return a.data + a.count; }
template <typename T> inline T* end(Array<T>&& a) { return a.data + a.count; }
2015-09-27 14:03:47 -04:00
2015-12-02 10:06:36 -05:00
2015-09-27 15:43:22 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Hash Table //
// //
2015-09-27 15:43:22 -04:00
////////////////////////////////
2015-09-27 14:03:47 -04:00
2015-12-02 10:06:36 -05:00
// Hash table for POD types only with a u64 key
2015-09-27 14:03:47 -04:00
template <typename T>
2015-09-27 15:43:22 -04:00
struct Hash_Table
2015-09-27 14:03:47 -04:00
{
2015-11-29 10:53:59 -05:00
using Type = T;
2015-09-27 15:43:22 -04:00
struct Entry
2015-09-27 14:03:47 -04:00
{
2015-09-27 15:43:22 -04:00
u64 key;
s64 next;
T value;
};
2015-09-27 14:03:47 -04:00
2015-09-27 15:43:22 -04:00
Array<s64> hashes;
Array<Entry> entries;
2015-09-27 14:03:47 -04:00
2015-10-31 20:45:54 -04:00
Hash_Table();
explicit Hash_Table(Allocator* a);
2015-12-17 07:26:24 -05:00
Hash_Table(Hash_Table<T> const& other);
2015-11-18 19:35:56 -05:00
Hash_Table(Hash_Table<T>&& other);
2015-09-27 15:43:22 -04:00
~Hash_Table() = default;
2015-11-18 19:35:56 -05:00
2015-12-17 07:26:24 -05:00
Hash_Table<T>& operator=(Hash_Table<T> const& other);
2015-11-18 19:35:56 -05:00
Hash_Table<T>& operator=(Hash_Table<T>&& other);
2015-09-27 15:43:22 -04:00
};
2015-09-27 14:03:47 -04:00
namespace hash_table
{
2015-12-02 10:06:36 -05:00
// Helper function to make a hash table
template <typename T> Hash_Table<T> make(Allocator* a);
2015-12-02 10:06:36 -05:00
// Return `true` if the specified key exist in the hash table
2015-12-17 07:26:24 -05:00
template <typename T> bool has(Hash_Table<T> const& h, u64 key);
2015-12-02 10:06:36 -05:00
// Returns the value stored at the key, or a `default_value` if the key is not found in the hash table
2015-12-17 07:26:24 -05:00
template <typename T> T const& get(Hash_Table<T> const& h, u64 key, T const& default_value);
2015-12-02 10:06:36 -05:00
// Sets the value for the key in the hash table
2015-12-17 07:26:24 -05:00
template <typename T> void set(Hash_Table<T>* h, u64 key, T const& value);
template <typename T> void set(Hash_Table<T>* h, u64 key, T&& value);
2015-12-02 10:06:36 -05:00
// Removes the key from the hash table if it exists
template <typename T> void remove(Hash_Table<T>* h, u64 key);
2015-12-02 10:06:36 -05:00
// Resizes the hash table's lookup table to the specified size
template <typename T> void reserve(Hash_Table<T>* h, usize capacity);
2015-12-02 10:06:36 -05:00
// Remove all elements from the hash table
template <typename T> void clear(Hash_Table<T>* h);
} // namespace hash_table
2015-12-02 10:06:36 -05:00
// Used to iterate over the array with a C++11 for loop - in random order
2015-12-17 07:26:24 -05:00
template <typename T> typename Hash_Table<T>::Entry const* begin(Hash_Table<T> const& h);
template <typename T> typename Hash_Table<T>::Entry const* end(Hash_Table<T> const& h);
namespace multi_hash_table
{
2015-12-02 10:06:36 -05:00
// Outputs all the items that with the specified key
2015-12-17 07:26:24 -05:00
template <typename T> void get(Hash_Table<T> const& h, u64 key, Array<T>& items);
2015-12-02 10:06:36 -05:00
// Returns the count of entries with the specified key
2015-12-17 07:26:24 -05:00
template <typename T> usize count(Hash_Table<T> const& h, u64 key);
2015-12-02 10:06:36 -05:00
// Finds the first entry with specified key in the hash table
2015-12-17 07:26:24 -05:00
template <typename T> typename Hash_Table<T>::Entry const* find_first(Hash_Table<T> const& h, u64 key);
2015-12-02 10:06:36 -05:00
// Finds the next entry with same key as `e`
2015-12-17 07:26:24 -05:00
template <typename T> typename Hash_Table<T>::Entry const* find_next(Hash_Table<T> const& h, typename Hash_Table<T>::Entry const* e);
2015-12-02 10:06:36 -05:00
// Inserts the `value` as an additional value for the specified key
2015-12-17 07:26:24 -05:00
template <typename T> void insert(Hash_Table<T>* h, u64 key, T const& value);
template <typename T> void insert(Hash_Table<T>* h, u64 key, T&& value);
2015-12-02 10:06:36 -05:00
// Removes a specified entry `e` from the hash table
2015-12-17 07:26:24 -05:00
template <typename T> void remove_entry(Hash_Table<T>* h, typename Hash_Table<T>::Entry const* e);
2015-12-02 10:06:36 -05:00
// Removes all entries with from the hash table with the specified key
template <typename T> void remove_all(Hash_Table<T>* h, u64 key);
} // namespace multi_hash_table
2015-12-02 10:06:36 -05:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Hash //
// //
////////////////////////////////
namespace hash
{
2015-12-17 07:26:24 -05:00
u32 adler32(void const* key, u32 num_bytes);
2015-12-17 07:26:24 -05:00
u32 crc32(void const* key, u32 num_bytes);
u64 crc64(void const* key, usize num_bytes);
2015-12-17 07:26:24 -05:00
u32 fnv32(void const* key, usize num_bytes);
u64 fnv64(void const* key, usize num_bytes);
u32 fnv32a(void const* key, usize num_bytes);
u64 fnv64a(void const* key, usize num_bytes);
2015-12-17 07:26:24 -05:00
u32 murmur32(void const* key, u32 num_bytes, u32 seed = 0x9747b28c);
u64 murmur64(void const* key, usize num_bytes, u64 seed = 0x9747b28c);
} // namespace hash
2015-12-02 10:06:36 -05:00
2015-12-02 15:34:35 -05:00
////////////////////////////////
// //
// Sort //
// //
////////////////////////////////
namespace sort
{
// Comparison_Function
// NOTE(bill): Similar to str(n)cmp
// a < b --> -1
// a == b --> 0
// a > b --> +1
// Quick Sort (Qsort)
2015-12-02 15:34:35 -05:00
template <typename T, typename Comparison_Function>
void quick(T* array, usize count, Comparison_Function compare);
// TODO(bill): Implement other sorting algorithms
2015-12-02 15:34:35 -05:00
} // namespace sort
2015-12-02 10:06:36 -05:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Time //
// //
////////////////////////////////
struct Time
{
s64 microseconds;
};
2015-12-17 07:26:24 -05:00
extern Time const TIME_ZERO;
namespace time
{
Time now();
void sleep(Time time);
Time seconds(f32 s);
Time milliseconds(s32 ms);
Time microseconds(s64 us);
f32 as_seconds(Time t);
s32 as_milliseconds(Time t);
s64 as_microseconds(Time t);
} // namespace time
bool operator==(Time left, Time right);
bool operator!=(Time left, Time right);
bool operator<(Time left, Time right);
bool operator>(Time left, Time right);
bool operator<=(Time left, Time right);
bool operator>=(Time left, Time right);
2015-11-19 17:27:57 -05:00
Time operator+(Time right);
Time operator-(Time right);
Time operator+(Time left, Time right);
Time operator-(Time left, Time right);
Time& operator+=(Time& left, Time right);
Time& operator-=(Time& left, Time right);
Time operator*(Time left, f32 right);
Time operator*(Time left, s64 right);
Time operator*(f32 left, Time right);
Time operator*(s64 left, Time right);
Time& operator*=(Time& left, f32 right);
Time& operator*=(Time& left, s64 right);
Time operator/(Time left, f32 right);
Time operator/(Time left, s64 right);
Time& operator/=(Time& left, f32 right);
Time& operator/=(Time& left, s64 right);
f32 operator/(Time left, Time right);
Time operator%(Time left, Time right);
Time& operator%=(Time& left, Time right);
2015-12-02 10:06:36 -05:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// OS //
// //
////////////////////////////////
2015-11-18 19:35:56 -05:00
// TODO(bill): Should this be system:: vs os:: ?
namespace os
{
2015-12-02 10:06:36 -05:00
u64 rdtsc();
} // namespace os
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
2015-12-02 15:34:35 -05:00
// Template Implementations //
2015-12-02 10:06:36 -05:00
// //
////////////////////////////////
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Array //
// //
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-09-28 15:31:26 -04:00
template <typename T>
2015-10-18 07:18:54 -04:00
inline
Array<T>::Array(Allocator* a, usize count_)
: allocator(a)
2015-10-18 07:18:54 -04:00
, count(0)
, capacity(0)
, data(nullptr)
{
if (count_ > 0)
2015-09-27 14:03:47 -04:00
{
2015-09-28 15:31:26 -04:00
data = alloc_array<T>(a, count_);
if (data)
2015-10-18 07:18:54 -04:00
count = capacity = count_;
2015-09-28 15:31:26 -04:00
}
}
2015-09-27 14:03:47 -04:00
2015-10-18 07:18:54 -04:00
template <typename T>
2015-10-28 14:32:42 -04:00
inline
2015-12-17 07:26:24 -05:00
Array<T>::Array(Array<T> const& other)
2015-10-18 07:18:54 -04:00
: allocator(other.allocator)
, count(0)
, capacity(0)
, data(nullptr)
{
2015-12-17 07:26:24 -05:00
auto new_count = other.count;
2015-11-28 17:55:24 -05:00
array::set_capacity(this, new_count);
memory::copy_array(other.data, new_count, data);
this->count = new_count;
2015-10-18 07:18:54 -04:00
}
2015-09-28 15:31:26 -04:00
template <typename T>
inline
Array<T>::Array(Array<T>&& other)
: allocator(nullptr)
, count(0)
, capacity(0)
, data(nullptr)
{
*this = move(other);
}
2015-10-18 07:18:54 -04:00
template <typename T>
2015-10-28 14:32:42 -04:00
inline
Array<T>::~Array()
2015-10-18 07:18:54 -04:00
{
2015-10-31 20:45:54 -04:00
if (allocator && capacity > 0)
free(allocator, data);
2015-10-18 07:18:54 -04:00
}
template <typename T>
Array<T>&
2015-12-17 07:26:24 -05:00
Array<T>::operator=(Array<T> const& other)
2015-10-18 07:18:54 -04:00
{
2015-10-31 20:45:54 -04:00
if (allocator == nullptr)
allocator = other.allocator;
2015-12-17 07:26:24 -05:00
auto new_count = other.count;
2015-11-28 17:55:24 -05:00
array::resize(this, new_count);
memory::copy_array(other.data, new_count, data);
2015-10-18 07:18:54 -04:00
return *this;
}
template <typename T>
Array<T>&
Array<T>::operator=(Array<T>&& other)
{
if (this != &other)
{
if (allocator && capacity > 0)
free(allocator, data);
2015-11-18 19:35:56 -05:00
allocator = other.allocator;
count = other.count;
capacity = other.capacity;
data = other.data;
other.allocator = nullptr;
other.count = 0;
other.capacity = 0;
other.data = nullptr;
}
return *this;
}
template <typename T>
2015-12-17 07:26:24 -05:00
inline T const&
Array<T>::operator[](usize index) const
{
#if GB_ARRAY_BOUND_CHECKING
GB_ASSERT(index < static_cast<usize>(capacity), "Array out of bounds");
#endif
return data[index];
}
template <typename T>
inline T&
Array<T>::operator[](usize index)
{
#if GB_ARRAY_BOUND_CHECKING
GB_ASSERT(index < static_cast<usize>(capacity), "Array out of bounds");
#endif
return data[index];
}
2015-10-18 07:18:54 -04:00
namespace array
{
2015-09-28 15:31:26 -04:00
template <typename T>
inline Array<T>
make(Allocator* allocator, usize count)
2015-09-27 14:03:47 -04:00
{
2015-10-31 20:45:54 -04:00
Array<T> array{allocator};
2015-09-28 15:31:26 -04:00
if (count > 0)
2015-09-27 14:03:47 -04:00
{
2015-09-28 15:31:26 -04:00
array.data = alloc_array<T>(allocator, count);
if (array.data)
2015-10-18 07:18:54 -04:00
array.count = array.capacity = count;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
return array;
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
free(Array<T>* a)
2015-09-27 14:03:47 -04:00
{
2015-10-31 20:45:54 -04:00
if (a->allocator)
free(a->allocator, a->data);
a->count = 0;
2015-11-18 07:18:33 -05:00
a->capacity = 0;
a->data = nullptr;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
2015-12-17 07:26:24 -05:00
append(Array<T>* a, T const& item)
2015-09-28 15:31:26 -04:00
{
if (a->capacity < a->count + 1)
array::grow(a);
a->data[a->count++] = item;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
template <typename T>
inline void
append(Array<T>* a, T&& item)
{
if (a->capacity < a->count + 1)
array::grow(a);
a->data[a->count++] = move(item);
}
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
2015-12-17 07:26:24 -05:00
append(Array<T>* a, T const* items, usize count)
2015-09-27 14:03:47 -04:00
{
2015-10-31 20:45:54 -04:00
if (a->capacity <= a->count + static_cast<s64>(count))
2015-10-28 14:32:42 -04:00
array::grow(a, a->count + count);
2015-09-27 14:03:47 -04:00
memory::copy_array(items, count, &a->data[a->count]);
a->count += count;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
2015-11-18 19:35:56 -05:00
pop(Array<T>* a)
2015-09-27 14:03:47 -04:00
{
GB_ASSERT(a->count > 0);
2015-09-27 14:03:47 -04:00
a->count--;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
clear(Array<T>* a)
2015-09-27 14:03:47 -04:00
{
a->count = 0;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
resize(Array<T>* a, usize count)
2015-09-28 15:31:26 -04:00
{
if (a->capacity < static_cast<s64>(count))
array::grow(a, count);
a->count = count;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
reserve(Array<T>* a, usize capacity)
2015-09-28 15:31:26 -04:00
{
if (a->capacity < static_cast<s64>(capacity))
array::set_capacity(a, capacity);
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
set_capacity(Array<T>* a, usize capacity)
2015-09-28 15:31:26 -04:00
{
if (static_cast<s64>(capacity) == a->capacity)
2015-09-28 15:31:26 -04:00
return;
2015-09-27 14:03:47 -04:00
if (static_cast<s64>(capacity) < a->count)
array::resize(a, capacity);
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
T* data = nullptr;
2015-10-18 07:18:54 -04:00
if (capacity > 0)
2015-09-28 15:31:26 -04:00
{
data = alloc_array<T>(a->allocator, capacity);
memory::copy_array(a->data, a->count, data);
2015-09-28 15:31:26 -04:00
}
free(a->allocator, a->data);
a->data = data;
a->capacity = capacity;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
grow(Array<T>* a, usize min_capacity)
2015-09-28 15:31:26 -04:00
{
// TODO(bill): Decide on decent growing formula for Array
usize capacity = 2 * a->capacity + 8;
2015-10-18 07:18:54 -04:00
if (capacity < min_capacity)
capacity = min_capacity;
set_capacity(a, capacity);
2015-09-28 15:31:26 -04:00
}
2015-10-18 07:18:54 -04:00
} // namespace array
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Hash Table //
// //
2015-09-28 15:31:26 -04:00
////////////////////////////////
2015-10-31 20:45:54 -04:00
template <typename T>
inline
Hash_Table<T>::Hash_Table()
: hashes()
, entries()
{
}
2015-10-18 07:18:54 -04:00
template <typename T>
inline
Hash_Table<T>::Hash_Table(Allocator* a)
: hashes(a)
, entries(a)
{
}
template <typename T>
2015-10-28 14:32:42 -04:00
inline
2015-12-17 07:26:24 -05:00
Hash_Table<T>::Hash_Table(Hash_Table<T> const& other)
: hashes(other.hashes)
, entries(other.entries)
{
}
2015-11-18 19:35:56 -05:00
template <typename T>
inline
Hash_Table<T>::Hash_Table(Hash_Table<T>&& other)
: hashes(move(other.hashes))
, entries(move(other.entries))
{
}
template <typename T>
inline Hash_Table<T>&
2015-12-17 07:26:24 -05:00
Hash_Table<T>::operator=(Hash_Table<T> const& other)
2015-10-18 07:18:54 -04:00
{
2015-10-31 20:45:54 -04:00
hashes = other.hashes;
entries = other.entries;
return *this;
2015-10-18 07:18:54 -04:00
}
2015-11-18 19:35:56 -05:00
template <typename T>
inline Hash_Table<T>&
Hash_Table<T>::operator=(Hash_Table<T>&& other)
{
hashes = move(other.hashes);
entries = move(other.entries);
return *this;
}
2015-10-18 07:18:54 -04:00
namespace hash_table
{
template <typename T>
inline Hash_Table<T>
make(Allocator* a)
2015-10-18 07:18:54 -04:00
{
return Hash_Table<T>{a};
2015-10-18 07:18:54 -04:00
}
2015-09-28 15:31:26 -04:00
namespace impl
{
struct Find_Result
{
s64 hash_index;
s64 data_prev;
s64 entry_index;
2015-09-28 15:31:26 -04:00
};
2015-09-27 14:03:47 -04:00
template <typename T> usize add_entry(Hash_Table<T>* h, u64 key);
2015-12-17 07:26:24 -05:00
template <typename T> void erase(Hash_Table<T>* h, Find_Result const& fr);
template <typename T> Find_Result find_result_from_key(Hash_Table<T> const& h, u64 key);
template <typename T> Find_Result find_result_from_entry(Hash_Table<T> const& h, typename Hash_Table<T>::Entry const* e);
2015-11-19 18:22:21 -05:00
template <typename T> s64 make_entry(Hash_Table<T>* h, u64 key);
template <typename T> void find_and_erase_entry(Hash_Table<T>* h, u64 key);
2015-12-17 07:26:24 -05:00
template <typename T> s64 find_entry_or_fail(Hash_Table<T> const& h, u64 key);
2015-11-19 18:22:21 -05:00
template <typename T> s64 find_or_make_entry(Hash_Table<T>* h, u64 key);
template <typename T> void rehash(Hash_Table<T>* h, usize new_capacity);
template <typename T> void grow(Hash_Table<T>* h);
template <typename T> bool is_full(Hash_Table<T>* h);
2015-10-18 07:18:54 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
usize
add_entry(Hash_Table<T>* h, u64 key)
2015-09-28 15:31:26 -04:00
{
typename Hash_Table<T>::Entry e = {};
2015-09-28 15:31:26 -04:00
e.key = key;
e.next = -1;
usize e_index = h->entries.count;
array::append(&h->entries, e);
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
return e_index;
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
void
2015-12-17 07:26:24 -05:00
erase(Hash_Table<T>* h, Find_Result const& fr)
2015-09-28 15:31:26 -04:00
{
if (fr.data_prev < 0)
h->hashes[fr.hash_index] = h->entries[fr.entry_index].next;
2015-09-28 15:31:26 -04:00
else
h->entries[fr.data_prev].next = h->entries[fr.entry_index].next;
2015-09-27 14:03:47 -04:00
2015-11-19 18:22:21 -05:00
array::pop(&h->entries); // Update array count
2015-09-27 14:03:47 -04:00
if (fr.entry_index == h->entries.count)
2015-09-28 15:31:26 -04:00
return;
2015-09-27 14:03:47 -04:00
h->entries[fr.entry_index] = h->entries[h->entries.count];
2015-09-27 14:03:47 -04:00
2015-11-19 18:22:21 -05:00
auto last = impl::find_result_from_key(*h, h->entries[fr.entry_index].key);
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
if (last.data_prev < 0)
h->hashes[last.hash_index] = fr.entry_index;
2015-09-28 15:31:26 -04:00
else
h->entries[last.entry_index].next = fr.entry_index;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
Find_Result
2015-12-17 07:26:24 -05:00
find_result_from_key(Hash_Table<T> const& h, u64 key)
2015-09-28 15:31:26 -04:00
{
Find_Result fr = {};
2015-11-19 18:22:21 -05:00
fr.hash_index = -1;
fr.data_prev = -1;
fr.entry_index = -1;
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
if (h.hashes.count == 0)
return fr;
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
fr.hash_index = key % h.hashes.count;
fr.entry_index = h.hashes[fr.hash_index];
while (fr.entry_index >= 0)
2015-09-28 15:31:26 -04:00
{
if (h.entries[fr.entry_index].key == key)
2015-09-28 15:31:26 -04:00
return fr;
fr.data_prev = fr.entry_index;
fr.entry_index = h.entries[fr.entry_index].next;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
return fr;
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
Find_Result
2015-12-17 07:26:24 -05:00
find_result_from_entry(Hash_Table<T> const& h, typename Hash_Table<T>::Entry const* e)
2015-09-28 15:31:26 -04:00
{
Find_Result fr = {};
2015-11-19 18:22:21 -05:00
fr.hash_index = -1;
fr.data_prev = -1;
fr.entry_index = -1;
2015-10-05 16:58:47 -04:00
if (h.hashes.count == 0 || !e)
return fr;
2015-11-19 18:22:21 -05:00
fr.hash_index = e->key % h.hashes.count;
fr.entry_index = h.hashes[fr.hash_index];
while (fr.entry_index >= 0)
2015-10-05 16:58:47 -04:00
{
if (&h.entries[fr.entry_index] == e)
2015-10-05 16:58:47 -04:00
return fr;
fr.data_prev = fr.entry_index;
fr.entry_index = h.entries[fr.entry_index].next;
2015-10-05 16:58:47 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
return fr;
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
2015-10-18 07:18:54 -04:00
s64
make_entry(Hash_Table<T>* h, u64 key)
2015-09-28 15:31:26 -04:00
{
2015-12-17 07:26:24 -05:00
Find_Result const fr = impl::find_result_from_key(*h, key);
s64 const index = impl::add_entry(h, key);
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
if (fr.data_prev < 0)
h->hashes[fr.hash_index] = index;
2015-09-28 15:31:26 -04:00
else
h->entries[fr.data_prev].next = index;
2015-09-27 14:03:47 -04:00
h->entries[index].next = fr.entry_index;
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
return index;
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
void
find_and_erase_entry(Hash_Table<T>* h, u64 key)
2015-09-28 15:31:26 -04:00
{
2015-12-17 07:26:24 -05:00
Find_Result const fr = impl::find_result_from_key(*h, key);
if (fr.entry_index >= 0)
2015-11-19 18:22:21 -05:00
impl::erase(h, fr);
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
s64
2015-12-17 07:26:24 -05:00
find_entry_or_fail(Hash_Table<T> const& h, u64 key)
2015-09-28 15:31:26 -04:00
{
2015-11-19 18:22:21 -05:00
return impl::find_result_from_key(h, key).entry_index;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
s64
find_or_make_entry(Hash_Table<T>* h, u64 key)
2015-09-28 15:31:26 -04:00
{
2015-12-17 07:26:24 -05:00
auto const fr = impl::find_result_from_key(*h, key);
if (fr.entry_index >= 0)
return fr.entry_index;
2015-09-27 14:03:47 -04:00
2015-10-18 07:18:54 -04:00
s64 index = impl::add_entry(h, key);
2015-09-28 15:31:26 -04:00
if (fr.data_prev < 0)
h->hashes[fr.hash_index] = index;
2015-09-28 15:31:26 -04:00
else
h->entries[fr.data_prev].next = index;
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
return index;
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
void
rehash(Hash_Table<T>* h, usize new_capacity)
2015-09-27 14:03:47 -04:00
{
auto nh = hash_table::make<T>(h->hashes.allocator);
array::resize(&nh.hashes, new_capacity);
array::reserve(&nh.entries, h->entries.count);
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
for (usize i = 0; i < new_capacity; i++)
nh.hashes[i] = -1;
2015-09-27 14:03:47 -04:00
for (u32 i = 0; i < h->entries.count; i++)
2015-09-28 15:31:26 -04:00
{
2015-12-17 07:26:24 -05:00
auto const* e = &h->entries[i];
multi_hash_table::insert(&nh, e->key, e->value);
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-10-31 20:45:54 -04:00
Hash_Table<T> empty_ht{h->hashes.allocator};
h->~Hash_Table<T>();
2015-09-27 14:03:47 -04:00
memory::copy_array(&nh, 1, h);
memory::copy_array(&empty_ht, 1, &nh);
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
void
grow(Hash_Table<T>* h)
2015-09-28 15:31:26 -04:00
{
const usize new_capacity = 2 * h->entries.count + 8;
2015-10-18 07:18:54 -04:00
impl::rehash(h, new_capacity);
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
bool
is_full(Hash_Table<T>* h)
2015-09-28 15:31:26 -04:00
{
// Make sure that there is enough space
2015-12-17 07:26:24 -05:00
f32 const maximum_load_coefficient = 0.75f;
return h->entries.count >= maximum_load_coefficient * h->hashes.count;
2015-09-28 15:31:26 -04:00
}
} // namespace impl
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline bool
2015-12-17 07:26:24 -05:00
has(Hash_Table<T> const& h, u64 key)
2015-09-28 15:31:26 -04:00
{
2015-10-18 07:18:54 -04:00
return impl::find_entry_or_fail(h, key) >= 0;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
2015-12-17 07:26:24 -05:00
inline T const&
get(Hash_Table<T> const& h, u64 key, T const& default_value)
2015-09-28 15:31:26 -04:00
{
2015-12-17 07:26:24 -05:00
s64 const index = impl::find_entry_or_fail(h, key);
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
if (index < 0)
return default_value;
return h.entries[index].value;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
2015-12-17 07:26:24 -05:00
set(Hash_Table<T>* h, u64 key, T const& value)
2015-09-28 15:31:26 -04:00
{
if (h->hashes.count == 0)
2015-10-18 07:18:54 -04:00
impl::grow(h);
2015-09-27 14:03:47 -04:00
2015-12-17 07:26:24 -05:00
s64 const index = impl::find_or_make_entry(h, key);
h->entries[index].value = value;
2015-10-18 07:18:54 -04:00
if (impl::is_full(h))
impl::grow(h);
2015-09-28 15:31:26 -04:00
}
2015-09-27 15:43:22 -04:00
template <typename T>
inline void
set(Hash_Table<T>* h, u64 key, T&& value)
{
if (h->hashes.count == 0)
impl::grow(h);
2015-12-17 07:26:24 -05:00
s64 const index = impl::find_or_make_entry(h, key);
h->entries[index].value = move(value);
if (impl::is_full(h))
impl::grow(h);
}
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
remove(Hash_Table<T>* h, u64 key)
2015-09-28 15:31:26 -04:00
{
2015-10-18 07:18:54 -04:00
impl::find_and_erase_entry(h, key);
2015-09-28 15:31:26 -04:00
}
2015-09-27 15:43:22 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
reserve(Hash_Table<T>* h, usize capacity)
2015-09-28 15:31:26 -04:00
{
2015-10-18 07:18:54 -04:00
impl::rehash(h, capacity);
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
clear(Hash_Table<T>* h)
2015-09-28 15:31:26 -04:00
{
array::clear(&h->hashes);
array::clear(&h->entries);
2015-09-28 15:31:26 -04:00
}
2015-10-18 07:18:54 -04:00
} // namespace hash_table
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
2015-12-17 07:26:24 -05:00
inline typename Hash_Table<T>::Entry const*
begin(Hash_Table<T> const& h)
2015-09-28 15:31:26 -04:00
{
return begin(h.entries);
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
2015-12-17 07:26:24 -05:00
inline typename Hash_Table<T>::Entry const*
end(Hash_Table<T> const& h)
2015-09-28 15:31:26 -04:00
{
return end(h.entries);
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-10-18 07:18:54 -04:00
namespace multi_hash_table
{
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
2015-12-17 07:26:24 -05:00
get(Hash_Table<T> const& h, u64 key, Array<T>& items)
2015-09-28 15:31:26 -04:00
{
2015-10-18 07:18:54 -04:00
auto e = multi_hash_table::find_first(h, key);
2015-09-28 15:31:26 -04:00
while (e)
{
2015-10-18 07:18:54 -04:00
array::append(items, e->value);
e = multi_hash_table::find_next(h, e);
2015-09-28 15:31:26 -04:00
}
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline usize
2015-12-17 07:26:24 -05:00
count(Hash_Table<T> const& h, u64 key)
2015-09-28 15:31:26 -04:00
{
usize count = 0;
2015-10-18 07:18:54 -04:00
auto e = multi_hash_table::find_first(h, key);
2015-09-28 15:31:26 -04:00
while (e)
{
2015-10-02 08:11:18 -04:00
count++;
2015-10-18 07:18:54 -04:00
e = multi_hash_table::find_next(h, e);
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
return count;
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
2015-12-17 07:26:24 -05:00
inline typename Hash_Table<T>::Entry const*
find_first(Hash_Table<T> const& h, u64 key)
2015-09-28 15:31:26 -04:00
{
2015-12-17 07:26:24 -05:00
s64 const index = hash_table::impl::find_entry_or_fail(h, key);
2015-09-28 15:31:26 -04:00
if (index < 0)
return nullptr;
return &h.entries[index];
2015-09-28 15:31:26 -04:00
}
template <typename T>
2015-12-17 07:26:24 -05:00
typename Hash_Table<T>::Entry const*
find_next(Hash_Table<T> const& h, typename Hash_Table<T>::Entry const* e)
2015-09-28 15:31:26 -04:00
{
if (!e)
return nullptr;
auto index = e->next;
while (index >= 0)
{
2015-11-19 18:22:21 -05:00
if (h.entries[index].key == e->key)
return &h.entries[index];
index = h.entries[index].next;
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
return nullptr;
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
2015-12-17 07:26:24 -05:00
insert(Hash_Table<T>* h, u64 key, T const& value)
2015-09-28 15:31:26 -04:00
{
if (h->hashes.count == 0)
2015-10-18 07:18:54 -04:00
hash_table::impl::grow(h);
2015-09-27 14:03:47 -04:00
2015-10-18 07:18:54 -04:00
auto next = hash_table::impl::make_entry(h, key);
h->entries[next].value = value;
2015-09-27 14:03:47 -04:00
2015-10-18 07:18:54 -04:00
if (hash_table::impl::is_full(h))
hash_table::impl::grow(h);
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
template <typename T>
inline void
insert(Hash_Table<T>* h, u64 key, T&& value)
{
if (h->hashes.count == 0)
hash_table::impl::grow(h);
auto next = hash_table::impl::make_entry(h, key);
h->entries[next].value = move(value);
if (hash_table::impl::is_full(h))
hash_table::impl::grow(h);
}
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
2015-12-17 07:26:24 -05:00
remove_entry(Hash_Table<T>* h, typename Hash_Table<T>::Entry const* e)
2015-09-28 15:31:26 -04:00
{
2015-12-17 07:26:24 -05:00
auto const fr = hash_table::impl::find_result_from_entry(*h, e);
if (fr.entry_index >= 0)
2015-10-18 07:18:54 -04:00
hash_table::impl::erase(h, fr);
2015-09-28 15:31:26 -04:00
}
2015-09-27 14:03:47 -04:00
2015-09-28 15:31:26 -04:00
template <typename T>
inline void
remove_all(Hash_Table<T>* h, u64 key)
2015-09-28 15:31:26 -04:00
{
2015-11-19 18:22:21 -05:00
while (hash_table::has(*h, key))
2015-10-18 07:18:54 -04:00
hash_table::remove(h, key);
2015-09-28 15:31:26 -04:00
}
2015-10-18 07:18:54 -04:00
} // namespace multi_hash_table
2015-09-27 14:03:47 -04:00
2015-10-28 14:32:42 -04:00
2015-11-19 17:27:57 -05:00
namespace memory
{
template <typename T>
inline T*
zero_struct(T* ptr)
{
return static_cast<T*>(memory::zero(ptr, sizeof(T)));
}
2015-11-29 13:04:33 -05:00
template <typename T>
inline T*
zero_array(T* ptr, usize count)
{
return static_cast<T*>(memory::zero(ptr, count * sizeof(T)));
}
2015-11-19 17:27:57 -05:00
template <typename T>
inline T*
2015-12-17 07:26:24 -05:00
copy_array(T const* src_array, usize count, T* dest_array)
2015-11-19 17:27:57 -05:00
{
return static_cast<T*>(memory::copy(src_array, count * sizeof(T), dest_array));
2015-11-19 17:27:57 -05:00
}
template <typename T>
inline void
swap(T* a, T* b)
{
2015-12-02 15:34:35 -05:00
T c = __GB_NAMESPACE_PREFIX::move(*a);
*a = __GB_NAMESPACE_PREFIX::move(*b);
*b = __GB_NAMESPACE_PREFIX::move(c);
}
template <typename T, usize N>
inline void
swap(T (& a)[N], T (& b)[N])
{
for (usize i = 0; i < N; i++)
math::swap(&a[i], &b[i]);
}
2015-11-19 17:27:57 -05:00
} // namespace memory
2015-10-28 14:32:42 -04:00
2015-12-02 15:34:35 -05:00
////////////////////////////////
// //
// Sort //
// //
////////////////////////////////
namespace sort
{
template <typename T, typename Comparison_Function>
void
quick(T* array, usize count, Comparison_Function compare)
{
if (count < 2) return;
2015-12-17 07:26:24 -05:00
T const& mid = array[count/2];
2015-12-02 15:34:35 -05:00
s64 i = 0;
s64 j = count-1;
while (true)
{
while (compare(array[i], mid) < 0) i++;
while (compare(mid, array[j]) < 0) j--;
if (i >= j) break;
memory::swap(&array[i], &array[j]);
i++;
j--;
}
quick(array, i, compare);
quick(array+i, count-i, compare);
}
} // namespace sort
2015-10-28 14:32:42 -04:00
2015-10-18 07:18:54 -04:00
__GB_NAMESPACE_END
2015-09-29 10:11:18 -04:00
2015-09-28 15:31:26 -04:00
#endif // GB_INCLUDE_GB_HPP
2015-09-27 15:43:22 -04:00
2015-12-02 15:34:35 -05:00
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
//
2015-12-02 10:06:36 -05:00
// It's turtles all the way down!
2015-12-02 15:34:35 -05:00
//
//
//
//
//
2015-09-28 15:31:26 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Implemenation //
// //
2015-09-28 15:31:26 -04:00
////////////////////////////////
2015-12-02 10:38:18 -05:00
#if defined(GB_IMPLEMENTATION)
2015-12-02 10:38:18 -05:00
#if defined(GB_SYSTEM_WINDOWS)
2015-12-17 07:26:24 -05:00
#if !defined(GB_NO_STDIO)
#include <dbghelp.h>
#pragma comment(lib, "dbghelp.lib") // TODO(bill): Should this be pragma included or not?
2015-12-02 10:38:18 -05:00
2015-12-17 07:26:24 -05:00
internal_linkage void
gb__print_call_stack(FILE* out_stream)
{
SymInitialize(GetCurrentProcess(), nullptr, true);
SymSetOptions(SYMOPT_LOAD_LINES | SYMOPT_UNDNAME);
DWORD mtype = {};
CONTEXT ctx = {};
ctx.ContextFlags = CONTEXT_CONTROL;
RtlCaptureContext(&ctx);
STACKFRAME64 stack = {};
#if defined(_M_IX86)
mtype = IMAGE_FILE_MACHINE_I386;
stack.AddrPC.Offset = ctx.Eip;
stack.AddrPC.Mode = AddrModeFlat;
stack.AddrFrame.Offset = ctx.Ebp;
stack.AddrFrame.Mode = AddrModeFlat;
stack.AddrStack.Offset = ctx.Esp;
stack.AddrStack.Mode = AddrModeFlat;
#elif defined(_M_X64)
mtype = IMAGE_FILE_MACHINE_AMD64;
stack.AddrPC.Offset = ctx.Rip;
stack.AddrPC.Mode = AddrModeFlat;
stack.AddrFrame.Offset = ctx.Rsp;
stack.AddrFrame.Mode = AddrModeFlat;
stack.AddrStack.Offset = ctx.Rsp;
stack.AddrStack.Mode = AddrModeFlat;
#else
#error Unknown Windows Platform
#endif
2015-12-02 10:38:18 -05:00
2015-12-17 07:26:24 -05:00
DWORD ldsp = 0;
IMAGEHLP_LINE64 line = {};
line.SizeOfStruct = sizeof(IMAGEHLP_LINE64);
2015-12-02 10:38:18 -05:00
2015-12-17 07:26:24 -05:00
char buf[sizeof(SYMBOL_INFO) + (MAX_SYM_NAME * sizeof(TCHAR))];
2015-12-02 10:38:18 -05:00
2015-12-17 07:26:24 -05:00
SYMBOL_INFO* sym = reinterpret_cast<SYMBOL_INFO*>(buf);
sym->SizeOfStruct = sizeof(SYMBOL_INFO);
sym->MaxNameLen = MAX_SYM_NAME;
2015-12-02 10:38:18 -05:00
2015-12-17 07:26:24 -05:00
UINT layer_count = 0;
while (StackWalk64(mtype,
GetCurrentProcess(), GetCurrentThread(),
&stack, &ctx, nullptr,
SymFunctionTableAccess64, SymGetModuleBase64, nullptr))
{
if (stack.AddrPC.Offset == 0)
break;
2015-12-02 10:38:18 -05:00
2015-12-17 07:26:24 -05:00
BOOL result = SymGetLineFromAddr64(GetCurrentProcess(), stack.AddrPC.Offset, &ldsp, &line);
result = result && SymFromAddr(GetCurrentProcess(), stack.AddrPC.Offset, 0, sym);
2015-12-02 10:38:18 -05:00
2015-12-17 07:26:24 -05:00
if (result)
{
fprintf(out_stream,
"\t[%u] `%s` (%s:%d)\n",
layer_count, sym->Name, line.FileName, line.LineNumber);
}
else
{
fprintf(out_stream,
"\t[%u] 0x%p\n",
layer_count, stack.AddrPC.Offset);
}
layer_count++;
2015-12-02 10:38:18 -05:00
}
2015-12-17 07:26:24 -05:00
SymCleanup(GetCurrentProcess());
2015-12-02 10:38:18 -05:00
}
2015-12-17 07:26:24 -05:00
#endif
2015-12-02 10:38:18 -05:00
#else
#error gb__print_call_stack() not implemeneted
// TODO(bill): Implemenet gb__print_call_stack()
#endif
// Helper function used as a better alternative to assert which allows for
// optional printf style error messages
inline void
2015-12-17 07:26:24 -05:00
gb__assert_handler(bool condition, char const* condition_str,
char const* filename, size_t line,
char const* error_text, ...)
2015-12-02 10:38:18 -05:00
{
if (condition)
return;
2015-12-17 07:26:24 -05:00
#if !defined(GB_NO_STDIO)
2015-12-02 10:38:18 -05:00
FILE* out_stream = stderr;
fprintf(out_stream, "ASSERT! %s(%lu): %s", filename, line, condition_str);
if (error_text)
{
fprintf(out_stream, " - ");
va_list args;
va_start(args, error_text);
vfprintf(out_stream, error_text, args);
va_end(args);
}
fprintf(out_stream, "\n");
fprintf(out_stream, "Stacktrack:\n");
gb__print_call_stack(out_stream);
2015-12-17 07:26:24 -05:00
#endif
2015-12-02 10:38:18 -05:00
// TODO(bill): Are these decent breaking functions???
#if defined(GB_COMPILER_MSVC)
__debugbreak();
#elif defined(GB_COMPILER_GNU_GCC)
__builtin_trap();
#else
#error Implement aborting function
#endif
}
__GB_NAMESPACE_START
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Memory //
// //
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-10-28 14:32:42 -04:00
namespace mutex
{
2015-11-19 17:27:57 -05:00
inline Mutex
2015-10-28 14:32:42 -04:00
make()
2015-09-27 14:03:47 -04:00
{
2015-10-28 14:32:42 -04:00
Mutex m = {};
2015-10-04 15:59:40 -04:00
#if defined(GB_SYSTEM_WINDOWS)
2015-10-28 14:32:42 -04:00
m.win32_mutex = CreateMutex(0, false, 0);
2015-09-27 14:03:47 -04:00
#else
2015-10-28 14:32:42 -04:00
pthread_mutex_init(&m.posix_mutex, nullptr);
2015-09-27 14:03:47 -04:00
#endif
2015-10-28 14:32:42 -04:00
return m;
2015-09-27 14:03:47 -04:00
}
2015-11-19 17:27:57 -05:00
inline void
2015-10-28 14:32:42 -04:00
destroy(Mutex* m)
2015-09-27 14:03:47 -04:00
{
2015-10-04 15:59:40 -04:00
#if defined(GB_SYSTEM_WINDOWS)
2015-10-28 14:32:42 -04:00
CloseHandle(m->win32_mutex);
2015-09-27 14:03:47 -04:00
#else
2015-10-28 14:32:42 -04:00
pthread_mutex_destroy(&m->posix_mutex);
2015-09-27 14:03:47 -04:00
#endif
}
2015-10-28 14:32:42 -04:00
2015-11-19 17:27:57 -05:00
inline void
lock(Mutex* m)
2015-09-27 14:03:47 -04:00
{
2015-10-04 15:59:40 -04:00
#if defined(GB_SYSTEM_WINDOWS)
2015-10-28 14:32:42 -04:00
WaitForSingleObject(m->win32_mutex, INFINITE);
2015-09-27 14:03:47 -04:00
#else
2015-10-28 14:32:42 -04:00
pthread_mutex_lock(&m->posix_mutex);
2015-09-27 14:03:47 -04:00
#endif
}
2015-11-19 17:27:57 -05:00
inline bool
try_lock(Mutex* m)
2015-09-27 14:03:47 -04:00
{
2015-10-04 15:59:40 -04:00
#if defined(GB_SYSTEM_WINDOWS)
2015-10-28 14:32:42 -04:00
return WaitForSingleObject(m->win32_mutex, 0) == WAIT_OBJECT_0;
2015-09-27 14:03:47 -04:00
#else
2015-10-28 14:32:42 -04:00
return pthread_mutex_trylock(&m->posix_mutex) == 0;
2015-09-27 14:03:47 -04:00
#endif
}
2015-11-19 17:27:57 -05:00
inline void
unlock(Mutex* m)
2015-09-27 14:03:47 -04:00
{
2015-10-04 15:59:40 -04:00
#if defined(GB_SYSTEM_WINDOWS)
2015-10-28 14:32:42 -04:00
ReleaseMutex(m->win32_mutex);
2015-09-27 14:03:47 -04:00
#else
2015-10-28 14:32:42 -04:00
pthread_mutex_unlock(&m->posix_mutex);
2015-09-27 14:03:47 -04:00
#endif
}
2015-10-18 07:18:54 -04:00
} // namespace mutex
2015-09-27 14:03:47 -04:00
2015-12-17 07:26:24 -05:00
2015-10-03 10:26:29 -04:00
// Atomics
namespace atomic
{
#if defined(_MSC_VER)
inline u32
2015-12-17 07:26:24 -05:00
load(Atomic32 const volatile* object)
2015-10-03 10:26:29 -04:00
{
2015-10-04 15:59:40 -04:00
return object->nonatomic;
2015-10-03 10:26:29 -04:00
}
inline void
2015-12-17 07:26:24 -05:00
store(Atomic32 volatile* object, u32 value)
2015-10-03 10:26:29 -04:00
{
2015-10-04 15:59:40 -04:00
object->nonatomic = value;
2015-10-03 10:26:29 -04:00
}
inline u32
2015-12-17 07:26:24 -05:00
compare_exchange_strong(Atomic32 volatile* object, u32 expected, u32 desired)
2015-10-03 10:26:29 -04:00
{
2015-12-17 07:26:24 -05:00
return _InterlockedCompareExchange(reinterpret_cast<long volatile*>(object), desired, expected);
2015-10-03 10:26:29 -04:00
}
inline u32
2015-12-17 07:26:24 -05:00
exchanged(Atomic32 volatile* object, u32 desired)
2015-10-03 10:26:29 -04:00
{
2015-12-17 07:26:24 -05:00
return _InterlockedExchange(reinterpret_cast<long volatile*>(object), desired);
2015-10-03 10:26:29 -04:00
}
inline u32
2015-12-17 07:26:24 -05:00
fetch_add(Atomic32 volatile* object, s32 operand)
2015-10-03 10:26:29 -04:00
{
2015-12-17 07:26:24 -05:00
return _InterlockedExchangeAdd(reinterpret_cast<long volatile*>(object), operand);
2015-10-03 10:26:29 -04:00
}
inline u32
2015-12-17 07:26:24 -05:00
fetch_and(Atomic32 volatile* object, u32 operand)
2015-10-03 10:26:29 -04:00
{
2015-12-17 07:26:24 -05:00
return _InterlockedAnd(reinterpret_cast<long volatile*>(object), operand);
2015-10-03 10:26:29 -04:00
}
inline u32
2015-12-17 07:26:24 -05:00
fetch_or_32(Atomic32 volatile* object, u32 operand)
2015-10-03 10:26:29 -04:00
{
2015-12-17 07:26:24 -05:00
return _InterlockedOr(reinterpret_cast<long volatile*>(object), operand);
2015-10-03 10:26:29 -04:00
}
inline u64
2015-12-17 07:26:24 -05:00
load(Atomic64 const volatile* object)
2015-10-03 10:26:29 -04:00
{
2015-10-04 15:59:40 -04:00
#if defined(GB_ARCH_64_BIT)
return object->nonatomic;
2015-10-03 10:26:29 -04:00
#else
// NOTE(bill): The most compatible way to get an atomic 64-bit load on x86 is with cmpxchg8b
u64 result;
__asm
{
mov esi, object;
mov ebx, eax;
mov ecx, edx;
2015-10-05 16:30:55 -04:00
lock cmpxchg8b [esi];
mov dword ptr result, eax;
mov dword ptr result[4], edx;
2015-10-03 10:26:29 -04:00
}
return result;
#endif
}
inline void
2015-12-17 07:26:24 -05:00
store(Atomic64 volatile* object, u64 value)
2015-10-03 10:26:29 -04:00
{
2015-10-04 15:59:40 -04:00
#if defined(GB_ARCH_64_BIT)
object->nonatomic = value;
2015-10-03 10:26:29 -04:00
#else
2015-11-18 19:35:56 -05:00
// NOTE(bill): The most compatible way to get an atomic 64-bit store on x86 is with cmpxchg8b
2015-10-03 10:26:29 -04:00
__asm
{
mov esi, object;
mov ebx, dword ptr value;
mov ecx, dword ptr value[4];
retry:
cmpxchg8b [esi];
jne retry;
}
#endif
}
inline u64
2015-12-17 07:26:24 -05:00
compare_exchange_strong(Atomic64 volatile* object, u64 expected, u64 desired)
2015-10-03 10:26:29 -04:00
{
2015-12-17 07:26:24 -05:00
return _InterlockedCompareExchange64(reinterpret_cast<s64 volatile*>(object), desired, expected);
2015-10-03 10:26:29 -04:00
}
inline u64
2015-12-17 07:26:24 -05:00
exchanged(Atomic64 volatile* object, u64 desired)
2015-10-03 10:26:29 -04:00
{
2015-10-04 15:59:40 -04:00
#if defined(GB_ARCH_64_BIT)
2015-12-17 07:26:24 -05:00
return _InterlockedExchange64(reinterpret_cast<s64 volatile*>(object), desired);
2015-10-03 10:26:29 -04:00
#else
u64 expected = object->nonatomic;
while (true)
{
2015-12-17 07:26:24 -05:00
u64 original = _InterlockedCompareExchange64(reinterpret_cast<s64 volatile*>(object), desired, expected);
2015-10-03 10:26:29 -04:00
if (original == expected)
return original;
expected = original;
}
#endif
}
inline u64
2015-12-17 07:26:24 -05:00
fetch_add(Atomic64 volatile* object, s64 operand)
2015-10-03 10:26:29 -04:00
{
2015-10-04 15:59:40 -04:00
#if defined(GB_ARCH_64_BIT)
2015-12-17 07:26:24 -05:00
return _InterlockedExchangeAdd64(reinterpret_cast<s64 volatile*>(object), operand);
2015-10-03 10:26:29 -04:00
#else
u64 expected = object->nonatomic;
while (true)
{
2015-12-17 07:26:24 -05:00
u64 original = _InterlockedExchange64(reinterpret_cast<s64 volatile*>(object), expected + operand, expected);
2015-10-03 10:26:29 -04:00
if (original == expected)
return original;
expected = original;
}
#endif
}
inline u64
2015-12-17 07:26:24 -05:00
fetch_and(Atomic64 volatile* object, u64 operand)
2015-10-03 10:26:29 -04:00
{
2015-10-04 15:59:40 -04:00
#if defined(GB_ARCH_64_BIT)
2015-12-17 07:26:24 -05:00
return _InterlockedAnd64(reinterpret_cast<s64 volatile*>(object), operand);
2015-10-03 10:26:29 -04:00
#else
u64 expected = object->nonatomic;
while (true)
{
2015-12-17 07:26:24 -05:00
u64 original = _InterlockedCompareExchange64(reinterpret_cast<s64 volatile*>(object), expected & operand, expected);
2015-10-03 10:26:29 -04:00
if (original == expected)
return original;
expected = original;
}
#endif
}
inline u64
2015-12-17 07:26:24 -05:00
fetch_or(Atomic64 volatile* object, u64 operand)
2015-10-03 10:26:29 -04:00
{
2015-10-04 15:59:40 -04:00
#if defined(GB_ARCH_64_BIT)
2015-12-17 07:26:24 -05:00
return _InterlockedAnd64(reinterpret_cast<s64 volatile*>(object), operand);
2015-10-03 10:26:29 -04:00
#else
u64 expected = object->nonatomic;
while (true)
{
2015-12-17 07:26:24 -05:00
u64 original = _InterlockedCompareExchange64(reinterpret_cast<s64 volatile*>(object), expected | operand, expected);
2015-10-03 10:26:29 -04:00
if (original == expected)
return original;
expected = original;
}
#endif
}
#else
#error TODO(bill): Implement atomics for this platform
#endif
} // namespace atomic
2015-12-17 07:26:24 -05:00
2015-10-28 14:32:42 -04:00
namespace semaphore
{
Semaphore
make()
{
Semaphore semaphore = {};
#if defined(GB_SYSTEM_WINDOWS)
semaphore.win32_handle = CreateSemaphore(nullptr, 0, GB_S32_MAX, nullptr);
2015-12-17 07:26:24 -05:00
2015-10-28 14:32:42 -04:00
GB_ASSERT(semaphore.win32_handle != nullptr, "CreateSemaphore: GetLastError = %d", GetLastError());
#else
semaphore.count = 0;
s32 result = pthread_cond_init(&semaphore.cond, nullptr);
GB_ASSERT(result == 0, "pthread_cond_init: errno = %d", result);
semaphore.mutex = mutex::make();
#endif
return semaphore;
}
void
destroy(Semaphore* semaphore)
{
#if defined(GB_SYSTEM_WINDOWS)
BOOL err = CloseHandle(semaphore->win32_handle);
GB_ASSERT(err != 0, "CloseHandle: GetLastError = %d", GetLastError());
#else
s32 result = pthread_cond_destroy(&semaphore->cond);
GB_ASSERT(result == 0, "pthread_cond_destroy: errno = %d", result);
mutex::destroy(&semaphore->mutex);
#endif
}
void
post(Semaphore* semaphore, u32 count)
{
#if defined(GB_SYSTEM_WINDOWS)
BOOL err = ReleaseSemaphore(semaphore->win32_handle, count, nullptr);
GB_ASSERT(err != 0, "ReleaseSemaphore: GetLastError = %d", GetLastError());
#else
mutex::lock(&semaphore->mutex);
2015-10-28 14:32:42 -04:00
for (u32 i = 0; i < count; i++)
{
s32 result = pthread_cond_signal(&semaphore->cond);
GB_ASSERT(result == 0, "pthread_cond_signal: errno = %d", result);
}
semaphore->count += count;
2015-12-02 10:06:36 -05:00
mutex::unlock(&semaphore->mutex);
2015-10-28 14:32:42 -04:00
#endif
}
void
wait(Semaphore* semaphore)
{
#if defined(GB_SYSTEM_WINDOWS)
DWORD result = WaitForSingleObject(semaphore->win32_handle, INFINITE);
GB_ASSERT(result == WAIT_OBJECT_0, "WaitForSingleObject: GetLastError = %d", GetLastError());
#else
mutex::lock(&semaphore->mutex);
2015-10-28 14:32:42 -04:00
while (count <= 0)
{
s32 result = pthread_cond_wait(&semaphore->cond, &semaphore->mutex.posix_mutex);
GB_ASSERT(result == 0, "pthread_cond_wait: errno = %d", result);
}
count--;
2015-12-02 10:06:36 -05:00
mutex::unlock(&semaphore->mutex);
2015-10-28 14:32:42 -04:00
#endif
}
} // namespace semaphore
namespace thread
{
Thread
make()
{
2015-10-31 20:45:54 -04:00
Thread t = {};
2015-10-28 14:32:42 -04:00
#if defined(GB_SYSTEM_WINDOWS)
2015-10-31 20:45:54 -04:00
t.win32_handle = INVALID_HANDLE_VALUE;
2015-10-28 14:32:42 -04:00
#else
2015-10-31 20:45:54 -04:00
t.posix_handle = 0;
2015-10-28 14:32:42 -04:00
#endif
t.function = nullptr;
t.data = nullptr;
2015-10-31 20:45:54 -04:00
t.stack_size = 0;
t.is_running = false;
t.semaphore = semaphore::make();
2015-10-28 14:32:42 -04:00
2015-10-31 20:45:54 -04:00
return t;
2015-10-28 14:32:42 -04:00
}
void
2015-10-31 20:45:54 -04:00
destroy(Thread* t)
2015-10-28 14:32:42 -04:00
{
2015-10-31 20:45:54 -04:00
if (t->is_running)
thread::join(t);
2015-10-28 14:32:42 -04:00
2015-10-31 20:45:54 -04:00
semaphore::destroy(&t->semaphore);
2015-10-28 14:32:42 -04:00
}
2015-11-18 19:35:56 -05:00
internal_linkage void
2015-10-31 20:45:54 -04:00
run(Thread* t)
2015-10-28 14:32:42 -04:00
{
2015-10-31 20:45:54 -04:00
semaphore::post(&t->semaphore);
2015-11-18 19:35:56 -05:00
t->function(t->data);
2015-10-28 14:32:42 -04:00
}
#if defined(GB_SYSTEM_WINDOWS)
2015-11-17 18:45:30 -05:00
internal_linkage DWORD WINAPI
2015-10-28 14:32:42 -04:00
thread_proc(void* arg)
{
2015-11-18 19:35:56 -05:00
thread::run(static_cast<Thread*>(arg));
return 0;
2015-10-28 14:32:42 -04:00
}
#else
2015-11-17 18:45:30 -05:00
internal_linkage void*
2015-10-28 14:32:42 -04:00
thread_proc(void* arg)
{
2015-11-18 19:35:56 -05:00
thread::run(static_cast<Thread*>(arg));
return nullptr;
2015-10-28 14:32:42 -04:00
}
#endif
void
2015-12-02 15:34:35 -05:00
start(Thread* t, Thread_Procedure* func, void* data, usize stack_size)
2015-10-28 14:32:42 -04:00
{
2015-10-31 20:45:54 -04:00
GB_ASSERT(!t->is_running);
2015-10-28 14:32:42 -04:00
GB_ASSERT(func != nullptr);
2015-10-31 20:45:54 -04:00
t->function = func;
t->data = data;
t->stack_size = stack_size;
2015-10-28 14:32:42 -04:00
#if defined(GB_SYSTEM_WINDOWS)
2015-10-31 20:45:54 -04:00
t->win32_handle = CreateThread(nullptr, stack_size, thread_proc, t, 0, nullptr);
GB_ASSERT(t->win32_handle != nullptr,
2015-10-28 14:32:42 -04:00
"CreateThread: GetLastError = %d", GetLastError());
#else
pthread_attr_t attr;
s32 result = pthread_attr_init(&attr);
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);
GB_ASSERT(result == 0, "pthread_attr_init: errno = %d", result);
2015-10-31 20:45:54 -04:00
if (t->stack_size != 0)
2015-10-28 14:32:42 -04:00
{
2015-10-31 20:45:54 -04:00
result = pthread_attr_setstacksize(&attr, t->stack_size);
2015-10-28 14:32:42 -04:00
GB_ASSERT(result == 0, "pthread_attr_setstacksize: errno = %d", result);
}
2015-10-31 20:45:54 -04:00
result = pthread_create(&t->posix_handle, &attr, thread_proc, thread);
2015-10-28 14:32:42 -04:00
GB_ASSERT(result == 0, "pthread_create: errno = %d", result);
// NOTE(bill): Free attr memory
result = pthread_attr_destroy(&attr);
GB_ASSERT(result == 0, "pthread_attr_destroy: errno = %d", result);
// NOTE(bill): So much boiler patch compared to windows.h (for once)
#endif
2015-10-31 20:45:54 -04:00
t->is_running = true;
semaphore::wait(&t->semaphore);
2015-10-28 14:32:42 -04:00
}
void
join(Thread* t)
2015-10-28 14:32:42 -04:00
{
2015-12-02 10:06:36 -05:00
if (!t->is_running) return;
2015-10-28 14:32:42 -04:00
#if defined(GB_SYSTEM_WINDOWS)
2015-10-31 20:45:54 -04:00
WaitForSingleObject(t->win32_handle, INFINITE);
CloseHandle(t->win32_handle);
t->win32_handle = INVALID_HANDLE_VALUE;
2015-10-28 14:32:42 -04:00
#else
2015-10-31 20:45:54 -04:00
int result = pthread_join(t->posix_handle, nullptr);
t->posix_handle = 0;
2015-10-28 14:32:42 -04:00
#endif
2015-10-31 20:45:54 -04:00
t->is_running = false;
2015-10-28 14:32:42 -04:00
}
2015-11-18 19:35:56 -05:00
inline bool
2015-12-17 07:26:24 -05:00
is_running(Thread const& thread)
2015-10-28 14:32:42 -04:00
{
return thread.is_running != 0;
}
2015-11-18 19:35:56 -05:00
inline u32
current_id()
{
u32 thread_id;
#if defined(GB_SYSTEM_WINDOWS)
u8* thread_local_storage = reinterpret_cast<u8*>(__readgsqword(0x30));
thread_id = *reinterpret_cast<u32*>(thread_local_storage + 0x48);
#elif defined(GB_SYSTEM_OSX) && defined(GB_ARCH_64_BIT)
asm("mov %%gs:0x00,%0" : "=r"(thread_id));
#elif defined(GB_ARCH_32_BIT)
asm("mov %%gs:0x08,%0" : "=r"(thread_id));
#elif defined(GB_ARCH_64_BIT)
asm("mov %%gs:0x10,%0" : "=r"(thread_id));
#else
#error Unsupported architecture for thread::current_id()
#endif
return thread_id;
}
2015-10-28 14:32:42 -04:00
} // namespace thread
2015-09-27 14:03:47 -04:00
namespace heap
2015-11-29 10:53:59 -05:00
{
namespace functions
{
internal_linkage void*
alloc(Allocator* a, usize size, usize align)
2015-09-27 14:03:47 -04:00
{
Heap* heap = reinterpret_cast<Heap*>(a);
2015-11-29 10:53:59 -05:00
2015-12-02 10:06:36 -05:00
if (heap->use_mutex) mutex::lock(&heap->mutex);
2015-11-29 13:04:33 -05:00
usize total = size + align - (size % align);
#if defined (GB_SYSTEM_WINDOWS)
total += sizeof(Heap::Header);
2015-11-29 13:04:33 -05:00
void* data = HeapAlloc(heap->win32_heap_handle, 0, total);
Heap::Header* h = static_cast<Heap::Header*>(data);
h->size = total;
data = (h + 1);
#else
// TODO(bill): Find a better malloc alternative for this platform
void* data = malloc(total);
#endif
2015-09-27 14:03:47 -04:00
2015-11-29 10:53:59 -05:00
heap->total_allocated_count += total;
heap->allocation_count++;
2015-09-27 14:03:47 -04:00
2015-12-02 10:06:36 -05:00
if (heap->use_mutex) mutex::unlock(&heap->mutex);
2015-11-29 13:04:33 -05:00
2015-09-27 14:03:47 -04:00
return data;
}
2015-11-29 10:53:59 -05:00
internal_linkage void
free(Allocator* a, void* ptr)
2015-09-27 14:03:47 -04:00
{
if (!ptr)
return;
Heap* heap = reinterpret_cast<Heap*>(a);
2015-11-29 10:53:59 -05:00
2015-12-02 10:06:36 -05:00
if (heap->use_mutex) mutex::lock(&heap->mutex);
2015-11-29 13:04:33 -05:00
2015-11-29 10:53:59 -05:00
heap->total_allocated_count -= allocated_size(heap, ptr);
heap->allocation_count--;
2015-09-27 14:03:47 -04:00
#if defined (GB_SYSTEM_WINDOWS)
auto* header = static_cast<Heap::Header*>(ptr) - 1;
2015-11-29 13:04:33 -05:00
HeapFree(heap->win32_heap_handle, 0, header);
#else
2015-11-29 13:04:33 -05:00
::free(ptr);
#endif
2015-11-29 13:04:33 -05:00
2015-12-02 10:06:36 -05:00
if (heap->use_mutex) mutex::unlock(&heap->mutex);
2015-09-27 14:03:47 -04:00
}
2015-11-19 17:27:57 -05:00
inline s64
2015-12-17 07:26:24 -05:00
allocated_size(Allocator* a, void const* ptr)
2015-09-27 14:03:47 -04:00
{
#if defined(GB_SYSTEM_WINDOWS)
auto* heap = reinterpret_cast<Heap*>(a);
2015-11-29 13:04:33 -05:00
2015-12-02 10:06:36 -05:00
if (heap->use_mutex) mutex::lock(&heap->mutex);
2015-12-17 07:26:24 -05:00
auto const* h = static_cast<Heap::Header const*>(ptr) - 1;
s64 result = h->size;
2015-11-29 13:04:33 -05:00
2015-12-02 10:06:36 -05:00
if (heap->use_mutex) mutex::unlock(&heap->mutex);
2015-11-29 13:04:33 -05:00
return static_cast<s64>(result);
#elif defined(GB_SYSTEM_OSX)
return static_cast<s64>(malloc_size(ptr));
#elif defined(GB_SYSTEM_LINUX)
return static_cast<s64>(malloc_usable_size(ptr));
#else
#error Implement Heap::allocated_size
#endif
2015-09-27 14:03:47 -04:00
}
2015-11-19 17:27:57 -05:00
inline s64
2015-11-29 10:53:59 -05:00
total_allocated(Allocator* a)
2015-09-27 14:03:47 -04:00
{
auto* heap = reinterpret_cast<Heap*>(a);
2015-11-29 13:04:33 -05:00
2015-12-02 10:06:36 -05:00
if (heap->use_mutex) mutex::lock(&heap->mutex);
2015-11-29 13:04:33 -05:00
s64 result = heap->total_allocated_count;
2015-12-02 10:06:36 -05:00
if (heap->use_mutex) mutex::unlock(&heap->mutex);
2015-11-29 13:04:33 -05:00
return result;
2015-09-27 14:03:47 -04:00
}
2015-11-29 10:53:59 -05:00
} // namespace functions
2015-09-27 14:03:47 -04:00
Heap
2015-11-29 13:04:33 -05:00
make(bool use_mutex)
2015-09-29 10:11:18 -04:00
{
Heap heap = {};
2015-09-29 10:11:18 -04:00
2015-11-29 13:04:33 -05:00
heap.use_mutex = use_mutex;
2015-12-02 10:06:36 -05:00
if (use_mutex) heap.mutex = mutex::make();
2015-11-29 13:04:33 -05:00
2015-11-29 10:53:59 -05:00
#if defined(GB_SYSTEM_WINDOWS)
2015-11-29 13:04:33 -05:00
heap.win32_heap_handle = HeapCreate(0, 0, 0);
2015-11-29 10:53:59 -05:00
#endif
heap.alloc = functions::alloc;
2015-12-02 10:06:36 -05:00
heap.free = functions::free;
2015-11-29 10:53:59 -05:00
heap.allocated_size = functions::allocated_size;
heap.total_allocated = functions::total_allocated;
2015-09-27 14:03:47 -04:00
2015-11-29 10:53:59 -05:00
return heap;
}
void
destroy(Heap* heap)
2015-10-03 10:26:29 -04:00
{
2015-12-02 10:06:36 -05:00
if (heap->use_mutex) mutex::destroy(&heap->mutex);
2015-11-29 13:04:33 -05:00
2015-11-29 10:53:59 -05:00
#if defined (GB_SYSTEM_WINDOWS)
2015-11-29 13:04:33 -05:00
HeapDestroy(heap->win32_heap_handle);
2015-11-29 10:53:59 -05:00
#endif
}
} // namespace heap
2015-10-03 10:26:29 -04:00
namespace arena
2015-11-29 10:53:59 -05:00
{
namespace functions
2015-10-03 10:26:29 -04:00
{
2015-11-29 10:53:59 -05:00
internal_linkage void*
alloc(Allocator* a, usize size, usize align)
{
Arena* arena = reinterpret_cast<Arena*>(a);
2015-11-29 10:53:59 -05:00
2015-10-03 10:26:29 -04:00
s64 actual_size = size + align;
2015-11-29 10:53:59 -05:00
if (arena->total_allocated_count + actual_size > arena->total_size)
{
GB_ASSERT(arena->total_allocated_count + actual_size <= arena->total_size,
"Arena has no more space for allocation");
2015-10-03 10:26:29 -04:00
return nullptr;
}
2015-09-27 14:03:47 -04:00
2015-11-29 10:53:59 -05:00
void* ptr = memory::align_forward(memory::pointer_add(arena->physical_start, arena->total_allocated_count), align);
2015-09-27 14:03:47 -04:00
2015-11-29 10:53:59 -05:00
arena->total_allocated_count += actual_size;
2015-09-27 14:03:47 -04:00
return ptr;
}
inline void free(Allocator*, void*) {} // NOTE(bill): Arenas free all at once
2015-09-29 10:11:18 -04:00
2015-12-17 07:26:24 -05:00
inline s64 allocated_size(Allocator*, void const*) { return -1; }
2015-09-27 14:03:47 -04:00
2015-11-29 10:53:59 -05:00
inline s64
total_allocated(Allocator* a)
{
return reinterpret_cast<Arena*>(a)->total_allocated_count;
2015-11-29 10:53:59 -05:00
}
} // namespace functions
2015-09-27 14:03:47 -04:00
Arena
2015-11-29 10:53:59 -05:00
make(Allocator* backing, usize size)
{
Arena arena = {};
2015-11-29 10:53:59 -05:00
arena.backing = backing;
arena.physical_start = nullptr;
arena.total_size = size;
arena.temp_count = 0;
arena.total_allocated_count = 0;
arena.physical_start = alloc(arena.backing, size);
arena.alloc = functions::alloc;
arena.free = functions::free;
2015-11-29 10:53:59 -05:00
arena.allocated_size = functions::allocated_size;
arena.total_allocated = functions::total_allocated;
return arena;
}
Arena
2015-11-29 10:53:59 -05:00
make(void* start, usize size)
{
Arena arena = {};
2015-11-29 10:53:59 -05:00
arena.backing = nullptr;
arena.physical_start = start;
arena.total_size = size;
arena.temp_count = 0;
arena.total_allocated_count = 0;
arena.alloc = functions::alloc;
arena.free = functions::free;
2015-11-29 10:53:59 -05:00
arena.allocated_size = functions::allocated_size;
arena.total_allocated = functions::total_allocated;
return arena;
}
void
destroy(Arena* arena)
{
2015-11-29 10:53:59 -05:00
if (arena->backing)
free(arena->backing, arena->physical_start);
2015-11-29 10:53:59 -05:00
clear(arena);
2015-11-29 10:53:59 -05:00
}
inline void
clear(Arena* arena)
{
GB_ASSERT(arena->temp_count == 0,
"%ld Temporary_Arena_Memory have not be cleared", arena->temp_count);
arena->total_allocated_count = 0;
}
} // namespace arena
namespace temporary_arena_memory
{
inline Temporary_Arena_Memory
make(Arena* arena)
{
Temporary_Arena_Memory tmp = {};
tmp.arena = arena;
tmp.original_count = arena->total_allocated_count;
2015-11-18 19:35:56 -05:00
arena->temp_count++;
return tmp;
}
inline void
free(Temporary_Arena_Memory tmp)
{
GB_ASSERT(total_allocated(tmp.arena) >= tmp.original_count);
tmp.arena->total_allocated_count = tmp.original_count;
GB_ASSERT(tmp.arena->temp_count > 0);
tmp.arena->temp_count--;
}
} // namespace temporary_arena_memory
2015-12-02 10:06:36 -05:00
namespace pool
2015-12-02 10:06:36 -05:00
{
namespace functions
{
internal_linkage void*
alloc(Allocator* a, usize size, usize align)
{
Pool* pool = reinterpret_cast<Pool*>(a);
2015-12-02 10:06:36 -05:00
GB_ASSERT(size == pool->block_size, "Size must match block size");
GB_ASSERT(align == pool->block_align, "Align must match block align");
GB_ASSERT(pool->free_list != nullptr, "Pool out of memory");
2015-12-02 10:06:36 -05:00
uintptr next_free = *reinterpret_cast<uintptr*>(pool->free_list);
void* ptr = pool->free_list;
pool->free_list = reinterpret_cast<void*>(next_free);
pool->total_size += pool->block_size;
return ptr;
}
internal_linkage void
free(Allocator* a, void* ptr)
{
if (!ptr) return;
Pool* pool = reinterpret_cast<Pool*>(a);
2015-12-02 10:06:36 -05:00
uintptr* next = static_cast<uintptr*>(ptr);
*next = reinterpret_cast<uintptr>(pool->free_list);
pool->free_list = ptr;
pool->total_size -= pool->block_size;
}
internal_linkage s64
2015-12-17 07:26:24 -05:00
allocated_size(Allocator*, void const*)
2015-12-02 10:06:36 -05:00
{
return -1;
}
internal_linkage s64
total_allocated(Allocator* a)
{
Pool* pool = reinterpret_cast<Pool*>(a);
2015-12-02 10:06:36 -05:00
return pool->total_size;
}
} // namespace functions
Pool
2015-12-02 10:06:36 -05:00
make(Allocator* backing, usize num_blocks, usize block_size, usize block_align)
{
Pool pool = {};
2015-12-02 10:06:36 -05:00
pool.backing = backing;
pool.block_size = block_size;
pool.block_align = block_align;
usize actual_block_size = block_size + block_align;
usize pool_size = num_blocks * actual_block_size;
u8* data = static_cast<u8*>(alloc(backing, pool_size, block_align));
// Init intrusive freelist
u8* curr = data;
for (usize block_index = 0; block_index < num_blocks-1; block_index++)
{
uintptr* next = reinterpret_cast<uintptr*>(curr);
*next = reinterpret_cast<uintptr>(curr) + actual_block_size;
curr += actual_block_size;
}
uintptr* end = reinterpret_cast<uintptr*>(curr);
*end = reinterpret_cast<uintptr>(nullptr);
pool.physical_start = data;
pool.free_list = data;
// Set functions pointers
pool.alloc = functions::alloc;
pool.free = functions::free;
pool.allocated_size = functions::allocated_size;
pool.total_allocated = functions::total_allocated;
return pool;
}
inline void
destroy(Pool* pool)
2015-12-02 10:06:36 -05:00
{
free(pool->backing, pool->physical_start);
}
} // namespace pool
2015-12-02 10:06:36 -05:00
2015-11-19 17:27:57 -05:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Memory //
// //
2015-11-19 17:27:57 -05:00
////////////////////////////////
namespace memory
{
inline void*
align_forward(void* ptr, usize align)
{
GB_ASSERT(GB_IS_POWER_OF_TWO(align),
"Alignment must be a power of two and not zero -- %llu", align);
uintptr p = uintptr(ptr);
const usize modulo = p % align;
2015-12-02 10:06:36 -05:00
if (modulo) p += (align - modulo);
2015-11-19 17:27:57 -05:00
return reinterpret_cast<void*>(p);
}
inline void*
pointer_add(void* ptr, usize bytes)
{
return static_cast<void*>(static_cast<u8*>(ptr) + bytes);
}
2015-12-17 07:26:24 -05:00
inline void const*
pointer_add(void const* ptr, usize bytes)
2015-11-19 17:27:57 -05:00
{
2015-12-17 07:26:24 -05:00
return static_cast<void const*>(static_cast<u8 const*>(ptr) + bytes);
2015-11-19 17:27:57 -05:00
}
inline void*
pointer_sub(void* ptr, usize bytes)
{
return static_cast<void*>(static_cast<u8*>(ptr) - bytes);
}
2015-12-17 07:26:24 -05:00
inline void const*
pointer_sub(void const* ptr, usize bytes)
2015-11-19 17:27:57 -05:00
{
2015-12-17 07:26:24 -05:00
return static_cast<void const*>(static_cast<u8 const*>(ptr) - bytes);
2015-11-19 17:27:57 -05:00
}
GB_FORCE_INLINE void*
2015-11-29 13:04:33 -05:00
set(void* ptr, usize bytes, u8 value)
2015-11-19 17:27:57 -05:00
{
return memset(ptr, value, bytes);
}
GB_FORCE_INLINE void*
zero(void* ptr, usize bytes)
{
2015-11-29 13:04:33 -05:00
return memory::set(ptr, bytes, 0);
2015-11-19 17:27:57 -05:00
}
GB_FORCE_INLINE void*
2015-12-17 07:26:24 -05:00
copy(void const* src, usize bytes, void* dest)
2015-11-19 17:27:57 -05:00
{
return memcpy(dest, src, bytes);
}
GB_FORCE_INLINE void*
2015-12-17 07:26:24 -05:00
move(void const* src, usize bytes, void* dest)
2015-11-19 17:27:57 -05:00
{
return memmove(dest, src, bytes);
}
GB_FORCE_INLINE bool
2015-12-17 07:26:24 -05:00
equals(void const* a, void const* b, usize bytes)
2015-11-19 17:27:57 -05:00
{
return (memcmp(a, b, bytes) == 0);
}
} // namespace memory
inline void*
alloc(Allocator* a, usize size, usize align)
{
GB_ASSERT(a != nullptr);
return a->alloc(a, size, align);
}
inline void
free(Allocator* a, void* ptr)
{
GB_ASSERT(a != nullptr);
2015-12-02 10:06:36 -05:00
if (ptr) a->free(a, ptr);
}
inline s64
2015-12-17 07:26:24 -05:00
allocated_size(Allocator* a, void const* ptr)
{
GB_ASSERT(a != nullptr);
return a->allocated_size(a, ptr);
}
inline s64
total_allocated(Allocator* a)
{
GB_ASSERT(a != nullptr);
return a->total_allocated(a);
}
2015-09-28 16:58:33 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// String //
// //
2015-09-28 16:58:33 -04:00
////////////////////////////////
2015-10-18 07:18:54 -04:00
namespace string
{
2015-11-19 17:27:57 -05:00
inline String
2015-12-17 07:26:24 -05:00
make(Allocator* a, char const* str)
2015-09-28 16:58:33 -04:00
{
2015-10-18 07:18:54 -04:00
return string::make(a, str, (string::Size)strlen(str));
2015-09-28 16:58:33 -04:00
}
2015-11-19 17:27:57 -05:00
String
2015-12-17 07:26:24 -05:00
make(Allocator* a, void const* init_str, Size len)
2015-09-28 16:58:33 -04:00
{
2015-10-18 07:18:54 -04:00
usize header_size = sizeof(string::Header);
2015-09-28 16:58:33 -04:00
void* ptr = alloc(a, header_size + len + 1);
2015-12-02 10:06:36 -05:00
if (!ptr) return nullptr;
2015-12-02 10:06:36 -05:00
if (!init_str) memory::zero(ptr, header_size + len + 1);
2015-09-28 16:58:33 -04:00
2015-10-05 16:30:55 -04:00
String str = static_cast<char*>(ptr) + header_size;
2015-10-18 07:18:54 -04:00
string::Header* header = string::header(str);
header->allocator = a;
header->length = len;
header->capacity = len;
2015-09-28 16:58:33 -04:00
if (len && init_str)
memory::copy(init_str, len, str);
2015-09-28 16:58:33 -04:00
str[len] = '\0';
return str;
}
2015-11-29 13:04:33 -05:00
inline void
2015-11-19 17:27:57 -05:00
free(String str)
2015-09-28 16:58:33 -04:00
{
2015-12-02 10:06:36 -05:00
if (str == nullptr) return;
2015-11-28 17:55:24 -05:00
2015-10-18 07:18:54 -04:00
string::Header* h = string::header(str);
2015-11-28 17:55:24 -05:00
2015-12-02 10:06:36 -05:00
if (h->allocator) free(h->allocator, h);
2015-09-28 16:58:33 -04:00
}
2015-11-19 17:27:57 -05:00
inline String
2015-12-17 07:26:24 -05:00
duplicate(Allocator* a, String const str)
2015-09-28 16:58:33 -04:00
{
2015-10-18 07:18:54 -04:00
return string::make(a, str, string::length(str));
2015-09-28 16:58:33 -04:00
}
2015-11-19 17:27:57 -05:00
inline Size
2015-12-17 07:26:24 -05:00
length(String const str)
2015-09-28 16:58:33 -04:00
{
return string::header(str)->length;
2015-09-28 16:58:33 -04:00
}
2015-11-19 17:27:57 -05:00
inline Size
2015-12-17 07:26:24 -05:00
capacity(String const str)
2015-09-28 16:58:33 -04:00
{
return string::header(str)->capacity;
2015-09-28 16:58:33 -04:00
}
2015-11-19 17:27:57 -05:00
inline Size
2015-12-17 07:26:24 -05:00
available_space(String const str)
2015-09-28 16:58:33 -04:00
{
2015-10-18 07:18:54 -04:00
string::Header* h = string::header(str);
if (h->capacity > h->length)
return h->capacity - h->length;
2015-09-28 16:58:33 -04:00
return 0;
}
2015-11-19 17:27:57 -05:00
inline void
clear(String str)
2015-09-28 16:58:33 -04:00
{
string::header(str)->length = 0;
2015-09-28 16:58:33 -04:00
str[0] = '\0';
}
2015-11-19 17:27:57 -05:00
void
append(String* str, char c)
{
Size curr_len = string::length(*str);
string::make_space_for(str, 1);
2015-12-02 10:06:36 -05:00
if (str == nullptr) return;
(*str)[curr_len] = c;
(*str)[curr_len + 1] = '\0';
string::header(*str)->length = curr_len + 1;
}
2015-11-19 17:27:57 -05:00
inline void
2015-12-17 07:26:24 -05:00
append(String* str, String const other)
2015-09-28 16:58:33 -04:00
{
2015-10-18 07:18:54 -04:00
string::append(str, other, string::length(other));
2015-09-28 16:58:33 -04:00
}
2015-11-19 17:27:57 -05:00
inline void
2015-12-17 07:26:24 -05:00
append_cstring(String* str, char const* other)
2015-09-28 16:58:33 -04:00
{
2015-10-18 07:18:54 -04:00
string::append(str, other, (Size)strlen(other));
2015-09-28 16:58:33 -04:00
}
2015-11-19 17:27:57 -05:00
void
2015-12-17 07:26:24 -05:00
append(String* str, void const* other, Size other_len)
2015-09-28 16:58:33 -04:00
{
Size curr_len = string::length(*str);
2015-09-28 16:58:33 -04:00
2015-10-18 07:18:54 -04:00
string::make_space_for(str, other_len);
2015-09-28 16:58:33 -04:00
if (str == nullptr)
return;
memory::copy(other, other_len, (*str) + curr_len);
(*str)[curr_len + other_len] = '\0';
string::header(*str)->length = curr_len + other_len;
2015-09-28 16:58:33 -04:00
}
namespace impl
{
// NOTE(bill): ptr _must_ be allocated with Allocator* a
2015-11-17 18:45:30 -05:00
internal_linkage inline void*
string_realloc(Allocator* a, void* ptr, usize old_size, usize new_size)
2015-09-28 16:58:33 -04:00
{
if (!ptr)
return alloc(a, new_size);
if (new_size < old_size)
new_size = old_size;
if (old_size == new_size)
return ptr;
void* new_ptr = alloc(a, new_size);
if (!new_ptr)
return nullptr;
memory::copy(ptr, old_size, new_ptr);
2015-09-28 16:58:33 -04:00
free(a, ptr);
2015-09-28 16:58:33 -04:00
return new_ptr;
}
} // namespace impl
2015-10-28 14:32:42 -04:00
void
make_space_for(String* str, Size add_len)
2015-09-28 16:58:33 -04:00
{
Size len = string::length(*str);
2015-10-18 07:18:54 -04:00
Size new_len = len + add_len;
2015-09-28 16:58:33 -04:00
Size available = string::available_space(*str);
2015-09-28 16:58:33 -04:00
if (available >= add_len) // Return if there is enough space left
return;
2015-11-29 10:53:59 -05:00
void* ptr = string::header(*str);
usize old_size = sizeof(string::Header) + string::length(*str) + 1;
2015-10-18 07:18:54 -04:00
usize new_size = sizeof(string::Header) + new_len + 1;
2015-09-28 16:58:33 -04:00
Allocator* a = string::header(*str)->allocator;
void* new_ptr = impl::string_realloc(a, ptr, old_size, new_size);
2015-09-28 16:58:33 -04:00
if (new_ptr == nullptr)
return;
2015-11-29 10:53:59 -05:00
string::Header* header = static_cast<string::Header*>(new_ptr);
header->allocator = a;
header->length = len;
header->capacity = new_len;
2015-11-29 10:53:59 -05:00
*str = reinterpret_cast<String>(header + 1);
2015-09-28 16:58:33 -04:00
}
2015-10-28 14:32:42 -04:00
usize
2015-12-17 07:26:24 -05:00
allocation_size(String const str)
2015-09-28 16:58:33 -04:00
{
2015-10-18 07:18:54 -04:00
Size cap = string::capacity(str);
return sizeof(string::Header) + cap;
2015-09-28 16:58:33 -04:00
}
2015-10-28 14:32:42 -04:00
bool
2015-12-17 07:26:24 -05:00
equals(String const lhs, String const rhs)
2015-09-28 16:58:33 -04:00
{
2015-10-18 07:18:54 -04:00
Size lhs_len = string::length(lhs);
Size rhs_len = string::length(rhs);
2015-09-28 16:58:33 -04:00
if (lhs_len != rhs_len)
return false;
2015-10-18 07:18:54 -04:00
for (Size i = 0; i < lhs_len; i++)
2015-09-28 16:58:33 -04:00
{
if (lhs[i] != rhs[i])
return false;
}
return true;
}
2015-10-28 14:32:42 -04:00
int
2015-12-17 07:26:24 -05:00
compare(String const lhs, String const rhs) // NOTE(bill): three-way comparison
2015-10-28 14:32:42 -04:00
{
// Treat as cstring
2015-12-17 07:26:24 -05:00
char const* str1 = lhs;
char const* str2 = rhs;
2015-10-28 14:32:42 -04:00
int s1;
int s2;
do
{
s1 = *str1++;
s2 = *str2++;
if (s1 == 0)
break;
}
while (s1 == s2);
return (s1 < s2) ? -1 : (s1 > s2);
}
void
2015-12-17 07:26:24 -05:00
trim(String* str, char const* cut_set)
2015-09-28 16:58:33 -04:00
{
char* start;
char* end;
char* start_pos;
char* end_pos;
2015-10-28 14:32:42 -04:00
start_pos = start = *str;
end_pos = end = *str + string::length(*str) - 1;
2015-09-28 16:58:33 -04:00
while (start_pos <= end && strchr(cut_set, *start_pos))
start_pos++;
while (end_pos > start_pos && strchr(cut_set, *end_pos))
end_pos--;
2015-10-18 07:18:54 -04:00
Size len = static_cast<Size>((start_pos > end_pos) ? 0 : ((end_pos - start_pos)+1));
2015-09-28 16:58:33 -04:00
2015-10-28 14:32:42 -04:00
if (*str != start_pos)
memory::move(start_pos, len, *str);
2015-10-28 14:32:42 -04:00
(*str)[len] = '\0';
2015-09-28 16:58:33 -04:00
string::header(*str)->length = len;
2015-09-28 16:58:33 -04:00
}
inline void
trim_space(String* str)
{
trim(str, " \n\r\t\v\f");
}
2015-10-18 07:18:54 -04:00
} // namespace string
2015-09-28 16:58:33 -04:00
2015-10-28 14:32:42 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Hash //
// //
2015-10-28 14:32:42 -04:00
////////////////////////////////
namespace hash
{
u32
2015-12-17 07:26:24 -05:00
adler32(void const* key, u32 num_bytes)
2015-10-28 14:32:42 -04:00
{
const u32 MOD_ADLER = 65521;
u32 a = 1;
u32 b = 0;
2015-12-17 07:26:24 -05:00
u8 const* bytes = static_cast<u8 const*>(key);
2015-10-28 14:32:42 -04:00
for (u32 i = 0; i < num_bytes; i++)
{
a = (a + bytes[i]) % MOD_ADLER;
b = (b + a) % MOD_ADLER;
}
return (b << 16) | a;
}
2015-11-17 18:45:30 -05:00
global_variable const u32 GB_CRC32_TABLE[256] = {
2015-10-28 14:32:42 -04:00
0x00000000, 0x77073096, 0xee0e612c, 0x990951ba,
2015-09-28 16:58:33 -04:00
0x076dc419, 0x706af48f, 0xe963a535, 0x9e6495a3,
0x0edb8832, 0x79dcb8a4, 0xe0d5e91e, 0x97d2d988,
0x09b64c2b, 0x7eb17cbd, 0xe7b82d07, 0x90bf1d91,
0x1db71064, 0x6ab020f2, 0xf3b97148, 0x84be41de,
0x1adad47d, 0x6ddde4eb, 0xf4d4b551, 0x83d385c7,
0x136c9856, 0x646ba8c0, 0xfd62f97a, 0x8a65c9ec,
0x14015c4f, 0x63066cd9, 0xfa0f3d63, 0x8d080df5,
0x3b6e20c8, 0x4c69105e, 0xd56041e4, 0xa2677172,
0x3c03e4d1, 0x4b04d447, 0xd20d85fd, 0xa50ab56b,
0x35b5a8fa, 0x42b2986c, 0xdbbbc9d6, 0xacbcf940,
0x32d86ce3, 0x45df5c75, 0xdcd60dcf, 0xabd13d59,
0x26d930ac, 0x51de003a, 0xc8d75180, 0xbfd06116,
0x21b4f4b5, 0x56b3c423, 0xcfba9599, 0xb8bda50f,
0x2802b89e, 0x5f058808, 0xc60cd9b2, 0xb10be924,
0x2f6f7c87, 0x58684c11, 0xc1611dab, 0xb6662d3d,
0x76dc4190, 0x01db7106, 0x98d220bc, 0xefd5102a,
0x71b18589, 0x06b6b51f, 0x9fbfe4a5, 0xe8b8d433,
0x7807c9a2, 0x0f00f934, 0x9609a88e, 0xe10e9818,
0x7f6a0dbb, 0x086d3d2d, 0x91646c97, 0xe6635c01,
0x6b6b51f4, 0x1c6c6162, 0x856530d8, 0xf262004e,
0x6c0695ed, 0x1b01a57b, 0x8208f4c1, 0xf50fc457,
0x65b0d9c6, 0x12b7e950, 0x8bbeb8ea, 0xfcb9887c,
0x62dd1ddf, 0x15da2d49, 0x8cd37cf3, 0xfbd44c65,
0x4db26158, 0x3ab551ce, 0xa3bc0074, 0xd4bb30e2,
0x4adfa541, 0x3dd895d7, 0xa4d1c46d, 0xd3d6f4fb,
0x4369e96a, 0x346ed9fc, 0xad678846, 0xda60b8d0,
0x44042d73, 0x33031de5, 0xaa0a4c5f, 0xdd0d7cc9,
0x5005713c, 0x270241aa, 0xbe0b1010, 0xc90c2086,
0x5768b525, 0x206f85b3, 0xb966d409, 0xce61e49f,
0x5edef90e, 0x29d9c998, 0xb0d09822, 0xc7d7a8b4,
0x59b33d17, 0x2eb40d81, 0xb7bd5c3b, 0xc0ba6cad,
0xedb88320, 0x9abfb3b6, 0x03b6e20c, 0x74b1d29a,
0xead54739, 0x9dd277af, 0x04db2615, 0x73dc1683,
0xe3630b12, 0x94643b84, 0x0d6d6a3e, 0x7a6a5aa8,
0xe40ecf0b, 0x9309ff9d, 0x0a00ae27, 0x7d079eb1,
0xf00f9344, 0x8708a3d2, 0x1e01f268, 0x6906c2fe,
0xf762575d, 0x806567cb, 0x196c3671, 0x6e6b06e7,
0xfed41b76, 0x89d32be0, 0x10da7a5a, 0x67dd4acc,
0xf9b9df6f, 0x8ebeeff9, 0x17b7be43, 0x60b08ed5,
0xd6d6a3e8, 0xa1d1937e, 0x38d8c2c4, 0x4fdff252,
0xd1bb67f1, 0xa6bc5767, 0x3fb506dd, 0x48b2364b,
0xd80d2bda, 0xaf0a1b4c, 0x36034af6, 0x41047a60,
0xdf60efc3, 0xa867df55, 0x316e8eef, 0x4669be79,
0xcb61b38c, 0xbc66831a, 0x256fd2a0, 0x5268e236,
0xcc0c7795, 0xbb0b4703, 0x220216b9, 0x5505262f,
0xc5ba3bbe, 0xb2bd0b28, 0x2bb45a92, 0x5cb36a04,
0xc2d7ffa7, 0xb5d0cf31, 0x2cd99e8b, 0x5bdeae1d,
0x9b64c2b0, 0xec63f226, 0x756aa39c, 0x026d930a,
0x9c0906a9, 0xeb0e363f, 0x72076785, 0x05005713,
0x95bf4a82, 0xe2b87a14, 0x7bb12bae, 0x0cb61b38,
0x92d28e9b, 0xe5d5be0d, 0x7cdcefb7, 0x0bdbdf21,
0x86d3d2d4, 0xf1d4e242, 0x68ddb3f8, 0x1fda836e,
0x81be16cd, 0xf6b9265b, 0x6fb077e1, 0x18b74777,
0x88085ae6, 0xff0f6a70, 0x66063bca, 0x11010b5c,
0x8f659eff, 0xf862ae69, 0x616bffd3, 0x166ccf45,
0xa00ae278, 0xd70dd2ee, 0x4e048354, 0x3903b3c2,
0xa7672661, 0xd06016f7, 0x4969474d, 0x3e6e77db,
0xaed16a4a, 0xd9d65adc, 0x40df0b66, 0x37d83bf0,
0xa9bcae53, 0xdebb9ec5, 0x47b2cf7f, 0x30b5ffe9,
0xbdbdf21c, 0xcabac28a, 0x53b39330, 0x24b4a3a6,
0xbad03605, 0xcdd70693, 0x54de5729, 0x23d967bf,
0xb3667a2e, 0xc4614ab8, 0x5d681b02, 0x2a6f2b94,
0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d,
};
2015-11-17 18:45:30 -05:00
global_variable const u64 GB_CRC64_TABLE[256] = {
2015-09-28 16:58:33 -04:00
0x0000000000000000ull, 0x42F0E1EBA9EA3693ull, 0x85E1C3D753D46D26ull, 0xC711223CFA3E5BB5ull,
0x493366450E42ECDFull, 0x0BC387AEA7A8DA4Cull, 0xCCD2A5925D9681F9ull, 0x8E224479F47CB76Aull,
0x9266CC8A1C85D9BEull, 0xD0962D61B56FEF2Dull, 0x17870F5D4F51B498ull, 0x5577EEB6E6BB820Bull,
0xDB55AACF12C73561ull, 0x99A54B24BB2D03F2ull, 0x5EB4691841135847ull, 0x1C4488F3E8F96ED4ull,
0x663D78FF90E185EFull, 0x24CD9914390BB37Cull, 0xE3DCBB28C335E8C9ull, 0xA12C5AC36ADFDE5Aull,
0x2F0E1EBA9EA36930ull, 0x6DFEFF5137495FA3ull, 0xAAEFDD6DCD770416ull, 0xE81F3C86649D3285ull,
0xF45BB4758C645C51ull, 0xB6AB559E258E6AC2ull, 0x71BA77A2DFB03177ull, 0x334A9649765A07E4ull,
0xBD68D2308226B08Eull, 0xFF9833DB2BCC861Dull, 0x388911E7D1F2DDA8ull, 0x7A79F00C7818EB3Bull,
0xCC7AF1FF21C30BDEull, 0x8E8A101488293D4Dull, 0x499B3228721766F8ull, 0x0B6BD3C3DBFD506Bull,
0x854997BA2F81E701ull, 0xC7B97651866BD192ull, 0x00A8546D7C558A27ull, 0x4258B586D5BFBCB4ull,
0x5E1C3D753D46D260ull, 0x1CECDC9E94ACE4F3ull, 0xDBFDFEA26E92BF46ull, 0x990D1F49C77889D5ull,
0x172F5B3033043EBFull, 0x55DFBADB9AEE082Cull, 0x92CE98E760D05399ull, 0xD03E790CC93A650Aull,
0xAA478900B1228E31ull, 0xE8B768EB18C8B8A2ull, 0x2FA64AD7E2F6E317ull, 0x6D56AB3C4B1CD584ull,
0xE374EF45BF6062EEull, 0xA1840EAE168A547Dull, 0x66952C92ECB40FC8ull, 0x2465CD79455E395Bull,
0x3821458AADA7578Full, 0x7AD1A461044D611Cull, 0xBDC0865DFE733AA9ull, 0xFF3067B657990C3Aull,
0x711223CFA3E5BB50ull, 0x33E2C2240A0F8DC3ull, 0xF4F3E018F031D676ull, 0xB60301F359DBE0E5ull,
0xDA050215EA6C212Full, 0x98F5E3FE438617BCull, 0x5FE4C1C2B9B84C09ull, 0x1D14202910527A9Aull,
0x93366450E42ECDF0ull, 0xD1C685BB4DC4FB63ull, 0x16D7A787B7FAA0D6ull, 0x5427466C1E109645ull,
0x4863CE9FF6E9F891ull, 0x0A932F745F03CE02ull, 0xCD820D48A53D95B7ull, 0x8F72ECA30CD7A324ull,
0x0150A8DAF8AB144Eull, 0x43A04931514122DDull, 0x84B16B0DAB7F7968ull, 0xC6418AE602954FFBull,
0xBC387AEA7A8DA4C0ull, 0xFEC89B01D3679253ull, 0x39D9B93D2959C9E6ull, 0x7B2958D680B3FF75ull,
0xF50B1CAF74CF481Full, 0xB7FBFD44DD257E8Cull, 0x70EADF78271B2539ull, 0x321A3E938EF113AAull,
0x2E5EB66066087D7Eull, 0x6CAE578BCFE24BEDull, 0xABBF75B735DC1058ull, 0xE94F945C9C3626CBull,
0x676DD025684A91A1ull, 0x259D31CEC1A0A732ull, 0xE28C13F23B9EFC87ull, 0xA07CF2199274CA14ull,
0x167FF3EACBAF2AF1ull, 0x548F120162451C62ull, 0x939E303D987B47D7ull, 0xD16ED1D631917144ull,
0x5F4C95AFC5EDC62Eull, 0x1DBC74446C07F0BDull, 0xDAAD56789639AB08ull, 0x985DB7933FD39D9Bull,
0x84193F60D72AF34Full, 0xC6E9DE8B7EC0C5DCull, 0x01F8FCB784FE9E69ull, 0x43081D5C2D14A8FAull,
0xCD2A5925D9681F90ull, 0x8FDAB8CE70822903ull, 0x48CB9AF28ABC72B6ull, 0x0A3B7B1923564425ull,
0x70428B155B4EAF1Eull, 0x32B26AFEF2A4998Dull, 0xF5A348C2089AC238ull, 0xB753A929A170F4ABull,
0x3971ED50550C43C1ull, 0x7B810CBBFCE67552ull, 0xBC902E8706D82EE7ull, 0xFE60CF6CAF321874ull,
0xE224479F47CB76A0ull, 0xA0D4A674EE214033ull, 0x67C58448141F1B86ull, 0x253565A3BDF52D15ull,
0xAB1721DA49899A7Full, 0xE9E7C031E063ACECull, 0x2EF6E20D1A5DF759ull, 0x6C0603E6B3B7C1CAull,
0xF6FAE5C07D3274CDull, 0xB40A042BD4D8425Eull, 0x731B26172EE619EBull, 0x31EBC7FC870C2F78ull,
0xBFC9838573709812ull, 0xFD39626EDA9AAE81ull, 0x3A28405220A4F534ull, 0x78D8A1B9894EC3A7ull,
0x649C294A61B7AD73ull, 0x266CC8A1C85D9BE0ull, 0xE17DEA9D3263C055ull, 0xA38D0B769B89F6C6ull,
0x2DAF4F0F6FF541ACull, 0x6F5FAEE4C61F773Full, 0xA84E8CD83C212C8Aull, 0xEABE6D3395CB1A19ull,
0x90C79D3FEDD3F122ull, 0xD2377CD44439C7B1ull, 0x15265EE8BE079C04ull, 0x57D6BF0317EDAA97ull,
0xD9F4FB7AE3911DFDull, 0x9B041A914A7B2B6Eull, 0x5C1538ADB04570DBull, 0x1EE5D94619AF4648ull,
0x02A151B5F156289Cull, 0x4051B05E58BC1E0Full, 0x87409262A28245BAull, 0xC5B073890B687329ull,
0x4B9237F0FF14C443ull, 0x0962D61B56FEF2D0ull, 0xCE73F427ACC0A965ull, 0x8C8315CC052A9FF6ull,
0x3A80143F5CF17F13ull, 0x7870F5D4F51B4980ull, 0xBF61D7E80F251235ull, 0xFD913603A6CF24A6ull,
0x73B3727A52B393CCull, 0x31439391FB59A55Full, 0xF652B1AD0167FEEAull, 0xB4A25046A88DC879ull,
0xA8E6D8B54074A6ADull, 0xEA16395EE99E903Eull, 0x2D071B6213A0CB8Bull, 0x6FF7FA89BA4AFD18ull,
0xE1D5BEF04E364A72ull, 0xA3255F1BE7DC7CE1ull, 0x64347D271DE22754ull, 0x26C49CCCB40811C7ull,
0x5CBD6CC0CC10FAFCull, 0x1E4D8D2B65FACC6Full, 0xD95CAF179FC497DAull, 0x9BAC4EFC362EA149ull,
0x158E0A85C2521623ull, 0x577EEB6E6BB820B0ull, 0x906FC95291867B05ull, 0xD29F28B9386C4D96ull,
0xCEDBA04AD0952342ull, 0x8C2B41A1797F15D1ull, 0x4B3A639D83414E64ull, 0x09CA82762AAB78F7ull,
0x87E8C60FDED7CF9Dull, 0xC51827E4773DF90Eull, 0x020905D88D03A2BBull, 0x40F9E43324E99428ull,
0x2CFFE7D5975E55E2ull, 0x6E0F063E3EB46371ull, 0xA91E2402C48A38C4ull, 0xEBEEC5E96D600E57ull,
0x65CC8190991CB93Dull, 0x273C607B30F68FAEull, 0xE02D4247CAC8D41Bull, 0xA2DDA3AC6322E288ull,
0xBE992B5F8BDB8C5Cull, 0xFC69CAB42231BACFull, 0x3B78E888D80FE17Aull, 0x7988096371E5D7E9ull,
0xF7AA4D1A85996083ull, 0xB55AACF12C735610ull, 0x724B8ECDD64D0DA5ull, 0x30BB6F267FA73B36ull,
0x4AC29F2A07BFD00Dull, 0x08327EC1AE55E69Eull, 0xCF235CFD546BBD2Bull, 0x8DD3BD16FD818BB8ull,
0x03F1F96F09FD3CD2ull, 0x41011884A0170A41ull, 0x86103AB85A2951F4ull, 0xC4E0DB53F3C36767ull,
0xD8A453A01B3A09B3ull, 0x9A54B24BB2D03F20ull, 0x5D45907748EE6495ull, 0x1FB5719CE1045206ull,
0x919735E51578E56Cull, 0xD367D40EBC92D3FFull, 0x1476F63246AC884Aull, 0x568617D9EF46BED9ull,
0xE085162AB69D5E3Cull, 0xA275F7C11F7768AFull, 0x6564D5FDE549331Aull, 0x279434164CA30589ull,
0xA9B6706FB8DFB2E3ull, 0xEB46918411358470ull, 0x2C57B3B8EB0BDFC5ull, 0x6EA7525342E1E956ull,
0x72E3DAA0AA188782ull, 0x30133B4B03F2B111ull, 0xF7021977F9CCEAA4ull, 0xB5F2F89C5026DC37ull,
0x3BD0BCE5A45A6B5Dull, 0x79205D0E0DB05DCEull, 0xBE317F32F78E067Bull, 0xFCC19ED95E6430E8ull,
0x86B86ED5267CDBD3ull, 0xC4488F3E8F96ED40ull, 0x0359AD0275A8B6F5ull, 0x41A94CE9DC428066ull,
0xCF8B0890283E370Cull, 0x8D7BE97B81D4019Full, 0x4A6ACB477BEA5A2Aull, 0x089A2AACD2006CB9ull,
0x14DEA25F3AF9026Dull, 0x562E43B4931334FEull, 0x913F6188692D6F4Bull, 0xD3CF8063C0C759D8ull,
0x5DEDC41A34BBEEB2ull, 0x1F1D25F19D51D821ull, 0xD80C07CD676F8394ull, 0x9AFCE626CE85B507ull,
};
2015-10-28 14:32:42 -04:00
u32
2015-12-17 07:26:24 -05:00
crc32(void const* key, u32 num_bytes)
2015-09-28 16:58:33 -04:00
{
2015-10-05 16:30:55 -04:00
u32 result = static_cast<u32>(~0);
2015-12-17 07:26:24 -05:00
u8 const* c = reinterpret_cast<u8 const*>(key);
2015-12-02 10:06:36 -05:00
2015-09-28 18:57:43 -04:00
for (u32 remaining = num_bytes; remaining--; c++)
2015-09-28 16:58:33 -04:00
result = (result >> 8) ^ (GB_CRC32_TABLE[(result ^ *c) & 0xff]);
return ~result;
}
2015-10-28 14:32:42 -04:00
u64
2015-12-17 07:26:24 -05:00
crc64(void const* key, usize num_bytes)
2015-09-28 16:58:33 -04:00
{
2015-10-28 14:32:42 -04:00
u64 result = static_cast<u64>(~0);
2015-12-17 07:26:24 -05:00
u8 const* c = reinterpret_cast<u8 const*>(key);
2015-09-28 16:58:33 -04:00
for (usize remaining = num_bytes; remaining--; c++)
result = (result >> 8) ^ (GB_CRC64_TABLE[(result ^ *c) & 0xff]);
return ~result;
}
inline u32
2015-12-17 07:26:24 -05:00
fnv32(void const* key, usize num_bytes)
{
u32 h = 0x811c9dc5;
2015-12-17 07:26:24 -05:00
u8 const* buffer = static_cast<u8 const*>(key);
2015-09-28 16:58:33 -04:00
for (usize i = 0; i < num_bytes; i++)
{
h = (h * 0x01000193) ^ buffer[i];
}
2015-09-28 16:58:33 -04:00
return h;
}
2015-09-28 16:58:33 -04:00
inline u64
2015-12-17 07:26:24 -05:00
fnv64(void const* key, usize num_bytes)
{
u64 h = 0xcbf29ce484222325ull;
2015-12-17 07:26:24 -05:00
u8 const* buffer = static_cast<u8 const*>(key);
2015-09-28 16:58:33 -04:00
for (usize i = 0; i < num_bytes; i++)
{
h = (h * 0x100000001b3ll) ^ buffer[i];
}
2015-09-28 16:58:33 -04:00
return h;
}
2015-09-28 16:58:33 -04:00
inline u32
2015-12-17 07:26:24 -05:00
fnv32a(void const* key, usize num_bytes)
{
u32 h = 0x811c9dc5;
2015-12-17 07:26:24 -05:00
u8 const* buffer = static_cast<u8 const*>(key);
2015-09-28 16:58:33 -04:00
for (usize i = 0; i < num_bytes; i++)
{
h = (h ^ buffer[i]) * 0x01000193;
}
return h;
}
inline u64
2015-12-17 07:26:24 -05:00
fnv64a(void const* key, usize num_bytes)
{
u64 h = 0xcbf29ce484222325ull;
2015-12-17 07:26:24 -05:00
u8 const* buffer = static_cast<u8 const*>(key);
for (usize i = 0; i < num_bytes; i++)
{
h = (h ^ buffer[i]) * 0x100000001b3ll;
}
return h;
}
2015-09-28 16:58:33 -04:00
2015-10-28 14:32:42 -04:00
u32
2015-12-17 07:26:24 -05:00
murmur32(void const* key, u32 num_bytes, u32 seed)
2015-09-28 16:58:33 -04:00
{
const u32 c1 = 0xcc9e2d51;
const u32 c2 = 0x1b873593;
const u32 r1 = 15;
const u32 r2 = 13;
const u32 m = 5;
const u32 n = 0xe6546b64;
2015-09-28 16:58:33 -04:00
u32 hash = seed;
const usize nblocks = num_bytes / 4;
2015-10-05 16:30:55 -04:00
const u32* blocks = static_cast<const u32*>(key);
2015-09-28 16:58:33 -04:00
for (usize i = 0; i < nblocks; i++) {
u32 k = blocks[i];
k *= c1;
k = (k << r1) | (k >> (32 - r1));
k *= c2;
hash ^= k;
hash = ((hash << r2) | (hash >> (32 - r2))) * m + n;
}
2015-12-17 07:26:24 -05:00
u8 const* tail = (static_cast<u8 const*>(key)) + nblocks * 4;
2015-09-28 16:58:33 -04:00
u32 k1 = 0;
switch (num_bytes & 3) {
case 3:
k1 ^= tail[2] << 16;
case 2:
k1 ^= tail[1] << 8;
case 1:
k1 ^= tail[0];
k1 *= c1;
k1 = (k1 << r1) | (k1 >> (32 - r1));
k1 *= c2;
hash ^= k1;
}
hash ^= num_bytes;
hash ^= (hash >> 16);
hash *= 0x85ebca6b;
hash ^= (hash >> 13);
hash *= 0xc2b2ae35;
hash ^= (hash >> 16);
return hash;
}
2015-10-04 15:59:40 -04:00
#if defined(GB_ARCH_64_BIT)
u64
2015-12-17 07:26:24 -05:00
murmur64(void const* key, usize num_bytes, u64 seed)
{
const u64 m = 0xc6a4a7935bd1e995ULL;
const s32 r = 47;
2015-09-28 16:58:33 -04:00
u64 h = seed ^ (num_bytes * m);
2015-09-28 16:58:33 -04:00
const u64* data = static_cast<const u64*>(key);
const u64* end = data + (num_bytes / 8);
2015-09-28 16:58:33 -04:00
while (data != end)
{
u64 k = *data++;
2015-09-28 16:58:33 -04:00
k *= m;
k ^= k >> r;
k *= m;
2015-09-28 16:58:33 -04:00
h ^= k;
h *= m;
}
2015-09-28 16:58:33 -04:00
2015-12-17 07:26:24 -05:00
u8 const* data2 = reinterpret_cast<u8 const*>(data);
2015-09-28 16:58:33 -04:00
switch (num_bytes & 7)
{
case 7: h ^= static_cast<u64>(data2[6]) << 48;
case 6: h ^= static_cast<u64>(data2[5]) << 40;
case 5: h ^= static_cast<u64>(data2[4]) << 32;
case 4: h ^= static_cast<u64>(data2[3]) << 24;
case 3: h ^= static_cast<u64>(data2[2]) << 16;
case 2: h ^= static_cast<u64>(data2[1]) << 8;
case 1: h ^= static_cast<u64>(data2[0]);
h *= m;
};
h ^= h >> r;
2015-09-28 16:58:33 -04:00
h *= m;
h ^= h >> r;
2015-09-28 16:58:33 -04:00
return h;
}
2015-09-28 16:58:33 -04:00
#elif GB_ARCH_32_BIT
u64
2015-12-17 07:26:24 -05:00
murmur64(void const* key, usize num_bytes, u64 seed)
{
const u32 m = 0x5bd1e995;
const s32 r = 24;
2015-09-28 16:58:33 -04:00
u32 h1 = static_cast<u32>(seed) ^ static_cast<u32>(num_bytes);
u32 h2 = static_cast<u32>(seed >> 32);
2015-09-28 16:58:33 -04:00
const u32* data = static_cast<const u32*>(key);
2015-09-28 16:58:33 -04:00
while (num_bytes >= 8)
{
u32 k1 = *data++;
k1 *= m;
k1 ^= k1 >> r;
k1 *= m;
h1 *= m;
h1 ^= k1;
num_bytes -= 4;
u32 k2 = *data++;
k2 *= m;
k2 ^= k2 >> r;
k2 *= m;
h2 *= m;
h2 ^= k2;
num_bytes -= 4;
}
2015-09-28 16:58:33 -04:00
if (num_bytes >= 4)
{
u32 k1 = *data++;
k1 *= m;
k1 ^= k1 >> r;
k1 *= m;
h1 *= m;
h1 ^= k1;
num_bytes -= 4;
}
2015-09-28 16:58:33 -04:00
switch (num_bytes)
{
2015-12-17 07:26:24 -05:00
case 3: h2 ^= reinterpret_cast<u8 const*>(data)[2] << 16;
case 2: h2 ^= reinterpret_cast<u8 const*>(data)[1] << 8;
case 1: h2 ^= reinterpret_cast<u8 const*>(data)[0] << 0;
h2 *= m;
};
2015-09-28 16:58:33 -04:00
h1 ^= h2 >> 18;
h1 *= m;
h2 ^= h1 >> 22;
h2 *= m;
h1 ^= h2 >> 17;
h1 *= m;
h2 ^= h1 >> 19;
2015-09-28 16:58:33 -04:00
h2 *= m;
u64 h = h1;
2015-09-28 16:58:33 -04:00
h = (h << 32) | h2;
2015-09-28 16:58:33 -04:00
return h;
}
2015-09-28 16:58:33 -04:00
#else
#error murmur64 function not supported on this architecture
2015-09-28 16:58:33 -04:00
#endif
} // namespace hash
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// Time //
// //
2015-09-27 14:03:47 -04:00
////////////////////////////////
2015-10-18 07:18:54 -04:00
2015-12-17 07:26:24 -05:00
Time const TIME_ZERO = time::seconds(0);
2015-10-18 07:18:54 -04:00
2015-10-28 14:32:42 -04:00
namespace time
{
2015-10-04 15:59:40 -04:00
#if defined(GB_SYSTEM_WINDOWS)
internal_linkage LARGE_INTEGER
win32_get_frequency()
{
LARGE_INTEGER f;
QueryPerformanceFrequency(&f);
return f;
}
2015-09-28 15:31:26 -04:00
Time
now()
{
// NOTE(bill): std::chrono does not have a good enough precision in MSVC12
// and below. This may have been fixed in MSVC14 but unsure as of yet.
2015-09-28 15:31:26 -04:00
// Force the following code to run on first core
// NOTE(bill): See
// http://msdn.microsoft.com/en-us/library/windows/desktop/ms644904(v=vs.85).aspx
HANDLE currentThread = GetCurrentThread();
DWORD_PTR previousMask = SetThreadAffinityMask(currentThread, 1);
2015-09-28 15:31:26 -04:00
// Get the frequency of the performance counter
// It is constant across the program's lifetime
local_persist LARGE_INTEGER s_frequency = win32_get_frequency();
2015-09-28 15:31:26 -04:00
// Get the current time
LARGE_INTEGER t;
QueryPerformanceCounter(&t);
2015-09-28 15:31:26 -04:00
// Restore the thread affinity
SetThreadAffinityMask(currentThread, previousMask);
2015-09-28 15:31:26 -04:00
return time::microseconds(1000000ll * t.QuadPart / s_frequency.QuadPart);
}
2015-09-28 15:31:26 -04:00
void
sleep(Time t)
{
if (t.microseconds <= 0)
return;
2015-09-28 15:31:26 -04:00
// Get the supported timer resolutions on this system
TIMECAPS tc;
timeGetDevCaps(&tc, sizeof(TIMECAPS));
// Set the timer resolution to the minimum for the Sleep call
timeBeginPeriod(tc.wPeriodMin);
2015-09-28 15:31:26 -04:00
// Wait...
::Sleep(time::as_milliseconds(t));
2015-09-28 15:31:26 -04:00
// Reset the timer resolution back to the system default
timeBeginPeriod(tc.wPeriodMin);
}
2015-09-28 15:31:26 -04:00
#else
Time
now()
{
#if defined(GB_SYSTEM_OSX)
s64 t = static_cast<s64>(mach_absolute_time());
return microseconds(t);
#else
struct timeval t;
gettimeofday(&t, nullptr);
2015-09-28 15:31:26 -04:00
return microseconds((t.tv_sec * 1000000ll) + (t.tv_usec * 1ll));
#endif
}
2015-09-28 15:31:26 -04:00
void
sleep(Time t)
{
if (t.microseconds <= 0)
return;
2015-09-28 15:31:26 -04:00
struct timespec spec = {};
spec.tv_sec = static_cast<s64>(as_seconds(t));
spec.tv_nsec = 1000ll * (as_microseconds(t) % 1000000ll);
2015-09-28 15:31:26 -04:00
nanosleep(&spec, nullptr);
}
2015-09-28 15:31:26 -04:00
#endif
2015-10-18 07:18:54 -04:00
Time seconds(f32 s) { return {static_cast<s64>(s * 1000000ll)}; }
Time milliseconds(s32 ms) { return {static_cast<s64>(ms * 1000ll)}; }
2015-09-29 10:11:18 -04:00
Time microseconds(s64 us) { return {us}; }
f32 as_seconds(Time t) { return static_cast<f32>(t.microseconds / 1000000.0f); }
s32 as_milliseconds(Time t) { return static_cast<s32>(t.microseconds / 1000ll); }
s64 as_microseconds(Time t) { return t.microseconds; }
2015-10-28 14:32:42 -04:00
} // namespace time
2015-10-28 14:32:42 -04:00
bool operator==(Time left, Time right) { return left.microseconds == right.microseconds; }
bool operator!=(Time left, Time right) { return !operator==(left, right); }
2015-09-28 15:31:26 -04:00
2015-10-28 14:32:42 -04:00
bool operator<(Time left, Time right) { return left.microseconds < right.microseconds; }
bool operator>(Time left, Time right) { return left.microseconds > right.microseconds; }
2015-09-28 15:31:26 -04:00
2015-10-28 14:32:42 -04:00
bool operator<=(Time left, Time right) { return left.microseconds <= right.microseconds; }
bool operator>=(Time left, Time right) { return left.microseconds >= right.microseconds; }
2015-09-28 15:31:26 -04:00
2015-11-19 17:27:57 -05:00
Time operator+(Time right) { return {+right.microseconds}; }
2015-10-28 14:32:42 -04:00
Time operator-(Time right) { return {-right.microseconds}; }
2015-09-28 15:31:26 -04:00
2015-10-28 14:32:42 -04:00
Time operator+(Time left, Time right) { return {left.microseconds + right.microseconds}; }
Time operator-(Time left, Time right) { return {left.microseconds - right.microseconds}; }
2015-09-28 15:31:26 -04:00
2015-10-28 14:32:42 -04:00
Time& operator+=(Time& left, Time right) { return (left = left + right); }
Time& operator-=(Time& left, Time right) { return (left = left - right); }
2015-09-28 15:31:26 -04:00
2015-10-28 14:32:42 -04:00
Time operator*(Time left, f32 right) { return time::seconds(time::as_seconds(left) * right); }
Time operator*(Time left, s64 right) { return time::microseconds(time::as_microseconds(left) * right); }
Time operator*(f32 left, Time right) { return time::seconds(time::as_seconds(right) * left); }
Time operator*(s64 left, Time right) { return time::microseconds(time::as_microseconds(right) * left); }
2015-09-28 15:31:26 -04:00
2015-10-28 14:32:42 -04:00
Time& operator*=(Time& left, f32 right) { return (left = left * right); }
Time& operator*=(Time& left, s64 right) { return (left = left * right); }
2015-09-28 15:31:26 -04:00
2015-10-28 14:32:42 -04:00
Time operator/(Time left, f32 right) { return time::seconds(time::as_seconds(left) / right); }
Time operator/(Time left, s64 right) { return time::microseconds(time::as_microseconds(left) / right); }
f32 operator/(Time left, Time right) { return time::as_seconds(left) / time::as_seconds(right); }
2015-09-28 15:31:26 -04:00
2015-10-28 14:32:42 -04:00
Time& operator/=(Time& left, f32 right) { return (left = left / right); }
Time& operator/=(Time& left, s64 right) { return (left = left / right); }
2015-09-28 15:31:26 -04:00
2015-10-28 14:32:42 -04:00
Time operator%(Time left, Time right) { return time::microseconds(time::as_microseconds(left) % time::as_microseconds(right)); }
Time& operator%=(Time& left, Time right) { return (left = left % right); }
2015-09-28 15:31:26 -04:00
2015-10-28 14:32:42 -04:00
////////////////////////////////
2015-12-02 10:06:36 -05:00
// //
// OS //
// //
2015-10-28 14:32:42 -04:00
////////////////////////////////
namespace os
{
GB_FORCE_INLINE u64
2015-12-02 10:06:36 -05:00
rdtsc()
{
#if GB_SYSTEM_WINDOWS
2015-12-02 10:06:36 -05:00
return ::__rdtsc();
#else
// TODO(bill): Check that rdtsc() works
2015-12-02 10:06:36 -05:00
return ::rdtsc();
#endif
}
} // namespace os
2015-10-18 07:18:54 -04:00
__GB_NAMESPACE_END
2015-09-27 14:03:47 -04:00
#endif // GB_IMPLEMENTATION