| Index: base/allocator/malloc_zone_functions_mac.cc | 
| diff --git a/base/allocator/malloc_zone_functions_mac.cc b/base/allocator/malloc_zone_functions_mac.cc | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..e64719da9e63fa485e2a3a6aca2aa7644e458fba | 
| --- /dev/null | 
| +++ b/base/allocator/malloc_zone_functions_mac.cc | 
| @@ -0,0 +1,110 @@ | 
| +// Copyright 2017 The Chromium Authors. All rights reserved. | 
| +// Use of this source code is governed by a BSD-style license that can be | 
| +// found in the LICENSE file. | 
| + | 
| +#include "base/allocator/malloc_zone_functions_mac.h" | 
| + | 
| +#include "base/atomicops.h" | 
| +#include "base/synchronization/lock.h" | 
| + | 
| +namespace base { | 
| +namespace allocator { | 
| + | 
| +MallocZoneFunctions* g_malloc_zones = nullptr; | 
| +MallocZoneFunctions::MallocZoneFunctions() {} | 
| + | 
| +void StoreZoneFunctions(const ChromeMallocZone* zone, | 
| +                        MallocZoneFunctions* functions) { | 
| +  functions->malloc = zone->malloc; | 
| +  functions->calloc = zone->calloc; | 
| +  functions->valloc = zone->valloc; | 
| +  functions->free = zone->free; | 
| +  functions->realloc = zone->realloc; | 
| +  functions->size = zone->size; | 
| +  CHECK(functions->malloc && functions->calloc && functions->valloc && | 
| +        functions->free && functions->realloc && functions->size); | 
| + | 
| +  // These functions might be nullptr. | 
| +  functions->batch_malloc = zone->batch_malloc; | 
| +  functions->batch_free = zone->batch_free; | 
| + | 
| +  if (zone->version >= 5) { | 
| +    // Not all custom malloc zones have a memalign. | 
| +    functions->memalign = zone->memalign; | 
| +  } | 
| +  if (zone->version >= 6) { | 
| +    // This may be nullptr. | 
| +    functions->free_definite_size = zone->free_definite_size; | 
| +  } | 
| + | 
| +  functions->context = zone; | 
| +} | 
| + | 
| +namespace { | 
| + | 
| +// All modifications to g_malloc_zones are gated behind this lock. | 
| +// Dispatch to a malloc zone does not need to acquire this lock. | 
| +base::Lock& GetLock() { | 
| +  static base::Lock* g_lock = new base::Lock; | 
| +  return *g_lock; | 
| +} | 
| + | 
| +void EnsureMallocZonesInitializedLocked() { | 
| +  GetLock().AssertAcquired(); | 
| +  if (!g_malloc_zones) { | 
| +    g_malloc_zones = reinterpret_cast<base::allocator::MallocZoneFunctions*>( | 
| +        calloc(kMaxZoneCount, sizeof(MallocZoneFunctions))); | 
| +  } | 
| +} | 
| + | 
| +int g_zone_count = 0; | 
| + | 
| +bool IsMallocZoneAlreadyStoredLocked(ChromeMallocZone* zone) { | 
| +  EnsureMallocZonesInitializedLocked(); | 
| +  GetLock().AssertAcquired(); | 
| +  for (int i = 0; i < g_zone_count; ++i) { | 
| +    if (g_malloc_zones[i].context == reinterpret_cast<void*>(zone)) | 
| +      return true; | 
| +  } | 
| +  return false; | 
| +} | 
| + | 
| +}  // namespace | 
| + | 
| +void StoreMallocZone(ChromeMallocZone* zone) { | 
| +  base::AutoLock l(GetLock()); | 
| +  EnsureMallocZonesInitializedLocked(); | 
| +  if (IsMallocZoneAlreadyStoredLocked(zone)) | 
| +    return; | 
| + | 
| +  if (g_zone_count == kMaxZoneCount) | 
| +    return; | 
| + | 
| +  StoreZoneFunctions(zone, &g_malloc_zones[g_zone_count]); | 
| +  ++g_zone_count; | 
| + | 
| +  // No other thread can possibly see these stores at this point. The code that | 
| +  // reads these values is triggered after this function returns. so we want to | 
| +  // guarantee that they are committed at this stage" | 
| +  base::subtle::MemoryBarrier(); | 
| +} | 
| + | 
| +bool IsMallocZoneAlreadyStored(ChromeMallocZone* zone) { | 
| +  base::AutoLock l(GetLock()); | 
| +  return IsMallocZoneAlreadyStoredLocked(zone); | 
| +} | 
| + | 
| +int GetMallocZoneCountForTesting() { | 
| +  base::AutoLock l(GetLock()); | 
| +  return g_zone_count; | 
| +} | 
| + | 
| +void ClearAllMallocZonesForTesting() { | 
| +  base::AutoLock l(GetLock()); | 
| +  EnsureMallocZonesInitializedLocked(); | 
| +  memset(g_malloc_zones, 0, kMaxZoneCount * sizeof(MallocZoneFunctions)); | 
| +  g_zone_count = 0; | 
| +} | 
| + | 
| +}  // namespace allocator | 
| +}  // namespace base | 
|  |