best_fit_allocator.cc 5.8 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
// Copyright (c) 2018 PaddlePaddle Authors. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

#include "paddle/fluid/memory/allocation/best_fit_allocator.h"
16
#include <cmath>
17 18 19 20 21 22 23 24 25

namespace paddle {
namespace memory {
namespace allocation {

static int HighestBitPos(size_t N) {
  if (UNLIKELY(N == 0)) {
    return 0;
  } else {
Y
Yu Yang 已提交
26
#ifdef __GNUCC__
S
sneaxiy 已提交
27 28
    return sizeof(unsigned int) * 8 - __builtin_clz(N);
#else
29
    return static_cast<int>(std::log2(N) + 1);
S
sneaxiy 已提交
30
#endif
31 32 33 34 35 36 37 38 39 40
  }
}

BestFitAllocator::BestFitAllocator(Allocation* allocation)
    : allocation_(allocation) {
  details::Chunk chunk;
  chunk.size_ = allocation_->size();
  chunk.offset_ = 0;
  chunk.is_free = true;
  chunks_.emplace_back(chunk);
Y
Yu Yang 已提交
41 42
  free_chunks_[HighestBitPos(chunk.size_)].insert(
      {chunk.size_, chunks_.begin()});
43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60
}

size_t BestFitAllocator::FreeSize() const {
  size_t acc = 0;
  for (auto& array_item : free_chunks_) {
    for (auto& pair : array_item) {
      acc += pair.second->size_;
    }
  }
  return acc;
}

BestFitAllocator::ListIt BestFitAllocator::SplitChunk(size_t request_size,
                                                      size_t free_chunk_offset,
                                                      MapIt bin_iterator) {
  auto to_split_it = bin_iterator->second;
  free_chunks_[free_chunk_offset].erase(bin_iterator);

61 62 63 64 65 66 67
  PADDLE_ENFORCE_EQ(to_split_it->is_free, true,
                    platform::errors::PreconditionNotMet(
                        "The memory chunk to split is not free"));
  PADDLE_ENFORCE_GE(to_split_it->size_, request_size,
                    platform::errors::PreconditionNotMet(
                        "The size of memory chunk to split is "
                        "not larger than size of request memory"));
68 69 70 71 72 73

  auto remaining_size = to_split_it->size_ - request_size;
  details::Chunk to_use;
  details::Chunk remaining;
  to_use.size_ = request_size;
  to_use.is_free = false;
Y
Yu Yang 已提交
74 75 76
  remaining.size_ = remaining_size;
  remaining.is_free = true;

77 78
  // calc offsets
  to_use.offset_ = to_split_it->offset_;
Y
Yu Yang 已提交
79
  remaining.offset_ = to_use.offset_ + to_use.size_;
80 81 82

  // insert to chunk list
  auto to_use_it = chunks_.insert(to_split_it, to_use);
Y
Yu Yang 已提交
83 84 85 86
  if (remaining.size_ != 0) {
    auto bit_size = static_cast<size_t>(HighestBitPos(remaining.size_));
    free_chunks_[bit_size].insert(
        {remaining.size_, chunks_.insert(to_split_it, remaining)});
87 88 89 90 91
  }
  chunks_.erase(to_split_it);
  return to_use_it;
}

Y
Yu Yang 已提交
92 93 94 95 96 97 98 99 100 101 102 103
void BestFitAllocator::InsertFreeNode(const ListIt& it) {
  auto pos = static_cast<size_t>(HighestBitPos(it->size_));
  auto& free_map = free_chunks_[pos];
  free_map.insert({it->size_, it});
}
void BestFitAllocator::EraseFreeNode(const ListIt& it) {
  size_t pos = static_cast<size_t>(HighestBitPos(it->size_));
  auto& free_map = free_chunks_[pos];
  auto map_it = free_map.find(it->size_);
  while (map_it->second != it && map_it != free_map.end()) {
    ++map_it;
  }
104 105 106
  PADDLE_ENFORCE_NE(
      map_it, free_map.end(),
      platform::errors::NotFound("The node to erase is not found in map"));
Y
Yu Yang 已提交
107 108 109 110 111 112 113 114 115
  free_map.erase(map_it);
}
size_t BestFitAllocator::NumFreeChunks() const {
  size_t num = 0;
  for (auto& array_item : free_chunks_) {
    num += array_item.size();
  }
  return num;
}
Z
Zeng Jinle 已提交
116
void BestFitAllocator::FreeImpl(Allocation* allocation) {
Y
Yu Yang 已提交
117
  auto* bf_allocation = dynamic_cast<BestFitAllocation*>(allocation);
118 119 120 121
  PADDLE_ENFORCE_NOT_NULL(
      bf_allocation,
      platform::errors::InvalidArgument(
          "The input allocation is not type of BestFitAllocation."));
122
  auto chunk_it = bf_allocation->ChunkIterator();
123 124 125
  PADDLE_ENFORCE_EQ(chunk_it->is_free, false,
                    platform::errors::PreconditionNotMet(
                        "The chunk of allocation to free is freed already"));
126
  chunk_it->is_free = true;
Y
Yu Yang 已提交
127
  if (chunk_it != chunks_.begin()) {
128 129 130 131
    auto prev_it = chunk_it;
    --prev_it;

    if (prev_it->is_free) {
Y
Yu Yang 已提交
132
      // Merge Left.
133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148
      EraseFreeNode(prev_it);
      prev_it->size_ += chunk_it->size_;
      chunks_.erase(chunk_it);
      chunk_it = prev_it;
    }
  }

  auto next_it = chunk_it;
  ++next_it;
  if (next_it != chunks_.end() && next_it->is_free) {
    EraseFreeNode(next_it);
    chunk_it->size_ += next_it->size_;
    chunks_.erase(next_it);
  }

  InsertFreeNode(chunk_it);
Y
Yu Yang 已提交
149
  delete allocation;
150
}
151
Allocation* BestFitAllocator::AllocateImpl(size_t size) {
Y
Yu Yang 已提交
152 153 154 155 156 157 158
  auto highest_set_bit = static_cast<size_t>(HighestBitPos(size));
  MapIt map_it;
  for (; highest_set_bit < free_chunks_.size(); ++highest_set_bit) {
    map_it = free_chunks_[highest_set_bit].lower_bound(size);
    if (map_it != free_chunks_[highest_set_bit].end()) {
      break;
    }
159
  }
Y
Yu Yang 已提交
160
  if (UNLIKELY(highest_set_bit == free_chunks_.size())) {
161 162
    PADDLE_THROW_BAD_ALLOC(platform::errors::ResourceExhausted(
        "Cannot allocate %d, All fragments size is %d.", size, FreeSize()));
163
  }
Y
Yu Yang 已提交
164 165
  auto chunk_it = SplitChunk(size, highest_set_bit, map_it);
  return new BestFitAllocation(this, chunk_it);
166 167 168 169 170
}

BestFitAllocation::BestFitAllocation(
    paddle::memory::allocation::BestFitAllocator* allocator,
    typename details::ChunkList::iterator chunk_it)
Y
Yu Yang 已提交
171 172 173 174
    : Allocation(reinterpret_cast<void*>(
                     reinterpret_cast<uintptr_t>(allocator->BasePtr()) +
                     chunk_it->offset_),
                 chunk_it->size_, allocator->Place()),
175 176 177 178
      chunk_it_(chunk_it) {}
}  // namespace allocation
}  // namespace memory
}  // namespace paddle