|  | /* | 
|  | * 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 | 
|  | } |