diff --git a/3party/bsdiff-courgette/bsdiff/bsdiff.h b/3party/bsdiff-courgette/bsdiff/bsdiff.h index a4c76f7163..5cbd32a1d7 100644 --- a/3party/bsdiff-courgette/bsdiff/bsdiff.h +++ b/3party/bsdiff-courgette/bsdiff/bsdiff.h @@ -501,6 +501,9 @@ BSDiffStatus ApplyBinaryPatch(OldReader & old_reader, NewSink & new_sink, return UNEXPECTED_ERROR; } + if (cancellable.IsCancelled()) + return CANCELLED; + return OK; } } // namespace bsdiff diff --git a/storage/diff_scheme/diff_manager.cpp b/storage/diff_scheme/diff_manager.cpp index 9e5746a135..5c2ef89b1b 100644 --- a/storage/diff_scheme/diff_manager.cpp +++ b/storage/diff_scheme/diff_manager.cpp @@ -102,10 +102,6 @@ void Manager::ApplyDiff(ApplyDiffParams && p, base::Cancellable const & cancella case DiffApplicationResult::Ok: { diffFile->DeleteFromDisk(MapOptions::Diff); - std::lock_guard lock(m_mutex); - auto it = m_diffs.find(diffFile->GetCountryName()); - CHECK(it != m_diffs.end(), ()); - it->second.m_status = SingleDiffStatus::Applied; break; } case DiffApplicationResult::Cancelled: @@ -163,25 +159,13 @@ bool Manager::HasDiffFor(storage::CountryId const & countryId) const return WithDiff(countryId, [](DiffInfo const &) {}); } -void Manager::RemoveAppliedDiffs() -{ - std::lock_guard lock(m_mutex); - for (auto it = m_diffs.begin(); it != m_diffs.end();) - { - if (it->second.m_status == SingleDiffStatus::Applied) - it = m_diffs.erase(it); - else - ++it; - } - - if (m_diffs.empty()) - m_status = Status::NotAvailable; -} - void Manager::RemoveDiffForCountry(storage::CountryId const & countryId) { std::lock_guard lock(m_mutex); m_diffs.erase(countryId); + + if (m_diffs.empty()) + m_status = Status::NotAvailable; } void Manager::AbortDiffScheme() diff --git a/storage/diff_scheme/diff_manager.hpp b/storage/diff_scheme/diff_manager.hpp index fbe075695d..6dfcf5e882 100644 --- a/storage/diff_scheme/diff_manager.hpp +++ b/storage/diff_scheme/diff_manager.hpp @@ -60,7 +60,6 @@ public: // has been downloaded. bool HasDiffFor(storage::CountryId const & countryId) const; - void RemoveAppliedDiffs(); void RemoveDiffForCountry(storage::CountryId const & countryId); void AbortDiffScheme(); diff --git a/storage/storage.cpp b/storage/storage.cpp index efc04d7aa0..def9edde07 100644 --- a/storage/storage.cpp +++ b/storage/storage.cpp @@ -630,7 +630,6 @@ void Storage::DownloadNextCountryFromQueue() if (m_queue.empty()) { - m_diffManager.RemoveAppliedDiffs(); m_downloadingPolicy->ScheduleRetry(m_failedCountries, [this](CountriesSet const & needReload) { for (auto const & country : needReload) { @@ -1299,7 +1298,10 @@ bool Storage::DeleteCountryFilesFromDownloader(CountryId const & countryId) return false; if (m_latestDiffRequest && m_latestDiffRequest == countryId) + { m_diffsCancellable.Cancel(); + m_diffsBeingApplied.erase(countryId); + } MapOptions const opt = queuedCountry->GetInitOptions(); if (IsCountryFirstInQueue(countryId)) @@ -1571,6 +1573,7 @@ void Storage::ApplyDiff(CountryId const & countryId, functionGetPath(MapOptions::Map)); + m_diffManager.RemoveDiffForCountry(countryId); fn(true); break; } @@ -1621,6 +1632,7 @@ void Storage::ApplyDiff(CountryId const & countryId, function m_latestDiffRequest; + // Since the diff manager runs on a different thread, the result + // of diff application may return "Ok" when in fact the diff was + // cancelled. However, the storage thread knows for sure whether the + // latest request was to apply or to cancel the diff, and this knowledge + // is represented by |m_diffsBeingApplied|. + std::set m_diffsBeingApplied; + DownloadingPolicy m_defaultDownloadingPolicy; DownloadingPolicy * m_downloadingPolicy = &m_defaultDownloadingPolicy;