2023-08-19 06:48:47 +03:00
|
|
|
/*
|
|
|
|
Copyright (c) 2009-2010 Mikko Mononen memon@inside.org
|
|
|
|
recast4j copyright (c) 2015-2019 Piotr Piastucki piotr@jtilia.org
|
|
|
|
DotRecast Copyright (c) 2023 Choi Ikpil ikpil@naver.com
|
|
|
|
|
|
|
|
This software is provided 'as-is', without any express or implied
|
|
|
|
warranty. In no event will the authors be held liable for any damages
|
|
|
|
arising from the use of this software.
|
|
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
|
|
including commercial applications, and to alter it and redistribute it
|
|
|
|
freely, subject to the following restrictions:
|
|
|
|
1. The origin of this software must not be misrepresented; you must not
|
|
|
|
claim that you wrote the original software. If you use this software
|
|
|
|
in a product, an acknowledgment in the product documentation would be
|
|
|
|
appreciated but is not required.
|
|
|
|
2. Altered source versions must be plainly marked as such, and must not be
|
|
|
|
misrepresented as being the original software.
|
|
|
|
3. This notice may not be removed or altered from any source distribution.
|
|
|
|
*/
|
|
|
|
|
|
|
|
using System;
|
|
|
|
using System.Collections.Concurrent;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Threading.Tasks;
|
2023-10-16 17:55:34 +03:00
|
|
|
using DotRecast.Core.Numerics;
|
2023-08-19 06:48:47 +03:00
|
|
|
using DotRecast.Detour.TileCache.Io.Compress;
|
|
|
|
using DotRecast.Recast;
|
|
|
|
using DotRecast.Recast.Geom;
|
|
|
|
|
|
|
|
namespace DotRecast.Detour.TileCache
|
|
|
|
{
|
|
|
|
public class DtTileCacheLayerBuilder
|
|
|
|
{
|
2023-09-02 09:49:10 +03:00
|
|
|
private readonly IDtTileCacheCompressorFactory _compFactory;
|
2023-08-19 06:48:47 +03:00
|
|
|
|
|
|
|
public DtTileCacheLayerBuilder(IDtTileCacheCompressorFactory compFactory)
|
|
|
|
{
|
|
|
|
_compFactory = compFactory;
|
|
|
|
}
|
|
|
|
|
2023-08-19 07:16:29 +03:00
|
|
|
public List<DtTileCacheLayerBuildResult> Build(IInputGeomProvider geom, RcConfig cfg, DtTileCacheStorageParams storageParams, int threads, int tw, int th)
|
2023-08-19 06:48:47 +03:00
|
|
|
{
|
|
|
|
if (threads == 1)
|
|
|
|
{
|
2023-08-19 07:16:29 +03:00
|
|
|
return BuildSingleThread(geom, cfg, storageParams, tw, th);
|
2023-08-19 06:48:47 +03:00
|
|
|
}
|
|
|
|
|
2023-08-19 07:16:29 +03:00
|
|
|
return BuildMultiThread(geom, cfg, storageParams, tw, th, threads);
|
2023-08-19 06:48:47 +03:00
|
|
|
}
|
|
|
|
|
2023-08-19 07:16:29 +03:00
|
|
|
private List<DtTileCacheLayerBuildResult> BuildSingleThread(IInputGeomProvider geom, RcConfig cfg, DtTileCacheStorageParams storageParams, int tw, int th)
|
2023-08-19 06:48:47 +03:00
|
|
|
{
|
2023-08-19 07:03:00 +03:00
|
|
|
var results = new List<DtTileCacheLayerBuildResult>();
|
2023-08-19 06:48:47 +03:00
|
|
|
for (int y = 0; y < th; ++y)
|
|
|
|
{
|
|
|
|
for (int x = 0; x < tw; ++x)
|
|
|
|
{
|
2023-08-19 07:16:29 +03:00
|
|
|
var result = BuildTileCacheLayer(geom, cfg, x, y, storageParams);
|
2023-08-19 07:03:00 +03:00
|
|
|
results.Add(result);
|
2023-08-19 06:48:47 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-19 07:03:00 +03:00
|
|
|
return results;
|
2023-08-19 06:48:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-08-19 07:16:29 +03:00
|
|
|
private List<DtTileCacheLayerBuildResult> BuildMultiThread(IInputGeomProvider geom, RcConfig cfg, DtTileCacheStorageParams storageParams, int tw, int th, int threads)
|
2023-08-19 06:48:47 +03:00
|
|
|
{
|
2023-08-19 07:03:00 +03:00
|
|
|
var results = new List<Task<DtTileCacheLayerBuildResult>>();
|
2023-08-19 06:48:47 +03:00
|
|
|
for (int y = 0; y < th; ++y)
|
|
|
|
{
|
|
|
|
for (int x = 0; x < tw; ++x)
|
|
|
|
{
|
|
|
|
int tx = x;
|
|
|
|
int ty = y;
|
2023-08-19 07:16:29 +03:00
|
|
|
var task = Task.Run(() => BuildTileCacheLayer(geom, cfg, tx, ty, storageParams));
|
2023-08-19 07:03:00 +03:00
|
|
|
results.Add(task);
|
2023-08-19 06:48:47 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return results
|
2023-08-19 07:03:00 +03:00
|
|
|
.Select(x => x.Result)
|
2023-08-19 06:48:47 +03:00
|
|
|
.ToList();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected virtual RcHeightfieldLayerSet BuildHeightfieldLayerSet(IInputGeomProvider geom, RcConfig cfg, int tx, int ty)
|
|
|
|
{
|
2023-09-22 17:42:21 +03:00
|
|
|
RcBuilder rcBuilder = new RcBuilder();
|
2023-08-19 06:48:47 +03:00
|
|
|
RcVec3f bmin = geom.GetMeshBoundsMin();
|
|
|
|
RcVec3f bmax = geom.GetMeshBoundsMax();
|
2023-09-22 17:42:21 +03:00
|
|
|
RcBuilderConfig builderCfg = new RcBuilderConfig(cfg, bmin, bmax, tx, ty);
|
2023-08-19 06:48:47 +03:00
|
|
|
RcHeightfieldLayerSet lset = rcBuilder.BuildLayers(geom, builderCfg);
|
|
|
|
return lset;
|
|
|
|
}
|
|
|
|
|
2023-08-19 07:16:29 +03:00
|
|
|
protected virtual DtTileCacheLayerBuildResult BuildTileCacheLayer(IInputGeomProvider geom, RcConfig cfg, int tx, int ty, DtTileCacheStorageParams storageParams)
|
2023-08-19 06:48:47 +03:00
|
|
|
{
|
|
|
|
RcHeightfieldLayerSet lset = BuildHeightfieldLayerSet(geom, cfg, tx, ty);
|
|
|
|
List<byte[]> result = new List<byte[]>();
|
|
|
|
if (lset != null)
|
|
|
|
{
|
|
|
|
DtTileCacheBuilder builder = new DtTileCacheBuilder();
|
|
|
|
for (int i = 0; i < lset.layers.Length; ++i)
|
|
|
|
{
|
|
|
|
RcHeightfieldLayer layer = lset.layers[i];
|
|
|
|
|
|
|
|
// Store header
|
|
|
|
DtTileCacheLayerHeader header = new DtTileCacheLayerHeader();
|
|
|
|
header.magic = DtTileCacheLayerHeader.DT_TILECACHE_MAGIC;
|
|
|
|
header.version = DtTileCacheLayerHeader.DT_TILECACHE_VERSION;
|
|
|
|
|
|
|
|
// Tile layer location in the navmesh.
|
|
|
|
header.tx = tx;
|
|
|
|
header.ty = ty;
|
|
|
|
header.tlayer = i;
|
|
|
|
header.bmin = layer.bmin;
|
|
|
|
header.bmax = layer.bmax;
|
|
|
|
|
|
|
|
// Tile info.
|
|
|
|
header.width = layer.width;
|
|
|
|
header.height = layer.height;
|
|
|
|
header.minx = layer.minx;
|
|
|
|
header.maxx = layer.maxx;
|
|
|
|
header.miny = layer.miny;
|
|
|
|
header.maxy = layer.maxy;
|
|
|
|
header.hmin = layer.hmin;
|
|
|
|
header.hmax = layer.hmax;
|
|
|
|
|
2023-08-19 09:31:51 +03:00
|
|
|
var comp = _compFactory.Create(storageParams.Compatibility ? 0 : 1);
|
2023-08-19 07:16:29 +03:00
|
|
|
var bytes = builder.CompressTileCacheLayer(header, layer.heights, layer.areas, layer.cons, storageParams.Order, storageParams.Compatibility, comp);
|
2023-08-19 06:48:47 +03:00
|
|
|
result.Add(bytes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-19 07:03:00 +03:00
|
|
|
return new DtTileCacheLayerBuildResult(tx, ty, result);
|
2023-08-19 06:48:47 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|