PHY

From Valve Developer Community
Jump to navigation Jump to search

PHY is the extension for Source Source's proprietary collision model data format. It stores data for the rigid or jointed (ragdoll) collision model used by the MDL. In addition, it stores some $keyvalues about a model such as which gib-models to use if destroyed.

Note.pngNote:The PHY file is completely independent from the rest of the model's binaries, such as MDL or VVD. As long as the filename and skeleton structure matches the data within the MDL file, the collision data can be changed without any adverse affects. A collision mesh could be added to a model without one initially, but a recompile would be needed to set the correct flags within the MDL file and to add $keyvalues data.

Also if you're optimizing your map because you have too many edicts it's useful to know that the PHY file counts as one edict (adds to Entity_limit) for unknown reasons. You can delete the PHY file to overcome this problem if the custom model doesn't need the collision data.

File Format

In brief, a PHY file is formed by:

  • A main header, which specifies the number of solids
  • A series of VPHY/collision metadata sections, one per solid
  • A collection of triangles and vertices composing the collider meshes
  • A single text data section, variously-ordered block elements that specify which solid they apply to

Main header

The first bytes of the file form the main header:

// this structure can be found in <mod folder>/src/public/phyfile.h
struct phyheader_t
{
	int		size;           // Size of this header section (generally 16)
	int		id;             // Often zero, unknown purpose.
	int		solidCount;     // Number of solids in file
	long	        checkSum;	// checksum of source .mdl file (4-bytes)
};

Collision data sections

There will be a series of these sections, back-to-back, numbering the same as the header's solidCount.

// new phy format
struct compactsurfaceheader_t
{
	int	size;			// Size of the content after this byte
	int	vphysicsID;		// Generally the ASCII for "VPHY" in newer files
	short	version;
	short	modelType;
	int	surfaceSize;
	Vector	dragAxisAreas;
	int	axisMapSize;
};

// old style phy format
struct legacysurfaceheader_t
{
	int	size;
	float	mass_center[3];
	float	rotation_inertia[3];
	float	upper_limit_radius;
	int	max_deviation : 8;
	int	byte_size : 24;
	int	offset_ledgetree_root;
	int	dummy[3]; 		// dummy[2] is "IVPS" or 0
};
Note.pngNote:Probably it's not complete description of solid section

Each surface has a list of convex solids that ends at the vertices_offset address.

struct convexsolidheader_t {
	int vertices_offset; // For each convexsolidheader_t it will point to the same address of the file
	int bone_index;
	int flags;
	int triangles_count;
}

After the convex solid data there's a list of triangles:

struct triangledata_t {
	byte vertex_index;
	byte unused1;
	unsigned short unused2;

	short vertex1_index;
	short unused3;
	short vertex2_index;
	short unused4;
	short vertex3_index;
	short unused5;
}

From the vertices_offset address comes the full list of vertices in vec4

Todo: Describe the data that comes after the vertices section

Text section

The text section occurs near the end of the file, and is one single string with no size header. Among the text blocks it may contain:

  • solid blocks define solids, using an index value to distinguish which set of collision data they refer to.
  • ragdollconstraint blocks define relationships between parent and child solids.
  • An editparams block with general information about the entire object.
  • Various break blocks with information about rigid or jointed gibs created when the model is destroyed.

Here is an incomplete sample from the Combine Strider, formatted for readability.

solid
{
	"index" "8"
	"name" "Combine_Strider.Leg_Hind_Bone1"
	"parent" "Combine_Strider.Leg_Hind_Bone"
	"mass" "137.739014"
	"surfaceprop" "strider"
	"damping" "0.000000"
	"rotdamping" "1.000000"
	"inertia" "1.000000"
	"volume" "109397.476563"
} 
ragdollconstraint
{
	"parent" "7"
	"child" "8"
	"xmin" "0.000000"
	"xmax" "0.000000"
	"xfriction" "2.000000"
	"ymin" "0.000000"
	"ymax" "0.000000"
	"yfriction" "2.000000"
	"zmin" "-32.000000"
	"zmax" "138.000000"
	"zfriction" "0.200000"
}
editparams
{
	"rootname" "combine_strider.body_bone"
	"totalmass" "2000.000000"
} 
break
{
	"model" "Gibs/Strider_Gib1" 
	"health" "0" 
	"fadetime" "0" 
	"placementbone" "Combine_strider.body_bone"
}
break
{
	"ragdoll" "Gibs/Strider_Head" 
	"health" "0" 
	"fadetime" "0"  
}

See also