Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright 2013 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "base/memory/discardable_memory.h" | |
| 6 | |
| 7 #include "base/memory/discardable_memory_provider.h" | |
| 8 | |
| 9 using base::internal::DiscardableMemoryProvider; | |
| 10 | |
| 11 namespace base { | |
| 12 | |
| 13 // static | |
| 14 bool DiscardableMemory::SupportedNatively() { | |
| 15 return false; | |
| 16 } | |
| 17 | |
| 18 DiscardableMemory::~DiscardableMemory() { | |
| 19 if (is_locked()) | |
| 20 Unlock(); | |
| 21 if (memory_) | |
| 22 Deallocate(); | |
| 23 DiscardableMemoryProvider::GetInstance()->Unregister(this); | |
| 24 } | |
| 25 | |
| 26 bool DiscardableMemory::InitializeAndLock(size_t size) { | |
| 27 DiscardableMemoryProvider::GetInstance()->Register(this); | |
| 28 size_ = size; | |
| 29 return Lock() == DISCARDABLE_MEMORY_PURGED; | |
| 30 } | |
| 31 | |
| 32 LockDiscardableMemoryStatus DiscardableMemory::Lock() { | |
| 33 DCHECK(!is_locked_); | |
| 34 | |
| 35 // This can fail if we have not been registered. That is, it's an error to | |
| 36 // |Lock| before calling |InitializeAndLock|. | |
| 37 if (!DiscardableMemoryProvider::GetInstance()->DidAccess(this)) { | |
| 38 NOTREACHED(); | |
| 39 return DISCARDABLE_MEMORY_FAILED; | |
| 40 } | |
| 41 | |
| 42 is_locked_ = true; | |
| 43 if (memory_) | |
| 44 return DISCARDABLE_MEMORY_SUCCESS; | |
| 45 | |
| 46 if (Allocate()) | |
| 47 return DISCARDABLE_MEMORY_PURGED; | |
| 48 | |
| 49 // Attempt to purge to get Allocate to succeed. | |
| 50 DiscardableMemoryProvider::GetInstance()->PurgeAll(); | |
|
willchan no longer on Chromium
2013/10/01 18:12:05
This is kind of a big hammer. No need to premature
reveman
2013/10/09 22:40:24
My latest patch is purging the minimal amount of L
| |
| 51 if (Allocate()) | |
| 52 return DISCARDABLE_MEMORY_PURGED; | |
| 53 | |
| 54 is_locked_ = false; | |
| 55 return DISCARDABLE_MEMORY_FAILED; | |
| 56 } | |
| 57 | |
| 58 void DiscardableMemory::Unlock() { | |
| 59 DCHECK(is_locked_); | |
| 60 DiscardableMemoryProvider::GetInstance()->DidAccess(this); | |
| 61 is_locked_ = false; | |
| 62 } | |
| 63 | |
| 64 bool DiscardableMemory::Allocate() { | |
| 65 DCHECK(!memory_); | |
| 66 DiscardableMemoryProvider* provider = | |
| 67 DiscardableMemoryProvider::GetInstance(); | |
| 68 | |
| 69 if (size_ == 0) | |
|
willchan no longer on Chromium
2013/10/01 18:12:05
Interesting. I hadn't considered this case before.
| |
| 70 return false; | |
| 71 | |
| 72 size_t limit = provider->discardable_memory_limit(); | |
| 73 if (limit > 0 && size_ > limit) | |
| 74 return false; | |
| 75 | |
| 76 memory_ = malloc(size_); | |
| 77 if (memory_) | |
| 78 provider->DidAllocate(size_); | |
| 79 | |
| 80 return !!memory_; | |
| 81 } | |
| 82 | |
| 83 void DiscardableMemory::Deallocate() { | |
| 84 DCHECK(memory_); | |
| 85 free(memory_); | |
| 86 memory_ = NULL; | |
| 87 DiscardableMemoryProvider::GetInstance()->DidDeallocate(size_); | |
| 88 } | |
| 89 | |
| 90 // static | |
| 91 bool DiscardableMemory::PurgeForTestingSupported() { | |
| 92 return true; | |
| 93 } | |
| 94 | |
| 95 // static | |
| 96 void DiscardableMemory::PurgeForTesting() { | |
| 97 DiscardableMemoryProvider::GetInstance()->PurgeAll(); | |
| 98 } | |
| 99 | |
| 100 } // namespace base | |
| OLD | NEW |