/* * * Copyright 2017 gRPC authors. * * 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 GRPC_CORE_LIB_GPRPP_MEMORY_H #define GRPC_CORE_LIB_GPRPP_MEMORY_H #include #include #include #include namespace grpc_core { // The alignment of memory returned by gpr_malloc(). constexpr size_t kAllignmentForDefaultAllocationInBytes = 8; // Alternative to new, since we cannot use it (for fear of libstdc++) template inline T* New(Args&&... args) { void* p = alignof(T) > kAllignmentForDefaultAllocationInBytes ? gpr_malloc_aligned(sizeof(T), alignof(T)) : gpr_malloc(sizeof(T)); return new (p) T(std::forward(args)...); } // Alternative to delete, since we cannot use it (for fear of libstdc++) template inline void Delete(T* p) { p->~T(); if (alignof(T) > kAllignmentForDefaultAllocationInBytes) { gpr_free_aligned(p); } else { gpr_free(p); } } template class DefaultDelete { public: void operator()(T* p) { Delete(p); } }; template > using UniquePtr = std::unique_ptr; template inline UniquePtr MakeUnique(Args&&... args) { return UniquePtr(New(std::forward(args)...)); } // an allocator that uses gpr_malloc/gpr_free template class Allocator { public: typedef T value_type; typedef T* pointer; typedef const T* const_pointer; typedef T& reference; typedef const T& const_reference; typedef std::size_t size_type; typedef std::ptrdiff_t difference_type; typedef std::false_type propagate_on_container_move_assignment; template struct rebind { typedef Allocator other; }; typedef std::true_type is_always_equal; pointer address(reference x) const { return &x; } const_pointer address(const_reference x) const { return &x; } pointer allocate(std::size_t n, std::allocator::const_pointer hint = nullptr) { return static_cast(gpr_malloc(n * sizeof(T))); } void deallocate(T* p, std::size_t n) { gpr_free(p); } size_t max_size() const { return std::numeric_limits::max() / sizeof(value_type); } void construct(pointer p, const_reference val) { new ((void*)p) T(val); } template void construct(U* p, Args&&... args) { ::new ((void*)p) U(std::forward(args)...); } void destroy(pointer p) { p->~T(); } template void destroy(U* p) { p->~U(); } }; } // namespace grpc_core #endif /* GRPC_CORE_LIB_GPRPP_MEMORY_H */