Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Use search_pool to control iterator->Next() #1008

Merged

Conversation

alwayslove2013
Copy link
Collaborator

@alwayslove2013 alwayslove2013 commented Dec 26, 2024

issue: #997

The current AnnIterator function utilizes the search pool for concurrency control only when initializing the Iterator. Once the returned iterator is handed over to the upper layer, the ->next() calls are not subjected to any thread restrictions, which may lead to issues such as OMP conflicts.

To address this, we propose the following considerations:

  • All iterators will accept a use_knowhere_search_pool parameter during construction.
    • When set to True (the default), the iterator->next() will be scheduled by the knowhere_search_thread_pool.
    • When set to False, iterator->next() will not involve thread scheduling internally, so please take caution.
  • The initialization of iterator in the AnnIterator function will no longer be concurrent, which helps to avoid potential deadlocks.
    • Furthermore, to enhance performance for large_nq, we will try to streamline the initialization process for all iterators by deferring heavier pre-computation tasks to the first call of ->next().

Copy link

mergify bot commented Dec 26, 2024

@alwayslove2013 🔍 Important: PR Classification Needed!

For efficient project management and a seamless review process, it's essential to classify your PR correctly. Here's how:

  1. If you're fixing a bug, label it as kind/bug.
  2. For small tweaks (less than 20 lines without altering any functionality), please use kind/improvement.
  3. Significant changes that don't modify existing functionalities should be tagged as kind/enhancement.
  4. Adjusting APIs or changing functionality? Go with kind/feature.

For any PR outside the kind/improvement category, ensure you link to the associated issue using the format: “issue: #”.

Thanks for your efforts and contribution to the community!.

@alwayslove2013 alwayslove2013 force-pushed the iterator_next_with_search_pool branch 3 times, most recently from ee16112 to 3c70c4b Compare December 26, 2024 07:05
@@ -202,7 +203,7 @@ class IndexNode : public Object {
return GenResultDataSet(nq, std::move(range_search_result));
}

auto its_or = AnnIterator(dataset, std::move(cfg), bitset);
auto its_or = AnnIterator(dataset, std::move(cfg), bitset, false);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please add a comment about why this parameter is false

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

added.

The range_search function has utilized the search_pool to concurrently handle various queries.
To prevent potential deadlocks, the iterator for a single query no longer requires additional thread control over the next() call.

// If use_knowhere_search_pool is True (the default), the iterator->Next() will be scheduled by the
// knowhere_search_thread_pool.
// If False, will Not involve thread scheduling internally, so please take caution.
template <bool use_knowhere_search_pool = true>
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What are the justifications for adding use_knowhere_search_pool as a template parameter instead of making a regular IndexIterator::use_knowhere_search_pool field? I'm not sure I follow

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just to reduce the judgment of use_knowhere_search_pool in each iterator->next() call.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've re-read the code again and I don't believe I see a reason for making use_knowhere_search_pool a template parameter. Basically, it leads to increase of the size of the compiled code, also this parameter is not generic enough (like T in std::vector<T>) and it does not have a critical-performance impact. Please consider demoting use_knowhere_search_pool to a variable. Or I can accept the code as is.

