Rocket.Chat.ReactNative/ios/Pods/Flipper-RSocket/yarpl/Refcounted.h

86 lines
2.5 KiB
C++

// Copyright (c) Facebook, Inc. and its affiliates.
//
// 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.
#pragma once
#include <folly/Synchronized.h>
#include <atomic>
namespace yarpl {
template <typename T>
struct AtomicReference {
folly::Synchronized<std::shared_ptr<T>, std::mutex> ref;
AtomicReference() = default;
AtomicReference(std::shared_ptr<T>&& r) {
*(ref.lock()) = std::move(r);
}
};
template <typename T>
std::shared_ptr<T> atomic_load(AtomicReference<T>* ar) {
return *(ar->ref.lock());
}
template <typename T>
std::shared_ptr<T> atomic_exchange(
AtomicReference<T>* ar,
std::shared_ptr<T> r) {
auto refptr = ar->ref.lock();
auto old = std::move(*refptr);
*refptr = std::move(r);
return old;
}
template <typename T>
std::shared_ptr<T> atomic_exchange(AtomicReference<T>* ar, std::nullptr_t) {
return atomic_exchange(ar, std::shared_ptr<T>());
}
template <typename T>
void atomic_store(AtomicReference<T>* ar, std::shared_ptr<T> r) {
*ar->ref.lock() = std::move(r);
}
class enable_get_ref : public std::enable_shared_from_this<enable_get_ref> {
private:
virtual void dummy_internal_get_ref() {}
protected:
// materialize a reference to 'this', but a type even further derived from
// Derived, because C++ doesn't have covariant return types on methods
template <typename As>
std::shared_ptr<As> ref_from_this(As* ptr) {
// at runtime, ensure that the most derived class can indeed be
// converted into an 'as'
(void)ptr; // silence 'unused parameter' errors in Release builds
return std::static_pointer_cast<As>(this->shared_from_this());
}
template <typename As>
std::shared_ptr<As> ref_from_this(As const* ptr) const {
// at runtime, ensure that the most derived class can indeed be
// converted into an 'as'
(void)ptr; // silence 'unused parameter' errors in Release builds
return std::static_pointer_cast<As const>(this->shared_from_this());
}
public:
virtual ~enable_get_ref() = default;
};
} /* namespace yarpl */