| /* |
| * 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 <stdlib.h> // for free, malloc |
| #include <string.h> // for memcpy |
| |
| #include "rtc_base/checks.h" |
| |
| #ifdef _WIN32 |
| #include <windows.h> |
| #else |
| #include <stdint.h> |
| #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<uintptr_t>(pointer); |
| return reinterpret_cast<void*>(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<uintptr_t>(memory_pointer); |
| align_start_pos += sizeof(uintptr_t); |
| uintptr_t aligned_pos = GetRightAlign(align_start_pos, alignment); |
| void* aligned_pointer = reinterpret_cast<void*>(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<void*>(header_pos); |
| uintptr_t memory_start = reinterpret_cast<uintptr_t>(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<uintptr_t>(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<uintptr_t*>(header_pos); |
| void* memory_start = reinterpret_cast<void*>(memory_start_pos); |
| free(memory_start); |
| } |
| |
| } // namespace webrtc |