2016-07-21 06:06:41 +08:00
|
|
|
//===-- sanitizer_allocator_local_cache.h -----------------------*- C++ -*-===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2016-07-21 06:06:41 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Part of the Sanitizer Allocator.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef SANITIZER_ALLOCATOR_H
|
|
|
|
#error This file must be included inside sanitizer_allocator.h
|
|
|
|
#endif
|
|
|
|
|
2016-08-23 08:30:43 +08:00
|
|
|
// Cache used by SizeClassAllocator64.
|
|
|
|
template <class SizeClassAllocator>
|
|
|
|
struct SizeClassAllocator64LocalCache {
|
|
|
|
typedef SizeClassAllocator Allocator;
|
|
|
|
|
|
|
|
void Init(AllocatorGlobalStats *s) {
|
|
|
|
stats_.Init();
|
|
|
|
if (s)
|
|
|
|
s->Register(&stats_);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Destroy(SizeClassAllocator *allocator, AllocatorGlobalStats *s) {
|
|
|
|
Drain(allocator);
|
|
|
|
if (s)
|
|
|
|
s->Unregister(&stats_);
|
|
|
|
}
|
|
|
|
|
|
|
|
void *Allocate(SizeClassAllocator *allocator, uptr class_id) {
|
|
|
|
CHECK_NE(class_id, 0UL);
|
|
|
|
CHECK_LT(class_id, kNumClasses);
|
|
|
|
PerClass *c = &per_class_[class_id];
|
2017-06-27 06:54:10 +08:00
|
|
|
if (UNLIKELY(c->count == 0)) {
|
|
|
|
if (UNLIKELY(!Refill(c, allocator, class_id)))
|
|
|
|
return nullptr;
|
2018-02-06 02:06:45 +08:00
|
|
|
DCHECK_GT(c->count, 0);
|
2017-06-27 06:54:10 +08:00
|
|
|
}
|
2016-08-25 05:20:10 +08:00
|
|
|
CompactPtrT chunk = c->chunks[--c->count];
|
2018-02-06 02:06:45 +08:00
|
|
|
stats_.Add(AllocatorStatAllocated, c->class_size);
|
|
|
|
return reinterpret_cast<void *>(allocator->CompactPtrToPointer(
|
2016-08-25 05:20:10 +08:00
|
|
|
allocator->GetRegionBeginBySizeClass(class_id), chunk));
|
2016-08-23 08:30:43 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void Deallocate(SizeClassAllocator *allocator, uptr class_id, void *p) {
|
|
|
|
CHECK_NE(class_id, 0UL);
|
|
|
|
CHECK_LT(class_id, kNumClasses);
|
|
|
|
// If the first allocator call on a new thread is a deallocation, then
|
|
|
|
// max_count will be zero, leading to check failure.
|
|
|
|
PerClass *c = &per_class_[class_id];
|
2018-02-06 02:06:45 +08:00
|
|
|
InitCache(c);
|
2016-08-23 08:30:43 +08:00
|
|
|
if (UNLIKELY(c->count == c->max_count))
|
2016-08-25 05:20:10 +08:00
|
|
|
Drain(c, allocator, class_id, c->max_count / 2);
|
|
|
|
CompactPtrT chunk = allocator->PointerToCompactPtr(
|
|
|
|
allocator->GetRegionBeginBySizeClass(class_id),
|
|
|
|
reinterpret_cast<uptr>(p));
|
|
|
|
c->chunks[c->count++] = chunk;
|
2018-02-06 02:06:45 +08:00
|
|
|
stats_.Sub(AllocatorStatAllocated, c->class_size);
|
2016-08-23 08:30:43 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void Drain(SizeClassAllocator *allocator) {
|
[sanitizer] Size class map & local cache improvements
Summary:
- Reland rL324263, this time allowing for a compile-time decision as to whether
or not use the 32-bit division. A single test is using a class map covering
a maximum size greater than 4GB, this can be checked via the template
parameters, and allows SizeClassAllocator64PopulateFreeListOOM to pass;
- `MaxCachedHint` is always called on a class id for which we have already
computed the size, but we still recompute `Size(class_id)`. Change the
prototype of the function to work on sizes instead of class ids. This also
allows us to get rid of the `kBatchClassID` special case. Update the callers
accordingly;
- `InitCache` and `Drain` will start iterating at index 1: index 0 contents are
unused and can safely be left to be 0. Plus we do not pay the cost of going
through an `UNLIKELY` in `MaxCachedHint`, and touching memory that is
otherwise not used;
- `const` some variables in the areas modified;
- Remove an spurious extra line at the end of a file.
Reviewers: alekseyshl, tl0gic, dberris
Reviewed By: alekseyshl, dberris
Subscribers: dberris, kubamracek, delcypher, llvm-commits, #sanitizers
Differential Revision: https://reviews.llvm.org/D43088
llvm-svn: 324906
2018-02-13 00:59:17 +08:00
|
|
|
for (uptr i = 1; i < kNumClasses; i++) {
|
2017-08-28 23:20:02 +08:00
|
|
|
PerClass *c = &per_class_[i];
|
2016-08-23 08:30:43 +08:00
|
|
|
while (c->count > 0)
|
2017-08-28 23:20:02 +08:00
|
|
|
Drain(c, allocator, i, c->count);
|
2016-08-23 08:30:43 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-28 23:20:02 +08:00
|
|
|
private:
|
|
|
|
typedef typename Allocator::SizeClassMapT SizeClassMap;
|
|
|
|
static const uptr kNumClasses = SizeClassMap::kNumClasses;
|
|
|
|
typedef typename Allocator::CompactPtrT CompactPtrT;
|
|
|
|
|
2016-08-23 08:30:43 +08:00
|
|
|
struct PerClass {
|
2016-08-25 05:20:10 +08:00
|
|
|
u32 count;
|
|
|
|
u32 max_count;
|
2017-04-14 00:49:16 +08:00
|
|
|
uptr class_size;
|
2016-08-25 05:20:10 +08:00
|
|
|
CompactPtrT chunks[2 * SizeClassMap::kMaxNumCachedHint];
|
2016-08-23 08:30:43 +08:00
|
|
|
};
|
|
|
|
PerClass per_class_[kNumClasses];
|
|
|
|
AllocatorStats stats_;
|
|
|
|
|
2018-02-06 02:06:45 +08:00
|
|
|
void InitCache(PerClass *c) {
|
|
|
|
if (LIKELY(c->max_count))
|
2016-08-23 08:30:43 +08:00
|
|
|
return;
|
[sanitizer] Size class map & local cache improvements
Summary:
- Reland rL324263, this time allowing for a compile-time decision as to whether
or not use the 32-bit division. A single test is using a class map covering
a maximum size greater than 4GB, this can be checked via the template
parameters, and allows SizeClassAllocator64PopulateFreeListOOM to pass;
- `MaxCachedHint` is always called on a class id for which we have already
computed the size, but we still recompute `Size(class_id)`. Change the
prototype of the function to work on sizes instead of class ids. This also
allows us to get rid of the `kBatchClassID` special case. Update the callers
accordingly;
- `InitCache` and `Drain` will start iterating at index 1: index 0 contents are
unused and can safely be left to be 0. Plus we do not pay the cost of going
through an `UNLIKELY` in `MaxCachedHint`, and touching memory that is
otherwise not used;
- `const` some variables in the areas modified;
- Remove an spurious extra line at the end of a file.
Reviewers: alekseyshl, tl0gic, dberris
Reviewed By: alekseyshl, dberris
Subscribers: dberris, kubamracek, delcypher, llvm-commits, #sanitizers
Differential Revision: https://reviews.llvm.org/D43088
llvm-svn: 324906
2018-02-13 00:59:17 +08:00
|
|
|
for (uptr i = 1; i < kNumClasses; i++) {
|
2016-08-23 08:30:43 +08:00
|
|
|
PerClass *c = &per_class_[i];
|
[sanitizer] Size class map & local cache improvements
Summary:
- Reland rL324263, this time allowing for a compile-time decision as to whether
or not use the 32-bit division. A single test is using a class map covering
a maximum size greater than 4GB, this can be checked via the template
parameters, and allows SizeClassAllocator64PopulateFreeListOOM to pass;
- `MaxCachedHint` is always called on a class id for which we have already
computed the size, but we still recompute `Size(class_id)`. Change the
prototype of the function to work on sizes instead of class ids. This also
allows us to get rid of the `kBatchClassID` special case. Update the callers
accordingly;
- `InitCache` and `Drain` will start iterating at index 1: index 0 contents are
unused and can safely be left to be 0. Plus we do not pay the cost of going
through an `UNLIKELY` in `MaxCachedHint`, and touching memory that is
otherwise not used;
- `const` some variables in the areas modified;
- Remove an spurious extra line at the end of a file.
Reviewers: alekseyshl, tl0gic, dberris
Reviewed By: alekseyshl, dberris
Subscribers: dberris, kubamracek, delcypher, llvm-commits, #sanitizers
Differential Revision: https://reviews.llvm.org/D43088
llvm-svn: 324906
2018-02-13 00:59:17 +08:00
|
|
|
const uptr size = Allocator::ClassIdToSize(i);
|
|
|
|
c->max_count = 2 * SizeClassMap::MaxCachedHint(size);
|
|
|
|
c->class_size = size;
|
2016-08-23 08:30:43 +08:00
|
|
|
}
|
2018-02-06 02:06:45 +08:00
|
|
|
DCHECK_NE(c->max_count, 0UL);
|
2016-08-23 08:30:43 +08:00
|
|
|
}
|
|
|
|
|
2017-06-27 06:54:10 +08:00
|
|
|
NOINLINE bool Refill(PerClass *c, SizeClassAllocator *allocator,
|
2016-08-25 05:20:10 +08:00
|
|
|
uptr class_id) {
|
2018-02-06 02:06:45 +08:00
|
|
|
InitCache(c);
|
|
|
|
const uptr num_requested_chunks = c->max_count / 2;
|
2017-06-27 06:54:10 +08:00
|
|
|
if (UNLIKELY(!allocator->GetFromAllocator(&stats_, class_id, c->chunks,
|
|
|
|
num_requested_chunks)))
|
|
|
|
return false;
|
2016-08-25 05:20:10 +08:00
|
|
|
c->count = num_requested_chunks;
|
2017-06-27 06:54:10 +08:00
|
|
|
return true;
|
2016-08-23 08:30:43 +08:00
|
|
|
}
|
|
|
|
|
2016-08-25 05:20:10 +08:00
|
|
|
NOINLINE void Drain(PerClass *c, SizeClassAllocator *allocator, uptr class_id,
|
|
|
|
uptr count) {
|
|
|
|
CHECK_GE(c->count, count);
|
2018-02-06 02:06:45 +08:00
|
|
|
const uptr first_idx_to_drain = c->count - count;
|
2016-08-25 05:20:10 +08:00
|
|
|
c->count -= count;
|
|
|
|
allocator->ReturnToAllocator(&stats_, class_id,
|
|
|
|
&c->chunks[first_idx_to_drain], count);
|
2016-08-23 08:30:43 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Cache used by SizeClassAllocator32.
|
|
|
|
template <class SizeClassAllocator>
|
|
|
|
struct SizeClassAllocator32LocalCache {
|
2016-07-21 06:06:41 +08:00
|
|
|
typedef SizeClassAllocator Allocator;
|
2016-08-06 09:24:11 +08:00
|
|
|
typedef typename Allocator::TransferBatch TransferBatch;
|
2016-07-21 06:06:41 +08:00
|
|
|
|
|
|
|
void Init(AllocatorGlobalStats *s) {
|
|
|
|
stats_.Init();
|
|
|
|
if (s)
|
|
|
|
s->Register(&stats_);
|
|
|
|
}
|
|
|
|
|
2017-08-28 23:20:02 +08:00
|
|
|
// Returns a TransferBatch suitable for class_id.
|
|
|
|
TransferBatch *CreateBatch(uptr class_id, SizeClassAllocator *allocator,
|
|
|
|
TransferBatch *b) {
|
|
|
|
if (uptr batch_class_id = per_class_[class_id].batch_class_id)
|
|
|
|
return (TransferBatch*)Allocate(allocator, batch_class_id);
|
|
|
|
return b;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Destroys TransferBatch b.
|
|
|
|
void DestroyBatch(uptr class_id, SizeClassAllocator *allocator,
|
|
|
|
TransferBatch *b) {
|
|
|
|
if (uptr batch_class_id = per_class_[class_id].batch_class_id)
|
|
|
|
Deallocate(allocator, batch_class_id, b);
|
|
|
|
}
|
|
|
|
|
2016-07-21 06:06:41 +08:00
|
|
|
void Destroy(SizeClassAllocator *allocator, AllocatorGlobalStats *s) {
|
|
|
|
Drain(allocator);
|
|
|
|
if (s)
|
|
|
|
s->Unregister(&stats_);
|
|
|
|
}
|
|
|
|
|
|
|
|
void *Allocate(SizeClassAllocator *allocator, uptr class_id) {
|
|
|
|
CHECK_NE(class_id, 0UL);
|
|
|
|
CHECK_LT(class_id, kNumClasses);
|
|
|
|
PerClass *c = &per_class_[class_id];
|
2017-06-22 08:02:37 +08:00
|
|
|
if (UNLIKELY(c->count == 0)) {
|
2018-02-06 02:06:45 +08:00
|
|
|
if (UNLIKELY(!Refill(c, allocator, class_id)))
|
2017-06-22 08:02:37 +08:00
|
|
|
return nullptr;
|
2018-02-06 02:06:45 +08:00
|
|
|
DCHECK_GT(c->count, 0);
|
2017-06-22 08:02:37 +08:00
|
|
|
}
|
2016-07-21 06:06:41 +08:00
|
|
|
void *res = c->batch[--c->count];
|
|
|
|
PREFETCH(c->batch[c->count - 1]);
|
2018-02-06 02:06:45 +08:00
|
|
|
stats_.Add(AllocatorStatAllocated, c->class_size);
|
2016-07-21 06:06:41 +08:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Deallocate(SizeClassAllocator *allocator, uptr class_id, void *p) {
|
|
|
|
CHECK_NE(class_id, 0UL);
|
|
|
|
CHECK_LT(class_id, kNumClasses);
|
|
|
|
// If the first allocator call on a new thread is a deallocation, then
|
|
|
|
// max_count will be zero, leading to check failure.
|
|
|
|
PerClass *c = &per_class_[class_id];
|
2018-02-06 02:06:45 +08:00
|
|
|
InitCache(c);
|
2016-07-21 06:06:41 +08:00
|
|
|
if (UNLIKELY(c->count == c->max_count))
|
2018-02-06 02:06:45 +08:00
|
|
|
Drain(c, allocator, class_id);
|
2016-07-21 06:06:41 +08:00
|
|
|
c->batch[c->count++] = p;
|
2018-02-06 02:06:45 +08:00
|
|
|
stats_.Sub(AllocatorStatAllocated, c->class_size);
|
2016-07-21 06:06:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void Drain(SizeClassAllocator *allocator) {
|
[sanitizer] Size class map & local cache improvements
Summary:
- Reland rL324263, this time allowing for a compile-time decision as to whether
or not use the 32-bit division. A single test is using a class map covering
a maximum size greater than 4GB, this can be checked via the template
parameters, and allows SizeClassAllocator64PopulateFreeListOOM to pass;
- `MaxCachedHint` is always called on a class id for which we have already
computed the size, but we still recompute `Size(class_id)`. Change the
prototype of the function to work on sizes instead of class ids. This also
allows us to get rid of the `kBatchClassID` special case. Update the callers
accordingly;
- `InitCache` and `Drain` will start iterating at index 1: index 0 contents are
unused and can safely be left to be 0. Plus we do not pay the cost of going
through an `UNLIKELY` in `MaxCachedHint`, and touching memory that is
otherwise not used;
- `const` some variables in the areas modified;
- Remove an spurious extra line at the end of a file.
Reviewers: alekseyshl, tl0gic, dberris
Reviewed By: alekseyshl, dberris
Subscribers: dberris, kubamracek, delcypher, llvm-commits, #sanitizers
Differential Revision: https://reviews.llvm.org/D43088
llvm-svn: 324906
2018-02-13 00:59:17 +08:00
|
|
|
for (uptr i = 1; i < kNumClasses; i++) {
|
2017-08-28 23:20:02 +08:00
|
|
|
PerClass *c = &per_class_[i];
|
2016-07-21 06:06:41 +08:00
|
|
|
while (c->count > 0)
|
2018-02-06 02:06:45 +08:00
|
|
|
Drain(c, allocator, i);
|
2016-07-21 06:06:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-28 23:20:02 +08:00
|
|
|
private:
|
|
|
|
typedef typename Allocator::SizeClassMapT SizeClassMap;
|
|
|
|
static const uptr kBatchClassID = SizeClassMap::kBatchClassID;
|
|
|
|
static const uptr kNumClasses = SizeClassMap::kNumClasses;
|
|
|
|
// If kUseSeparateSizeClassForBatch is true, all TransferBatch objects are
|
|
|
|
// allocated from kBatchClassID size class (except for those that are needed
|
|
|
|
// for kBatchClassID itself). The goal is to have TransferBatches in a totally
|
|
|
|
// different region of RAM to improve security.
|
|
|
|
static const bool kUseSeparateSizeClassForBatch =
|
|
|
|
Allocator::kUseSeparateSizeClassForBatch;
|
|
|
|
|
2016-07-21 06:06:41 +08:00
|
|
|
struct PerClass {
|
|
|
|
uptr count;
|
|
|
|
uptr max_count;
|
2017-04-14 00:49:16 +08:00
|
|
|
uptr class_size;
|
2017-08-28 23:20:02 +08:00
|
|
|
uptr batch_class_id;
|
2016-08-10 07:30:22 +08:00
|
|
|
void *batch[2 * TransferBatch::kMaxNumCached];
|
2016-07-21 06:06:41 +08:00
|
|
|
};
|
|
|
|
PerClass per_class_[kNumClasses];
|
|
|
|
AllocatorStats stats_;
|
|
|
|
|
2018-02-06 02:06:45 +08:00
|
|
|
void InitCache(PerClass *c) {
|
|
|
|
if (LIKELY(c->max_count))
|
2016-07-21 06:06:41 +08:00
|
|
|
return;
|
2017-08-28 23:20:02 +08:00
|
|
|
const uptr batch_class_id = SizeClassMap::ClassID(sizeof(TransferBatch));
|
[sanitizer] Size class map & local cache improvements
Summary:
- Reland rL324263, this time allowing for a compile-time decision as to whether
or not use the 32-bit division. A single test is using a class map covering
a maximum size greater than 4GB, this can be checked via the template
parameters, and allows SizeClassAllocator64PopulateFreeListOOM to pass;
- `MaxCachedHint` is always called on a class id for which we have already
computed the size, but we still recompute `Size(class_id)`. Change the
prototype of the function to work on sizes instead of class ids. This also
allows us to get rid of the `kBatchClassID` special case. Update the callers
accordingly;
- `InitCache` and `Drain` will start iterating at index 1: index 0 contents are
unused and can safely be left to be 0. Plus we do not pay the cost of going
through an `UNLIKELY` in `MaxCachedHint`, and touching memory that is
otherwise not used;
- `const` some variables in the areas modified;
- Remove an spurious extra line at the end of a file.
Reviewers: alekseyshl, tl0gic, dberris
Reviewed By: alekseyshl, dberris
Subscribers: dberris, kubamracek, delcypher, llvm-commits, #sanitizers
Differential Revision: https://reviews.llvm.org/D43088
llvm-svn: 324906
2018-02-13 00:59:17 +08:00
|
|
|
for (uptr i = 1; i < kNumClasses; i++) {
|
2016-07-21 06:06:41 +08:00
|
|
|
PerClass *c = &per_class_[i];
|
[sanitizer] Size class map & local cache improvements
Summary:
- Reland rL324263, this time allowing for a compile-time decision as to whether
or not use the 32-bit division. A single test is using a class map covering
a maximum size greater than 4GB, this can be checked via the template
parameters, and allows SizeClassAllocator64PopulateFreeListOOM to pass;
- `MaxCachedHint` is always called on a class id for which we have already
computed the size, but we still recompute `Size(class_id)`. Change the
prototype of the function to work on sizes instead of class ids. This also
allows us to get rid of the `kBatchClassID` special case. Update the callers
accordingly;
- `InitCache` and `Drain` will start iterating at index 1: index 0 contents are
unused and can safely be left to be 0. Plus we do not pay the cost of going
through an `UNLIKELY` in `MaxCachedHint`, and touching memory that is
otherwise not used;
- `const` some variables in the areas modified;
- Remove an spurious extra line at the end of a file.
Reviewers: alekseyshl, tl0gic, dberris
Reviewed By: alekseyshl, dberris
Subscribers: dberris, kubamracek, delcypher, llvm-commits, #sanitizers
Differential Revision: https://reviews.llvm.org/D43088
llvm-svn: 324906
2018-02-13 00:59:17 +08:00
|
|
|
const uptr size = Allocator::ClassIdToSize(i);
|
|
|
|
const uptr max_cached = TransferBatch::MaxCached(size);
|
2017-04-24 22:53:38 +08:00
|
|
|
c->max_count = 2 * max_cached;
|
[sanitizer] Size class map & local cache improvements
Summary:
- Reland rL324263, this time allowing for a compile-time decision as to whether
or not use the 32-bit division. A single test is using a class map covering
a maximum size greater than 4GB, this can be checked via the template
parameters, and allows SizeClassAllocator64PopulateFreeListOOM to pass;
- `MaxCachedHint` is always called on a class id for which we have already
computed the size, but we still recompute `Size(class_id)`. Change the
prototype of the function to work on sizes instead of class ids. This also
allows us to get rid of the `kBatchClassID` special case. Update the callers
accordingly;
- `InitCache` and `Drain` will start iterating at index 1: index 0 contents are
unused and can safely be left to be 0. Plus we do not pay the cost of going
through an `UNLIKELY` in `MaxCachedHint`, and touching memory that is
otherwise not used;
- `const` some variables in the areas modified;
- Remove an spurious extra line at the end of a file.
Reviewers: alekseyshl, tl0gic, dberris
Reviewed By: alekseyshl, dberris
Subscribers: dberris, kubamracek, delcypher, llvm-commits, #sanitizers
Differential Revision: https://reviews.llvm.org/D43088
llvm-svn: 324906
2018-02-13 00:59:17 +08:00
|
|
|
c->class_size = size;
|
2017-08-28 23:20:02 +08:00
|
|
|
// Precompute the class id to use to store batches for the current class
|
|
|
|
// id. 0 means the class size is large enough to store a batch within one
|
|
|
|
// of the chunks. If using a separate size class, it will always be
|
|
|
|
// kBatchClassID, except for kBatchClassID itself.
|
|
|
|
if (kUseSeparateSizeClassForBatch) {
|
|
|
|
c->batch_class_id = (i == kBatchClassID) ? 0 : kBatchClassID;
|
|
|
|
} else {
|
[sanitizer] Size class map & local cache improvements
Summary:
- Reland rL324263, this time allowing for a compile-time decision as to whether
or not use the 32-bit division. A single test is using a class map covering
a maximum size greater than 4GB, this can be checked via the template
parameters, and allows SizeClassAllocator64PopulateFreeListOOM to pass;
- `MaxCachedHint` is always called on a class id for which we have already
computed the size, but we still recompute `Size(class_id)`. Change the
prototype of the function to work on sizes instead of class ids. This also
allows us to get rid of the `kBatchClassID` special case. Update the callers
accordingly;
- `InitCache` and `Drain` will start iterating at index 1: index 0 contents are
unused and can safely be left to be 0. Plus we do not pay the cost of going
through an `UNLIKELY` in `MaxCachedHint`, and touching memory that is
otherwise not used;
- `const` some variables in the areas modified;
- Remove an spurious extra line at the end of a file.
Reviewers: alekseyshl, tl0gic, dberris
Reviewed By: alekseyshl, dberris
Subscribers: dberris, kubamracek, delcypher, llvm-commits, #sanitizers
Differential Revision: https://reviews.llvm.org/D43088
llvm-svn: 324906
2018-02-13 00:59:17 +08:00
|
|
|
c->batch_class_id = (size <
|
2017-04-24 22:53:38 +08:00
|
|
|
TransferBatch::AllocationSizeRequiredForNElements(max_cached)) ?
|
2017-08-28 23:20:02 +08:00
|
|
|
batch_class_id : 0;
|
|
|
|
}
|
2016-07-21 06:06:41 +08:00
|
|
|
}
|
2018-02-06 02:06:45 +08:00
|
|
|
DCHECK_NE(c->max_count, 0UL);
|
2016-07-21 06:06:41 +08:00
|
|
|
}
|
|
|
|
|
2018-02-06 02:06:45 +08:00
|
|
|
NOINLINE bool Refill(PerClass *c, SizeClassAllocator *allocator,
|
|
|
|
uptr class_id) {
|
|
|
|
InitCache(c);
|
2016-08-06 09:24:11 +08:00
|
|
|
TransferBatch *b = allocator->AllocateBatch(&stats_, this, class_id);
|
2017-06-22 08:02:37 +08:00
|
|
|
if (UNLIKELY(!b))
|
|
|
|
return false;
|
2016-08-03 08:14:10 +08:00
|
|
|
CHECK_GT(b->Count(), 0);
|
2016-08-10 04:54:50 +08:00
|
|
|
b->CopyToArray(c->batch);
|
2016-08-03 08:14:10 +08:00
|
|
|
c->count = b->Count();
|
2016-07-22 02:47:53 +08:00
|
|
|
DestroyBatch(class_id, allocator, b);
|
2017-06-22 08:02:37 +08:00
|
|
|
return true;
|
2016-07-21 06:06:41 +08:00
|
|
|
}
|
|
|
|
|
2018-02-06 02:06:45 +08:00
|
|
|
NOINLINE void Drain(PerClass *c, SizeClassAllocator *allocator,
|
|
|
|
uptr class_id) {
|
|
|
|
const uptr count = Min(c->max_count / 2, c->count);
|
|
|
|
const uptr first_idx_to_drain = c->count - count;
|
2016-08-06 09:24:11 +08:00
|
|
|
TransferBatch *b = CreateBatch(
|
|
|
|
class_id, allocator, (TransferBatch *)c->batch[first_idx_to_drain]);
|
2017-06-22 08:02:37 +08:00
|
|
|
// Failure to allocate a batch while releasing memory is non recoverable.
|
|
|
|
// TODO(alekseys): Figure out how to do it without allocating a new batch.
|
2018-06-21 01:10:33 +08:00
|
|
|
if (UNLIKELY(!b)) {
|
|
|
|
Report("FATAL: Internal error: %s's allocator failed to allocate a "
|
|
|
|
"transfer batch.\n", SanitizerToolName);
|
|
|
|
Die();
|
|
|
|
}
|
2018-05-04 03:14:07 +08:00
|
|
|
b->SetFromArray(&c->batch[first_idx_to_drain], count);
|
2018-02-06 02:06:45 +08:00
|
|
|
c->count -= count;
|
2016-07-21 06:06:41 +08:00
|
|
|
allocator->DeallocateBatch(&stats_, class_id, b);
|
|
|
|
}
|
|
|
|
};
|