| /* |
| * Copyright 2015 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "SkVarAlloc.h" |
| |
| // We use non-standard malloc diagnostic methods to make sure our allocations are sized well. |
| #if defined(SK_BUILD_FOR_MAC) |
| #include <malloc/malloc.h> |
| #elif defined(SK_BUILD_FOR_UNIX) || defined(SK_BUILD_FOR_WIN32) |
| #include <malloc.h> |
| #endif |
| |
| struct SkVarAlloc::Block { |
| Block* prev; |
| char* data() { return (char*)(this + 1); } |
| |
| static Block* Alloc(Block* prev, size_t size, unsigned flags) { |
| SkASSERT(size >= sizeof(Block)); |
| Block* b = (Block*)sk_malloc_flags(size, flags); |
| b->prev = prev; |
| return b; |
| } |
| }; |
| |
| SkVarAlloc::SkVarAlloc(size_t minLgSize) |
| : fBytesAllocated(0) |
| , fByte(NULL) |
| , fRemaining(0) |
| , fLgSize(minLgSize) |
| , fBlock(NULL) {} |
| |
| SkVarAlloc::SkVarAlloc(size_t minLgSize, char* storage, size_t len) |
| : fBytesAllocated(0) |
| , fByte(storage) |
| , fRemaining(len) |
| , fLgSize(minLgSize) |
| , fBlock(NULL) {} |
| |
| SkVarAlloc::~SkVarAlloc() { |
| Block* b = fBlock; |
| while (b) { |
| Block* prev = b->prev; |
| sk_free(b); |
| b = prev; |
| } |
| } |
| |
| void SkVarAlloc::makeSpace(size_t bytes, unsigned flags) { |
| SkASSERT(SkIsAlignPtr(bytes)); |
| |
| size_t alloc = 1<<fLgSize++; |
| while (alloc < bytes + sizeof(Block)) { |
| alloc *= 2; |
| } |
| fBytesAllocated += alloc; |
| fBlock = Block::Alloc(fBlock, alloc, flags); |
| fByte = fBlock->data(); |
| fRemaining = alloc - sizeof(Block); |
| |
| #if defined(SK_BUILD_FOR_MAC) |
| SkASSERT(alloc == malloc_good_size(alloc)); |
| #elif defined(SK_BUILD_FOR_UNIX) && !defined(__UCLIBC__) |
| // TODO(mtklein): tune so we can assert something like this |
| //SkASSERT(alloc == malloc_usable_size(fBlock)); |
| #endif |
| } |