// Copyright (c) 2009 The Chromium Embedded Framework Authors. All rights // reserved. Use of this source code is governed by a BSD-style license that // can be found in the LICENSE file. #ifndef CEF_LIBCEF_DLL_CTOCPP_CTOCPP_REF_COUNTED_H_ #define CEF_LIBCEF_DLL_CTOCPP_CTOCPP_REF_COUNTED_H_ #pragma once #include "include/base/cef_logging.h" #include "include/capi/cef_base_capi.h" #include "include/cef_api_hash.h" #include "include/cef_base.h" #include "libcef_dll/wrapper_types.h" // Wrap a C structure with a C++ class. This is used when the implementation // exists on the other side of the DLL boundary but will have methods called on // this side of the DLL boundary. template class CefCToCppRefCounted : public BaseName { public: CefCToCppRefCounted(const CefCToCppRefCounted&) = delete; CefCToCppRefCounted& operator=(const CefCToCppRefCounted&) = delete; // Create a new wrapper instance for a structure reference received from the // other side. static CefRefPtr Wrap(StructName* s); // Retrieve the underlying structure reference from a wrapper instance for // return back to the other side. static StructName* Unwrap(CefRefPtr c); // CefBaseRefCounted methods increment/decrement reference counts on both this // object and the underlying wrapped structure. void AddRef() const { UnderlyingAddRef(); ref_count_.AddRef(); } bool Release() const; bool HasOneRef() const { return UnderlyingHasOneRef(); } bool HasAtLeastOneRef() const { return UnderlyingHasAtLeastOneRef(); } protected: CefCToCppRefCounted() = default; virtual ~CefCToCppRefCounted() = default; // If returning the structure across the DLL boundary use Unwrap() instead. StructName* GetStruct() const { WrapperStruct* wrapperStruct = GetWrapperStruct(this); return wrapperStruct->struct_; } private: // Used to associate this wrapper object and the structure reference received // from the other side. struct WrapperStruct; static WrapperStruct* GetWrapperStruct(const BaseName* obj, bool require_exact_type = true); // Unwrap as the derived type. static StructName* UnwrapDerived(CefWrapperType type, BaseName* c); // Increment/decrement reference counts on only the underlying class. NO_SANITIZE("cfi-icall") void UnderlyingAddRef() const { cef_base_ref_counted_t* base = reinterpret_cast(GetStruct()); if (base->add_ref) { base->add_ref(base); } } NO_SANITIZE("cfi-icall") bool UnderlyingRelease() const { cef_base_ref_counted_t* base = reinterpret_cast(GetStruct()); if (!base->release) { return false; } return base->release(base) ? true : false; } NO_SANITIZE("cfi-icall") bool UnderlyingHasOneRef() const { cef_base_ref_counted_t* base = reinterpret_cast(GetStruct()); if (!base->has_one_ref) { return false; } return base->has_one_ref(base) ? true : false; } NO_SANITIZE("cfi-icall") bool UnderlyingHasAtLeastOneRef() const { cef_base_ref_counted_t* base = reinterpret_cast(GetStruct()); if (!base->has_at_least_one_ref) { return false; } return base->has_at_least_one_ref(base) ? true : false; } CefRefCount ref_count_; static CefWrapperType kWrapperType; }; template struct CefCToCppRefCounted::WrapperStruct { CefWrapperType type_; StructName* struct_; ClassName wrapper_; }; template CefRefPtr CefCToCppRefCounted::Wrap( StructName* s) { if (!s) { return nullptr; } const auto size = reinterpret_cast(s)->size; if (size != sizeof(StructName)) { LOG(FATAL) << "Cannot wrap struct with invalid base.size value (got " << size << ", expected " << sizeof(StructName) << ") at API version " #if defined(WRAPPING_CEF_SHARED) << CEF_API_VERSION; #else << cef_api_version(); #endif } // Wrap their structure with the CefCToCppRefCounted object. WrapperStruct* wrapperStruct = new WrapperStruct; wrapperStruct->type_ = kWrapperType; wrapperStruct->struct_ = s; // Put the wrapper object in a smart pointer. CefRefPtr wrapperPtr(&wrapperStruct->wrapper_); // Release the reference that was added to the CefCppToC wrapper object on // the other side before their structure was passed to us. wrapperStruct->wrapper_.UnderlyingRelease(); // Return the smart pointer. return wrapperPtr; } template StructName* CefCToCppRefCounted::Unwrap( CefRefPtr c) { if (!c.get()) { return nullptr; } WrapperStruct* wrapperStruct = GetWrapperStruct(c.get(), /*require_exact_type=*/false); // If the type does not match this object then we need to unwrap as the // derived type. if (wrapperStruct->type_ != kWrapperType) { return UnwrapDerived(wrapperStruct->type_, c.get()); } // Add a reference to the CefCppToC wrapper object on the other side that // will be released once the structure is received. wrapperStruct->wrapper_.UnderlyingAddRef(); // Return their original structure. return wrapperStruct->struct_; } template bool CefCToCppRefCounted::Release() const { UnderlyingRelease(); if (ref_count_.Release()) { WrapperStruct* wrapperStruct = GetWrapperStruct(this); delete wrapperStruct; return true; } return false; } template typename CefCToCppRefCounted::WrapperStruct* CefCToCppRefCounted::GetWrapperStruct( const BaseName* obj, bool require_exact_type) { // Offset using the WrapperStruct size instead of individual member sizes to // avoid problems due to platform/compiler differences in structure padding. auto* wrapperStruct = reinterpret_cast( reinterpret_cast(const_cast(obj)) - (sizeof(WrapperStruct) - sizeof(ClassName))); if (require_exact_type) { // Verify that the wrapper offset was calculated correctly. CHECK_EQ(kWrapperType, wrapperStruct->type_); } return wrapperStruct; } #endif // CEF_LIBCEF_DLL_CTOCPP_CTOCPP_REF_COUNTED_H_