Inconsistent Mesh Collider Results (Fixed)

Unity Version: 2019.4.2f1
Unity Physics Version: preview.5 - 0.4.0
Entities Version: preview.4 - 0.11.1
Summary:

Collisions between a Capsule Collider and Mesh Colliders is inconsistent.

Details:

I have a custom terrain system which generates its own terrain meshes, in segments of 64x64 units. The terrain uses Mesh Colliders to add physics. These colliders work for some segments, but do not work for others.

I have tried, and failed, to discern what may be causing this inconsistent behaviour. It occurs even when the terrain is flat, and even when the same mesh is used to build the collider for all of the segments. Some will generate collisions, others will not and the player falls through. Interestingly, sometimes direction entering/traversing the terrain matters. Going from segment A to B may fail, but C to B works.

Some details about these meshes:

  • Approximately 7,000 to 28,000 vertices with a third that many triangles, depending on segment complexity.
  • Have tried both a mesh of only the terrain surface as well as a “complete” mesh (all sides).
  • All terrain entities have the same components, including: PhysicsMass, RenderBounds, PhysicsCollider, Rotation, Translation, etc.

The only components that vary in value between the segments are: WorldRenderBounds, LocalToWorld, Translation, and ChunkWorldRenderBounds. As would be expected.

Raycasts also fail to return any hits for these problem segments as well, in addition to the player falling through.

I have a bunch of code surrounding my implementation, so it is hard to give a good, simple code snippet. However the most relevant section is below.

protected override void OnUpdate()
{
    Entities.ForEach((
        Entity entity,
        ref TerrainColliderProxyComponent proxy,
        ref Translation position,
        ref RenderBounds bounds) =>
    {
        if (proxy.RequiresColliderUpdate)
        {
            Material material = GetMaterial(entity);
            CollisionFilter filter = GetCollisionFilter(entity);
            TerrainSegment segment = ...;

            UpdatePhysicsMeshCollider(entity, segment.TerrainMesh, material, filter);

            proxy.RequiresColliderUpdate = false;
        }
    });
}

private void UpdatePhysicsMeshCollider(Entity entity, CustomMesh mesh, Material material, CollisionFilter filter)
{
    mesh.FetchPhysicsData(out NativeArray<float3> vertices, out NativeArray<int3> triangles);
 
    var collider = MeshCollider.Create(vertices, triangles, filter, material);
    var component = new PhysicsCollider() { Value = collider };

    if (!EntityManager.HasComponent<PhysicsCollider>(entity))
    {
        PostUpdateCommands.AddComponent(entity, component);
    }
    else
    {
        EntityManager.SetComponentData(entity, component);
    }

    triangles.Dispose();
    vertices.Dispose();
}

private Material GetMaterial(Entity entity)
{
    if (EntityManager.HasComponent<PhysicsMaterialProxyComponent>(entity))
    {
        return EntityManager.GetComponentData<PhysicsMaterialProxyComponent>(entity).ToMaterial();
    }

    return Material.Default;
}

private CollisionFilter GetCollisionFilter(Entity entity)
{
    if (EntityManager.HasComponent<CollisionFilterProxyComponent>(entity))
    {
        return EntityManager.GetComponentData<CollisionFilterProxyComponent>(entity).ToCollisionFilter();
    }

    return CollisionFilter.Default;
}

And finally, a GIF. It shows where four segments meet, and the top two segments are OK while the bottom two the collisions fail.

Questions:

  • Any ideas why the same mesh which works as the source of the Mesh Collider for Entity A but it fails for Entity B?
  • Are there any limitations to the mesh data supplied to the MeshCollider.Create method? For example, Unity Meshes have a limit of 65536 vertices. Though I do not think this is the issue since the identical meshes work for some entities, but fails for others.
  • The documentation states that the Mesh Collider is a “triangle soup.” Can the mesh have holes? Can it be a single plane (ie terrain surface)?
  • Are there any limitations/concerns with when a mesh collider is made/timing of creating multiple mesh colliders? I assume it is safe to do so within a system?
  • Does winding order matter for mesh colliders? From my experience it does not seem so.

Thank you for your time.

Have you used the PhysicsDebugDisplay component to see what the actual colliders look like?

Also for terrain you should use the TerrainCollider. Takes heights as input it’s far more optimized for this use case.

