// Copyright 2007-2010 Baptiste Lepilleur
// Distributed under MIT license, or public domain if desired and
// recognized in your jurisdiction.
// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE

// included by json_value.cpp

namespace Json {

// //////////////////////////////////////////////////////////////////
// //////////////////////////////////////////////////////////////////
// //////////////////////////////////////////////////////////////////
// class ValueInternalArray
// //////////////////////////////////////////////////////////////////
// //////////////////////////////////////////////////////////////////
// //////////////////////////////////////////////////////////////////

ValueArrayAllocator::~ValueArrayAllocator() {}

// //////////////////////////////////////////////////////////////////
// class DefaultValueArrayAllocator
// //////////////////////////////////////////////////////////////////
#ifdef JSON_USE_SIMPLE_INTERNAL_ALLOCATOR
class DefaultValueArrayAllocator : public ValueArrayAllocator {
public: // overridden from ValueArrayAllocator
  virtual ~DefaultValueArrayAllocator() {}

  virtual ValueInternalArray* newArray() { return new ValueInternalArray(); }

  virtual ValueInternalArray* newArrayCopy(const ValueInternalArray& other) {
    return new ValueInternalArray(other);
  }

  virtual void destructArray(ValueInternalArray* array) { delete array; }

  virtual void
  reallocateArrayPageIndex(Value**& indexes,
                           ValueInternalArray::PageIndex& indexCount,
                           ValueInternalArray::PageIndex minNewIndexCount) {
    ValueInternalArray::PageIndex newIndexCount = (indexCount * 3) / 2 + 1;
    if (minNewIndexCount > newIndexCount)
      newIndexCount = minNewIndexCount;
    void* newIndexes = realloc(indexes, sizeof(Value*) * newIndexCount);
    JSON_ASSERT_MESSAGE(newIndexes, "Couldn't realloc.");
    indexCount = newIndexCount;
    indexes = static_cast<Value**>(newIndexes);
  }
  virtual void releaseArrayPageIndex(Value** indexes,
                                     ValueInternalArray::PageIndex indexCount) {
    if (indexes)
      free(indexes);
  }

  virtual Value* allocateArrayPage() {
    return static_cast<Value*>(
        malloc(sizeof(Value) * ValueInternalArray::itemsPerPage));
  }

  virtual void releaseArrayPage(Value* value) {
    if (value)
      free(value);
  }
};

#else  // #ifdef JSON_USE_SIMPLE_INTERNAL_ALLOCATOR
/// @todo make this thread-safe (lock when accessign batch allocator)
class DefaultValueArrayAllocator : public ValueArrayAllocator {
public: // overridden from ValueArrayAllocator
  virtual ~DefaultValueArrayAllocator() {}

  virtual ValueInternalArray* newArray() {
    ValueInternalArray* array = arraysAllocator_.allocate();
    new (array) ValueInternalArray(); // placement new
    return array;
  }

  virtual ValueInternalArray* newArrayCopy(const ValueInternalArray& other) {
    ValueInternalArray* array = arraysAllocator_.allocate();
    new (array) ValueInternalArray(other); // placement new
    return array;
  }

  virtual void destructArray(ValueInternalArray* array) {
    if (array) {
      array->~ValueInternalArray();
      arraysAllocator_.release(array);
    }
  }

  virtual void
  reallocateArrayPageIndex(Value**& indexes,
                           ValueInternalArray::PageIndex& indexCount,
                           ValueInternalArray::PageIndex minNewIndexCount) {
    ValueInternalArray::PageIndex newIndexCount = (indexCount * 3) / 2 + 1;
    if (minNewIndexCount > newIndexCount)
      newIndexCount = minNewIndexCount;
    void* newIndexes = realloc(indexes, sizeof(Value*) * newIndexCount);
    JSON_ASSERT_MESSAGE(newIndexes, "Couldn't realloc.");
    indexCount = newIndexCount;
    indexes = static_cast<Value**>(newIndexes);
  }
  virtual void releaseArrayPageIndex(Value** indexes,
                                     ValueInternalArray::PageIndex indexCount) {
    if (indexes)
      free(indexes);
  }

  virtual Value* allocateArrayPage() {
    return static_cast<Value*>(pagesAllocator_.allocate());
  }

