| /* |
| Licensed to the Apache Software Foundation (ASF) under one |
| or more contributor license agreements. See the NOTICE file |
| distributed with this work for additional information |
| regarding copyright ownership. The ASF licenses this file |
| to you 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 CV_SHARED_PTR_H_INCLUDED |
| #define CV_SHARED_PTR_H_INCLUDED |
| |
| #ifdef _WIN32 |
| #include <Windows.h> |
| #endif |
| |
| namespace |
| { |
| |
| struct AtomicCounter |
| { |
| inline AtomicCounter(long x) : val(x) |
| { |
| } |
| |
| inline long operator++() |
| { |
| #ifdef __GNUC__ |
| return __sync_add_and_fetch(&val, 1); |
| #elif _MSC_VER |
| return InterlockedIncrement(&val); |
| #endif |
| } |
| |
| inline long operator--() |
| { |
| #ifdef __GNUC__ |
| return __sync_sub_and_fetch(&val, 1); |
| #elif _MSC_VER |
| return InterlockedDecrement(&val); |
| #endif |
| } |
| |
| private: |
| long val; |
| }; |
| |
| typedef AtomicCounter RefCounterT; |
| |
| struct Holder |
| { |
| inline Holder(void* t) : target(t), count(1) |
| { |
| } |
| |
| inline void inc() |
| { |
| ++count; |
| } |
| |
| template <class T> inline void dec() |
| { |
| if (--count <= 0) |
| { |
| delete (T*) target; |
| delete this; |
| } |
| } |
| |
| void* target; |
| RefCounterT count; |
| }; |
| } |
| |
| template <class From, class To> inline bool RequireConvertible() |
| { |
| To* oth = (From*)0; |
| return oth == NULL; |
| } |
| |
| /* |
| * TODO: support alias-constructed pointers, add relational operators, |
| * support deleters, add the rest of the standard methods (swap, reset, use_count, etc.) |
| */ |
| template <class T> class shared_ptr |
| { |
| public: |
| |
| template <typename U> friend class shared_ptr; |
| |
| typedef T element_type; |
| |
| shared_ptr() : holder(NULL) |
| { |
| } |
| |
| explicit shared_ptr(element_type* t) : holder(new Holder(t)) |
| { |
| } |
| |
| shared_ptr(const shared_ptr& x, T* p) : holder(x.holder) |
| { |
| if (holder) |
| { |
| holder->inc(); |
| } |
| } |
| |
| ~shared_ptr() |
| { |
| if (holder) |
| { |
| holder->dec<element_type>(); |
| } |
| } |
| |
| shared_ptr(const shared_ptr& x) : holder(x.holder) |
| { |
| if (holder) |
| { |
| holder->inc(); |
| } |
| } |
| |
| template <class U> shared_ptr(const shared_ptr<U>& x) |
| { |
| RequireConvertible<U, element_type>(); |
| holder = x.holder; |
| if (holder) |
| { |
| holder->inc(); |
| } |
| } |
| |
| shared_ptr& operator=(const shared_ptr& x) |
| { |
| if (x.holder) |
| { |
| x.holder->inc(); |
| } |
| if (holder) |
| { |
| holder->dec<T>(); |
| } |
| holder = x.holder; |
| return *this; |
| } |
| |
| template <class U> shared_ptr& operator=(const shared_ptr<U>& x) |
| { |
| RequireConvertible<U, element_type>(); |
| if (x.holder) |
| { |
| x.holder->inc(); |
| } |
| if (holder) |
| { |
| holder->dec<T>(); |
| } |
| holder = x.holder; |
| return *this; |
| } |
| |
| element_type* operator->() const |
| { |
| if (!holder) |
| { |
| return NULL; |
| } |
| return (element_type*)holder->target; |
| } |
| |
| element_type& operator*() const |
| { |
| return *operator->(); |
| } |
| |
| operator bool() const |
| { |
| if (!holder) |
| { |
| return false; |
| } |
| return holder->target != NULL; |
| } |
| |
| element_type* get() const |
| { |
| return operator->(); |
| } |
| |
| private: |
| Holder* holder; |
| }; |
| |
| template<typename T> shared_ptr<T> make_shared() { return shared_ptr<T>(new T()); } |
| template<typename T, typename Arg1> shared_ptr<T> make_shared(const Arg1& arg1) { return shared_ptr<T>(new T(arg1)); } |
| template<typename T, typename Arg1, typename Arg2> shared_ptr<T> make_shared(const Arg1& arg1, const Arg2& arg2) { return shared_ptr<T>(new T(arg1, arg2)); } |
| template<typename T, typename Arg1, typename Arg2, typename Arg3> shared_ptr<T> make_shared(const Arg1& arg1, const Arg2& arg2, const Arg3& arg3) { return shared_ptr<T>(new T(arg1, arg2, arg3)); } |
| template<typename T, typename Arg1, typename Arg2, typename Arg3, typename Arg4> shared_ptr<T> make_shared(const Arg1& arg1, const Arg2& arg2, const Arg3& arg3, const Arg4& arg4) { return shared_ptr<T>(new T(arg1, arg2, arg3, arg4)); } |
| |
| template <class T, class U> inline shared_ptr<T> static_pointer_cast(const shared_ptr<U>& x) |
| { |
| return shared_ptr<T>(x, static_cast<T*>(x.get())); |
| } |
| |
| template <class T, class U> inline shared_ptr<T> const_pointer_cast(const shared_ptr<U>& x) |
| { |
| return shared_ptr<T>(x, const_cast<T*>(x.get())); |
| } |
| |
| template <class T, class U> inline shared_ptr<T> dynamic_pointer_cast(const shared_ptr<U>& x) |
| { |
| if (T* p = dynamic_cast<T*>(x.get())) |
| { |
| return shared_ptr<T>(x, p); |
| } |
| return shared_ptr<T>(); |
| } |
| |
| #endif // CV_SHARED_PTR_H_INCLUDED |
| |