Oh my gosh!!!
I found some ground-breaking research on how CSG works on Rōblox:
Here are some example unions:
CSGv3.rbxmx (17,0 KB)
CSGv2.rbxmx (21,0 KB)
Geometry made with Studio versions from before 2022M used CSGv2, whereas current builds exclusively use CSGv3.
In both cases, they’re stored in SharedString
form. In the CSGv2
file, the mesh data is as follows:
<BinaryString name="MeshData"><![CDATA[FX0pFXVsMgQ0aWkWPjJuYy1W ... AWcA
GmAuaR1S]]></BinaryString>
Long.
Here’s what happens when I decode the base64 from the XML file:
15 7d 29 15 75 6c 32 04 ...
No discernible magic header. I was stuck … until I found the revolutionary
rbx_mesh
!!!
WHAT THE HACK! THEY’RE USING XOR ENCRYPTION; I SHOULD’VE SÈÉN THAT COMING!!!
pub const OBFUSCATION_NOISE_CYCLE_XOR:[u8;31]=[86,46,110,88,49,32,48,4,52,105,12,119,12,1,94,0,26,96,55,105,29,82,43,7,79,36,89,101,83,4,122];
The XOR block has 31 entries and goes in a loop .
And then it begins with:
CSGMDL\x02 ...
CSGv3 unions are similar, but have \x04
instead of \x02
.
CSGv2
#[binrw::binrw]
#[brw(little)]
#[derive(Debug,Clone)]
pub struct MeshData2{
#[brw(magic=b"CSGMDL\x02\0\0\0")]
pub hash:Hash,
pub mesh:Mesh2,
}
CSGv3
#[binrw::binrw]
#[brw(little)]
#[derive(Debug,Clone)]
pub struct MeshData4{
#[brw(magic=b"CSGMDL\x04\0\0\0")]
pub hash:Hash,
pub mesh:Mesh2,
pub _unknown1_count:u32,
#[br(count=_unknown1_count)]
pub _unknown1_list:Vec<u32>,
}
What does Mesh2
mean? Glad you asked:
#[binrw::binrw]
#[brw(little)]
#[derive(Debug,Clone)]
pub struct Mesh2{
pub vertex_count:u32,
// vertex data length
#[brw(magic=84u32)]
#[br(count=vertex_count)]
pub vertices:Vec<Vertex>,
pub face_count:u32,
#[br(count=face_count/3)]
pub faces:Vec<[VertexId;3]>,
}
Note how similar their structures look. Both MeshData2
and MeshData4
begin with a magic header, then the same Hash
, then the same Mesh2
.
Only difference being that aside from that is the presence of some unknown values after all the faces
in MeshData4
. Even Krakow10 doesn’t know yet.
More research: