Batching rework and optimization

This commit is contained in:
Nahuel Rocchetti 2023-08-01 03:42:47 -03:00
parent 1c7589027c
commit cca1ae128b
2 changed files with 41 additions and 53 deletions

View file

@ -12,14 +12,26 @@ namespace OpenTS2.Rendering
{ {
public Material Material; public Material Material;
public Mesh Mesh; public Mesh Mesh;
public MeshRenderer Renderer;
public Transform Transform; public Transform Transform;
public int SubMesh; public BatchCandidate(Material material, Mesh mesh, MeshRenderer renderer, Transform transform)
public BatchCandidate(Material material, Mesh mesh, Transform transform, int submesh)
{ {
Material = material; Material = material;
Mesh = mesh; Mesh = mesh;
Transform = transform; Transform = transform;
SubMesh = submesh; Renderer = renderer;
}
}
public class BatchResult
{
public GameObject BatchedObjectsParent;
public List<MeshRenderer> DisabledRenderers = new List<MeshRenderer>();
public void RestoreVisibility()
{
foreach (var renderer in DisabledRenderers)
renderer.enabled = true;
} }
} }
@ -61,49 +73,44 @@ namespace OpenTS2.Rendering
/// <param name="parent">Parent transform that contains all meshes.</param> /// <param name="parent">Parent transform that contains all meshes.</param>
/// <param name="flipFaces">Whether to flip faces. Set this to true if the meshes are using TS2's coordinate system.</param> /// <param name="flipFaces">Whether to flip faces. Set this to true if the meshes are using TS2's coordinate system.</param>
/// <returns>Parent GameObject containing all batched meshes.</returns> /// <returns>Parent GameObject containing all batched meshes.</returns>
public static GameObject Batch(Transform parent, bool flipFaces = false) public static BatchResult Batch(Transform parent, bool flipFaces = false)
{ {
var result = new BatchResult();
// TODO - Maybe make this function multithreaded. // TODO - Maybe make this function multithreaded.
var candidatesByMaterial = new Dictionary<Material, List<BatchCandidate>>(); var candidatesByMaterial = new Dictionary<Material, List<BatchCandidate>>();
var nonBatchedCandidates = new List<BatchCandidate>();
var renderers = parent.GetComponentsInChildren<MeshRenderer>(); var renderers = parent.GetComponentsInChildren<MeshRenderer>();
foreach (var element in renderers) foreach (var renderer in renderers)
{ {
var filter = element.GetComponent<MeshFilter>(); var filter = renderer.GetComponent<MeshFilter>();
if (filter == null) if (filter == null)
continue; continue;
var materials = element.sharedMaterials; if (filter.sharedMesh == null)
continue;
for (var i = 0; i < materials.Length; i++) if (renderer.sharedMaterial == null)
continue;
if (filter.sharedMesh.subMeshCount > 1)
continue;
var mesh = filter.sharedMesh;
var material = renderer.sharedMaterial;
if (!CanBatchShader(material.shader))
continue;
var candidate = new BatchCandidate(material, mesh, renderer, filter.transform);
if (!candidatesByMaterial.TryGetValue(material, out List<BatchCandidate> candidates))
{ {
var mat = materials[i]; candidates = new List<BatchCandidate>();
if (filter.sharedMesh.subMeshCount <= i) candidatesByMaterial[material] = candidates;
continue;
var candidate = new BatchCandidate(mat, filter.sharedMesh, filter.transform, i);
if (!CanBatchShader(mat.shader))
{
nonBatchedCandidates.Add(candidate);
continue;
}
if (!candidatesByMaterial.TryGetValue(mat, out List<BatchCandidate> candidates))
{
candidates = new List<BatchCandidate>();
candidatesByMaterial[mat] = candidates;
}
candidates.Add(candidate);
} }
candidates.Add(candidate);
} }
var finalGameObject = new GameObject("Batched Objects"); var finalGameObject = new GameObject("Batched Objects");
result.BatchedObjectsParent = finalGameObject;
foreach (var material in candidatesByMaterial) foreach (var material in candidatesByMaterial)
{ {
if (material.Value.Count == 1) if (material.Value.Count == 1)
{
MakeNonBatched(material.Value[0].Transform, material.Value[0].Mesh, material.Key);
continue; continue;
}
var mesh = new Mesh(); var mesh = new Mesh();
var vertAmount = 0; var vertAmount = 0;
foreach(var candidate in material.Value) foreach (var candidate in material.Value)
{ {
vertAmount += candidate.Mesh.vertexCount; vertAmount += candidate.Mesh.vertexCount;
} }
@ -127,18 +134,20 @@ namespace OpenTS2.Rendering
foreach (var candidate in batchCandidates) foreach (var candidate in batchCandidates)
{ {
result.DisabledRenderers.Add(candidate.Renderer);
candidate.Renderer.enabled = false;
var vertices = new List<Vector3>(); var vertices = new List<Vector3>();
var normals = new List<Vector3>(); var normals = new List<Vector3>();
var uv = new List<Vector2>(); var uv = new List<Vector2>();
candidate.Mesh.GetVertices(vertices); candidate.Mesh.GetVertices(vertices);
var triangles = candidate.Mesh.GetTriangles(candidate.SubMesh); var triangles = candidate.Mesh.GetTriangles(0);
candidate.Mesh.GetNormals(normals); candidate.Mesh.GetNormals(normals);
candidate.Mesh.GetUVs(0, uv); candidate.Mesh.GetUVs(0, uv);
for (var i = 0; i < vertices.Count; i++) for (var i = 0; i < vertices.Count; i++)
{ {
vertices[i] = candidate.Transform.TransformPoint(vertices[i]); vertices[i] = candidate.Transform.TransformPoint(vertices[i]);
} }
for(var i=0;i<normals.Count;i++) for (var i = 0; i < normals.Count; i++)
{ {
normals[i] = candidate.Transform.TransformDirection(normals[i]); normals[i] = candidate.Transform.TransformDirection(normals[i]);
} }
@ -161,24 +170,7 @@ namespace OpenTS2.Rendering
mesh.RecalculateBounds(); mesh.RecalculateBounds();
mesh.Optimize(); mesh.Optimize();
} }
foreach(var nonBatched in nonBatchedCandidates) return result;
{
MakeNonBatched(nonBatched.Transform, nonBatched.Mesh, nonBatched.Material);
}
return finalGameObject;
void MakeNonBatched(Transform transform, Mesh mesh, Material material)
{
var candidateGameObject = new GameObject($"Non-Batched: {material.name}", typeof(MeshRenderer), typeof(MeshFilter));
candidateGameObject.transform.SetParent(finalGameObject.transform);
candidateGameObject.transform.position = transform.position;
candidateGameObject.transform.rotation = transform.rotation;
candidateGameObject.transform.localScale = transform.lossyScale;
var renderer = candidateGameObject.GetComponent<MeshRenderer>();
var filter = candidateGameObject.GetComponent<MeshFilter>();
renderer.sharedMaterial = material;
filter.sharedMesh = mesh;
}
} }
} }
} }

View file

@ -61,10 +61,6 @@ namespace OpenTS2.Scenes
var batchedDeco = Batching.Batch(_decorationsParent, flipFaces: true); var batchedDeco = Batching.Batch(_decorationsParent, flipFaces: true);
var batchedLots = Batching.Batch(_lotsParent, flipFaces: true); var batchedLots = Batching.Batch(_lotsParent, flipFaces: true);
var batchedRoads = Batching.Batch(_roadsParent, flipFaces: false); var batchedRoads = Batching.Batch(_roadsParent, flipFaces: false);
_decorationsParent.gameObject.SetActive(false);
_roadsParent.gameObject.SetActive(false);
_lotsParent.gameObject.SetActive(false);
} }
// Clean up the road materials we created. Textures should get garbage collected. // Clean up the road materials we created. Textures should get garbage collected.