This file is indexed.

/usr/include/SurgSim/Collision/SegmentSelfContact.h is in libopensurgsim-dev 0.7.0-6ubuntu1.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  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
// This file is a part of the OpenSurgSim project.
// Copyright 2013-2015, SimQuest Solutions Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

#ifndef SURGSIM_COLLISION_SEGMENTSELFCONTACT_H
#define SURGSIM_COLLISION_SEGMENTSELFCONTACT_H

#include "SurgSim/Collision/ShapeShapeContactCalculation.h"
#include "SurgSim/DataStructures/AabbTree.h"
#include "SurgSim/Framework/Logger.h"
#include "SurgSim/Math/SegmentMeshShape.h"

namespace SurgSim
{
namespace Collision
{

class CollisionPair;

/// SegmentSelfContact computes the self collisions among a SegmentMesh under motion at two
/// time points parametrized over the time interval [0,1]. An initial phase uses the AABB tree to
/// select a set of potentially colliding segments from the SegmentMesh. For each of these
/// candidate segment pairs, the goal is to determine the point of earliest contact should any exist.
///
/// At the highest level the actual collision detection of candidate segment pairs is a two phase
/// algorithm. First determine if there is contact at the start of an interval and report the contact if
/// found. If no contact is found at the start, subdivide the interval, determine which of the resulting
/// candidate subintervals may have collisions, and then recursively check those promising subintervals.
/// Note that a simple algorithm based on interval arithmetic (including the Interval, LinearMotion and
/// Polynomial interval classes) allows for a quick determination of promising subintervals allowing many
/// of the subintervals to be pruned during the subdivision step without forcing the recursion to bottom out.
///
/// \sa Interval, LinearMotion, Polynomial, SegmentSegmentCcdIntervalCheck
///
class SegmentSelfContact : public ShapeShapeContactCalculation<Math::SegmentMeshShape, Math::SegmentMeshShape>
{
public:
	/// Constructor.
	SegmentSelfContact();

	std::list<std::shared_ptr<Contact>> calculateCcdContact(
		const Math::SegmentMeshShape& segmentShape1AtTime0, const Math::RigidTransform3d& segmentPose1AtTime0,
		const Math::SegmentMeshShape& segmentShape1AtTime1, const Math::RigidTransform3d& segmentPose1AtTime1,
		const Math::SegmentMeshShape& segmentShape2AtTime0, const Math::RigidTransform3d& segmentPose2AtTime0,
		const Math::SegmentMeshShape& segmentShape2AtTime1, const Math::RigidTransform3d& segmentPose2AtTime1)
		const override;

	/// Set the minimum time precision allowed when deciding on the depth of recursion.
	/// \param precision the desired minimum time precision
	void setTimeMinPrecisionEpsilon(double precision);

	/// \return the minimum time precision allowed when deciding on the depth of recursion.
	double getTimeMinPrecisionEpsilon();

	/// Set the maximum time precision allowed when deciding on the depth of recursion.
	/// \param precision the desired maximum time precision
	void setTimeMaxPrecisionEpsilon(double precision);

	/// \return the maximum time precision allowed when deciding on the depth of recursion.
	double getTimeMaxPrecisionEpsilon();

	/// Set the maximum separation for which two points are considered the same.
	/// \param precision the desired maximum separation for which two points are considered the same.
	void setDistanceEpsilon(double precision);

	/// \return the maximum separation for which two points are considered the same.
	double distanceEpsilon();

	/// Function that returns the shapes between which this class performs collision detection.
	/// \return int std::pair containing the shape types.
	std::pair<int, int> getShapeTypes() override;

protected:
	/// Detect if two segments actually collide either at time t=0 (Fixed case) or within a movement phase.
	/// \param pt0Positions are the segment endpoints for the first segment at time t=0.
	/// \param pt1Positions are the segment endpoints for the first segment at time t=1.
	/// \param qt0Positions are the segment endpoints for the second segment at time t=0.
	/// \param qt1Positions are the segment endpoints for the second segment at time t=1.
	/// \param segmentRadius1 is the radius of the first segment.
	/// \param segmentRadius2 is the radius of the second segment.
	/// \param timePrecision is the minimum time delta. Recursion terminates at or below the timePrecision.
	/// \param r [out] parametric location of the collision point (if any) on segment 1.
	/// \param s [out] parametric location of the collision point (if any) on segment 2.
	/// \param t [out] parametric location of the time of any collision in the interval [0, 1].
	/// \param pToQDir [out] direction from p(s) to q(r)
	/// \param contactPtP [out] location of the contact point along segment 1.
	/// \param contactPtQ [out] location of the contact point along segment 2.
	/// \return true if a collision is detected between segment 1 and segment 2; false otherwise.
	bool detectCollision(
		const std::array<SurgSim::Math::Vector3d, 2>& pt0Positions,
		const std::array<SurgSim::Math::Vector3d, 2>& pt1Positions,
		const std::array<SurgSim::Math::Vector3d, 2>& qt0Positions,
		const std::array<SurgSim::Math::Vector3d, 2>& qt1Positions,
		double segmentRadius1, double segmentRadius2,
		double timePrecision,
		double* r, double* s, double* t,
		SurgSim::Math::Vector3d* pToQDir,
		SurgSim::Math::Vector3d* contactPtP,
		SurgSim::Math::Vector3d* contactPtQ) const;