distances_id.id = distances_id.id == -1 ? -1 : distances_id.id + xb_id_offset;
if (xb_id_offset != 0) {
for (auto& distances_id : distances_ids) {
distances_id.id = distances_id.id == -1 ? -1 : distances_id.id + xb_id_offset;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

just to confirm: this is evaluated as distances_id.id = a ? b : c, not `distances_id.id = (a ? b : c) + xb_id_offset'?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@cqy123456 help check~

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yes, -1 or distances_id.id + xb_id_offset

@alwayslove2013 alwayslove2013 force-pushed the iterator_next_with_search_pool branch 2 times, most recently from b331267 to 98048fc Compare December 28, 2024 04:54
Copy link

codecov bot commented Dec 28, 2024

Codecov Report

All modified and coverable lines are covered by tests ✅

Project coverage is 73.96%. Comparing base (3c46f4c) to head (22f8d9f).
Report is 282 commits behind head on main.

Additional details and impacted files

Impacted file tree graph

@@            Coverage Diff            @@
##           main    #1008       +/-   ##
=========================================
+ Coverage      0   73.96%   +73.96%     
=========================================
  Files         0       82       +82     
  Lines         0     6972     +6972     
=========================================
+ Hits          0     5157     +5157     
- Misses        0     1815     +1815     

see 82 files with indirect coverage changes

try {
for (int i = 0; i < nq; ++i) {
auto compute_dist_func = [=]() -> std::vector<DistId> {
auto xb = base_dataset->GetTensor();
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe add comments on why we cannot do it outside

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

added.

@mergify mergify bot added ci-passed and removed ci-passed labels Dec 30, 2024
@alexanderguzhva
Copy link
Collaborator

also, please squash the commits into a single one. Thanks.

@alwayslove2013 alwayslove2013 force-pushed the iterator_next_with_search_pool branch from 7a5a27d to 6ce1b7f Compare December 31, 2024 02:53
@mergify mergify bot removed the ci-passed label Dec 31, 2024
@alwayslove2013 alwayslove2013 force-pushed the iterator_next_with_search_pool branch from 6ce1b7f to 92f234b Compare December 31, 2024 02:56
@mergify mergify bot added the ci-passed label Dec 31, 2024
futs.emplace_back(
ThreadPool::GetGlobalSearchThreadPool()->push([&, idx = i]() { task_with_ordered_iterator(idx); }));
futs.emplace_back(ThreadPool::GetGlobalSearchThreadPool()->push([&, idx = i]() {
ThreadPool::ScopedSearchOmpSetter setter(1);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we rewrite the push to include this omp setter instead of calling it everywhere

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@cqy123456 Is this feasible? Are there any scenarios where we do not need the OmpSetter?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

OmpSetter only use in faiss index.

sign_(larger_is_closer ? -1 : 1) {
}

std::pair<int64_t, float>
Next() override {
if (!initialized_) {
throw std::runtime_error("Next should not be called before initialization");
initialize();
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Curios why we need this

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Throwing error is a bit too strict?

// dims out of 30000 total dims) sharing at least 1 common non-zero dimension.
results_.reserve(distances.size() * 0.3);
for (size_t i = 0; i < distances.size(); i++) {
if (distances[i] != 0) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why we do this? I think this PrecomputedDistanceIterator is used for both dense/sparse BF
@zhengbuqian

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

submitted a new commit to reduce misunderstanding.

@mergify mergify bot removed the ci-passed label Dec 31, 2024
@alwayslove2013 alwayslove2013 force-pushed the iterator_next_with_search_pool branch from 465f005 to cf9d25d Compare December 31, 2024 08:06
@mergify mergify bot added the ci-passed label Dec 31, 2024
@zilliztech zilliztech deleted a comment from alwayslove2013 Jan 2, 2025
@PwzXxm
Copy link
Collaborator

PwzXxm commented Jan 2, 2025

/lgtm
/approve

@PwzXxm
Copy link
Collaborator

PwzXxm commented Jan 2, 2025

/lgtm

@alwayslove2013
Copy link
Collaborator Author

issue: #997

@alwayslove2013
Copy link
Collaborator Author

/kind enhancement

Copy link
Collaborator

@foxspy foxspy left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

/lgtm

@foxspy
Copy link
Collaborator

foxspy commented Jan 2, 2025

/run-e2e

@sre-ci-robot
Copy link
Collaborator

[APPROVALNOTIFIER] This PR is APPROVED

This pull-request has been approved by: alexanderguzhva, alwayslove2013, foxspy, PwzXxm

The full list of commands accepted by this bot can be found here.

The pull request process is described here

Needs approval from an approver in each of these files:

Approvers can indicate their approval by writing /approve in a comment
Approvers can cancel approval by writing /approve cancel in a comment

@alwayslove2013 alwayslove2013 force-pushed the iterator_next_with_search_pool branch from 6ea0f92 to 22f8d9f Compare January 3, 2025 03:54
@sre-ci-robot sre-ci-robot removed the lgtm label Jan 3, 2025
@mergify mergify bot added the ci-passed label Jan 3, 2025
@cqy123456
Copy link
Collaborator

/lgtm

@sre-ci-robot sre-ci-robot merged commit 9a6a8df into zilliztech:main Jan 3, 2025
14 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

7 participants