diff options
author | Maksim Denisov <denisov.maksim@huawei.com> | 2024-07-01 16:54:24 +0200 |
---|---|---|
committer | Maksim Denisov <denisov.maksim@huawei.com> | 2024-07-05 13:32:39 +0200 |
commit | 18d26dea1bf9c29a073870988a190cdd7dbc8ed0 (patch) | |
tree | 5887e73048f7c576bece3fc23f401f7ddd4bdd00 | |
parent | 993245d862fcd44b1d8d404b970cbd774526ae2b (diff) | |
download | justbuild-18d26dea1bf9c29a073870988a190cdd7dbc8ed0.tar.gz |
Implement uplinking logic in a separate class.
-rw-r--r-- | src/buildtool/storage/TARGETS | 2 | ||||
-rw-r--r-- | src/buildtool/storage/uplinker.cpp | 117 | ||||
-rw-r--r-- | src/buildtool/storage/uplinker.hpp | 101 |
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 |