summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMaksim Denisov <denisov.maksim@huawei.com>2024-07-01 16:54:24 +0200
committerMaksim Denisov <denisov.maksim@huawei.com>2024-07-05 13:32:39 +0200
commit18d26dea1bf9c29a073870988a190cdd7dbc8ed0 (patch)
tree5887e73048f7c576bece3fc23f401f7ddd4bdd00
parent993245d862fcd44b1d8d404b970cbd774526ae2b (diff)
downloadjustbuild-18d26dea1bf9c29a073870988a190cdd7dbc8ed0.tar.gz
Implement uplinking logic in a separate class.
-rw-r--r--src/buildtool/storage/TARGETS2
-rw-r--r--src/buildtool/storage/uplinker.cpp117
-rw-r--r--src/buildtool/storage/uplinker.hpp101
3 files changed, 220 insertions, 0 deletions
diff --git a/src/buildtool/storage/TARGETS b/src/buildtool/storage/TARGETS
index 2e2fc549..00a5f2d2 100644
--- a/src/buildtool/storage/TARGETS
+++ b/src/buildtool/storage/TARGETS
@@ -31,6 +31,7 @@
, "large_object_cas.hpp"
, "large_object_cas.tpp"
, "compactifier.hpp"
+ , "uplinker.hpp"
]
, "private-hdrs": ["compactification_task.hpp"]
, "srcs":
@@ -38,6 +39,7 @@
, "garbage_collector.cpp"
, "compactifier.cpp"
, "compactification_task.cpp"
+ , "uplinker.cpp"
]
, "deps":
[ "config"
diff --git a/src/buildtool/storage/uplinker.cpp b/src/buildtool/storage/uplinker.cpp
new file mode 100644
index 00000000..42213b33
--- /dev/null
+++ b/src/buildtool/storage/uplinker.cpp
@@ -0,0 +1,117 @@
+// Copyright 2024 Huawei Cloud Computing Technology Co., Ltd.
+//
+// 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.
+
+#ifndef BOOTSTRAP_BUILD_TOOL
+
+#include "src/buildtool/storage/uplinker.hpp"
+
+#include <algorithm>
+#include <cstddef>
+
+#include "src/buildtool/common/bazel_types.hpp"
+#include "src/buildtool/file_system/object_type.hpp"
+#include "src/buildtool/storage/local_ac.hpp"
+#include "src/buildtool/storage/local_cas.hpp"
+#include "src/buildtool/storage/storage.hpp"
+#include "src/buildtool/storage/target_cache.hpp"
+#include "src/buildtool/storage/target_cache_entry.hpp"
+
+namespace {
+[[nodiscard]] auto CreateGenerations(
+ gsl::not_null<StorageConfig const*> const& storage_config) noexcept
+ -> std::vector<Generation> {
+ std::vector<Generation> generations;
+ generations.reserve(storage_config->NumGenerations());
+ for (std::size_t i = 0; i < storage_config->NumGenerations(); ++i) {
+ auto const gen_config = storage_config->CreateGenerationConfig(i);
+ generations.emplace_back(gen_config);
+ }
+ return generations;
+}
+} // namespace
+
+GlobalUplinker::GlobalUplinker(
+ gsl::not_null<StorageConfig const*> const& storage_config) noexcept
+ : storage_config_{*storage_config},
+ generations_{CreateGenerations(&storage_config_)} {}
+
+auto GlobalUplinker::UplinkBlob(bazel_re::Digest const& digest,
+ bool is_executable) const noexcept -> bool {
+ // Try to find blob in all generations.
+ auto const& latest = generations_[Generation::kYoungest].CAS();
+ return std::any_of(
+ generations_.begin(),
+ generations_.end(),
+ [&latest, &digest, is_executable](Generation const& generation) {
+ return generation.CAS().LocalUplinkBlob(latest,
+ digest,
+ is_executable,
+ /*skip_sync=*/true,
+ /*splice_result=*/true);
+ });
+}
+
+auto GlobalUplinker::UplinkTree(bazel_re::Digest const& digest) const noexcept
+ -> bool {
+ // Try to find blob in all generations.
+ auto const& latest = generations_[Generation::kYoungest].CAS();
+ return std::any_of(generations_.begin(),
+ generations_.end(),
+ [&latest, &digest](Generation const& generation) {
+ return generation.CAS().LocalUplinkTree(
+ latest, digest, /*splice_result=*/true);
+ });
+}
+
+auto GlobalUplinker::UplinkLargeBlob(
+ bazel_re::Digest const& digest) const noexcept -> bool {
+ // Try to find large entry in all generations.
+ auto const& latest = generations_[Generation::kYoungest].CAS();
+ return std::any_of(
+ generations_.begin(),
+ generations_.end(),
+ [&latest, &digest](Generation const& generation) {
+ return generation.CAS().LocalUplinkLargeObject<ObjectType::File>(
+ latest, digest);
+ });
+}
+
+auto GlobalUplinker::UplinkActionCacheEntry(
+ bazel_re::Digest const& action_id) const noexcept -> bool {
+ // Try to find action-cache entry in all generations.
+ auto const& latest = generations_[Generation::kYoungest].ActionCache();
+ return std::any_of(generations_.begin(),
+ generations_.end(),
+ [&latest, &action_id](Generation const& generation) {
+ return generation.ActionCache().LocalUplinkEntry(
+ latest, action_id);
+ });
+}
+
+auto GlobalUplinker::UplinkTargetCacheEntry(
+ TargetCacheKey const& key,
+ std::optional<std::string> const& shard) const noexcept -> bool {
+ // Try to find target-cache entry in all generations.
+ auto const& latest =
+ generations_[Generation::kYoungest].TargetCache().WithShard(shard);
+ return std::any_of(
+ generations_.begin(),
+ generations_.end(),
+ [&latest, &key, &shard](Generation const& generation) {
+ return generation.TargetCache().WithShard(shard).LocalUplinkEntry(
+ latest, key);
+ });
+}
+
+#endif // BOOTSTRAP_BUILD_TOOL
diff --git a/src/buildtool/storage/uplinker.hpp b/src/buildtool/storage/uplinker.hpp
new file mode 100644
index 00000000..c713155c
--- /dev/null
+++ b/src/buildtool/storage/uplinker.hpp
@@ -0,0 +1,101 @@
+// Copyright 2024 Huawei Cloud Computing Technology Co., Ltd.
+//
+// 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.
+
+#ifndef INCLUDED_SRC_BUILDTOOL_STORAGE_UPLINKER_HPP
+#define INCLUDED_SRC_BUILDTOOL_STORAGE_UPLINKER_HPP
+
+#include <optional>
+#include <string>
+#include <type_traits>
+#include <vector>
+
+#include "gsl/gsl"
+#include "src/buildtool/storage/config.hpp"
+
+template <bool>
+class LocalStorage;
+class TargetCacheKey;
+
+namespace build::bazel::remote::execution::v2 {
+class Digest;
+}
+namespace bazel_re = build::bazel::remote::execution::v2;
+
+/// \brief Global uplinker implementation.
+/// Responsible for uplinking objects across all generations to latest
+/// generation.
+class GlobalUplinker final {
+ public:
+ explicit GlobalUplinker(
+ gsl::not_null<StorageConfig const*> const& storage_config) noexcept;
+
+ /// \brief Uplink blob across LocalCASes from all generations to latest.
+ /// Note that blobs will NOT be synced between file/executable CAS.
+ /// \param digest Digest of the blob to uplink.
+ /// \param is_executable Indicate that blob is an executable.
+ /// \returns true if blob was found and successfully uplinked.
+ [[nodiscard]] auto UplinkBlob(bazel_re::Digest const& digest,
+ bool is_executable) const noexcept -> bool;
+
+ /// \brief Uplink tree across LocalCASes from all generations to latest.
+ /// Note that the tree will be deeply uplinked, i.e., all entries referenced
+ /// by this tree will be uplinked before (including sub-trees).
+ /// \param digest Digest of the tree to uplink.
+ /// \returns true if tree was found and successfully uplinked (deep).
+ [[nodiscard]] auto UplinkTree(bazel_re::Digest const& digest) const noexcept
+ -> bool;
+
+ /// \brief Uplink large blob entry across LocalCASes from all generations to
+ /// latest. This method does not splice the large object.
+ /// \param digest Digest of the large blob entry to uplink.
+ /// \returns true if large entry was found and successfully uplinked.
+ [[nodiscard]] auto UplinkLargeBlob(
+ bazel_re::Digest const& digest) const noexcept -> bool;
+
+ /// \brief Uplink entry from action cache across all generations to latest.
+ /// Note that the entry will be uplinked including all referenced items.
+ /// \param action_id Id of the action to uplink entry for.
+ /// \returns true if cache entry was found and successfully uplinked.
+ [[nodiscard]] auto UplinkActionCacheEntry(
+ bazel_re::Digest const& action_id) const noexcept -> bool;
+
+ /// \brief Uplink entry from target cache across all generations to latest.
+ /// Note that the entry will be uplinked including all referenced items.
+ /// \param key Target cache key to uplink entry for.
+ /// \param shard Optional explicit shard, if the default is not intended.
+ /// \returns true if cache entry was found and successfully uplinked.
+ [[nodiscard]] auto UplinkTargetCacheEntry(
+ TargetCacheKey const& key,
+ std::optional<std::string> const& shard = std::nullopt) const noexcept
+ -> bool;
+
+ private:
+ StorageConfig const& storage_config_;
+ std::vector<LocalStorage<false>> const generations_;
+};
+
+/// \brief An empty constructable Uplinker. Although it doesn't have any
+/// interface, it allows objects employing uplinking store the uplinker by
+/// reference instead of unobvious 'optional' raw pointers.
+class StubUplinker final {
+ public:
+ explicit StubUplinker(
+ gsl::not_null<StorageConfig const*> const& /*unused*/) noexcept {}
+};
+
+template <bool kDoGlobalUplink>
+using Uplinker =
+ std::conditional_t<kDoGlobalUplink, GlobalUplinker, StubUplinker>;
+
+#endif // INCLUDED_SRC_BUILDTOOL_STORAGE_UPLINKER_HPP