-
Notifications
You must be signed in to change notification settings - Fork 1.2k
/
identity.h
122 lines (98 loc) · 3.54 KB
/
identity.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
//
// Copyright 2016 Pixar
//
// Licensed under the terms set forth in the LICENSE.txt file available at
// https://openusd.org/license.
//
#ifndef PXR_USD_SDF_IDENTITY_H
#define PXR_USD_SDF_IDENTITY_H
#include "pxr/pxr.h"
#include "pxr/usd/sdf/api.h"
#include "pxr/usd/sdf/declareHandles.h"
#include "pxr/usd/sdf/path.h"
#include <memory>
PXR_NAMESPACE_OPEN_SCOPE
class Sdf_IdentityRegistry;
class Sdf_IdRegistryImpl;
SDF_DECLARE_HANDLES(SdfLayer);
/// \class Sdf_Identity
///
/// Identifies the logical object behind an SdfSpec.
///
/// This is simply the layer the spec belongs to and the path to the spec.
///
class Sdf_Identity {
Sdf_Identity(Sdf_Identity const &) = delete;
Sdf_Identity &operator=(Sdf_Identity const &) = delete;
public:
/// Returns the layer that this identity refers to.
SDF_API
const SdfLayerHandle &GetLayer() const;
/// Returns the path that this identity refers to.
const SdfPath &GetPath() const {
return _path;
}
private:
// Ref-counting ops manage _refCount.
friend void TfDelegatedCountIncrement(Sdf_Identity*);
friend void TfDelegatedCountDecrement(Sdf_Identity*) noexcept;
friend class Sdf_IdentityRegistry;
friend class Sdf_IdRegistryImpl;
Sdf_Identity(Sdf_IdRegistryImpl *regImpl, const SdfPath &path)
: _refCount(0), _path(path), _regImpl(regImpl) {}
SDF_API
static void _UnregisterOrDelete(Sdf_IdRegistryImpl *reg, Sdf_Identity *id)
noexcept;
void _Forget();
mutable std::atomic_int _refCount;
SdfPath _path;
Sdf_IdRegistryImpl *_regImpl;
};
// Specialize TfDelegatedCountPtr operations.
inline void TfDelegatedCountIncrement(PXR_NS::Sdf_Identity* p) {
++p->_refCount;
}
inline void TfDelegatedCountDecrement(PXR_NS::Sdf_Identity* p) noexcept {
// Once the count hits zero, p is liable to be destroyed at any point,
// concurrently, by its owning registry if it happens to be doing a cleanup
// pass. Cache 'this' and the impl ptr in local variables so we have them
// before decrementing the count.
Sdf_Identity *self = p;
Sdf_IdRegistryImpl *reg = p->_regImpl;
if (--p->_refCount == 0) {
// Cannot use 'p' anymore here.
Sdf_Identity::_UnregisterOrDelete(reg, self);
}
}
class Sdf_IdentityRegistry {
Sdf_IdentityRegistry(const Sdf_IdentityRegistry&) = delete;
Sdf_IdentityRegistry& operator=(const Sdf_IdentityRegistry&) = delete;
public:
Sdf_IdentityRegistry(const SdfLayerHandle &layer);
~Sdf_IdentityRegistry();
/// Returns the layer that owns this registry.
const SdfLayerHandle &GetLayer() const {
return _layer;
}
/// Return the identity associated with \a path, issuing a new
/// one if necessary. The registry will track the identity
/// and update it if the logical object it represents moves
/// in namespace.
Sdf_IdentityRefPtr Identify(const SdfPath &path);
/// Update identity in response to a namespace edit.
void MoveIdentity(const SdfPath &oldPath, const SdfPath &newPath);
private:
friend class Sdf_Identity;
friend class Sdf_IdRegistryImpl;
// Remove the identity mapping for \a path to \a id from the registry. This
// is invoked when an identity's refcount hits zero.
SDF_API
void _UnregisterOrDelete();
/// The layer that owns this registry, and on behalf of which
/// this registry tracks identities.
const SdfLayerHandle _layer;
// Private implementation.
const std::unique_ptr<Sdf_IdRegistryImpl> _impl;
};
PXR_NAMESPACE_CLOSE_SCOPE
#endif // PXR_USD_SDF_IDENTITY_H