* Copyright (c) 2022 Huawei Device Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef ECMASCRIPT_MEM_NATIVE_AREA_ALLOCATOR_H
#define ECMASCRIPT_MEM_NATIVE_AREA_ALLOCATOR_H
#include <atomic>
#include <cstddef>
#include "ecmascript/common.h"
#include "ecmascript/log_wrapper.h"
#include "ecmascript/mem/mem.h"
#include "ecmascript/mem/area.h"
namespace panda::ecmascript {
enum class NativeFlag : uint32_t {
NO_DIV,
ARRAY_BUFFER,
REGEXP_BTYECODE,
CHUNK_MEM,
WRAPPER_DATA,
};
class PUBLIC_API NativeAreaAllocator {
public:
NativeAreaAllocator() = default;
virtual ~NativeAreaAllocator()
{
if (cachedArea_ != nullptr) {
FreeArea(cachedArea_);
cachedArea_ = nullptr;
}
}
Area *AllocateArea(size_t capacity);
void FreeArea(Area *area);
void Free(void *mem, size_t size);
void *AllocateBuffer(size_t size);
void FreeBuffer(void *mem);
void *NativeAreaPageMap(size_t size);
void NativeAreaPageUnmap(void *mem, size_t size);
static void FreeBufferFunc(void *env, void* buffer, void* data);
template<class T>
static void FreeObjectFunc([[maybe_unused]] void *env, void* buffer, void* data)
{
if (buffer == nullptr || data == nullptr) {
return;
}
NativeAreaAllocator* allocator = reinterpret_cast<NativeAreaAllocator*>(data);
allocator->Delete<T>(static_cast<T *>(buffer));
}
template<typename T, typename... Args>
std::enable_if_t<!std::is_array_v<T>, T *> New(Args &&... args)
{
void *p = AllocateBuffer(sizeof(T));
if (UNLIKELY(p == nullptr)) {
return nullptr;
}
new (p) T(std::forward<Args>(args)...);
return reinterpret_cast<T *>(p);
}
template<class T>
void Delete(T *ptr)
{
if (ptr == nullptr) {
return;
}
if constexpr (std::is_class_v<T>) {
ptr->~T();
}
FreeBuffer(ptr);
}
void IncreaseNativeMemoryUsage(size_t bytes)
{
size_t current = nativeMemoryUsage_.fetch_add(bytes, std::memory_order_relaxed) + bytes;
size_t max = maxNativeMemoryUsage_.load(std::memory_order_relaxed);
while (current > max && !maxNativeMemoryUsage_.compare_exchange_weak(max, current, std::memory_order_relaxed)) {
}
}
void DecreaseNativeMemoryUsage(size_t bytes)
{
nativeMemoryUsage_.fetch_sub(bytes, std::memory_order_relaxed);
}
size_t GetNativeMemoryUsage() const
{
return nativeMemoryUsage_.load(std::memory_order_relaxed);
}
size_t GetMaxNativeMemoryUsage() const
{
return maxNativeMemoryUsage_.load(std::memory_order_relaxed);
}
size_t GetArrayBufferNativeSize() const
{
return arrayBufferNativeSize_;
}
size_t GetRegExpNativeSize() const
{
return regExpNativeSize_;
}
size_t GetChunkNativeSize() const
{
return chunkNativeSize_;
}
inline void IncreaseNativeSizeStats(size_t size, NativeFlag flag)
{
if (size == 0) {
return;
}
switch (flag) {
case NativeFlag::ARRAY_BUFFER:
arrayBufferNativeSize_ += size;
break;
case NativeFlag::REGEXP_BTYECODE:
regExpNativeSize_ += size;
break;
case NativeFlag::CHUNK_MEM:
chunkNativeSize_ += size;
break;
default:
break;
}
}
inline void DecreaseNativeSizeStats(size_t size, NativeFlag flag)
{
if (size == 0) {
return;
}
switch (flag) {
case NativeFlag::ARRAY_BUFFER:
arrayBufferNativeSize_ -= size;
break;
case NativeFlag::REGEXP_BTYECODE:
regExpNativeSize_ -= size;
break;
case NativeFlag::CHUNK_MEM:
chunkNativeSize_ -= size;
break;
default:
break;
}
}
void ModifyNativeSizeStats(size_t preSize, size_t nextSize, NativeFlag flag) {
if (flag == NativeFlag::NO_DIV) {
return;
}
DecreaseNativeSizeStats(preSize, flag);
IncreaseNativeSizeStats(nextSize, flag);
}
void *Allocate(size_t size)
{
if (size == 0) {
LOG_ECMA_MEM(FATAL) << "size must have a size bigger than 0";
UNREACHABLE();
}
void *ptr = malloc(size);
if (ptr == nullptr) {
LOG_ECMA_MEM(FATAL) << "malloc failed";
UNREACHABLE();
}
IncreaseNativeMemoryUsage(size);
return ptr;
}
static inline Area *AllocateSpace(size_t capacity)
{
size_t headerSize = sizeof(Area);
if (capacity < headerSize) {
LOG_ECMA_MEM(FATAL) << "capacity must have a size not less than sizeof Area.";
UNREACHABLE();
}
void *mem = malloc(capacity);
if (mem == nullptr) {
LOG_ECMA_MEM(FATAL) << "malloc failed";
UNREACHABLE();
}
uintptr_t begin = reinterpret_cast<uintptr_t>(mem) + headerSize;
capacity -= headerSize;
return new (mem) Area(begin, capacity);
}
static inline void FreeSpace(Area *area)
{
if (area == nullptr) {
return;
}
free(reinterpret_cast<std::byte *>(area));
}
private:
NO_COPY_SEMANTIC(NativeAreaAllocator);
NO_MOVE_SEMANTIC(NativeAreaAllocator);
Area *cachedArea_ {nullptr};
std::atomic<size_t> nativeMemoryUsage_ {0};
std::atomic<size_t> maxNativeMemoryUsage_ {0};
size_t arrayBufferNativeSize_ {0};
size_t regExpNativeSize_ {0};
size_t chunkNativeSize_ {0};
};
}
#endif