Reland "Update the SkSL pool interface to take an allocation size."

This is a reland of 22ef2257c8

This reland fixes an issue with ExternalValue nodes on 32-bit builds;
these should never be pooled, because we can't control their lifetimes.

Original change's description:
> Update the SkSL pool interface to take an allocation size.
>
> Since our end goal no longer has all IRNodes ending up as the exact same
> size in memory, it makes sense for the allocator function to take in a
> desired size. This also opens the door to separate "small" and "large"
> pools, if we want to add pooling support for large but semi-common
> things like Variables.
>
> Change-Id: If3dbe31588adeedede327c5967c344a19507b6fa
> Reviewed-on: https://skia-review.googlesource.com/c/skia/+/329961
> Reviewed-by: Brian Osman <brianosman@google.com>
> Commit-Queue: Brian Osman <brianosman@google.com>
> Auto-Submit: John Stiles <johnstiles@google.com>

Change-Id: If701ae4a5e18b66d4138bc09c9c8dc1a60579c90
Reviewed-on: https://skia-review.googlesource.com/c/skia/+/330105
Reviewed-by: Brian Osman <brianosman@google.com>
Commit-Queue: Brian Osman <brianosman@google.com>
Auto-Submit: John Stiles <johnstiles@google.com>
This commit is contained in:
John Stiles 2020-10-27 17:49:37 -04:00 committed by Skia Commit-Bot
parent 7b14f2f6e9
commit 270b5c04c2
4 changed files with 34 additions and 21 deletions

View File

@ -105,6 +105,17 @@ public:
return String("external<") + this->name() + ">"; return String("external<") + this->name() + ">";
} }
// Disable IRNode pooling on external value nodes. ExternalValue node lifetimes are controlled
// by the calling code; we can't guarantee that they will be destroyed before a Program is
// freed. (In fact, it's very unlikely that they would be.)
static void* operator new(const size_t size) {
return ::operator new(size);
}
static void operator delete(void* ptr) {
::operator delete(ptr);
}
private: private:
using INHERITED = Symbol; using INHERITED = Symbol;

View File

@ -14,9 +14,12 @@
namespace SkSL { namespace SkSL {
static constexpr int kSmallNodeSize = 120;
static constexpr int kNodesInPool = 512;
namespace { struct IRNodeData { namespace { struct IRNodeData {
union { union {
uint8_t fBuffer[sizeof(IRNode)]; uint8_t fBuffer[kSmallNodeSize];
IRNodeData* fFreeListNext; IRNodeData* fFreeListNext;
}; };
}; } }; }
@ -116,8 +119,6 @@ Pool::~Pool() {
} }
std::unique_ptr<Pool> Pool::Create() { std::unique_ptr<Pool> Pool::Create() {
constexpr int kNodesInPool = 512;
SkAutoMutexExclusive lock(recycled_pool_mutex()); SkAutoMutexExclusive lock(recycled_pool_mutex());
std::unique_ptr<Pool> pool; std::unique_ptr<Pool> pool;
if (sRecycledPool) { if (sRecycledPool) {
@ -159,23 +160,26 @@ void Pool::detachFromThread() {
set_thread_local_pool_data(nullptr); set_thread_local_pool_data(nullptr);
} }
void* Pool::AllocIRNode() { void* Pool::AllocIRNode(size_t size) {
// Is a pool attached? // Is a pool attached?
PoolData* poolData = get_thread_local_pool_data(); PoolData* poolData = get_thread_local_pool_data();
if (poolData) { if (poolData) {
// Does the pool contain a free node? // Can the requested size fit in a pool node?
IRNodeData* node = poolData->fFreeListHead; if (size <= kSmallNodeSize) {
if (node) { // Does the pool contain a free node?
// Yes. Take a node from the freelist. IRNodeData* node = poolData->fFreeListHead;
poolData->fFreeListHead = node->fFreeListNext; if (node) {
VLOG("ALLOC Pool:0x%016llX Index:%04d 0x%016llX\n", // Yes. Take a node from the freelist.
(uint64_t)poolData, poolData->nodeIndex(node), (uint64_t)node); poolData->fFreeListHead = node->fFreeListNext;
return node->fBuffer; VLOG("ALLOC Pool:0x%016llX Index:%04d 0x%016llX\n",
(uint64_t)poolData, poolData->nodeIndex(node), (uint64_t)node);
return node->fBuffer;
}
} }
} }
// The pool is detached or full; allocate nodes using malloc. // The pool can't be used for this allocation. Allocate nodes using the system allocator.
void* ptr = ::operator new(sizeof(IRNode)); void* ptr = ::operator new(size);
VLOG("ALLOC Pool:0x%016llX Index:____ malloc 0x%016llX\n", VLOG("ALLOC Pool:0x%016llX Index:____ malloc 0x%016llX\n",
(uint64_t)poolData, (uint64_t)ptr); (uint64_t)poolData, (uint64_t)ptr);
return ptr; return ptr;
@ -197,7 +201,7 @@ void Pool::FreeIRNode(void* node_v) {
} }
} }
// No pool is attached or the node was malloced; it must be freed. // We couldn't associate this node with our pool. Free it using the system allocator.
VLOG("FREE Pool:0x%016llX Index:____ free 0x%016llX\n", VLOG("FREE Pool:0x%016llX Index:____ free 0x%016llX\n",
(uint64_t)poolData, (uint64_t)node_v); (uint64_t)poolData, (uint64_t)node_v);
::operator delete(node_v); ::operator delete(node_v);

View File

@ -41,8 +41,9 @@ public:
// It is an error to call this while no pool is attached. // It is an error to call this while no pool is attached.
void detachFromThread(); void detachFromThread();
// Retrieves a node from the thread pool. If the pool is exhausted, this will allocate a node. // Retrieves a node from the thread pool. If the pool is exhausted, or if the requested size
static void* AllocIRNode(); // exceeds the size that we can deliver from a pool, this will just allocate memory.
static void* AllocIRNode(size_t size);
// Releases a node that was created by AllocIRNode. This will return it to the pool, or free it, // Releases a node that was created by AllocIRNode. This will return it to the pool, or free it,
// as appropriate. Make sure to free all nodes, since some of them may be real allocations. // as appropriate. Make sure to free all nodes, since some of them may be real allocations.

View File

@ -67,10 +67,7 @@ public:
// Override operator new and delete to allow us to control allocation behavior. // Override operator new and delete to allow us to control allocation behavior.
static void* operator new(const size_t size) { static void* operator new(const size_t size) {
if (size == sizeof(IRNode)) { return Pool::AllocIRNode(size);
return Pool::AllocIRNode();
}
return ::operator new(size);
} }
static void operator delete(void* ptr) { static void operator delete(void* ptr) {