/* * Copyright (c) 2011 The WebRTC project authors. All Rights Reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ #include "rtc_base/memory/aligned_malloc.h" #include // for free, malloc #include // for memcpy #include "rtc_base/checks.h" #ifdef _WIN32 #include #else #include #endif // Reference on memory alignment: // http://stackoverflow.com/questions/227897/solve-the-memory-alignment-in-c-interview-question-that-stumped-me namespace webrtc { uintptr_t GetRightAlign(uintptr_t start_pos, size_t alignment) { // The pointer should be aligned with `alignment` bytes. The - 1 guarantees // that it is aligned towards the closest higher (right) address. return (start_pos + alignment - 1) & ~(alignment - 1); } // Alignment must be an integer power of two. bool ValidAlignment(size_t alignment) { if (!alignment) { return false; } return (alignment & (alignment - 1)) == 0; } void* GetRightAlign(const void* pointer, size_t alignment) { if (!pointer) { return NULL; } if (!ValidAlignment(alignment)) { return NULL; } uintptr_t start_pos = reinterpret_cast(pointer); return reinterpret_cast(GetRightAlign(start_pos, alignment)); } void* AlignedMalloc(size_t size, size_t alignment) { if (size == 0) { return NULL; } if (!ValidAlignment(alignment)) { return NULL; } // The memory is aligned towards the lowest address that so only // alignment - 1 bytes needs to be allocated. // A pointer to the start of the memory must be stored so that it can be // retreived for deletion, ergo the sizeof(uintptr_t). void* memory_pointer = malloc(size + sizeof(uintptr_t) + alignment - 1); RTC_CHECK(memory_pointer) << "Couldn't allocate memory in AlignedMalloc"; // Aligning after the sizeof(uintptr_t) bytes will leave room for the header // in the same memory block. uintptr_t align_start_pos = reinterpret_cast(memory_pointer); align_start_pos += sizeof(uintptr_t); uintptr_t aligned_pos = GetRightAlign(align_start_pos, alignment); void* aligned_pointer = reinterpret_cast(aligned_pos); // Store the address to the beginning of the memory just before the aligned // memory. uintptr_t header_pos = aligned_pos - sizeof(uintptr_t); void* header_pointer = reinterpret_cast(header_pos); uintptr_t memory_start = reinterpret_cast(memory_pointer); memcpy(header_pointer, &memory_start, sizeof(uintptr_t)); return aligned_pointer; } void AlignedFree(void* mem_block) { if (mem_block == NULL) { return; } uintptr_t aligned_pos = reinterpret_cast(mem_block); uintptr_t header_pos = aligned_pos - sizeof(uintptr_t); // Read out the address of the AlignedMemory struct from the header. uintptr_t memory_start_pos = *reinterpret_cast(header_pos); void* memory_start = reinterpret_cast(memory_start_pos); free(memory_start); } } // namespace webrtc