/*
 * Copyright (c) Meta Platforms, Inc. and affiliates.
 *
 * This source code is licensed under the MIT license found in the
 * LICENSE file in the root directory of this source tree.
 */

/*
 * C++ support for heaps. The set of functions is tailored for efficient
 * similarity search.
 *
 * There is no specific object for a heap, and the functions that operate on a
 * single heap are inlined, because heaps are often small. More complex
 * functions are implemented in Heaps.cpp
 *
 * All heap functions rely on a C template class that define the type of the
 * keys and values and their ordering (increasing with CMax and decreasing with
 * Cmin). The C types are defined in ordered_key_value.h
 */

#include <stdint.h>

#include <cassert>
#include <climits>
#include <cmath>
#include <cstdio>
#include <cstring>
#include <limits>
#include <utility>

namespace faiss {

/*******************************************************************
 * C object: uniform handling of min and max heap
 *******************************************************************/

/** The C object gives the type T of the values of a key-value storage, the type
 *  of the keys, TI and the comparison that is done: CMax for a decreasing
 *  series and CMin for increasing series. In other words, for a given threshold
 *  threshold, an incoming value x is kept if
 *
 *      C::cmp(threshold, x)
 *
 *  is true.
 */

template <typename T_, typename TI_>
struct CMax;

template <typename T>
inline T cmin_nextafter(T x);
template <typename T>
inline T cmax_nextafter(T x);

// traits of minheaps = heaps where the minimum value is stored on top
// useful to find the *max* values of an array
template <typename T_, typename TI_>
struct CMin {
  typedef T_ T;
  typedef TI_ TI;
  typedef CMax<T_, TI_> Crev;  // reference to reverse comparison
  inline static bool cmp(T a, T b) { return a < b; }
  // Similar to cmp(), but also breaks ties
  // by comparing the second pair of arguments.
  inline static bool cmp2(T a1, T b1, TI a2, TI b2) {
    return (a1 < b1) || ((a1 == b1) && (a2 < b2));
  }
  inline static T neutral() { return std::numeric_limits<T>::lowest(); }
  static const bool is_max = false;

