Version 4.3.0.0.beta1, tag libreoffice-4.3.0.0.beta1
[LibreOffice.git] / include / rtl / ref.hxx
blobcccbc0105ff6f54102f0af0fb4ba0e79e4d5371a
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #ifndef INCLUDED_RTL_REF_HXX
21 #define INCLUDED_RTL_REF_HXX
23 #include <sal/types.h>
24 #include <osl/diagnose.h>
25 #include <osl/interlck.h>
27 namespace rtl
30 /** Interface for a reference type.
32 class IReference
34 public:
35 /** @see osl_incrementInterlockedCount.
37 virtual oslInterlockedCount SAL_CALL acquire() = 0;
39 /** @see osl_decrementInterlockedCount.
41 virtual oslInterlockedCount SAL_CALL release() = 0;
43 #if !defined _MSC_VER // public -> protected changes mangled names there
44 protected:
45 #endif
46 ~IReference() {}
47 // avoid warnings about virtual members and non-virtual dtor
51 /** Template reference class for reference type derived from IReference.
53 template <class reference_type>
54 class Reference
56 /** The <b>reference_type</b> body pointer.
58 reference_type * m_pBody;
61 public:
62 /** Constructor...
64 inline Reference()
65 : m_pBody (0)
69 /** Constructor...
71 inline Reference (reference_type * pBody)
72 : m_pBody (pBody)
74 if (m_pBody)
75 m_pBody->acquire();
79 /** Copy constructor...
81 inline Reference (const Reference<reference_type> & handle)
82 : m_pBody (handle.m_pBody)
84 if (m_pBody)
85 m_pBody->acquire();
89 /** Destructor...
91 inline ~Reference()
93 if (m_pBody)
94 m_pBody->release();
97 /** Set...
98 Similar to assignment.
100 inline Reference<reference_type> &
101 SAL_CALL set (reference_type * pBody)
103 if (pBody)
104 pBody->acquire();
105 reference_type * const pOld = m_pBody;
106 m_pBody = pBody;
107 if (pOld)
108 pOld->release();
109 return *this;
112 /** Assignment.
113 Unbinds this instance from its body (if bound) and
114 bind it to the body represented by the handle.
116 inline Reference<reference_type> &
117 SAL_CALL operator= (const Reference<reference_type> & handle)
119 return set( handle.m_pBody );
122 /** Assignment...
124 inline Reference<reference_type> &
125 SAL_CALL operator= (reference_type * pBody)
127 return set( pBody );
130 /** Unbind the body from this handle.
131 Note that for a handle representing a large body,
132 "handle.clear().set(new body());" _might_
133 perform a little bit better than "handle.set(new body());",
134 since in the second case two large objects exist in memory
135 (the old body and the new body).
137 inline Reference<reference_type> & SAL_CALL clear()
139 if (m_pBody)
141 reference_type * const pOld = m_pBody;
142 m_pBody = 0;
143 pOld->release();
145 return *this;
149 /** Get the body. Can be used instead of operator->().
150 I.e. handle->someBodyOp() and handle.get()->someBodyOp()
151 are the same.
153 inline reference_type * SAL_CALL get() const
155 return m_pBody;
159 /** Probably most common used: handle->someBodyOp().
161 inline reference_type * SAL_CALL operator->() const
163 OSL_PRECOND(m_pBody, "Reference::operator->() : null body");
164 return m_pBody;
168 /** Allows (*handle).someBodyOp().
170 inline reference_type & SAL_CALL operator*() const
172 OSL_PRECOND(m_pBody, "Reference::operator*() : null body");
173 return *m_pBody;
177 /** Returns True if the handle does point to a valid body.
179 inline bool SAL_CALL is() const
181 return (m_pBody != 0);
185 /** Returns True if this points to pBody.
187 inline bool SAL_CALL operator== (const reference_type * pBody) const
189 return (m_pBody == pBody);
193 /** Returns True if handle points to the same body.
195 inline bool
196 SAL_CALL operator== (const Reference<reference_type> & handle) const
198 return (m_pBody == handle.m_pBody);
202 /** Needed to place References into STL collection.
204 inline bool
205 SAL_CALL operator!= (const Reference<reference_type> & handle) const
207 return (m_pBody != handle.m_pBody);
211 /** Needed to place References into STL collection.
213 inline bool
214 SAL_CALL operator< (const Reference<reference_type> & handle) const
216 return (m_pBody < handle.m_pBody);
220 /** Needed to place References into STL collection.
222 inline bool
223 SAL_CALL operator> (const Reference<reference_type> & handle) const
225 return (m_pBody > handle.m_pBody);
229 /// @cond INTERNAL
230 /** Enables boost::mem_fn and boost::bind to recognize Reference.
232 template <typename T>
233 inline T * get_pointer( Reference<T> const& r )
235 return r.get();
237 /// @endcond
239 } // namespace rtl
241 #endif /* ! INCLUDED_RTL_REF_HXX */
243 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */