From: Patrick Donnelly Date: Thu, 27 Jul 2017 19:10:14 +0000 (-0700) Subject: common: add alloc_ptr smart pointer X-Git-Tag: v12.2.1~30^2~11 X-Git-Url: http://git.apps.os.sepia.ceph.com/?a=commitdiff_plain;h=1bf3b1939ccdf30c4645009887c93fcb03d029e8;p=ceph.git common: add alloc_ptr smart pointer This ptr is like a unique_ptr except it allocates the underlying object on access. The idea being that we can save memory if the object is only needed sometimes. Signed-off-by: Patrick Donnelly (cherry picked from commit 5fa557d2713558038af7579de8a3ca56d58b8d90) --- diff --git a/src/include/alloc_ptr.h b/src/include/alloc_ptr.h new file mode 100644 index 0000000000000..258c5833845a8 --- /dev/null +++ b/src/include/alloc_ptr.h @@ -0,0 +1,91 @@ +// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*- +// vim: ts=8 sw=2 smarttab +/* + * Ceph - scalable distributed file system + * + * Copyright (C) 2017 Red Hat, Inc. + * + * This is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License version 2.1, as published by the Free Software + * Foundation. See file COPYING. + * + */ + +#ifndef CEPH_ALLOC_PTR_H +#define CEPH_ALLOC_PTR_H + +#include + +template +class alloc_ptr +{ +public: + typedef typename std::pointer_traits< std::unique_ptr >::pointer pointer; + typedef typename std::pointer_traits< std::unique_ptr >::element_type element_type; + + alloc_ptr() : ptr() {} + + template + alloc_ptr(U&& u) : ptr(std::forward(u)) {} + + alloc_ptr(alloc_ptr&& rhs) : ptr(std::move(rhs.ptr)) {} + alloc_ptr(const alloc_ptr& rhs) = delete; + alloc_ptr& operator=(const alloc_ptr&& rhs) { + ptr = rhs.ptr; + } + alloc_ptr& operator=(const alloc_ptr& rhs) { + ptr = rhs.ptr; + } + + void swap (alloc_ptr& rhs) { + ptr.swap(rhs.ptr); + } + element_type* release() { + return ptr.release(); + } + void reset(element_type *p = nullptr) { + ptr.reset(p); + } + element_type* get() const { + if (!ptr) + ptr.reset(new element_type); + return ptr.get(); + } + element_type& operator*() const { + if (!ptr) + ptr.reset(new element_type); + return *ptr; + } + element_type* operator->() const { + if (!ptr) + ptr.reset(new element_type); + return ptr.get(); + } + operator bool() const { + return !!ptr; + } + + friend bool operator< (const alloc_ptr& lhs, const alloc_ptr& rhs) { + return std::less(*lhs, *rhs); + } + friend bool operator<=(const alloc_ptr& lhs, const alloc_ptr& rhs) { + return std::less_equal(*lhs, *rhs); + } + friend bool operator> (const alloc_ptr& lhs, const alloc_ptr& rhs) { + return std::greater(*lhs, *rhs); + } + friend bool operator>=(const alloc_ptr& lhs, const alloc_ptr& rhs) { + return std::greater_equal(*lhs, *rhs); + } + friend bool operator==(const alloc_ptr& lhs, const alloc_ptr& rhs) { + return *lhs == *rhs; + } + friend bool operator!=(const alloc_ptr& lhs, const alloc_ptr& rhs) { + return *lhs != *rhs; + } +private: + mutable std::unique_ptr ptr; +}; + +#endif