2019-06-26 22:58:51 +02:00
/*
MIT License
2020-05-20 13:03:57 +02:00
Copyright ( c ) 2018 - 2020 Jonathan Young
2019-06-26 22:58:51 +02:00
Permission is hereby granted , free of charge , to any person obtaining a copy
of this software and associated documentation files ( the " Software " ) , to deal
in the Software without restriction , including without limitation the rights
to use , copy , modify , merge , publish , distribute , sublicense , and / or sell
copies of the Software , and to permit persons to whom the Software is
furnished to do so , subject to the following conditions :
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software .
THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS OR
IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY ,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT . IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER
LIABILITY , WHETHER IN AN ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING FROM ,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE .
*/
/*
thekla_atlas
MIT License
https : //github.com/Thekla/thekla_atlas
Copyright ( c ) 2013 Thekla , Inc
Copyright NVIDIA Corporation 2006 - - Ignacio Castano < icastano @ nvidia . com >
*/
2018-09-29 14:46:26 +02:00
# pragma once
# ifndef XATLAS_H
# define XATLAS_H
2019-06-26 22:58:51 +02:00
# include <stdint.h>
2019-04-19 12:03:12 +02:00
2018-09-29 14:46:26 +02:00
namespace xatlas {
2019-12-22 11:23:44 +01:00
struct ChartType
2019-06-26 22:58:51 +02:00
{
2019-12-22 11:23:44 +01:00
enum Enum
2019-06-26 22:58:51 +02:00
{
2019-12-22 11:23:44 +01:00
Planar ,
Ortho ,
LSCM ,
2020-05-20 13:03:57 +02:00
Piecewise ,
Invalid
2019-06-26 22:58:51 +02:00
} ;
} ;
2018-09-29 14:46:26 +02:00
2019-06-26 22:58:51 +02:00
// A group of connected faces, belonging to a single atlas.
struct Chart
{
2019-08-28 03:12:21 +02:00
uint32_t * faceArray ;
2020-05-20 13:03:57 +02:00
uint32_t atlasIndex ; // Sub-atlas index.
2019-08-28 03:12:21 +02:00
uint32_t faceCount ;
2019-12-22 11:23:44 +01:00
ChartType : : Enum type ;
2020-05-20 13:03:57 +02:00
uint32_t material ;
2019-06-26 22:58:51 +02:00
} ;
2018-09-29 14:46:26 +02:00
2019-06-26 22:58:51 +02:00
// Output vertex.
struct Vertex
2019-04-19 12:03:12 +02:00
{
2019-06-26 22:58:51 +02:00
int32_t atlasIndex ; // Sub-atlas index. -1 if the vertex doesn't exist in any atlas.
int32_t chartIndex ; // -1 if the vertex doesn't exist in any chart.
float uv [ 2 ] ; // Not normalized - values are in Atlas width and height range.
uint32_t xref ; // Index of input vertex from which this output vertex originated.
2018-09-29 14:46:26 +02:00
} ;
2019-06-26 22:58:51 +02:00
// Output mesh.
struct Mesh
2019-04-19 12:03:12 +02:00
{
2019-06-26 22:58:51 +02:00
Chart * chartArray ;
uint32_t * indexArray ;
Vertex * vertexArray ;
2020-05-20 13:03:57 +02:00
uint32_t chartCount ;
uint32_t indexCount ;
2019-06-26 22:58:51 +02:00
uint32_t vertexCount ;
2018-09-29 14:46:26 +02:00
} ;
2019-08-28 03:12:21 +02:00
static const uint32_t kImageChartIndexMask = 0x1FFFFFFF ;
static const uint32_t kImageHasChartIndexBit = 0x80000000 ;
static const uint32_t kImageIsBilinearBit = 0x40000000 ;
static const uint32_t kImageIsPaddingBit = 0x20000000 ;
2019-06-26 22:58:51 +02:00
// Empty on creation. Populated after charts are packed.
struct Atlas
2019-04-19 12:03:12 +02:00
{
2020-05-20 13:03:57 +02:00
uint32_t * image ;
Mesh * meshes ; // The output meshes, corresponding to each AddMesh call.
2019-06-26 22:58:51 +02:00
uint32_t width ; // Atlas width in texels.
uint32_t height ; // Atlas height in texels.
uint32_t atlasCount ; // Number of sub-atlases. Equal to 0 unless PackOptions resolution is changed from default (0).
uint32_t chartCount ; // Total number of charts in all meshes.
uint32_t meshCount ; // Number of output meshes. Equal to the number of times AddMesh was called.
float * utilization ; // Normalized atlas texel utilization array. E.g. a value of 0.8 means 20% empty space. atlasCount in length.
float texelsPerUnit ; // Equal to PackOptions texelsPerUnit if texelsPerUnit > 0, otherwise an estimated value to match PackOptions resolution.
2018-09-29 14:46:26 +02:00
} ;
2019-06-26 22:58:51 +02:00
// Create an empty atlas.
Atlas * Create ( ) ;
void Destroy ( Atlas * atlas ) ;
struct IndexFormat
2019-04-19 12:03:12 +02:00
{
enum Enum
{
2019-06-26 22:58:51 +02:00
UInt16 ,
UInt32
2018-09-29 14:46:26 +02:00
} ;
} ;
2019-06-26 22:58:51 +02:00
// Input mesh declaration.
struct MeshDecl
2019-04-19 12:03:12 +02:00
{
2019-06-26 22:58:51 +02:00
const void * vertexPositionData = nullptr ;
const void * vertexNormalData = nullptr ; // optional
const void * vertexUvData = nullptr ; // optional. The input UVs are provided as a hint to the chart generator.
const void * indexData = nullptr ; // optional
// Optional. indexCount / 3 (triangle count) in length.
// Don't atlas faces set to true. Ignored faces still exist in the output meshes, Vertex uv is set to (0, 0) and Vertex atlasIndex to -1.
const bool * faceIgnoreData = nullptr ;
2020-05-20 13:03:57 +02:00
uint32_t vertexCount = 0 ;
uint32_t vertexPositionStride = 0 ;
uint32_t vertexNormalStride = 0 ; // optional
uint32_t vertexUvStride = 0 ; // optional
uint32_t indexCount = 0 ;
int32_t indexOffset = 0 ; // optional. Add this offset to all indices.
IndexFormat : : Enum indexFormat = IndexFormat : : UInt16 ;
2019-06-26 22:58:51 +02:00
// Vertex positions within epsilon distance of each other are considered colocal.
float epsilon = 1.192092896e-07 F ;
2018-09-29 14:46:26 +02:00
} ;
2019-06-26 22:58:51 +02:00
struct AddMeshError
2019-04-19 12:03:12 +02:00
{
enum Enum
{
2019-06-26 22:58:51 +02:00
Success , // No error.
Error , // Unspecified error.
IndexOutOfRange , // An index is >= MeshDecl vertexCount.
InvalidIndexCount // Not evenly divisible by 3 - expecting triangles.
2018-09-29 14:46:26 +02:00
} ;
} ;
2019-06-26 22:58:51 +02:00
// Add a mesh to the atlas. MeshDecl data is copied, so it can be freed after AddMesh returns.
AddMeshError : : Enum AddMesh ( Atlas * atlas , const MeshDecl & meshDecl , uint32_t meshCountHint = 0 ) ;
2018-09-29 14:46:26 +02:00
2019-06-26 22:58:51 +02:00
// Wait for AddMesh async processing to finish. ComputeCharts / Generate call this internally.
void AddMeshJoin ( Atlas * atlas ) ;
2018-09-29 14:46:26 +02:00
2019-06-26 22:58:51 +02:00
struct UvMeshDecl
{
2020-05-20 13:03:57 +02:00
const void * vertexUvData = nullptr ;
const void * indexData = nullptr ; // optional
const uint32_t * faceMaterialData = nullptr ; // Optional. Faces with different materials won't be assigned to the same chart. Must be indexCount / 3 in length.
2019-06-26 22:58:51 +02:00
uint32_t vertexCount = 0 ;
uint32_t vertexStride = 0 ;
uint32_t indexCount = 0 ;
int32_t indexOffset = 0 ; // optional. Add this offset to all indices.
IndexFormat : : Enum indexFormat = IndexFormat : : UInt16 ;
bool rotateCharts = true ;
2018-09-29 14:46:26 +02:00
} ;
2019-06-26 22:58:51 +02:00
AddMeshError : : Enum AddUvMesh ( Atlas * atlas , const UvMeshDecl & decl ) ;
struct ChartOptions
2019-04-19 12:03:12 +02:00
{
2019-06-26 22:58:51 +02:00
float maxChartArea = 0.0f ; // Don't grow charts to be larger than this. 0 means no limit.
float maxBoundaryLength = 0.0f ; // Don't grow charts to have a longer boundary than this. 0 means no limit.
// Weights determine chart growth. Higher weights mean higher cost for that metric.
2020-05-20 13:03:57 +02:00
float normalDeviationWeight = 2.0f ; // Angle between face and average chart normal.
float roundnessWeight = 0.01f ;
float straightnessWeight = 6.0f ;
float normalSeamWeight = 4.0f ; // If > 1000, normal seams are fully respected.
float textureSeamWeight = 0.5f ;
2019-06-26 22:58:51 +02:00
2020-05-20 13:03:57 +02:00
float maxCost = 2.0f ; // If total of all metrics * weights > maxCost, don't grow chart. Lower values result in more charts.
2019-06-26 22:58:51 +02:00
uint32_t maxIterations = 1 ; // Number of iterations of the chart growing and seeding phases. Higher values result in better charts.
2018-09-29 14:46:26 +02:00
} ;
2019-06-26 22:58:51 +02:00
// Call after all AddMesh calls. Can be called multiple times to recompute charts with different options.
2020-05-20 13:03:57 +02:00
void ComputeCharts ( Atlas * atlas , ChartOptions options = ChartOptions ( ) ) ;
2019-06-26 22:58:51 +02:00
// Custom parameterization function. texcoords initial values are an orthogonal parameterization.
typedef void ( * ParameterizeFunc ) ( const float * positions , float * texcoords , uint32_t vertexCount , const uint32_t * indices , uint32_t indexCount ) ;
2020-05-20 13:03:57 +02:00
struct ParameterizeOptions
{
ParameterizeFunc func = nullptr ;
bool closeHoles = true ; // If the custom parameterization function works with multiple boundaries, this can be set to false to improve performance.
bool fixTJunctions = true ; // If meshes don't have T-junctions, this can be set to false to improve performance.
} ;
2019-06-26 22:58:51 +02:00
// Call after ComputeCharts. Can be called multiple times to re-parameterize charts with a different ParameterizeFunc.
2020-05-20 13:03:57 +02:00
void ParameterizeCharts ( Atlas * atlas , ParameterizeOptions options = ParameterizeOptions ( ) ) ;
2019-06-26 22:58:51 +02:00
struct PackOptions
2019-04-19 12:03:12 +02:00
{
2019-08-28 03:12:21 +02:00
// Leave space around charts for texels that would be sampled by bilinear filtering.
bool bilinear = true ;
// Align charts to 4x4 blocks. Also improves packing speed, since there are fewer possible chart locations to consider.
bool blockAlign = false ;
2019-06-26 22:58:51 +02:00
// Slower, but gives the best result. If false, use random chart placement.
bool bruteForce = false ;
// Create Atlas::image
bool createImage = false ;
2019-08-28 03:12:21 +02:00
// Charts larger than this will be scaled down. 0 means no limit.
uint32_t maxChartSize = 0 ;
// Number of pixels to pad charts with.
uint32_t padding = 0 ;
2019-06-26 22:58:51 +02:00
// Unit to texel scale. e.g. a 1x1 quad with texelsPerUnit of 32 will take up approximately 32x32 texels in the atlas.
// If 0, an estimated value will be calculated to approximately match the given resolution.
// If resolution is also 0, the estimated value will approximately match a 1024x1024 atlas.
float texelsPerUnit = 0.0f ;
// If 0, generate a single atlas with texelsPerUnit determining the final resolution.
// If not 0, and texelsPerUnit is not 0, generate one or more atlases with that exact resolution.
// If not 0, and texelsPerUnit is 0, texelsPerUnit is estimated to approximately match the resolution.
uint32_t resolution = 0 ;
2018-09-29 14:46:26 +02:00
} ;
2019-06-26 22:58:51 +02:00
// Call after ParameterizeCharts. Can be called multiple times to re-pack charts with different options.
void PackCharts ( Atlas * atlas , PackOptions packOptions = PackOptions ( ) ) ;
// Equivalent to calling ComputeCharts, ParameterizeCharts and PackCharts in sequence. Can be called multiple times to regenerate with different options.
2020-05-20 13:03:57 +02:00
void Generate ( Atlas * atlas , ChartOptions chartOptions = ChartOptions ( ) , ParameterizeOptions parameterizeOptions = ParameterizeOptions ( ) , PackOptions packOptions = PackOptions ( ) ) ;
2019-06-26 22:58:51 +02:00
// Progress tracking.
struct ProgressCategory
2019-04-19 12:03:12 +02:00
{
2019-06-26 22:58:51 +02:00
enum Enum
{
AddMesh ,
ComputeCharts ,
ParameterizeCharts ,
PackCharts ,
BuildOutputMeshes
} ;
2018-09-29 14:46:26 +02:00
} ;
2019-06-26 22:58:51 +02:00
// May be called from any thread. Return false to cancel.
typedef bool ( * ProgressFunc ) ( ProgressCategory : : Enum category , int progress , void * userData ) ;
void SetProgressCallback ( Atlas * atlas , ProgressFunc progressFunc = nullptr , void * progressUserData = nullptr ) ;
// Custom memory allocation.
typedef void * ( * ReallocFunc ) ( void * , size_t ) ;
2019-08-28 03:12:21 +02:00
typedef void ( * FreeFunc ) ( void * ) ;
void SetAlloc ( ReallocFunc reallocFunc , FreeFunc freeFunc = nullptr ) ;
2019-06-26 22:58:51 +02:00
// Custom print function.
typedef int ( * PrintFunc ) ( const char * , . . . ) ;
void SetPrint ( PrintFunc print , bool verbose ) ;
// Helper functions for error messages.
const char * StringForEnum ( AddMeshError : : Enum error ) ;
const char * StringForEnum ( ProgressCategory : : Enum category ) ;
2018-09-29 14:46:26 +02:00
} // namespace xatlas
# endif // XATLAS_H