_xiaofang/xiaofang/Assets/Obi/Resources/Compute/CapsuleShape.compute

122 lines
4.0 KiB
Plaintext
Raw Normal View History

2024-12-18 02:18:45 +08:00
#include "ColliderDefinitions.cginc"
#include "ContactHandling.cginc"
#include "Transform.cginc"
#include "Simplex.cginc"
#include "Bounds.cginc"
#include "SolverParameters.cginc"
#include "Optimization.cginc"
#pragma kernel GenerateContacts
StructuredBuffer<float4> positions;
StructuredBuffer<quaternion> orientations;
StructuredBuffer<float4> principalRadii;
StructuredBuffer<float4> velocities;
StructuredBuffer<int> simplices;
StructuredBuffer<transform> transforms;
StructuredBuffer<shape> shapes;
StructuredBuffer<uint2> contactPairs;
StructuredBuffer<int> contactOffsetsPerType;
RWStructuredBuffer<contact> contacts;
RWStructuredBuffer<uint> dispatchBuffer;
StructuredBuffer<transform> worldToSolver;
uint maxContacts;
struct Capsule : IDistanceFunction
{
shape s;
transform colliderToSolver;
void Evaluate(in float4 pos, in float4 radii, in quaternion orientation, inout SurfacePoint projectedPoint)
{
float4 center = s.center * colliderToSolver.scale;
float4 pnt = colliderToSolver.InverseTransformPointUnscaled(pos) - center;
if (s.is2D())
pnt[2] = 0;
int direction = (int)s.size.z;
float height;
float radius;
float4 halfVector = float4(0,0,0,0);
if (direction == 0)
{
radius = s.size.x * max(colliderToSolver.scale[1], colliderToSolver.scale[2]);
height = max(radius, s.size.y * 0.5f * colliderToSolver.scale[0]);
halfVector[0] = height - radius;
}
else if (direction == 1)
{
radius = s.size.x * max(colliderToSolver.scale[2], colliderToSolver.scale[0]);
height = max(radius, s.size.y * 0.5f * colliderToSolver.scale[1]);
halfVector[1] = height - radius;
}
else
{
radius = s.size.x * max(colliderToSolver.scale[0], colliderToSolver.scale[1]);
height = max(radius, s.size.y * 0.5f * colliderToSolver.scale[2]);
halfVector[2] = height - radius;
}
float mu;
float4 centerLine = NearestPointOnEdge(-halfVector, halfVector, pnt, mu);
float4 centerToPoint = pnt - centerLine;
float distanceToCenter = length(centerToPoint);
float4 normal = centerToPoint / (distanceToCenter + EPSILON);
projectedPoint.pos = colliderToSolver.TransformPointUnscaled(center + centerLine + normal * (radius + s.contactOffset));
projectedPoint.normal = colliderToSolver.TransformDirection(normal);
projectedPoint.bary = float4(1,0,0,0);
}
};
[numthreads(128, 1, 1)]
void GenerateContacts (uint3 id : SV_DispatchThreadID)
{
uint i = id.x;
// entry #11 in the dispatch buffer is the amount of pairs for the first shape type.
if (i >= dispatchBuffer[11 + 4*CAPSULE_SHAPE]) return;
uint count = contacts.IncrementCounter();
if (count < maxContacts)
{
int firstPair = contactOffsetsPerType[CAPSULE_SHAPE];
int simplexIndex = contactPairs[firstPair + i].x;
int colliderIndex = contactPairs[firstPair + i].y;
contact c = (contact)0;
Capsule capsuleShape;
capsuleShape.colliderToSolver = worldToSolver[0].Multiply(transforms[colliderIndex]);
capsuleShape.s = shapes[colliderIndex];
int simplexSize;
int simplexStart = GetSimplexStartAndSize(simplexIndex, simplexSize);
float4 simplexBary = BarycenterForSimplexOfSize(simplexSize);
float4 simplexPoint;
SurfacePoint surfacePoint = Optimize(capsuleShape, positions, orientations, principalRadii,
simplices, simplexStart, simplexSize, simplexBary, simplexPoint, surfaceCollisionIterations, surfaceCollisionTolerance);
c.pointB = surfacePoint.pos;
c.normal = surfacePoint.normal * capsuleShape.s.isInverted();
c.pointA = simplexBary;
c.bodyA = simplexIndex;
c.bodyB = colliderIndex;
contacts[count] = c;
InterlockedMax(dispatchBuffer[0],(count + 1) / 128 + 1);
InterlockedMax(dispatchBuffer[3], count + 1);
}
}