	/// Given a list of potentially intersecting AABB nodes, cull the list of any duplicates
	/// and return the uniques candidates as synchronized pairs.
	/// \param intersectionList list of potentially intersecting AABB node.
	/// \param segmentIds [out] paired unique matches
	void getUniqueCandidates(
		const std::list<SurgSim::DataStructures::AabbTree::TreeNodePairType>& intersectionList,
		std::set<std::pair<size_t, size_t>>* segmentIds) const;

	/// From the initial AABB tree collisions, there are some very simple filtering operations that we can
	/// do to eliminate a number of false positives. Most notably, we do not want to collide a single segment
	/// against itself, or against one of the segments with which it shares a vertex. These are trivial collisions
	/// and will always be present. Less obvious, we want to filter out segments with wild movement vectors.
	/// These segments cannot be appropriately processed and are likely to represent errors.
	/// \param segmentA the segment mesh at time t=0.
	/// \param segmentB the segment mesh at time t=1.
	/// \param segment1SegID the specific identifier of the candidate segment at time t=0.
	/// \param segment2SegID the specific identifier of the candidate segment at time t=1.
	/// \return true if this collision should be discarded, false if it should be processed further.
	bool removeInvalidCollisions(
		const Math::SegmentMeshShape& segmentA,
		const Math::SegmentMeshShape& segmentB,
		size_t segment1SegID, size_t segment2SegID) const;

	/// Verify the a given point at times t0 and t1 have remained within a reasonable neighborhood. Large
	/// movements make it impossible to accurately determine collisions.
	/// \param pt0 vertex coordinates at time t0
	/// \param pt1 vertex coordinates at time t1
	/// \param threshold distance threshold
	/// \return true if any point has exceeded the movement threshold
	bool detectExcessMovement(const SurgSim::Math::Vector3d& pt0,
							  const SurgSim::Math::Vector3d& pt1,
							  double threshold) const;

	/// Search the list of contacts for a match to the current contact.
	/// \param segmentShape shape to be interrogated to see if the contacts match
	/// \param contacts the current list of detected contacts
	/// \param t time of the contact under consideration.
	/// \param collisionType type of contact under consideration.
	/// \param segId1 segment 1 identifier.
	/// \param s1 parametric location of the contact point along segment 1.
	/// \param segId2 segment 2 identifier.
	/// \param s2 parametric location of the contact point along segment 2.
	/// \param timeEpsilon maximum allowed epsilon for time matches.
	/// \return true if the contacts match, return false otherwise.
	bool findSegSegContact(const Math::SegmentMeshShape& segmentShape,
						   const std::list<std::shared_ptr<Contact>>& contacts,
						   double t, Collision::CollisionDetectionType collisionType,
						   size_t segId1, double s1, size_t segId2, double s2, double timeEpsilon) const;

	/// Check for the same location among two parametric location specifications.
	/// \param segmentShape shape to be interrogated to see if the contacts match
	/// \param segId1 segment 1 identifier.
	/// \param s1 parametric location of the contact point along segment 1.
	/// \param segId2 segment 2 identifier.
	/// \param s2 parametric location of the contact point along segment 2.
	/// \return true if the contacts match, return false otherwise.
	bool isSameSegContactPoint(const Math::SegmentMeshShape& segmentShape,
							   size_t segId1, double s1, size_t segId2, double s2) const;

	/// Calculate the maximum time interval that guarantees that all collisions can be detected given the
	/// derived motions of the segment end points.
	/// \param pt0Positions are the segment endpoints for the first segment at time t=0.
	/// \param pt1Positions are the segment endpoints for the first segment at time t=1.
	/// \param qt0Positions are the segment endpoints for the second segment at time t=0.
	/// \param qt1Positions are the segment endpoints for the second segment at time t=1.
	/// \param effectiveThickness nearness criteria for declaring a contact.
	/// \return the maximum time interval that will still allow for the detection of all
	/// contacts within effectveThickness. This value is bounded from below by the value of
	/// the member variable m_timeMinPrecisionEpsilon and from above by m_timeMaxPrecisionEpsilon.
	double maxTimePrecision(
		const std::array<SurgSim::Math::Vector3d, 2>& pt0Positions,
		const std::array<SurgSim::Math::Vector3d, 2>& pt1Positions,
		const std::array<SurgSim::Math::Vector3d, 2>& qt0Positions,
		const std::array<SurgSim::Math::Vector3d, 2>& qt1Positions,
		double effectiveThickness) const;

private:
	/// Minimum distance precision epsilon used in continuous collision detection.
	double m_distanceEpsilon;

	/// Minimum time precision epsilon used in continuous collision detection.
	double m_timeMinPrecisionEpsilon;

	/// Maximum time precision epsilon used in continuous collision detection.
	double m_timeMaxPrecisionEpsilon;

	/// Maximum time precision epsilon used in continuous collision detection.
	const double m_maxMovementThreshold;

	/// Flag to determine if segment thickness is to be used in contact calculations
	const bool m_useSegmentThickness;

	/// Logger
	std::shared_ptr<SurgSim::Framework::Logger> m_logger;
};

}; // namespace Collision
}; // namespace SurgSim

#endif // SURGSIM_COLLISION_SEGMENTSELFCONTACT_H