forked from RobotLocomotion/drake
-
Notifications
You must be signed in to change notification settings - Fork 0
/
scene_graph_inspector.h
502 lines (436 loc) · 19.8 KB
/
scene_graph_inspector.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
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
#pragma once
#include <memory>
#include <optional>
#include <set>
#include <string>
#include <unordered_set>
#include <utility>
#include <vector>
#include "drake/common/drake_deprecated.h"
#include "drake/geometry/geometry_instance.h"
#include "drake/geometry/geometry_roles.h"
#include "drake/geometry/geometry_state.h"
#include "drake/geometry/internal_frame.h"
#include "drake/geometry/shape_specification.h"
namespace drake {
namespace geometry {
template <typename T>
class SceneGraph;
template <typename T>
class QueryObject;
/** The %SceneGraphInspector serves as a mechanism to query the topological
structure of a SceneGraph instance. The topological structure consists of all
of the SceneGraph data that does _not_ depend on input pose data. Including,
but not limited to:
- names of frames and geometries
- hierarchies (parents of geometries, parents of frames, etc.)
- geometry parameters (e.g., contact, rendering, visualization)
- fixed poses of geometries relative to frames
In contrast, the following pieces of data *do* depend on input pose data and
_cannot_ be performed with the %SceneGraphInspector (see the QueryObject
instead):
- world pose of frames or geometry
- collision queries
- proximity queries
A %SceneGraphInspector cannot be instantiated explicitly. Nor can it be copied
or moved. A _reference_ to a %SceneGraphInspector instance can be acquired from
- a SceneGraph instance (to inspect the state of the system's _model_), or
- a QueryObject instance (to inspect the state of the scene graph data stored
in the context).
The reference should not be persisted (and, as previously indicated, cannot
be copied). %SceneGraphInspector instances are cheap; they can be created,
queried, and thrown out. If there is any doubt about the valid lifespan of
a %SceneGraphInspector, throw out the old instance and request a new instance.
@tparam T The scalar of the associated SceneGraph instance. The template
parameter is provided for the sake of compatibility, although no queries (or
their results) depend on the scalar. */
template <typename T>
class SceneGraphInspector {
public:
DRAKE_NO_COPY_NO_MOVE_NO_ASSIGN(SceneGraphInspector)
// NOTE: An inspector should never be released into the wild without having
// set the state variable. Every query should start by demanding that state_
// is defined.
//----------------------------------------------------------------------------
/** @name Scene-graph wide data */
//@{
/** Reports the number of registered sources -- whether they have registered
frames/geometries or not. This will always be at least 1; the SceneGraph
itself counts as a source. */
int num_sources() const {
DRAKE_DEMAND(state_ != nullptr);
return state_->get_num_sources();
}
/** Reports the _total_ number of frames registered in the scene graph
(including the world frame). */
int num_frames() const {
DRAKE_DEMAND(state_ != nullptr);
return state_->get_num_frames();
}
/** Provides a range object for all of the frame ids in the scene graph. The
order is not generally guaranteed; but it will be consistent as long as there
are no changes to the topology. This is intended to be used as:
@code
for (FrameId id : inspector.all_frame_ids()) {
...
}
@endcode
This includes the id for the world frame. */
typename GeometryState<T>::FrameIdRange all_frame_ids() const {
DRAKE_DEMAND(state_ != nullptr);
return state_->get_frame_ids();
}
/** Reports the id for the world frame. */
FrameId world_frame_id() const {
DRAKE_DEMAND(state_ != nullptr);
return internal::InternalFrame::world_frame_id();
}
/** Reports the _total_ number of geometries in the scene graph. */
int num_geometries() const {
DRAKE_DEMAND(state_ != nullptr);
return state_->get_num_geometries();
}
/** Returns the set of all ids for registered geometries. The order is _not_
guaranteed to have any particular meaning. But the order is
guaranteed to remain fixed until a topological change is made (e.g., removal
or addition of geometry/frames). */
const std::vector<GeometryId> GetAllGeometryIds() const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetAllGeometryIds();
}
/** Reports the _total_ number of geometries in the scene graph with the
indicated role. */
int NumGeometriesWithRole(Role role) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->NumGeometriesWithRole(role);
}
/** Reports the total number of _dynamic_ geometries in the scene graph. */
int NumDynamicGeometries() const {
DRAKE_DEMAND(state_ != nullptr);
return state_->NumDynamicGeometries();
}
/** Reports the total number of _anchored_ geometries. This should provide
the same answer as calling NumGeometriesForFrame() with the world frame id.
*/
int NumAnchoredGeometries() const {
DRAKE_DEMAND(state_ != nullptr);
return state_->NumAnchoredGeometries();
}
/** Returns all pairs of geometries that are candidates for collision (in no
particular order). See SceneGraph::ExcludeCollisionsBetween() or
SceneGraph::ExcludeCollisionsWithin() for information on why a particular
pair may _not_ be a candidate. For candidate pair (A, B), the candidate is
always guaranteed to be reported in a fixed order (i.e., always (A, B) and
_never_ (B, A)). This is the same ordering as would be returned by, e.g.,
QueryObject::ComputePointPairPenetration(). */
std::set<std::pair<GeometryId, GeometryId>> GetCollisionCandidates()
const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetCollisionCandidates();
}
//@}
/** @name Sources and source-related data */
//@{
/** Reports `true` if the given `source_id` maps to a registered source. */
bool SourceIsRegistered(SourceId source_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->SourceIsRegistered(source_id);
}
/** Reports the name for the source with the given `source_id`.
@throws std::exception if `source_id` does not map to a registered source. */
const std::string& GetName(SourceId source_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetName(source_id);
}
/** Reports the name for the source with the given `id`.
@throws std::logic_error if `id` does not map to a registered source. */
DRAKE_DEPRECATED("2021-03-01", "Please use GetName(SourceId) instead.")
const std::string& GetSourceName(SourceId id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetName(id);
}
/** Reports the number of frames registered to the source with the given
`source_id`.
@throws std::logic_error if `source_id` does not map to a registered source.
*/
int NumFramesForSource(SourceId source_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->NumFramesForSource(source_id);
}
/** Reports the ids of all of the frames registered to the source with the
given source `source_id`.
@throws std::logic_error if `source_id` does not map to a registered source.
*/
const std::unordered_set<FrameId>& FramesForSource(SourceId source_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->FramesForSource(source_id);
}
//@}
/** @name Frames and their properties */
//@{
/** Reports if the frame with given `frame_id` was registered to the source
with the given `source_id`.
@param frame_id The query frame id.
@param source_id The query source id.
@returns True if `frame_id` was registered on `source_id`.
@throws std::logic_error If `frame_id` does not map to a registered frame
or `source_id` does not map to a registered source.
*/
bool BelongsToSource(FrameId frame_id, SourceId source_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->BelongsToSource(frame_id, source_id);
}
/** Reports the _name_ of the geometry source that registered the frame with
the given `frame_id`.
@throws std::logic_error If `frame_id` does not map to a registered frame.
*/
const std::string& GetOwningSourceName(FrameId frame_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetOwningSourceName(frame_id);
}
/** Reports the name of the frame with the given `frame_id`.
@throws std::logic_error if `frame_id` does not map to a registered frame.
*/
const std::string& GetName(FrameId frame_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetName(frame_id);
}
/** Reports the frame group for the frame with the given `frame_id`.
@throws std::logic_error if `frame_id` does not map to a registered frame.
@internal This value is equivalent to the old "model instance id". */
int GetFrameGroup(FrameId frame_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetFrameGroup(frame_id);
}
/** Reports the number of geometries affixed to the frame with the given
`frame_id`. This count does _not_ include geometries attached to frames that
are descendants of this frame.
@throws std::logic_error if `frame_id` does not map to a registered frame.
*/
int NumGeometriesForFrame(FrameId frame_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->NumGeometriesForFrame(frame_id);
}
/** Reports the total number of geometries with the given `role` directly
registered to the frame with the given `frame_id`. This count does _not_
include geometries attached to frames that are descendants of this frame.
@throws std::logic_error if `frame_id` does not map to a registered frame.
*/
int NumGeometriesForFrameWithRole(FrameId frame_id, Role role) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->NumGeometriesForFrameWithRole(frame_id, role);
}
/** Returns geometry ids that have been registered directly to the frame
indicated by `frame_id`. If a `role` is provided, only geometries with that
role assigned will be returned, otherwise all geometries will be returned.
@param frame_id The id of the frame in question.
@param role The requested role; if omitted, all geometries registered to the
frame are returned.
@returns The requested unique geometry ids in a consistent order.
@throws std::logic_error if `id` does not map to a registered frame. */
std::vector<GeometryId> GetGeometries(
FrameId frame_id, const std::optional<Role>& role = std::nullopt) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetGeometries(frame_id, role);
}
/** Reports the id of the geometry with the given `name` and `role`, attached
to the frame with the given frame `frame_id`.
@param frame_id The frame_id of the frame whose geometry is being
queried.
@param role The assigned role of the desired geometry.
@param name The name of the geometry to query for. The name will be
canonicalized prior to lookup (see
@ref canonicalized_geometry_names "GeometryInstance" for
details).
@return The id of the queried geometry.
@throws std::logic_error if no such geometry exists, multiple geometries have
that name, or if the `frame_id` does not map to a
registered frame. */
GeometryId GetGeometryIdByName(FrameId frame_id, Role role,
const std::string& name) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetGeometryIdByName(frame_id, role, name);
}
//@}
/** @name Geometries and their properties */
//@{
/** Reports if the given geometry id was registered to the source with the
given source id.
@param geometry_id The query geometry id.
@param source_id The query source id.
@returns True if `geometry_id` was registered on `source_id`.
@throws std::logic_error If `geometry_id` does not map to a registered
geometry or `source_id` does not map to a
registered source. */
bool BelongsToSource(GeometryId geometry_id, SourceId source_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->BelongsToSource(geometry_id, source_id);
}
/** Reports the _name_ of the geometry source that registered the geometry
with the given `geometry_id`.
@throws std::logic_error If `geometry_id` does not map to a registered
geometry. */
const std::string& GetOwningSourceName(GeometryId geometry_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetOwningSourceName(geometry_id);
}
/** Reports the id of the frame to which the given geometry with the given
`geometry_id` is registered.
@throws std::logic_error if `geometry_id` does not map to a registered
geometry. */
FrameId GetFrameId(GeometryId geometry_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetFrameId(geometry_id);
}
/** Reports the stored, canonical name of the geometry with the given
`geometry_id` (see @ref canonicalized_geometry_names "GeometryInstance" for
details).
@throws std::logic_error if `geometry_id` does not map to a registered
geometry. */
const std::string& GetName(GeometryId geometry_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetName(geometry_id);
}
/** Returns the shape specified for the geometry with the given `geometry_id`.
In order to extract the details of the shape, it should be passed through an
implementation of a ShapeReifier. */
const Shape& GetShape(GeometryId geometry_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetShape(geometry_id);
}
/** Reports the pose of the geometry G with the given `geometry_id` in its
registered _topological parent_ P, `X_PG`. That topological parent may be a
frame F or another geometry. If the geometry was registered directly to F,
then `X_PG = X_FG`.
@sa GetPoseInFrame()
@throws std::logic_error if `geometry_id` does not map to a registered
geometry. */
const math::RigidTransform<double>& GetPoseInParent(
GeometryId geometry_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetPoseInParent(geometry_id);
}
/** Reports the pose of the geometry G with the given `geometry_id` in its
registered frame F (regardless of whether its _topological parent_ is another
geometry P or not). If the geometry was registered directly to the frame F,
then `X_PG = X_FG`.
@sa GetPoseInParent()
@throws std::logic_error if `geometry_id` does not map to a registered
geometry. */
const math::RigidTransform<double>& GetPoseInFrame(
GeometryId geometry_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetPoseInFrame(geometry_id);
}
/** Return a pointer to the const properties indicated by `role` of the
geometry with the given `geometry_id`.
@param geometry_id The identifier for the queried geometry.
@param role The role whose properties are acquired.
@return A pointer to the properties (or `nullptr` if there are no such
properties).
@throws std::exception if `geometry_id` does not map to a registered
geometry. */
const GeometryProperties* GetProperties(GeometryId geometry_id,
Role role) const {
DRAKE_DEMAND(state_ != nullptr);
switch (role) {
case Role::kProximity:
return state_->GetProximityProperties(geometry_id);
case Role::kIllustration:
return state_->GetIllustrationProperties(geometry_id);
case Role::kPerception:
return state_->GetPerceptionProperties(geometry_id);
case Role::kUnassigned:
return nullptr;
}
return nullptr;
}
/** Returns a pointer to the const proximity properties of the geometry with
the given `geometry_id`.
@param geometry_id The identifier for the queried geometry.
@return A pointer to the properties (or `nullptr` if there are no such
properties).
@throws std::exception if `geometry_id` does not map to a registered
geometry. */
const ProximityProperties* GetProximityProperties(
GeometryId geometry_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetProximityProperties(geometry_id);
}
/** Returns a pointer to the const illustration properties of the geometry
with the given `geometry_id`.
@param geometry_id The identifier for the queried geometry.
@return A pointer to the properties (or `nullptr` if there are no such
properties).
@throws std::exception if `geometry_id` does not map to a registered
geometry. */
const IllustrationProperties* GetIllustrationProperties(
GeometryId geometry_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetIllustrationProperties(geometry_id);
}
/** Returns a pointer to the const perception properties of the geometry
with the given `geometry_id`.
@param geometry_id The identifier for the queried geometry.
@return A pointer to the properties (or `nullptr` if there are no such
properties).
@throws std::exception if `geometry_id` does not map to a registered
geometry. */
const PerceptionProperties* GetPerceptionProperties(
GeometryId geometry_id) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->GetPerceptionProperties(geometry_id);
}
/** Reports true if the two geometries with given ids `geometry_id1` and
`geometry_id2`, define a collision pair that has been filtered out.
@throws std::logic_error if either id does not map to a registered geometry
or if any of the geometries do not have a proximity
role. */
bool CollisionFiltered(GeometryId geometry_id1,
GeometryId geometry_id2) const {
DRAKE_DEMAND(state_ != nullptr);
return state_->CollisionFiltered(geometry_id1, geometry_id2);
}
/** Introspects the geometry indicated by the given `geometry_id`. The
geometry will be passed into the provided `reifier`. This is the mechanism by
which external code can discover and respond to the different types of
geometries stored in SceneGraph. See ShapeToString as an example.
@throws std::logic_error if the `geometry_id` does not refer to a valid
geometry. */
void Reify(GeometryId geometry_id, ShapeReifier* reifier) const {
DRAKE_DEMAND(state_ != nullptr);
state_->GetShape(geometry_id).Reify(reifier);
}
/** Obtains a new GeometryInstance that copies the geometry indicated by the
given `geometry_id`.
@return A new GeometryInstance that is ready to be added as a new geometry.
All roles/properties will be copied, the shape will be cloned based
off of the original, but the returned id() will completely unique.
@throws std::logic_error if the `geometry_id` does not refer to a valid
geometry. */
std::unique_ptr<GeometryInstance>
CloneGeometryInstance(GeometryId geometry_id) const;
/** Returns the geometry version that can be used to detect changes
to the geometry data associated with geometry roles. The reference returned
should not be persisted. If it needs to be persisted, it should be copied. */
const GeometryVersion& geometry_version() const {
return state_->geometry_version();
}
//@}
private:
// Only SceneGraph and QueryObject instances can construct
// SceneGraphInspector instances.
friend class SceneGraph<T>;
friend class QueryObject<T>;
// Testing utility.
friend class SceneGraphInspectorTester;
// Simple default constructor to be used by SceneGraph and QueryObject.
SceneGraphInspector() = default;
// Sets the scene graph data to inspect -- the inspector does _not_ own the
// data and expects that the lifespan of the provided data is longer than
// the inspector.
void set(const GeometryState<T>* state) { state_ = state; }
const GeometryState<T>* state_{nullptr};
};
} // namespace geometry
} // namespace drake