blob: 480e9ad0e551a433e54f12808ebdedebb1662cab [file] [log] [blame]
// Copyright 2019 The IREE Authors
//
// Licensed under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
#ifndef IREE_BASE_ALLOCATOR_H_
#define IREE_BASE_ALLOCATOR_H_
#include <memory.h>
#include <stdint.h>
#include <string.h>
#include "iree/base/alignment.h"
#include "iree/base/attributes.h"
#include "iree/base/config.h"
#include "iree/base/status.h"
#include "iree/base/target_platform.h"
#ifdef __cplusplus
extern "C" {
#endif // __cplusplus
//===----------------------------------------------------------------------===//
// Types and Enums
//===----------------------------------------------------------------------===//
#if IREE_STATISTICS_ENABLE
// Evalutes the expression code only if statistics are enabled.
//
// Example:
// struct {
// IREE_STATISTICS(uint32_t stats_only_value);
// } my_object;
// IREE_STATISTICS(my_object.stats_only_value = 5);
// IREE_STATISTICS({
// my_object.stats_only_value = 5;
// });
#define IREE_STATISTICS(expr) expr
#else
#define IREE_STATISTICS(expr)
#endif // IREE_STATISTICS_ENABLE
//===----------------------------------------------------------------------===//
// Byte buffers and memory utilities
//===----------------------------------------------------------------------===//
// A span of mutable bytes (ala std::span of uint8_t).
typedef struct iree_byte_span_t {
uint8_t* data;
iree_host_size_t data_length;
} iree_byte_span_t;
static inline iree_byte_span_t iree_make_byte_span(
void* data, iree_host_size_t data_length) {
iree_byte_span_t v = {(uint8_t*)data, data_length};
return v;
}
static inline iree_byte_span_t iree_byte_span_empty() {
iree_byte_span_t v = {NULL, 0};
return v;
}
static inline bool iree_byte_span_is_empty(iree_byte_span_t span) {
return span.data == NULL || span.data_length == 0;
}
// A span of constant bytes (ala std::span of const uint8_t).
typedef struct iree_const_byte_span_t {
const uint8_t* data;
iree_host_size_t data_length;
} iree_const_byte_span_t;
static inline iree_const_byte_span_t iree_make_const_byte_span(
const void* data, iree_host_size_t data_length) {
iree_const_byte_span_t v = {(const uint8_t*)data, data_length};
return v;
}
static inline iree_const_byte_span_t iree_const_byte_span_empty() {
iree_const_byte_span_t v = {NULL, 0};
return v;
}
static inline bool iree_const_byte_span_is_empty(iree_const_byte_span_t span) {
return span.data == NULL || span.data_length == 0;
}
static inline iree_const_byte_span_t iree_const_cast_byte_span(
iree_byte_span_t span) {
return iree_make_const_byte_span(span.data, span.data_length);
}
static inline iree_byte_span_t iree_cast_const_byte_span(
iree_const_byte_span_t span) {
return iree_make_byte_span((uint8_t*)span.data, span.data_length);
}
//===----------------------------------------------------------------------===//
// Totally shady stack allocation
//===----------------------------------------------------------------------===//
// TODO(benvanik): remove our uses of this or make them more explicit.
#if defined(IREE_PLATFORM_WINDOWS)
// The safe malloca that may fall back to heap in the case of stack overflows:
// https://docs.microsoft.com/en-us/cpp/c-runtime-library/reference/malloca?view=vs-2019
// Because that gets really annoying to deal with during error handling we just
// go for _alloca which may generate SEH exceptions if we blow the stack.
#include <malloc.h>
#define iree_alloca(sz) _alloca(sz)
#else
#include <alloca.h>
#define iree_alloca(sz) alloca(sz)
#endif // IREE_COMPILER_MSVC
//===----------------------------------------------------------------------===//
// iree_allocator_t (std::allocator-like interface)
//===----------------------------------------------------------------------===//
// Controls the behavior of an iree_allocator_ctl_fn_t callback function.
typedef enum iree_allocator_command_e {
// Allocates |byte_length| of memory and stores the pointer in |inout_ptr|.
// Systems should align to 16 byte boundaries (or otherwise their natural
// SIMD alignment). The runtime pools internally and small allocations
// (usually) won't be made through this interface.
//
// iree_allocator_ctl_fn_t:
// params: iree_allocator_alloc_params_t
// inout_ptr: set to allocated pointer
IREE_ALLOCATOR_COMMAND_MALLOC = 0,
// As with IREE_ALLOCATOR_COMMAND_MALLOC but zeros the memory.
//
// The contents of the allocation *must* be zeroed by the allocator prior to
// returning. Allocators may be able to elide the zeroing if they allocate
// fresh pages from the system. It is always safe to zero contents if the
// behavior of the allocator is not under our control.
//
// iree_allocator_ctl_fn_t:
// params: iree_allocator_alloc_params_t
// inout_ptr: set to allocated pointer
IREE_ALLOCATOR_COMMAND_CALLOC = 1,
// Tries to resize an allocation provided via |inout_ptr|, if possible.
// If the existing allocation is not reused then it is freed as if a call to
// iree_allocator_free had been called on it. If the allocation fails then
// the provided existing allocation is unmodified. Only pointers previously
// received from the iree_allocator_t are valid.
//
// iree_allocator_ctl_fn_t:
// params: iree_allocator_alloc_params_t
// inout_ptr: pointer of existing allocation; updated to realloced pointer
IREE_ALLOCATOR_COMMAND_REALLOC = 2,
// Frees the memory pointed to by |inout_ptr|.
//
// iree_allocator_ctl_fn_t:
// params: unused
// inout_ptr: pointer to free
IREE_ALLOCATOR_COMMAND_FREE = 3,
// TODO(benvanik): add optional IREE_ALLOCATOR_COMMAND_BIND like mbind:
// https://man7.org/linux/man-pages/man2/mbind.2.html
// This would take a pointer/length and a NUMA node ID to bind the memory to.
// We may want flags for controlling whether this is a new allocation getting
// bound or an existing one that is migrating to use MPOL_MF_MOVE.
} iree_allocator_command_t;
// Parameters for various allocation commands.
typedef struct iree_allocator_alloc_params_t {
// Minimum size, in bytes, of the allocation. The underlying allocator may
// pad the length out if needed.
iree_host_size_t byte_length;
} iree_allocator_alloc_params_t;
// Function pointer for an iree_allocator_t control function.
// |command| provides the operation to perform. Optionally some commands may use
// |params| to pass additional operation-specific parameters. |inout_ptr| usage
// is defined by each operation but is generally a pointer to the pointer to
// set to the newly allocated memory or a pointer to the pointer to free.
typedef iree_status_t(IREE_API_PTR* iree_allocator_ctl_fn_t)(
void* self, iree_allocator_command_t command, const void* params,
void** inout_ptr);
// An allocator for host-memory allocations.
// IREE will attempt to use this in place of the system malloc and free.
// Pass the iree_allocator_system() macro to use the system allocator.
typedef struct iree_allocator_t {
// Control function data.
void* self;
// ioctl-style control function servicing all allocator-related commands.
// See iree_allocator_command_t for more information.
iree_allocator_ctl_fn_t ctl;
} iree_allocator_t;
// Allocates a block of |byte_length| bytes from the given allocator.
// The contents of the returned memory is guaranteed to be zeroed.
IREE_API_EXPORT iree_status_t iree_allocator_malloc(
iree_allocator_t allocator, iree_host_size_t byte_length, void** out_ptr);
// Allocates a block of |byte_length| bytes from the given allocator.
// The content of the buffer returned is undefined: it may be zeros, a
// debug-fill pattern, or random memory from elsewhere in the process.
// Only use this when immediately overwriting all memory.
IREE_API_EXPORT iree_status_t iree_allocator_malloc_uninitialized(
iree_allocator_t allocator, iree_host_size_t byte_length, void** out_ptr);
// Reallocates |inout_ptr| to |byte_length| bytes with the given allocator.
// If the reallocation fails then the original |inout_ptr| is unmodified.
//
// WARNING: when extending the newly allocated bytes are undefined.
// TODO(benvanik): make them zeros; we should have an _uninitialized if needed.
IREE_API_EXPORT iree_status_t iree_allocator_realloc(
iree_allocator_t allocator, iree_host_size_t byte_length, void** inout_ptr);
// Duplicates the given byte block by allocating memory and copying it in.
IREE_API_EXPORT iree_status_t
iree_allocator_clone(iree_allocator_t allocator,
iree_const_byte_span_t source_bytes, void** out_ptr);
// Frees a previously-allocated block of memory to the given allocator.
IREE_API_EXPORT void iree_allocator_free(iree_allocator_t allocator, void* ptr);
//===----------------------------------------------------------------------===//
// Built-in iree_allocator_t implementations
//===----------------------------------------------------------------------===//
// Default C allocator controller using malloc/free.
IREE_API_EXPORT iree_status_t
iree_allocator_system_ctl(void* self, iree_allocator_command_t command,
const void* params, void** inout_ptr);
// Allocates using the iree_allocator_malloc and iree_allocator_free methods.
// These will usually be backed by malloc and free.
static inline iree_allocator_t iree_allocator_system(void) {
iree_allocator_t v = {NULL, iree_allocator_system_ctl};
return v;
}
// Does not perform any allocation or deallocation; used to wrap objects that
// are owned by external code/live in read-only memory/etc.
static inline iree_allocator_t iree_allocator_null(void) {
iree_allocator_t v = {NULL, NULL};
return v;
}
// Returns true if the allocator is `iree_allocator_null()`.
static inline bool iree_allocator_is_null(iree_allocator_t allocator) {
return allocator.ctl == NULL;
}
typedef struct {
iree_host_size_t capacity;
iree_host_size_t length;
iree_host_size_t head_size;
uint8_t* buffer;
} iree_allocator_inline_storage_t;
// Stack storage for an inline arena-style allocator.
//
// Usage:
// IREE_ALLOCATOR_INLINE_STORAGE(inline_storage, 2048);
// something_allocating(iree_allocator_inline_arena(&inline_storage.header));
#define IREE_ALLOCATOR_INLINE_STORAGE(var, storage_capacity) \
struct { \
iree_allocator_inline_storage_t header; \
uint8_t data[storage_capacity]; \
} var = { \
.header = \
{ \
.capacity = sizeof((var).data), \
.length = 0, \
.buffer = &(var).data[0], \
}, \
};
// Inline arena allocator controller used by iree_allocator_inline_arena.
IREE_API_EXPORT iree_status_t
iree_allocator_inline_arena_ctl(void* self, iree_allocator_command_t command,
const void* params, void** inout_ptr);
// Allocates with arena semantics within the given fixed-size |storage|.
// Frees are ignored and all allocations will fail once the allocated length
// exceeds the capacity. A special case for reallocations of the entire
// outstanding memory is supported to allow the arena to be implicitly reset.
static inline iree_allocator_t iree_allocator_inline_arena(
iree_allocator_inline_storage_t* storage) {
iree_allocator_t v = {storage, iree_allocator_inline_arena_ctl};
return v;
}
//===----------------------------------------------------------------------===//
// Aligned allocations via iree_allocator_t
//===----------------------------------------------------------------------===//
// Allocates memory of size |byte_length| where the byte starting at |offset|
// has a minimum alignment of |min_alignment|. In many cases |offset| can be 0.
//
// The |offset| can be used to ensure the alignment-sensitive portion of a
// combined allocation is aligned while any prefix metadata has system
// alignment. For example:
// typedef struct {
// uint32_t some_metadata;
// uint8_t data[];
// } buffer_t;
// buffer_t* buffer = NULL;
// iree_allocator_malloc_aligned(allocator, sizeof(buffer_t) + length,
// 4096, offsetof(buffer_t, data), &buffer);
// // `buffer` has system alignment, but the `data` will be aligned on at
// // least a 4096 boundary.
//
// The contents of the returned memory is guaranteed to be zeroed.
IREE_API_EXPORT iree_status_t iree_allocator_malloc_aligned(
iree_allocator_t allocator, iree_host_size_t byte_length,
iree_host_size_t min_alignment, iree_host_size_t offset, void** out_ptr);
// Reallocates memory to |byte_length|, growing or shrinking as needed.
// Only valid on memory allocated with iree_allocator_malloc_aligned.
// The newly reallocated memory will have the byte at |offset| aligned to at
// least |min_alignment|.
IREE_API_EXPORT iree_status_t iree_allocator_realloc_aligned(
iree_allocator_t allocator, iree_host_size_t byte_length,
iree_host_size_t min_alignment, iree_host_size_t offset, void** inout_ptr);
// Frees a |ptr| previously returned from iree_allocator_malloc_aligned.
IREE_API_EXPORT void iree_allocator_free_aligned(iree_allocator_t allocator,
void* ptr);
#ifdef __cplusplus
} // extern "C"
#endif // __cplusplus
#endif // IREE_BASE_ALLOCATOR_H_