forked from RobotLocomotion/drake
-
Notifications
You must be signed in to change notification settings - Fork 0
/
internal_geometry.h
342 lines (281 loc) · 13.5 KB
/
internal_geometry.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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
#pragma once
#include <memory>
#include <optional>
#include <string>
#include <unordered_map>
#include <unordered_set>
#include <utility>
#include "drake/common/copyable_unique_ptr.h"
#include "drake/common/drake_copyable.h"
#include "drake/geometry/geometry_ids.h"
#include "drake/geometry/geometry_roles.h"
#include "drake/geometry/internal_frame.h"
#include "drake/geometry/proximity/volume_mesh.h"
#include "drake/geometry/shape_specification.h"
#include "drake/math/rigid_transform.h"
namespace drake {
namespace geometry {
namespace internal {
// TODO(xuchenhan-tri): Consider splitting this class into RigidInternalGeometry
// and DeformableInternalGeometry because the two abstractions don't fit in the
// same class very nicely. For example, rigid geometries have the concept of
// "hangs-in" frame and parent geometry while deformable geometries have mesh
// representations. Right now we are manually disabling incompatible features.
/* The internal representation of the fixed portion of the scene graph
geometry. This includes those aspects that do *not* depend on computed values.
It includes things like the topology (which frame is a geometry attached to),
internal bookkeeping (how do we identify a proximity-engine representation of
the registered geometry), its name, and its *declared* geometry. */
class InternalGeometry {
public:
DRAKE_DEFAULT_COPY_AND_MOVE_AND_ASSIGN(InternalGeometry)
// TODO(SeanCurtis-TRI): Is this strictly required? Typically, I have this to
// be compatible with STL structures that need to default construct. Confirm
// and document if such is the case.
/* Default constructor. The geometry id will be invalid, the shape will
be nullptr, and the pose will be uninitialized. */
InternalGeometry() {}
/* Constructs a rigid internal geometry without any assigned roles defined as
an *immediate* child of the frame. Therefore, it is assumed that X_FG = X_PG.
@param source_id The id for the source that registered this geometry.
@param shape The shape specification for this instance.
@param frame_id The id of the frame this belongs to.
@param geometry_id The identifier for _this_ geometry.
@param name The name of the geometry.
@param X_FG The pose of the geometry G in the parent frame F. */
InternalGeometry(SourceId source_id, std::unique_ptr<Shape> shape,
FrameId frame_id, GeometryId geometry_id, std::string name,
math::RigidTransform<double> X_FG);
/* Constructs a deformable internal geometry without any assigned
roles defined in the given frame. The position of the geometry is
characterized by the positions of the mesh vertices in the given frame
instead of "pose" of the geometry in the frame. X_FG and X_PG are always
identity.
@param source_id The id for the source that registered this
geometry.
@param shape The shape specification for this instance.
@param frame_id The id of the frame this belongs to.
@param geometry_id The identifier for _this_ geometry.
@param name The name of the geometry.
@param resolution_hint The resolution hint for the mesh representation of
the geometry. */
InternalGeometry(SourceId source_id, std::unique_ptr<Shape> shape,
FrameId frame_id, GeometryId geometry_id, std::string name,
double resolution_hint);
/* Compares two %InternalGeometry instances for "equality". Two internal
geometries are considered equal if they have the same geometry identifier.
*/
bool operator==(const InternalGeometry &other) const {
return id_ == other.id_;
}
/* Compares two %InternalGeometry instances for inequality. See operator==()
for the definition of equality. */
bool operator!=(const InternalGeometry &other) const {
return !(*this == other);
}
/* @name Geometry properties */
//@{
/* Returns the shape specification for this geometry. */
const Shape& shape() const { return *shape_spec_; }
/* Returns the globally unique identifier for this geometry. */
GeometryId id() const { return id_; }
/* Returns the name of this geometry. */
const std::string& name() const { return name_; }
/* Returns the source id that registered the geometry. */
SourceId source_id() const { return source_id_; }
/* Returns true if this geometry belongs to the source with the given id. */
bool belongs_to_source(SourceId id) const { return source_id_ == id; }
//@}
/* @name Scene Graph topology */
//@{
/* Reports the frame this internal geometry is rigidly attached to. */
FrameId frame_id() const { return frame_id_; }
/* Returns true if the geometry is affixed to the frame with the given
`frame_id`. */
bool is_child_of_frame(FrameId frame_id) const {
return frame_id == frame_id_;
}
/* Returns the pose of this geometry in the declared *parent* frame -- note
if this geometry was registered as a child of another geometry it will *not*
be the same as X_FG(). */
const math::RigidTransform<double>& X_PG() const { return X_PG_; }
/* Returns the pose of this geometry in the frame to which it is ultimately
rigidly attached. This is in contrast to X_PG(). */
const math::RigidTransform<double>& X_FG() const { return X_FG_; }
// TODO(SeanCurtis-TRI): Determine if tracking this parent geometry is
// necessary for now or if that only exists to facilitate removal later on.
/* Returns the declared parent geometry (if one exists). */
std::optional<GeometryId> parent_id() const { return parent_geometry_id_; }
/* Sets this geometry to have *another* geometry as parent. In this case,
the pose set in the constructor is assumed to be X_PG and the X_FG value must
be updated.
@param id The id of the parent geometry.
@param X_FG The new value for X_FG (assuming the constructed value is to be
interpreted as X_PG).
@pre this geometry is rigid. Deformable geometries don't have parent or child
geometries. */
void set_geometry_parent(GeometryId id, math::RigidTransform<double> X_FG) {
DRAKE_DEMAND(!is_deformable());
parent_geometry_id_ = id;
X_FG_ = std::move(X_FG);
}
/* Returns true if this geometry has a geometry parent and the parent has the
given `geometry_id`. */
bool is_child_of_geometry(GeometryId geometry_id) const {
return parent_geometry_id_ && *parent_geometry_id_ == geometry_id;
}
/* Returns a list of identifiers of geometries that have *this* geometry as
a parent. In other words, for an internal geometry `g`, `g.index()`
appears in this list iff `this->is_child_of_geometry(g.index())` returns
true. */
const std::unordered_set<GeometryId>& child_geometry_ids() const {
return child_geometry_ids_;
}
/* Returns true if this geometry has a child geometry with the given
`geometry_id`. */
bool has_child(GeometryId geometry_id) const {
return child_geometry_ids_.find(geometry_id) != child_geometry_ids_.end();
}
/* Adds a geometry with the given `geometry_id` to this geometry's set of
children.
@pre this geometry is rigid. Deformable geometries don't have parent or child
geometries. */
void add_child(GeometryId geometry_id) {
DRAKE_DEMAND(!is_deformable());
child_geometry_ids_.insert(geometry_id);
}
/* Removes the given `geometry_id` from the set of geometries that this frame
considers to be children. If the id is not in the set of children, nothing
happens. */
void remove_child(GeometryId geometry_id) {
DRAKE_ASSERT(child_geometry_ids_.count(geometry_id) > 0);
child_geometry_ids_.erase(geometry_id);
}
/* A geometry is deformable iff an internal mesh representation is required to
define its topology. For example, geometries participating in hydroelastic
contact usually have mesh representations, but those meshes are only used for
proximity query purposes and do not affect whether the geometry is
deformable. */
bool is_deformable() const { return reference_mesh_ != nullptr; }
/* Returns true if the geometry can move with respect to the World frame. That
includes any deformable geometry, and any rigid geometry attached to a frame
other than World. */
bool is_dynamic() const {
// NOTE: If I allow non-dynamic frames welded to the world, this will not
// be sufficient.
return (frame_id_ != InternalFrame::world_frame_id() || is_deformable());
}
//@}
/* @name Role management
These methods determine what the internal geometry *knows* about its role.
Updating a geometry's knowledge is not the same as actually applying that
role. It is the job of GeometryState to make sure that an individual
geometry's understanding of its role is kept in sync with the corresponding
engine's understanding as well. */
//@{
/* Assigns a proximity role to this geometry, replacing any proximity
properties that were previously assigned. */
void SetRole(ProximityProperties properties) {
proximity_props_ = std::move(properties);
}
/* Assigns an illustration role to this geometry, replacing any illustration
properties that were previously assigned. */
void SetRole(IllustrationProperties properties) {
illustration_props_ = std::move(properties);
}
/* Assigns a perception role to this geometry. Throws if the geometry has
already had perception properties assigned. */
void SetRole(PerceptionProperties properties) {
if (perception_props_) {
throw std::logic_error("Geometry already has perception role assigned");
}
perception_props_ = std::move(properties);
}
/* Reports if the geometry has the indicated `role`. */
bool has_role(Role role) const;
/* Reports if the geometry has a proximity role. */
bool has_proximity_role() const { return proximity_props_ != std::nullopt; }
/* Reports if the geometry has a illustration role. */
bool has_illustration_role() const {
return illustration_props_ != std::nullopt;
}
/* Reports if the geometry has a perception role. */
bool has_perception_role() const { return perception_props_ != std::nullopt; }
/* Returns a pointer to the geometry's proximity properties (if they are
defined. Nullptr otherwise. */
const ProximityProperties* proximity_properties() const {
if (proximity_props_) return &*proximity_props_;
return nullptr;
}
/* Returns a pointer to the geometry's illustration properties (if they are
defined. Nullptr otherwise. */
const IllustrationProperties* illustration_properties() const {
if (illustration_props_) return &*illustration_props_;
return nullptr;
}
/* Returns a pointer to the geometry's perception properties, or nullptr if
not defined. */
const PerceptionProperties* perception_properties() const {
if (perception_props_) return &*perception_props_;
return nullptr;
}
/* Removes the proximity role assigned to this geometry -- if there was
no proximity role previously, this has no effect. */
void RemoveProximityRole() {
proximity_props_ = std::nullopt;
}
/* Removes the illustration role assigned to this geometry -- if there was
no illustration role previously, this has no effect. */
void RemoveIllustrationRole() {
illustration_props_ = std::nullopt;
}
/* Removes the perception role assigned to this geometry -- if there was
no perception role previously, this has no effect. */
void RemovePerceptionRole() {
perception_props_ = std::nullopt;
}
//@}
/* Returns a pointer to the geometry's reference mesh if the geometry is
deformable, or nullptr otherwise. */
const VolumeMesh<double>* reference_mesh() const {
return reference_mesh_.get();
}
private:
// The specification for this instance's shape.
copyable_unique_ptr<Shape> shape_spec_;
// The identifier for this frame.
GeometryId id_;
// The name of the geometry. Must be unique among geometries attached to the
// same frame.
std::string name_;
// The source id that registered the geometry.
SourceId source_id_;
// The identifier of the frame to which this geometry belongs.
FrameId frame_id_;
// The pose of this geometry in the registered parent frame. The parent may be
// a frame or another registered geometry. Identity if the geometry is
// deformable.
math::RigidTransform<double> X_PG_;
// The pose of this geometry in the ultimate frame to which this geometry is
// rigidly affixed. If there is no parent geometry, X_PG_ == X_FG_. In
// particular, deformable geometries have no parent geometries.
math::RigidTransform<double> X_FG_;
// The identifier for this frame's parent frame.
std::optional<GeometryId> parent_geometry_id_{};
// The identifiers for the geometry hung on this frame.
std::unordered_set<GeometryId> child_geometry_ids_;
// TODO(SeanCurtis-TRI): Consider introducing a mechanism where these are
// defined at the frame level, and all child geometries inherit.
// The optional property sets tied to the roles that the geometry plays.
std::optional<ProximityProperties> proximity_props_{};
std::optional<IllustrationProperties> illustration_props_{};
std::optional<PerceptionProperties> perception_props_{};
// Optional mesh representation for deformable geometries. The geometry is
// considered as deformable if a mesh representation exists. The vertex
// positions corresponds to those of the mesh in its reference configuration
// expressed in `this` geometry's frame.
copyable_unique_ptr<VolumeMesh<double>> reference_mesh_;
};
} // namespace internal
} // namespace geometry
} // namespace drake