  virtual void releaseArrayPage(Value* value) {
    if (value)
      pagesAllocator_.release(value);
  }

private:
  BatchAllocator<ValueInternalArray, 1> arraysAllocator_;
  BatchAllocator<Value, ValueInternalArray::itemsPerPage> pagesAllocator_;
};
#endif // #ifdef JSON_USE_SIMPLE_INTERNAL_ALLOCATOR

static ValueArrayAllocator*& arrayAllocator() {
  static DefaultValueArrayAllocator defaultAllocator;
  static ValueArrayAllocator* arrayAllocator = &defaultAllocator;
  return arrayAllocator;
}

static struct DummyArrayAllocatorInitializer {
  DummyArrayAllocatorInitializer() {
    arrayAllocator(); // ensure arrayAllocator() statics are initialized before
                      // main().
  }
} dummyArrayAllocatorInitializer;

// //////////////////////////////////////////////////////////////////
// class ValueInternalArray
// //////////////////////////////////////////////////////////////////
bool ValueInternalArray::equals(const IteratorState& x,
                                const IteratorState& other) {
  return x.array_ == other.array_ &&
         x.currentItemIndex_ == other.currentItemIndex_ &&
         x.currentPageIndex_ == other.currentPageIndex_;
}

void ValueInternalArray::increment(IteratorState& it) {
  JSON_ASSERT_MESSAGE(
      it.array_ && (it.currentPageIndex_ - it.array_->pages_) * itemsPerPage +
                           it.currentItemIndex_ !=
                       it.array_->size_,
      "ValueInternalArray::increment(): moving iterator beyond end");
  ++(it.currentItemIndex_);
  if (it.currentItemIndex_ == itemsPerPage) {
    it.currentItemIndex_ = 0;
    ++(it.currentPageIndex_);
  }
}

void ValueInternalArray::decrement(IteratorState& it) {
  JSON_ASSERT_MESSAGE(
      it.array_ && it.currentPageIndex_ == it.array_->pages_ &&
          it.currentItemIndex_ == 0,
      "ValueInternalArray::decrement(): moving iterator beyond end");
  if (it.currentItemIndex_ == 0) {
    it.currentItemIndex_ = itemsPerPage - 1;
    --(it.currentPageIndex_);
  } else {
    --(it.currentItemIndex_);
  }
}

Value& ValueInternalArray::unsafeDereference(const IteratorState& it) {
  return (*(it.currentPageIndex_))[it.currentItemIndex_];
}

Value& ValueInternalArray::dereference(const IteratorState& it) {
  JSON_ASSERT_MESSAGE(
      it.array_ && (it.currentPageIndex_ - it.array_->pages_) * itemsPerPage +
                           it.currentItemIndex_ <
                       it.array_->size_,
      "ValueInternalArray::dereference(): dereferencing invalid iterator");
  return unsafeDereference(it);
}

void ValueInternalArray::makeBeginIterator(IteratorState& it) const {
  it.array_ = const_cast<ValueInternalArray*>(this);
  it.currentItemIndex_ = 0;
  it.currentPageIndex_ = pages_;
}

void ValueInternalArray::makeIterator(IteratorState& it,
                                      ArrayIndex index) const {
  it.array_ = const_cast<ValueInternalArray*>(this);
  it.currentItemIndex_ = index % itemsPerPage;
  it.currentPageIndex_ = pages_ + index / itemsPerPage;
}

void ValueInternalArray::makeEndIterator(IteratorState& it) const {
  makeIterator(it, size_);
}

ValueInternalArray::ValueInternalArray() : pages_(0), size_(0), pageCount_(0) {}

ValueInternalArray::ValueInternalArray(const ValueInternalArray& other)
    : pages_(0), size_(other.size_), pageCount_(0) {
  PageIndex minNewPages = other.size_ / itemsPerPage;
  arrayAllocator()->reallocateArrayPageIndex(pages_, pageCount_, minNewPages);
  JSON_ASSERT_MESSAGE(pageCount_ >= minNewPages,
                      "ValueInternalArray::reserve(): bad reallocation");
  IteratorState itOther;
  other.makeBeginIterator(itOther);
  Value* value;
  for (ArrayIndex index = 0; index < size_; ++index, increment(itOther)) {
    if (index % itemsPerPage == 0) {
      PageIndex pageIndex = index / itemsPerPage;
      value = arrayAllocator()->allocateArrayPage();
      pages_[pageIndex] = value;
    }
    new (value) Value(dereference(itOther));
  }
}

ValueInternalArray& ValueInternalArray::operator=(ValueInternalArray other) {
  swap(other);
  return *this;
}

ValueInternalArray::~ValueInternalArray() {
  // destroy all constructed items
  IteratorState it;
  IteratorState itEnd;
  makeBeginIterator(it);
  makeEndIterator(itEnd);
  for (; !equals(it, itEnd); increment(it)) {
    Value* value = &dereference(it);
    value->~Value();
  }
  // release all pages
  PageIndex lastPageIndex = size_ / itemsPerPage;
  for (PageIndex pageIndex = 0; pageIndex < lastPageIndex; ++pageIndex)
    arrayAllocator()->releaseArrayPage(pages_[pageIndex]);
  // release pages index
  arrayAllocator()->releaseArrayPageIndex(pages_, pageCount_);
}

void ValueInternalArray::swap(ValueInternalArray& other) {
  Value** tempPages = pages_;
  pages_ = other.pages_;
  other.pages_ = tempPages;
  ArrayIndex tempSize = size_;
  size_ = other.size_;
  other.size_ = tempSize;
  PageIndex tempPageCount = pageCount_;
  pageCount_ = other.pageCount_;
  other.pageCount_ = tempPageCount;
}

void ValueInternalArray::clear() {
  ValueInternalArray dummy;
  swap(dummy);
}

void ValueInternalArray::resize(ArrayIndex newSize) {
  if (newSize == 0)
    clear();
  else if (newSize < size_) {
    IteratorState it;
    IteratorState itEnd;
    makeIterator(it, newSize);
    makeIterator(itEnd, size_);
    for (; !equals(it, itEnd); increment(it)) {
      Value* value = &dereference(it);
      value->~Value();
    }
    PageIndex pageIndex = (newSize + itemsPerPage - 1) / itemsPerPage;
    PageIndex lastPageIndex = size_ / itemsPerPage;
    for (; pageIndex < lastPageIndex; ++pageIndex)
      arrayAllocator()->releaseArrayPage(pages_[pageIndex]);
    size_ = newSize;
  } else if (newSize > size_)
    resolveReference(newSize);
}

void ValueInternalArray::makeIndexValid(ArrayIndex index) {
  // Need to enlarge page index ?
  if (index >= pageCount_ * itemsPerPage) {
    PageIndex minNewPages = (index + 1) / itemsPerPage;
    arrayAllocator()->reallocateArrayPageIndex(pages_, pageCount_, minNewPages);
    JSON_ASSERT_MESSAGE(pageCount_ >= minNewPages,
                        "ValueInternalArray::reserve(): bad reallocation");
  }

  // Need to allocate new pages ?
  ArrayIndex nextPageIndex = (size_ % itemsPerPage) != 0
                                 ? size_ - (size_ % itemsPerPage) + itemsPerPage
                                 : size_;
  if (nextPageIndex <= index) {
    PageIndex pageIndex = nextPageIndex / itemsPerPage;
    PageIndex pageToAllocate = (index - nextPageIndex) / itemsPerPage + 1;
    for (; pageToAllocate-- > 0; ++pageIndex)
      pages_[pageIndex] = arrayAllocator()->allocateArrayPage();
  }

  // Initialize all new entries
  IteratorState it;
  IteratorState itEnd;
  makeIterator(it, size_);
  size_ = index + 1;
  makeIterator(itEnd, size_);
  for (; !equals(it, itEnd); increment(it)) {
    Value* value = &dereference(it);
    new (value) Value(); // Construct a default value using placement new
  }
}

Value& ValueInternalArray::resolveReference(ArrayIndex index) {
  if (index >= size_)
    makeIndexValid(index);
  return pages_[index / itemsPerPage][index % itemsPerPage];
}

Value* ValueInternalArray::find(ArrayIndex index) const {
  if (index >= size_)
    return 0;
  return &(pages_[index / itemsPerPage][index % itemsPerPage]);
}

ValueInternalArray::ArrayIndex ValueInternalArray::size() const {
  return size_;
}

int ValueInternalArray::distance(const IteratorState& x,
                                 const IteratorState& y) {
  return indexOf(y) - indexOf(x);
}

ValueInternalArray::ArrayIndex
ValueInternalArray::indexOf(const IteratorState& iterator) {
  if (!iterator.array_)
    return ArrayIndex(-1);
  return ArrayIndex((iterator.currentPageIndex_ - iterator.array_->pages_) *
                        itemsPerPage +
                    iterator.currentItemIndex_);
}

int ValueInternalArray::compare(const ValueInternalArray& other) const {
  int sizeDiff(size_ - other.size_);
  if (sizeDiff != 0)
    return sizeDiff;

  for (ArrayIndex index = 0; index < size_; ++index) {
    int diff = pages_[index / itemsPerPage][index % itemsPerPage].compare(
        other.pages_[index / itemsPerPage][index % itemsPerPage]);
    if (diff != 0)
      return diff;
  }
  return 0;
}

} // namespace Json