| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140 |
- // zlib open source license
- //
- // Copyright (c) 2017 to 2023 David Forsgren Piuva
- //
- // This software is provided 'as-is', without any express or implied
- // warranty. In no event will the authors be held liable for any damages
- // arising from the use of this software.
- //
- // Permission is granted to anyone to use this software for any purpose,
- // including commercial applications, and to alter it and redistribute it
- // freely, subject to the following restrictions:
- //
- // 1. The origin of this software must not be misrepresented; you must not
- // claim that you wrote the original software. If you use this software
- // in a product, an acknowledgment in the product documentation would be
- // appreciated but is not required.
- //
- // 2. Altered source versions must be plainly marked as such, and must not be
- // misrepresented as being the original software.
- //
- // 3. This notice may not be removed or altered from any source
- // distribution.
- #include "SafePointer.h"
- #include "../api/stringAPI.h"
- #ifdef SAFE_POINTER_CHECKS
- #include <thread>
- #include <mutex>
- #endif
- using namespace dsr;
- // Thread hash of memory without any specific owner.
- static uint64_t ANY_THREAD_HASH = 0xF986BA1496E872A5;
- #ifdef SAFE_POINTER_CHECKS
- // Hashed thread identity.
- std::hash<std::thread::id> hasher;
- thread_local uint64_t threadHash = hasher(std::this_thread::get_id());
- // Globally unique identifiers for memory allocations.
- // Different allocations can have the same address at different times when allocations are recycled,
- // so a globally unique identifier is needed to make sure that we access the same allocation.
- static std::mutex idLock;
- static uint64_t idCounter = 0xD13A98271E08BF57;
- static uint64_t createIdentity() {
- uint64_t result;
- idLock.lock();
- result = idCounter;
- idCounter++;
- idLock.unlock();
- return result;
- }
- AllocationHeader::AllocationHeader()
- : totalSize(0), threadHash(0), allocationIdentity(0) {}
- AllocationHeader::AllocationHeader(uintptr_t totalSize, bool threadLocal)
- : totalSize(totalSize), threadHash(threadLocal ? threadHash : ANY_THREAD_HASH), allocationIdentity(createIdentity()) {
- }
- #else
- AllocationHeader::AllocationHeader()
- : totalSize(0) {}
- AllocationHeader::AllocationHeader(uintptr_t totalSize, bool threadLocal)
- : totalSize(totalSize) {}
- #endif
- #ifdef SAFE_POINTER_CHECKS
- void dsr::assertNonNegativeSize(intptr_t size) {
- if (size < 0) {
- throwError(U"Negative size of SafePointer!\n");
- }
- }
- void dsr::assertInsideSafePointer(const char* method, const char* name, const uint8_t* pointer, const uint8_t* data, const uint8_t* regionStart, const uint8_t* regionEnd, const AllocationHeader *header, uint64_t allocationIdentity, intptr_t claimedSize, intptr_t elementSize) {
- // If the pointer has an allocation header, check that the identity matches the one stored in the pointer.
- if (header != nullptr) {
- // TODO: Print more useful information.
- try {
- // Both allocation identity and thread hash may match by mistake, but in most of the cases this will give more information about why it happened.
- uint64_t headerIdentity = header->allocationIdentity;
- uint64_t headerHash = header->threadHash;
- if (headerIdentity != allocationIdentity) {
- throwError(U"SafePointer exception! Accessing freed memory or currupted allocation header!\n");
- return;
- } else if (headerHash != ANY_THREAD_HASH && headerHash != threadHash) {
- throwError(U"SafePointer exception! Accessing another thread's private memory!\n");
- return;
- }
- } catch(...) {
- throwError(U"SafePointer exception! Tried to access memory not available to the application!\n");
- return;
- }
- }
- if (regionStart == nullptr) {
- throwError(U"SafePointer exception! Tried to use a null pointer!\n");
- return;
- }
- if (pointer < regionStart || pointer + claimedSize > regionEnd) {
- String message;
- string_append(message, U"\n _________________ SafePointer out of bound exception! _________________\n");
- string_append(message, U"/\n");
- string_append(message, U"| Name: ", name, U"\n");
- string_append(message, U"| Method: ", method, U"\n");
- string_append(message, U"| Region: ", (uintptr_t)regionStart, U" to ", (uintptr_t)regionEnd, U"\n");
- string_append(message, U"| Region size: ", (intptr_t)(regionEnd - regionStart), U" bytes\n");
- string_append(message, U"| Base pointer: ", (uintptr_t)data, U"\n");
- string_append(message, U"| Requested pointer: ", (uintptr_t)pointer, U"\n");
- string_append(message, U"| Requested size: ", claimedSize, U" bytes\n");
- intptr_t startOffset = (intptr_t)pointer - (intptr_t)regionStart;
- intptr_t baseOffset = (intptr_t)pointer - (intptr_t)data;
- // Index relative to allocation start
- // regionStart is the start of the accessible memory region
- if (startOffset != baseOffset) {
- string_append(message, U"| Start offset: ", startOffset, U" bytes\n");
- if (startOffset % elementSize == 0) {
- intptr_t index = startOffset / elementSize;
- intptr_t elementCount = ((intptr_t)regionEnd - (intptr_t)regionStart) / elementSize;
- string_append(message, U"| Start index: ", index, U" [0..", (elementCount - 1), U"]\n");
- }
- }
- // Base index relative to the stored pointer within the region
- // data is the base of the allocation at index zero
- string_append(message, U"| Base offset: ", baseOffset, U" bytes\n");
- if (baseOffset % elementSize == 0) {
- intptr_t index = baseOffset / elementSize;
- intptr_t elementCount = ((intptr_t)regionEnd - (intptr_t)data) / elementSize;
- string_append(message, U"| Base index: ", index, U" [0..", (elementCount - 1), U"]\n");
- }
- string_append(message, U"\\_______________________________________________________________________\n\n");
- throwError(message);
- return;
- }
- }
- #endif
|