First thing that comes to my mind (other than what @snacktime suggested) is that your system doesn’t have UpdateBefore or UpdateAfter so it could run at the random time… I would make it run before BuildPhysicsWorld, because that’s the only way to guarantee that physics world will pick up all these colliders. Now that might not be the reason you are seeing this, but it’s a start.

Thank you for your responses.

Using the PhysicsDebugDisplay unfortunately doesn’t provide much more insight. See the GIF below (sorry for the poor performance, the gizmos really bogged me down).

You can see that I can move/jump in the bottom right, but then I fall through once I move up.

As for the TerrainCollider, I used to use it but now I use a custom mesh with vertical walls, sort of like voxels. From my understanding the TerrainCollider would try to create slopes based off of the heightmap and lose the verticals.

And the snippet didn’t show it, but the system (UpdateTerrainSystem) is set in a group which is before BuildPhysicsWorld.

Strange. Maybe there is a clue in the closest geometry that it does find. Like the closest hit on a distance query might be useful?

If it’s a dynamic body maybe crank up solver iterations maybe it hit something it can’t solve completely?

I tried your suggestion, snacktime, and used CollisionWorld.CalculateDistance to get the nearest point of contact.

Unsurprisingly when I fall through the mesh the closest point is the nearest segment that is working. Again, see the GIF below (magenta line is line to nearest hit).

I also tried mucking around with the various settings in the Physics Step script, including:

  • Trying Havok Physics instead of Unity
  • Increase Solver Iteration Count from 4 to 8 to 32
  • Enable “Synchronize Collision World”
  • Enable “Contact Solver Stabilization Heuristic”

There must be something obvious I am missing.

I’ve been handling mesh colliders for custom meshes without issue by setting .sharedMesh of the MeshCollider to .sharedMesh of the MeshFilter. This assumes each mesh is unique - I haven’t tried it where more than one MeshFilter uses the same mesh.

MeshFilter mf = GetComponent<MeshFilter>();
MeshCollider mc = GetComponent<MeshCollider>();
mc.sharedMesh = mf.sharedMesh;

I don’t know how much surface normals matter in cast hits if at all. But I’m curious if an aabb query returns what the distance query won’t. That might narrow this down some. Other then that I’m out of ideas.

@adamgolden Unfortunately this is about the Unity Physics package (aka DOTS Physics) and not UnityEngine.PhysicsModule.

Thanks @snacktime , I am running out of ideas too. Actually, I ran out right before make this post! About half of my terrain is usable so might chock this up to Unity.Physics being an alpha preview package and hopefully this just fixes itself while I focus on other things.

Wonder if I can detect when a terrain has an invalid collider (no ray hit) and try recreating the collider. What was the definition of insanity again? Its doing the same thing over and over again because its bound to work eventually?

I am curious what the RayTraceCamera will show. RayTraceCamera does what it name tries to explain, it just shoots bunch of rays and for every hit it displays the pixel in the target canvas as on the pic below.

To enable this do the following:

  • copy RayTraceCamera from any sample demo (e.g. Hello World). It’s child of the MainCamera.
  • copy RayTraceUI from the same demo
  • In your scene drag MainCamera to Render Camera field of the RayTraceUI
  • In your scene drag RayTraceDrawer (child of RayTraceUI) into RayTraceCamera’s Display Target field
  • Make sure that RayTraceCamera is enabled
  • Play the demo and go to game view
2 Likes

@Sima_Havok

After playing with the camera some, I found some really interesting behavior. One second the segment mesh is there, the next it disappears!

