blob: 4b5068b32d9172e27fd4f1160a52ad01cb5dbc61 [file] [log] [blame]
/*
* Copyright (c) 2021 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifndef ISO_HEAP_INLINES_H
#define ISO_HEAP_INLINES_H
#include "iso_heap.h"
#include "iso_heap_config.h"
#include "iso_heap_innards.h"
#include "pas_deallocate.h"
#include "pas_get_allocation_size.h"
#include "pas_get_heap.h"
#include "pas_has_object.h"
#include "pas_try_allocate.h"
#include "pas_try_allocate_array.h"
#include "pas_try_allocate_intrinsic.h"
#include "pas_try_allocate_primitive.h"
#include "pas_try_reallocate.h"
#if PAS_ENABLE_ISO
PAS_BEGIN_EXTERN_C;
PAS_CREATE_TRY_ALLOCATE_INTRINSIC(
iso_try_allocate_common_primitive_impl,
ISO_HEAP_CONFIG,
&iso_intrinsic_runtime_config.base,
&iso_allocator_counts,
pas_allocation_result_set_errno,
&iso_common_primitive_heap,
&iso_common_primitive_heap_support,
pas_intrinsic_heap_is_not_designated);
/* Need to create a different set of allocation functions if we want to pass nontrivial alignment,
since in that case we do not want to use the fancy lookup path. */
PAS_CREATE_TRY_ALLOCATE_INTRINSIC(
iso_try_allocate_common_primitive_with_alignment_impl,
ISO_HEAP_CONFIG,
&iso_intrinsic_runtime_config.base,
&iso_allocator_counts,
pas_allocation_result_set_errno,
&iso_common_primitive_heap,
&iso_common_primitive_heap_support,
pas_intrinsic_heap_is_not_designated);
PAS_CREATE_TRY_ALLOCATE_INTRINSIC(
iso_allocate_common_primitive_impl,
ISO_HEAP_CONFIG,
&iso_intrinsic_runtime_config.base,
&iso_allocator_counts,
pas_allocation_result_crash_on_error,
&iso_common_primitive_heap,
&iso_common_primitive_heap_support,
pas_intrinsic_heap_is_not_designated);
PAS_CREATE_TRY_ALLOCATE_INTRINSIC(
iso_allocate_common_primitive_with_alignment_impl,
ISO_HEAP_CONFIG,
&iso_intrinsic_runtime_config.base,
&iso_allocator_counts,
pas_allocation_result_crash_on_error,
&iso_common_primitive_heap,
&iso_common_primitive_heap_support,
pas_intrinsic_heap_is_not_designated);
static PAS_ALWAYS_INLINE void* iso_try_allocate_common_primitive_inline(size_t size)
{
return (void*)iso_try_allocate_common_primitive_impl(size, 1).begin;
}
static PAS_ALWAYS_INLINE void*
iso_try_allocate_common_primitive_with_alignment_inline(size_t size, size_t alignment)
{
return (void*)iso_try_allocate_common_primitive_with_alignment_impl(size, alignment).begin;
}
static PAS_ALWAYS_INLINE void* iso_try_allocate_common_primitive_zeroed_inline(size_t size)
{
return (void*)pas_allocation_result_zero(
iso_try_allocate_common_primitive_impl(size, 1),
size).begin;
}
static PAS_ALWAYS_INLINE void* iso_allocate_common_primitive_inline(size_t size)
{
return (void*)iso_allocate_common_primitive_impl(size, 1).begin;
}
static PAS_ALWAYS_INLINE void*
iso_allocate_common_primitive_with_alignment_inline(size_t size, size_t alignment)
{
return (void*)iso_allocate_common_primitive_with_alignment_impl(size, alignment).begin;
}
static PAS_ALWAYS_INLINE void* iso_allocate_common_primitive_zeroed_inline(size_t size)
{
return (void*)pas_allocation_result_zero(
iso_allocate_common_primitive_impl(size, 1),
size).begin;
}
static PAS_ALWAYS_INLINE void*
iso_try_reallocate_common_primitive_inline(void* old_ptr, size_t new_size,
pas_reallocate_free_mode free_mode)
{
return (void*)pas_try_reallocate_intrinsic(
old_ptr,
&iso_common_primitive_heap,
new_size,
ISO_HEAP_CONFIG,
iso_try_allocate_common_primitive_impl_for_realloc,
pas_reallocate_allow_heap_teleport,
free_mode).begin;
}
static PAS_ALWAYS_INLINE void*
iso_reallocate_common_primitive_inline(void* old_ptr, size_t new_size,
pas_reallocate_free_mode free_mode)
{
return (void*)pas_try_reallocate_intrinsic(
old_ptr,
&iso_common_primitive_heap,
new_size,
ISO_HEAP_CONFIG,
iso_allocate_common_primitive_impl_for_realloc,
pas_reallocate_allow_heap_teleport,
free_mode).begin;
}
PAS_CREATE_TRY_ALLOCATE(
iso_try_allocate_impl,
ISO_HEAP_CONFIG,
&iso_typed_runtime_config.base,
&iso_allocator_counts,
pas_allocation_result_set_errno);
static PAS_ALWAYS_INLINE void* iso_try_allocate_inline(pas_heap_ref* heap_ref)
{
return iso_try_allocate_impl(heap_ref).ptr;
}
PAS_CREATE_TRY_ALLOCATE(
iso_allocate_impl,
ISO_HEAP_CONFIG,
&iso_typed_runtime_config.base,
&iso_allocator_counts,
pas_allocation_result_crash_on_error);
static PAS_ALWAYS_INLINE void* iso_allocate_inline(pas_heap_ref* heap_ref)
{
return iso_allocate_impl(heap_ref).ptr;
}
PAS_CREATE_TRY_ALLOCATE_ARRAY(
iso_try_allocate_array_impl,
ISO_HEAP_CONFIG,
&iso_typed_runtime_config.base,
&iso_allocator_counts,
pas_allocation_result_set_errno);
static PAS_ALWAYS_INLINE void*
iso_try_allocate_array_inline(pas_heap_ref* heap_ref, size_t count, size_t alignment)
{
return iso_try_allocate_array_impl(heap_ref, count, alignment).ptr;
}
PAS_CREATE_TRY_ALLOCATE_ARRAY(
iso_allocate_array_impl,
ISO_HEAP_CONFIG,
&iso_typed_runtime_config.base,
&iso_allocator_counts,
pas_allocation_result_crash_on_error);
static PAS_ALWAYS_INLINE void*
iso_allocate_array_inline(pas_heap_ref* heap_ref, size_t count, size_t alignment)
{
return iso_allocate_array_impl(heap_ref, count, alignment).ptr;
}
static PAS_ALWAYS_INLINE void* iso_try_allocate_array_zeroed_inline(
pas_heap_ref* heap_ref, size_t count, size_t alignment)
{
return pas_typed_allocation_result_zero(
iso_try_allocate_array_impl(heap_ref, count, alignment)).ptr;
}
static PAS_ALWAYS_INLINE void* iso_allocate_array_zeroed_inline(
pas_heap_ref* heap_ref, size_t count, size_t alignment)
{
return pas_typed_allocation_result_zero(iso_allocate_array_impl(heap_ref, count, alignment)).ptr;
}
static PAS_ALWAYS_INLINE void* iso_try_reallocate_array_inline(void* old_ptr, pas_heap_ref* heap_ref,
size_t new_count,
pas_reallocate_free_mode free_mode)
{
return pas_try_reallocate_array(
old_ptr,
heap_ref,
new_count,
ISO_HEAP_CONFIG,
iso_try_allocate_array_impl_for_realloc,
&iso_typed_runtime_config.base,
pas_reallocate_allow_heap_teleport,
free_mode).ptr;
}
static PAS_ALWAYS_INLINE void* iso_reallocate_array_inline(void* old_ptr, pas_heap_ref* heap_ref,
size_t new_count,
pas_reallocate_free_mode free_mode)
{
return pas_try_reallocate_array(
old_ptr,
heap_ref,
new_count,
ISO_HEAP_CONFIG,
iso_allocate_array_impl_for_realloc,
&iso_typed_runtime_config.base,
pas_reallocate_allow_heap_teleport,
free_mode).ptr;
}
PAS_CREATE_TRY_ALLOCATE_PRIMITIVE(
iso_try_allocate_primitive_impl,
ISO_HEAP_CONFIG,
&iso_primitive_runtime_config.base,
&iso_allocator_counts,
pas_allocation_result_set_errno);
static PAS_ALWAYS_INLINE void* iso_try_allocate_primitive_inline(pas_primitive_heap_ref* heap_ref,
size_t size)
{
return (void*)iso_try_allocate_primitive_impl(heap_ref, size, 1).begin;
}
PAS_CREATE_TRY_ALLOCATE_PRIMITIVE(
iso_allocate_primitive_impl,
ISO_HEAP_CONFIG,
&iso_primitive_runtime_config.base,
&iso_allocator_counts,
pas_allocation_result_crash_on_error);
static PAS_ALWAYS_INLINE void* iso_allocate_primitive_inline(pas_primitive_heap_ref* heap_ref,
size_t size)
{
return (void*)iso_allocate_primitive_impl(heap_ref, size, 1).begin;
}
static PAS_ALWAYS_INLINE void* iso_try_allocate_primitive_zeroed_inline(pas_primitive_heap_ref* heap_ref,
size_t size)
{
return (void*)pas_allocation_result_zero(
iso_try_allocate_primitive_impl(heap_ref, size, 1),
size).begin;
}
static PAS_ALWAYS_INLINE void* iso_allocate_primitive_zeroed_inline(pas_primitive_heap_ref* heap_ref,
size_t size)
{
return (void*)pas_allocation_result_zero(
iso_allocate_primitive_impl(heap_ref, size, 1),
size).begin;
}
static PAS_ALWAYS_INLINE void*
iso_try_allocate_primitive_with_alignment_inline(pas_primitive_heap_ref* heap_ref,
size_t size,
size_t alignment)
{
return (void*)iso_try_allocate_primitive_impl(heap_ref, size, alignment).begin;
}
static PAS_ALWAYS_INLINE void*
iso_allocate_primitive_with_alignment_inline(pas_primitive_heap_ref* heap_ref,
size_t size,
size_t alignment)
{
return (void*)iso_allocate_primitive_impl(heap_ref, size, alignment).begin;
}
static PAS_ALWAYS_INLINE void* iso_try_reallocate_primitive_inline(void* old_ptr,
pas_primitive_heap_ref* heap_ref,
size_t new_size,
pas_reallocate_free_mode free_mode)
{
return (void*)pas_try_reallocate_primitive(
old_ptr,
heap_ref,
new_size,
ISO_HEAP_CONFIG,
iso_try_allocate_primitive_impl_for_realloc,
&iso_primitive_runtime_config.base,
pas_reallocate_allow_heap_teleport,
free_mode).begin;
}
static PAS_ALWAYS_INLINE void* iso_reallocate_primitive_inline(void* old_ptr,
pas_primitive_heap_ref* heap_ref,
size_t new_size,
pas_reallocate_free_mode free_mode)
{
return (void*)pas_try_reallocate_primitive(
old_ptr,
heap_ref,
new_size,
ISO_HEAP_CONFIG,
iso_allocate_primitive_impl_for_realloc,
&iso_primitive_runtime_config.base,
pas_reallocate_allow_heap_teleport,
free_mode).begin;
}
PAS_CREATE_TRY_ALLOCATE_PRIMITIVE(
iso_try_allocate_for_flex_impl,
ISO_HEAP_CONFIG,
&iso_flex_runtime_config.base,
&iso_allocator_counts,
pas_allocation_result_set_errno);
static PAS_ALWAYS_INLINE void* iso_try_allocate_for_flex_inline(const void* cls, size_t size)
{
return (void*)pas_allocation_result_zero(
iso_try_allocate_for_flex_impl(
pas_dynamic_primitive_heap_map_find(
&iso_flex_dynamic_heap_map, cls, size),
size, 1),
size).begin;
}
static PAS_ALWAYS_INLINE bool iso_has_object_inline(void* ptr)
{
return pas_has_object(ptr, ISO_HEAP_CONFIG);
}
static PAS_ALWAYS_INLINE size_t iso_get_allocation_size_inline(void* ptr)
{
return pas_get_allocation_size(ptr, ISO_HEAP_CONFIG);
}
static PAS_ALWAYS_INLINE pas_heap* iso_get_heap_inline(void* ptr)
{
return pas_get_heap(ptr, ISO_HEAP_CONFIG);
}
static PAS_ALWAYS_INLINE void iso_deallocate_inline(void* ptr)
{
pas_deallocate(ptr, ISO_HEAP_CONFIG);
}
PAS_END_EXTERN_C;
#endif /* PAS_ENABLE_ISO */
#endif /* ISO_HEAP_INLINES_H */