  inline static T nextafter(T x) { return cmin_nextafter(x); }
};

template <typename T_, typename TI_>
struct CMax {
  typedef T_ T;
  typedef TI_ TI;
  typedef CMin<T_, TI_> Crev;
  inline static bool cmp(T a, T b) { return a > b; }
  // Similar to cmp(), but also breaks ties
  // by comparing the second pair of arguments.
  inline static bool cmp2(T a1, T b1, TI a2, TI b2) {
    return (a1 > b1) || ((a1 == b1) && (a2 > b2));
  }
  inline static T neutral() { return std::numeric_limits<T>::max(); }
  static const bool is_max = true;
  inline static T nextafter(T x) { return cmax_nextafter(x); }
};

template <>
inline float cmin_nextafter<float>(float x) {
  return std::nextafterf(x, -HUGE_VALF);
}

template <>
inline float cmax_nextafter<float>(float x) {
  return std::nextafterf(x, HUGE_VALF);
}

template <>
inline uint16_t cmin_nextafter<uint16_t>(uint16_t x) {
  return x - 1;
}

template <>
inline uint16_t cmax_nextafter<uint16_t>(uint16_t x) {
  return x + 1;
}

/*******************************************************************
 * Basic heap ops: push and pop
 *******************************************************************/

/** Pops the top element from the heap defined by bh_val[0..k-1] and
 * bh_ids[0..k-1].  on output the element at k-1 is undefined.
 */
template <class C>
inline void heap_pop(size_t k, typename C::T *bh_val, typename C::TI *bh_ids) {
  bh_val--; /* Use 1-based indexing for easier node->child translation */
  bh_ids--;
  typename C::T val = bh_val[k];
  typename C::TI id = bh_ids[k];
  size_t i = 1, i1, i2;
  while (1) {
    i1 = i << 1;
    i2 = i1 + 1;
    if (i1 > k) break;
    if ((i2 == k + 1) || C::cmp2(bh_val[i1], bh_val[i2], bh_ids[i1], bh_ids[i2])) {
      if (C::cmp2(val, bh_val[i1], id, bh_ids[i1])) {
        break;
      }
      bh_val[i] = bh_val[i1];
      bh_ids[i] = bh_ids[i1];
      i = i1;
    } else {
      if (C::cmp2(val, bh_val[i2], id, bh_ids[i2])) {
        break;
      }
      bh_val[i] = bh_val[i2];
      bh_ids[i] = bh_ids[i2];
      i = i2;
    }
  }
  bh_val[i] = bh_val[k];
  bh_ids[i] = bh_ids[k];
}

/** Pushes the element (val, ids) into the heap bh_val[0..k-2] and
 * bh_ids[0..k-2].  on output the element at k-1 is defined.
 */
template <class C>
inline void heap_push(size_t k, typename C::T *bh_val, typename C::TI *bh_ids, typename C::T val,
                      typename C::TI id) {
  bh_val--; /* Use 1-based indexing for easier node->child translation */
  bh_ids--;
  size_t i = k, i_father;
  while (i > 1) {
    i_father = i >> 1;
    if (!C::cmp2(val, bh_val[i_father], id, bh_ids[i_father])) {
      /* the heap structure is ok */
      break;
    }
    bh_val[i] = bh_val[i_father];
    bh_ids[i] = bh_ids[i_father];
    i = i_father;
  }
  bh_val[i] = val;
  bh_ids[i] = id;
}

/**
 * Removes the top element from the heap and maintains the heap property.
 * Maintains an index array that tracks the heap position of each element.
 */
template <class C>
inline void heap_pop_tracking(size_t k, typename C::T *bh_val, typename C::TI *bh_ids,
                              int16_t *index_map) {
  // Mark the top element (being removed) as removed
  if (bh_ids[0] != -1) {
    index_map[bh_ids[0]] = -1;
  }

  bh_val--; /* Use 1-based indexing for easier node->child translation */
  bh_ids--;
  typename C::T val = bh_val[k];
  typename C::TI id = bh_ids[k];
  size_t i = 1, i1, i2;
  while (1) {
    i1 = i << 1;
    i2 = i1 + 1;
    if (i1 > k - 1)  // k-1 because we're removing one element
      break;
    if ((i2 == k) ||  // i2 == k because we're removing one element
        C::cmp2(bh_val[i1], bh_val[i2], bh_ids[i1], bh_ids[i2])) {
      if (C::cmp2(val, bh_val[i1], id, bh_ids[i1])) {
        break;
      }
      // Update index map for the element moving up in the tree
      if (bh_ids[i1] != -1) {
        index_map[bh_ids[i1]] = (int16_t)(i - 1);  // Convert back to 0-based indexing
      }
      bh_val[i] = bh_val[i1];
      bh_ids[i] = bh_ids[i1];
      i = i1;
    } else {
      if (C::cmp2(val, bh_val[i2], id, bh_ids[i2])) {
        break;
      }
      // Update index map for the element moving up in the tree
      if (bh_ids[i2] != -1) {
        index_map[bh_ids[i2]] = (int16_t)(i - 1);  // Convert back to 0-based indexing
      }
      bh_val[i] = bh_val[i2];
      bh_ids[i] = bh_ids[i2];
      i = i2;
    }
  }
  bh_val[i] = val;
  bh_ids[i] = id;
  // Update index map for the element that moved from the end to position i
  if (id != -1) {
    index_map[id] = (int16_t)(i - 1);  // Convert back to 0-based indexing
  }
}

/** Pushes the element (val, ids) into the heap bh_val[0..k-2] and
 * bh_ids[0..k-2]. On output the element at k-1 is defined.
 * Maintains an index array that tracks the heap position of each element.
 */
template <class C>
inline void heap_push_tracking(size_t k, typename C::T *bh_val, typename C::TI *bh_ids,
                               typename C::T val, typename C::TI id, int16_t *index_map) {
  bh_val--; /* Use 1-based indexing for easier node->child translation */
  bh_ids--;
  size_t i = k, i_father;
  while (i > 1) {
    i_father = i >> 1;
    if (!C::cmp2(val, bh_val[i_father], id, bh_ids[i_father])) {
      /* the heap structure is ok */
      break;
    }
    // Update index map for the element moving down in the tree
    if (bh_ids[i_father] != -1) {
      index_map[bh_ids[i_father]] = (int16_t)(i - 1);  // Convert back to 0-based indexing
    }
    bh_val[i] = bh_val[i_father];
    bh_ids[i] = bh_ids[i_father];
    i = i_father;
  }
  bh_val[i] = val;
  bh_ids[i] = id;
  // Update index map for the newly inserted element
  index_map[id] = (int16_t)(i - 1);  // Convert back to 0-based indexing
}

/**
 * Replaces the top element from the heap defined by bh_val[0..k-1] and
 * bh_ids[0..k-1], and for identical bh_val[] values also sorts by bh_ids[]
 * values.
 */
template <class C>
inline void heap_replace_top(size_t k, typename C::T *bh_val, typename C::TI *bh_ids,
                             typename C::T val, typename C::TI id) {
  bh_val--; /* Use 1-based indexing for easier node->child translation */
  bh_ids--;
  size_t i = 1, i1, i2;
  while (1) {
    i1 = i << 1;
    i2 = i1 + 1;
    if (i1 > k) {
      break;
    }

    // Note that C::cmp2() is a bool function answering
    // `(a1 > b1) || ((a1 == b1) && (a2 > b2))` for max
    // heap and same with the `<` sign for min heap.
    if ((i2 == k + 1) || C::cmp2(bh_val[i1], bh_val[i2], bh_ids[i1], bh_ids[i2])) {
      if (C::cmp2(val, bh_val[i1], id, bh_ids[i1])) {
        break;
      }
      bh_val[i] = bh_val[i1];
      bh_ids[i] = bh_ids[i1];
      i = i1;
    } else {
      if (C::cmp2(val, bh_val[i2], id, bh_ids[i2])) {
        break;
      }
      bh_val[i] = bh_val[i2];
      bh_ids[i] = bh_ids[i2];
      i = i2;
    }
  }
  bh_val[i] = val;
  bh_ids[i] = id;
}

/* Partial instanciation for heaps with TI = int64_t */

template <typename T>
inline void minheap_pop(size_t k, T *bh_val, int64_t *bh_ids) {
  heap_pop<CMin<T, int64_t>>(k, bh_val, bh_ids);
}

template <typename T>
inline void minheap_push(size_t k, T *bh_val, int64_t *bh_ids, T val, int64_t ids) {
  heap_push<CMin<T, int64_t>>(k, bh_val, bh_ids, val, ids);
}

template <typename T>
inline void minheap_replace_top(size_t k, T *bh_val, int64_t *bh_ids, T val, int64_t ids) {
  heap_replace_top<CMin<T, int64_t>>(k, bh_val, bh_ids, val, ids);
}

template <typename T>
inline void maxheap_pop(size_t k, T *bh_val, int64_t *bh_ids) {
  heap_pop<CMax<T, int64_t>>(k, bh_val, bh_ids);
}

template <typename T>
inline void maxheap_push(size_t k, T *bh_val, int64_t *bh_ids, T val, int64_t ids) {
  heap_push<CMax<T, int64_t>>(k, bh_val, bh_ids, val, ids);
}

template <typename T>
inline void maxheap_replace_top(size_t k, T *bh_val, int64_t *bh_ids, T val, int64_t ids) {
  heap_replace_top<CMax<T, int64_t>>(k, bh_val, bh_ids, val, ids);
}

/*******************************************************************
 * Basic heap<std:pair<>> ops: push and pop
 *******************************************************************/

// This section contains a heap implementation that works with
//   std::pair<Priority, Value> elements.

/** Pops the top element from the heap defined by bh_val[0..k-1] and
 * bh_ids[0..k-1].  on output the element at k-1 is undefined.
 */
template <class C>
inline void heap_pop(size_t k, std::pair<typename C::T, typename C::TI> *bh) {
  bh--; /* Use 1-based indexing for easier node->child translation */
  typename C::T val = bh[k].first;
  typename C::TI id = bh[k].second;
  size_t i = 1, i1, i2;
  while (1) {
    i1 = i << 1;
    i2 = i1 + 1;
    if (i1 > k) break;
    if ((i2 == k + 1) || C::cmp2(bh[i1].first, bh[i2].first, bh[i1].second, bh[i2].second)) {
      if (C::cmp2(val, bh[i1].first, id, bh[i1].second)) {
        break;
      }
      bh[i] = bh[i1];
      i = i1;
    } else {
      if (C::cmp2(val, bh[i2].first, id, bh[i2].second)) {
        break;
      }
      bh[i] = bh[i2];
      i = i2;
    }
  }
  bh[i] = bh[k];
}

/** Pushes the element (val, ids) into the heap bh_val[0..k-2] and
 * bh_ids[0..k-2].  on output the element at k-1 is defined.
 */
template <class C>
inline void heap_push(size_t k, std::pair<typename C::T, typename C::TI> *bh, typename C::T val,
                      typename C::TI id) {
  bh--; /* Use 1-based indexing for easier node->child translation */
  size_t i = k, i_father;
  while (i > 1) {
    i_father = i >> 1;
    auto bh_v = bh[i_father];
    if (!C::cmp2(val, bh_v.first, id, bh_v.second)) {
      /* the heap structure is ok */
      break;
    }
    bh[i] = bh_v;
    i = i_father;
  }
  bh[i] = std::make_pair(val, id);
}

/**
 * Replaces the top element from the heap defined by bh_val[0..k-1] and
 * bh_ids[0..k-1], and for identical bh_val[] values also sorts by bh_ids[]
 * values.
 */
template <class C>
inline void heap_replace_top(size_t k, std::pair<typename C::T, typename C::TI> *bh,
                             typename C::T val, typename C::TI id) {
  bh--; /* Use 1-based indexing for easier node->child translation */
  size_t i = 1, i1, i2;
  while (1) {
    i1 = i << 1;
    i2 = i1 + 1;
    if (i1 > k) {
      break;
    }

    // Note that C::cmp2() is a bool function answering
    // `(a1 > b1) || ((a1 == b1) && (a2 > b2))` for max
    // heap and same with the `<` sign for min heap.
    if ((i2 == k + 1) || C::cmp2(bh[i1].first, bh[i2].first, bh[i1].second, bh[i2].second)) {
      if (C::cmp2(val, bh[i1].first, id, bh[i1].second)) {
        break;
      }
      bh[i] = bh[i1];
      i = i1;
    } else {
      if (C::cmp2(val, bh[i2].first, id, bh[i2].second)) {
        break;
      }
      bh[i] = bh[i2];
      i = i2;
    }
  }
  bh[i] = std::make_pair(val, id);
}

/**
 * Helper function to sift down an element at given index to restore heap property.
 * This is similar to the logic in heap_pop but starts from an arbitrary position.
 */
template <class C>
inline void heap_sift_down(size_t k, typename C::T *bh_val, typename C::TI *bh_ids,
                           size_t start_idx) {
  // Convert to 1-based indexing for consistency with existing FAISS code
  bh_val--;
  bh_ids--;

  typename C::T val = bh_val[start_idx + 1];
  typename C::TI id = bh_ids[start_idx + 1];
  size_t i = start_idx + 1;  // Convert to 1-based

  while (1) {
    size_t i1 = i << 1;  // left child
    size_t i2 = i1 + 1;  // right child

    if (i1 > k) break;  // no children

    // Find the child to compare with
    size_t child_idx;
    if ((i2 == k + 1) || C::cmp2(bh_val[i1], bh_val[i2], bh_ids[i1], bh_ids[i2])) {
      child_idx = i1;
    } else {
      child_idx = i2;
    }

    // If heap property is satisfied, we're done
    if (C::cmp2(val, bh_val[child_idx], id, bh_ids[child_idx])) {
      break;
    }

    // Move child up and continue sifting down
    bh_val[i] = bh_val[child_idx];
    bh_ids[i] = bh_ids[child_idx];
    i = child_idx;
  }

  // Place the original element in its final position
  bh_val[i] = val;
  bh_ids[i] = id;
}

/**
 * Restores the heap property to an entire array in O(n) time.
 * This is the classic "heapify" operation that takes an arbitrary array
 * and transforms it into a valid heap by sifting down from the last
 * non-leaf node to the root.
 *
 * @param k       size of the array
 * @param bh_val  array of values
 * @param bh_ids  array of corresponding IDs
 */
template <class C>
inline void heap_restore_property(size_t k, typename C::T *bh_val, typename C::TI *bh_ids) {
  // Start from the last non-leaf node and sift down to root
  // Last non-leaf node is at index (k-2)/2 in 0-based indexing
  for (int64_t i = (k - 2) / 2; i >= 0; i--) {
    heap_sift_down<C>(k, bh_val, bh_ids, i);
  }
}

/**
 * Restores the heap property when only one element at a specific index
 * might have violated it. This is more efficient than rebuilding the entire heap
 * as it only affects the path from the violated element to either a leaf (sift down)
 * or the root (sift up).
 *
 * @param k         size of the heap
 * @param bh_val    array of values
 * @param bh_ids    array of corresponding IDs
 * @param fix_idx   0-based index of the element that potentially broke heap property
 */
template <class C>
inline void heap_fix_single_element(size_t k, typename C::T *bh_val, typename C::TI *bh_ids,
                                    size_t fix_idx) {
  if (k == 0 || fix_idx >= k) return;

  // Convert to 1-based indexing for consistency with existing code
  bh_val--;
  bh_ids--;
  size_t idx_1based = fix_idx + 1;

  typename C::T val = bh_val[idx_1based];
  typename C::TI id = bh_ids[idx_1based];

  // First, try sifting up (towards root)
  size_t i = idx_1based;
  while (i > 1) {            // while not at root
    size_t parent = i >> 1;  // parent index

    // If heap property is satisfied with parent, stop sifting up
    if (C::cmp2(bh_val[parent], val, bh_ids[parent], id)) {
      break;
    }

    // Move parent down
    bh_val[i] = bh_val[parent];
    bh_ids[i] = bh_ids[parent];
    i = parent;
  }

  // If we moved up, place the element and we're done
  if (i != idx_1based) {
    bh_val[i] = val;
    bh_ids[i] = id;
    return;
  }

  // Element didn't move up, so try sifting down (towards leaves)
  while (1) {
    size_t i1 = i << 1;  // left child
    size_t i2 = i1 + 1;  // right child

    if (i1 > k) break;  // no children

    // Find the child to compare with
    size_t child_idx;
    if ((i2 == k + 1) || C::cmp2(bh_val[i1], bh_val[i2], bh_ids[i1], bh_ids[i2])) {
      child_idx = i1;
    } else {
      child_idx = i2;
    }

    // If heap property is satisfied, we're done
    if (C::cmp2(val, bh_val[child_idx], id, bh_ids[child_idx])) {
      break;
    }

    // Move child up and continue sifting down
    bh_val[i] = bh_val[child_idx];
    bh_ids[i] = bh_ids[child_idx];
    i = child_idx;
  }

  // Place the element in its final position
  bh_val[i] = val;
  bh_ids[i] = id;
}

/**
 * Restores the heap property when only one element at a specific index
 * might have violated it. This is more efficient than rebuilding the entire heap
 * as it only affects the path from the violated element to either a leaf (sift down)
 * or the root (sift up). Maintains an index array that tracks the heap position of each element.
 *
 * @param k         size of the heap
 * @param bh_val    array of values
 * @param bh_ids    array of corresponding IDs
 * @param fix_idx   0-based index of the element that potentially broke heap property
 * @param index_map array where index_map[bh_ids[i]] = i (heap position of element with id
 * bh_ids[i]), -1 for removed elements
 */
template <class C>
inline void heap_fix_single_element_tracking(size_t k, typename C::T *bh_val,
                                             typename C::TI *bh_ids, size_t fix_idx,
                                             int16_t *index_map) {
  if (k == 0 || fix_idx >= k) return;

  // Convert to 1-based indexing for consistency with existing code
  bh_val--;
  bh_ids--;
  size_t idx_1based = fix_idx + 1;

  typename C::T val = bh_val[idx_1based];
  typename C::TI id = bh_ids[idx_1based];

  // First, try sifting up (towards root)
  size_t i = idx_1based;
  while (i > 1) {            // while not at root
    size_t parent = i >> 1;  // parent index

    // If heap property is satisfied with parent, stop sifting up
    if (C::cmp2(bh_val[parent], val, bh_ids[parent], id)) {
      break;
    }

    // Update index map for the element moving down in the tree
    if (bh_ids[parent] != -1) {
      index_map[bh_ids[parent]] = (int16_t)(i - 1);  // Convert back to 0-based indexing
    }

    // Move parent down
    bh_val[i] = bh_val[parent];
    bh_ids[i] = bh_ids[parent];
    i = parent;
  }

  // If we moved up, place the element and we're done
  if (i != idx_1based) {
    bh_val[i] = val;
    bh_ids[i] = id;
    // Update index map for the element that moved up
    if (id != -1) {
      index_map[id] = (int16_t)(i - 1);  // Convert back to 0-based indexing
    }
    return;
  }

  // Element didn't move up, so try sifting down (towards leaves)
  while (1) {
    size_t i1 = i << 1;  // left child
    size_t i2 = i1 + 1;  // right child

    if (i1 > k) break;  // no children

    // Find the child to compare with
    size_t child_idx;
    if ((i2 == k + 1) || C::cmp2(bh_val[i1], bh_val[i2], bh_ids[i1], bh_ids[i2])) {
      child_idx = i1;
    } else {
      child_idx = i2;
    }

    // If heap property is satisfied, we're done
    if (C::cmp2(val, bh_val[child_idx], id, bh_ids[child_idx])) {
      break;
    }

    // Update index map for the element moving up in the tree
    if (bh_ids[child_idx] != -1) {
      index_map[bh_ids[child_idx]] = (int16_t)(i - 1);  // Convert back to 0-based indexing
    }

    // Move child up and continue sifting down
    bh_val[i] = bh_val[child_idx];
    bh_ids[i] = bh_ids[child_idx];
    i = child_idx;
  }

  // Place the element in its final position
  bh_val[i] = val;
  bh_ids[i] = id;
  // Update index map for the element in its final position
  if (id != -1) {
    index_map[id] = (int16_t)(i - 1);  // Convert back to 0-based indexing
  }
}

/*******************************************************************
 * Heap initialization
 *******************************************************************/

/* Initialization phase for the heap (with unconditionnal pushes).
 * Store k0 elements in a heap containing up to k values. Note that
 * (bh_val, bh_ids) can be the same as (x, ids) */
template <class C>
inline void heap_heapify(size_t k, typename C::T *bh_val, typename C::TI *bh_ids,
                         const typename C::T *x = nullptr, const typename C::TI *ids = nullptr,
                         size_t k0 = 0) {
  if (k0 > 0) assert(x);

  if (ids) {
    for (size_t i = 0; i < k0; i++) heap_push<C>(i + 1, bh_val, bh_ids, x[i], ids[i]);
  } else {
    for (size_t i = 0; i < k0; i++) heap_push<C>(i + 1, bh_val, bh_ids, x[i], i);
  }

  for (size_t i = k0; i < k; i++) {
    bh_val[i] = C::neutral();
    bh_ids[i] = -1;
  }
}

template <typename T>
inline void minheap_heapify(size_t k, T *bh_val, int64_t *bh_ids, const T *x = nullptr,
                            const int64_t *ids = nullptr, size_t k0 = 0) {
  heap_heapify<CMin<T, int64_t>>(k, bh_val, bh_ids, x, ids, k0);
}

template <typename T>
inline void maxheap_heapify(size_t k, T *bh_val, int64_t *bh_ids, const T *x = nullptr,
                            const int64_t *ids = nullptr, size_t k0 = 0) {
  heap_heapify<CMax<T, int64_t>>(k, bh_val, bh_ids, x, ids, k0);
}

/*******************************************************************
 * Add n elements to the heap
 *******************************************************************/

/* Add some elements to the heap  */
template <class C>
inline void heap_addn(size_t k, typename C::T *bh_val, typename C::TI *bh_ids,
                      const typename C::T *x, const typename C::TI *ids, size_t n) {
  size_t i;
  if (ids)
    for (i = 0; i < n; i++) {
      if (C::cmp(bh_val[0], x[i])) {
        heap_replace_top<C>(k, bh_val, bh_ids, x[i], ids[i]);
      }
    }
  else
    for (i = 0; i < n; i++) {
      if (C::cmp(bh_val[0], x[i])) {
        heap_replace_top<C>(k, bh_val, bh_ids, x[i], i);
      }
    }
}

/* Partial instanciation for heaps with TI = int64_t */

template <typename T>
inline void minheap_addn(size_t k, T *bh_val, int64_t *bh_ids, const T *x, const int64_t *ids,
                         size_t n) {
  heap_addn<CMin<T, int64_t>>(k, bh_val, bh_ids, x, ids, n);
}

template <typename T>
inline void maxheap_addn(size_t k, T *bh_val, int64_t *bh_ids, const T *x, const int64_t *ids,
                         size_t n) {
  heap_addn<CMax<T, int64_t>>(k, bh_val, bh_ids, x, ids, n);
}

/*******************************************************************
 * Heap finalization (reorder elements)
 *******************************************************************/

/* This function maps a binary heap into a sorted structure.
   It returns the number  */
template <typename C>
inline size_t heap_reorder(size_t k, typename C::T *bh_val, typename C::TI *bh_ids) {
  size_t i, ii;

  for (i = 0, ii = 0; i < k; i++) {
    /* top element should be put at the end of the list */
    typename C::T val = bh_val[0];
    typename C::TI id = bh_ids[0];

    /* boundary case: we will over-ride this value if not a true element */
    heap_pop<C>(k - i, bh_val, bh_ids);
    bh_val[k - ii - 1] = val;
    bh_ids[k - ii - 1] = id;
    if (id != -1) ii++;
  }
  /* Count the number of elements which are effectively returned */
  size_t nel = ii;

  memmove(bh_val, bh_val + k - ii, ii * sizeof(*bh_val));
  memmove(bh_ids, bh_ids + k - ii, ii * sizeof(*bh_ids));

  for (; ii < k; ii++) {
    bh_val[ii] = C::neutral();
    bh_ids[ii] = -1;
  }
  return nel;
}

template <typename T>
inline size_t minheap_reorder(size_t k, T *bh_val, int64_t *bh_ids) {
  return heap_reorder<CMin<T, int64_t>>(k, bh_val, bh_ids);
}

template <typename T>
inline size_t maxheap_reorder(size_t k, T *bh_val, int64_t *bh_ids) {
  return heap_reorder<CMax<T, int64_t>>(k, bh_val, bh_ids);
}

template <class C>
inline void indirect_heap_pop(size_t k, const typename C::T *bh_val, typename C::TI *bh_ids) {
  bh_ids--; /* Use 1-based indexing for easier node->child translation */
  typename C::T val = bh_val[bh_ids[k]];
  size_t i = 1;
  while (1) {
    size_t i1 = i << 1;
    size_t i2 = i1 + 1;
    if (i1 > k) break;
    typename C::TI id1 = bh_ids[i1], id2 = bh_ids[i2];
    if (i2 == k + 1 || C::cmp(bh_val[id1], bh_val[id2])) {
      if (C::cmp(val, bh_val[id1])) break;
      bh_ids[i] = id1;
      i = i1;
    } else {
      if (C::cmp(val, bh_val[id2])) break;
      bh_ids[i] = id2;
      i = i2;
    }
  }
  bh_ids[i] = bh_ids[k];
}

template <class C>
inline void indirect_heap_push(size_t k, const typename C::T *bh_val, typename C::TI *bh_ids,
                               typename C::TI id) {
  bh_ids--; /* Use 1-based indexing for easier node->child translation */
  typename C::T val = bh_val[id];
  size_t i = k;
  while (i > 1) {
    size_t i_father = i >> 1;
    if (!C::cmp(val, bh_val[bh_ids[i_father]])) break;
    bh_ids[i] = bh_ids[i_father];
    i = i_father;
  }
  bh_ids[i] = id;
}

}  // namespace faiss