(If the GIF doesnt show due to its size, it can be seen here: https://vertexfragment.s3.amazonaws.com/public/collisionissuesraycamera2.gif)

(I don’t know how to make GIFs small)

1 Like

I am totally puzzled around cases where rays hit the mesh where character is but the character goes true, as well as cases where casts stop hitting the mesh (although that may be because mesh is too close to camera).

The raytrace camera gives just 2 colors although based on the hit normal it should color hits on the river slopes differently. That is very strange.

Do you use scaling?

@Sima_Havok here is another capture where the camera never goes below the ground.

In this one you can clearly see segments which were fine one second then vanish the next.

(https://vertexfragment.s3.amazonaws.com/public/collisionissuesraycamera3.gif)

And I don’t use scaling. The meshes are generated at the appropriate 1:1 world scale and then their RenderBounds are set to match the AABB returned by the MeshCollider.

Below are all of the components attached to one of these segments. This one is segment (0, 0) at the world origin, which I was falling through at the time.

There used to be a PhysicsMass also attached, set as Kinematic, but removed the component to check if that was the cause for some reason (when first using Unity.Physics I didn’t have kinematic enabled so I was actually bouncing my terrain away).

If this was a render issue instead of a physics one, I would say it looks like incorrect bounds. But the bounds are correct.

Stupid question, since you have your own terrain system the creates tiles of meshes and based on gifs I would say that whole tile is appearing and disappearing. That let me to an experiment with the crappy code below. The component was added to letter H.

using System.ComponentModel;
using Unity.Entities;
using Unity.Physics;
using Unity.Physics.Systems;

[GenerateAuthoringComponent]
public struct AddRemovePhysicsExclude : IComponentData
{
    [DefaultValue(2)]
    public int counter;
}

[UpdateBefore(typeof(BuildPhysicsWorld))]
public class AddPemovePhysicsExcludeSystem : SystemBase
{
    private EndSimulationEntityCommandBufferSystem m_EntityCommandBufferSystem;

    protected override void OnCreate()
    {
        m_EntityCommandBufferSystem = World.GetOrCreateSystem<EndSimulationEntityCommandBufferSystem>();
    }

    protected override void OnUpdate()
    {
        var commandBuffer = m_EntityCommandBufferSystem.CreateCommandBuffer();

        Entities.WithoutBurst().ForEach((Entity e, ref AddRemovePhysicsExclude exclude) =>
        {
            if (exclude.counter == 0)
            {
                commandBuffer.AddComponent(e, new PhysicsExclude());
                exclude.counter = -5;
            }
            else if (exclude.counter == -1)
            {
                commandBuffer.RemoveComponent(e, typeof(PhysicsExclude));
                exclude.counter = 5;
            }
            else if (exclude.counter > 0)
            {
                exclude.counter--;
            }
            else
            {
                exclude.counter++;
            }

        }).Run();
    }
}

Resulting behavior is
6076629--659139--PhysicsExclude.gif

So do you enable and disable tiles that character controller is around during the simulation?

Hello,

I seem to be having a similar problem in my Unity project using MeshColliders and the new Unity Physics package.

Use case: I am generating a grid of entities with MeshColliders which verts/tris are generated at runtime to make 2D voxel “terrain”. I am going to use it in a similar way that Terraria does for its terrain; I also split the voxels into chunks for performant reasons. Everything seems to work great until I edit the voxels a certain amount. In order to produce 2D physics, I constrain the cube physics entities using linear and angular constraints on a PhysicsJoint component (locked z linear and xy angular) connected to a Entity.Null via a PhysicsConstrainedBodyPair entity.

I read the thread and I do not have my physics colliders for the voxels updating using UpdateBefore or UpdateAfter. However, I have a video that, unless I am not understanding how the physics works, suggests the issue an invalid pointer or memory allocation in the meshcollider backend regarding… no idea. Vertex / tri memory data maybe? Just a wild guess.

Video: mzuuqv

I edited the video to speed it up at points because when I incorporated the RayCastCamera from the demo project, it was eating up around 100ms per second in the entity debugger.

Background info for the video: In the video, the pink area in the left is the grid of voxel RenderMesh/MeshCollider entites (a few thousand verts/tris). If I click, I spawn 20 cubes entities in that location. If I hover my mouse over a voxel, it finds a voxel (via mouse pos and magic array magic) and updates the chunks mesh and collider. All of the entity data for the cubes and voxels are hard-coded (not prefab). However, I have four uniformly-scaled cubes surrounding the “play area”, one left, right, above, and below. These four cubes are gameobjects in the scene and are converted to entities and physics objects using the Unity “ConvertToEntity”/physics scripts. These are the only GameObject physics entities like this in my project. I never interact with the four cubes in the code that would have them exhibit the behavior in the video (such as using code like in the post above).

What happens: Up until 30 seconds in the video, everything works as intended: I spawn a bunch of cubes that interact with the runtime generated voxel MeshColliders along with being able to hover my mouse over some voxels and remove them (which in turn regenerates the mesh and colliders). Around the 30 second mark, I am still editing voxels and updating the collider; look at the raytracing camera near the top right of the grid of voxels- they start to phase in and out like the post above. Keep watching and magically the giant cube on the right disappears. Around 1:40ish, I zoom in the raytracing camera; the one chunk of voxels is phasing in and out. Could this be caused by not using UpdateBefore or UpdateAfter? Why does the chunk not phase out in the beginning, or why do the four surrounding cubes begin to phase out if I never mess with their colliders? Throughout the rest of the video, the physics system seems to get less and less stable the more I update the MeshCollider (see 2:30 in video).

I have tried spawning just cubes with no voxels. No bug. I try editing just the voxels without cubes. No bug. But the cubes interacting with the updating (?) colliders with the voxels seems to cause this issue. Furthermore, I originally opted to use the Havok backend for this project. But when whatever happens happens, the Havok physics engine does not freak out like above, it simply crashes. The crash happens in the editor or in the build. I have included the crash files.

Sorry if this is a lot of information or is disorganized, first post.

6077568–659310–havok_editor_crash.zip (511 KB)

@petarmHavok maybe a Havok expert can take a look

Let me ask a few simple questions:

  1. Can you try going to BuildPhysicsWorld.cs and locating a haveStaticBodiesChanged[0] = 0; line, converting that to haveStaticBodiesChanged[0] = 1; and commenting out a block below that tries to set this value. There is a logic that decides to skip building the static layer if there are no changes and I just want to make sure it doesn’t have issues that are causing your problems. This applies to both @ssellvf and @teasully .
  2. @teasully It seems like your dynamic boxes are also disappearing from time to time. What is your number of bodies? Since every body has a joint, you might be hitting a maximum number of joints, as explained in Scheduler.cs (the big comment at the top of the file). I mean it doesn’t look like you have that much, but still, worth asking. Same question (although slightly less likely for @ssellvf ).
  3. @teasully These voxels, are they all combined in a single mesh collider? Or they are all individually mesh colliders?

Also, please reply to @Sima_Havok 's question regarding PhysicsExclude tag. Again, applies to both @teasully and @ssellvf .

Let’s start with that and work our way through.

  • In BuildPhysicsWorld.cs, I edited line 137 to say “haveStaticBodiesChanged[0] = 1;”. Then I commented out lines 138-170. This did not have any changes; the bug still appears.
  • I’m not sure how to tell how many physics bodies there are, however, I would like to safely say that I have less than 16,777,216 rigid bodies and 32,767 joints. Using a simple counter on cube spawn, I replicated the issue with around 1,600 bodies each with a joint of its own, so 1,600 joints.
  • I distribute the voxels into chunks, so each chunk holds 8x8 voxels (64). Each chunk has its own MeshCollider components. In the video, there is a 10x10 grid of chunks (100 chunks = 100 MeshColliders). Each MeshCollider is updated as needed, they are not updated every frame.

As for Sima_Havok’s question, I do not use the PhysicsExclude tag.

Also, I stumbled upon this thread ( Havok crashing on terrain collider ) which is about a bug concerning the TerrainCollider and an issue using triangle collision. The solution was to change the Collision Method from triangles to vertexes; is there a similar setting for MeshColliders and is that issue related to this one?

@petarmHavok I do not currently actively cull any of the segments. So no use of PhysicsExclude component or the generic Disabled component.

I also tried your suggestion of modifying BuildPhysicsWorld.cs

haveStaticBodiesChanged[0] = 1;
/*{
    if (PhysicsWorld.NumStaticBodies != previousStaticBodyCount)
    {
        haveStaticBodiesChanged[0] = 1;
    }
    else
    {
        // Make a job to test for changes
        int numChunks;
        using (NativeArray<ArchetypeChunk> chunks = StaticEntityGroup.CreateArchetypeChunkArray(Allocator.TempJob))
        {
            numChunks = chunks.Length;
        }
        var chunksHaveChanges = new NativeArray<int>(numChunks, Allocator.TempJob);

        staticBodiesCheckHandle = new Jobs.CheckStaticBodyChangesJob
        {
            LocalToWorldType = localToWorldType,
            ParentType = parentType,
            PositionType = positionType,
            RotationType = rotationType,
            PhysicsColliderType = physicsColliderType,
            ChunkHasChangesOutput = chunksHaveChanges,
            m_LastSystemVersion = LastSystemVersion
        }.Schedule(StaticEntityGroup, Dependency);

        staticBodiesCheckHandle = new Jobs.CheckStaticBodyChangesReduceJob
        {
            ChunkHasChangesOutput = chunksHaveChanges,
            Result = haveStaticBodiesChanged
        }.Schedule(staticBodiesCheckHandle);
    }
}*/

Which unfortunately had no effect.

I just want to add that I greatly appreciate the time that various people have spent in this thread to help me.

The terrain only has 2 different collision options. Mesh is only about triangles.