| // Copyright 2012 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef BASE_MAC_SCOPED_AUTHORIZATIONREF_H_ |
| #define BASE_MAC_SCOPED_AUTHORIZATIONREF_H_ |
| |
| #include <Security/Authorization.h> |
| |
| #include <utility> |
| |
| #include "base/base_export.h" |
| #include "base/check.h" |
| |
| // `ScopedAuthorizationRef` maintains ownership of an `AuthorizationRef`. It is |
| // patterned after the `unique_ptr` interface. |
| |
| namespace base::mac { |
| |
| class BASE_EXPORT ScopedAuthorizationRef { |
| public: |
| explicit ScopedAuthorizationRef(AuthorizationRef authorization = nullptr) |
| : authorization_(authorization) {} |
| |
| ScopedAuthorizationRef(const ScopedAuthorizationRef&) = delete; |
| ScopedAuthorizationRef& operator=(const ScopedAuthorizationRef&) = delete; |
| |
| ScopedAuthorizationRef(ScopedAuthorizationRef&& that) |
| : authorization_(std::exchange(that.authorization_, nullptr)) {} |
| ScopedAuthorizationRef& operator=(ScopedAuthorizationRef&& that) { |
| authorization_ = std::exchange(that.authorization_, nullptr); |
| return *this; |
| } |
| |
| ~ScopedAuthorizationRef() { |
| if (authorization_) { |
| FreeInternal(); |
| } |
| } |
| |
| void reset(AuthorizationRef authorization = nullptr) { |
| if (authorization_ != authorization) { |
| if (authorization_) { |
| FreeInternal(); |
| } |
| authorization_ = authorization; |
| } |
| } |
| |
| bool operator==(AuthorizationRef that) const { |
| return authorization_ == that; |
| } |
| |
| bool operator!=(AuthorizationRef that) const { |
| return authorization_ != that; |
| } |
| |
| operator AuthorizationRef() const { |
| return authorization_; |
| } |
| |
| explicit operator bool() const { return authorization_ != nullptr; } |
| |
| // This is to be used only to take ownership of objects that are created |
| // by pass-by-pointer create functions. To enforce this, require that the |
| // object be reset to NULL before this may be used. |
| [[nodiscard]] AuthorizationRef* InitializeInto() { |
| DCHECK(!authorization_); |
| return &authorization_; |
| } |
| |
| AuthorizationRef get() const { |
| return authorization_; |
| } |
| |
| // ScopedAuthorizationRef::release() is like std::unique_ptr<>::release. It is |
| // NOT a wrapper for AuthorizationFree(). To force a ScopedAuthorizationRef |
| // object to call AuthorizationFree(), use ScopedAuthorizationRef::reset(). |
| [[nodiscard]] AuthorizationRef release() { |
| AuthorizationRef temp = authorization_; |
| authorization_ = nullptr; |
| return temp; |
| } |
| |
| private: |
| // Calling AuthorizationFree, defined in Security.framework, from an inline |
| // function, results in link errors when linking dynamically with |
| // libbase.dylib. So wrap the call in an un-inlined method. This method |
| // doesn't check if |authorization_| is null; that check should be in the |
| // inlined callers. |
| void FreeInternal(); |
| |
| AuthorizationRef authorization_; |
| }; |
| |
| } // namespace base::mac |
| |
| #endif // BASE_MAC_SCOPED_AUTHORIZATIONREF_H_ |