This repository contains code for comparison of various libraries for input and output of PLY files. PLY file format has been developed at Stanford University by Greg Turk as a part of the real-world object digitization project undertaken by Stanford University in mid-90s.
The task is to read and write a basic triangle mesh stored in a PLY. The triangle mesh is represented as a list of vertices and a list of triangles, indicating which triplets of vertices make a triangle.
The data structure we wish to populate is:
typedef struct vec3f
{
float x,y,z;
} Vec3f;
typedef struct tri
{
int index1, index2, index3;
} Tri;
typedef struct triangle_mesh
{
int n_verts;
int n_faces;
Vec3f* vertices;
Tri* faces;
} TriMesh;
The meshes we are testing were processed to contain only the position attribute and only triangle faces. As such this task aims to measure the speed at which each library is able to exchange simple triangular mesh data. Each of the meshes is tested using both binary and text(ASCII) formats, as supported by PLY file format. For binary we use little-endian.
This benchmark focuses on rather large meshes (15k - 28 million triangles). The use case this benchmark analyzes is to minimize the time taken to load such large meshes. If your task is to read a lot of smaller .ply files, then this benchmark might not be reflective of your situation.
For an alternative task, where a large number of smaller meshes is parsed, and where meshes might have more varied per-vertex attribute list, please see the excellent ply-parsing-perf benchmark by Vilya Harvey.
Given that in our task we process only triangular meshes, it would be good to let the application know this information. Some libraries (see below) allow passing the expected size of list properties, leading to non-negligible speed-up in parsing. As such, where applicable, this feature has been enabled.
The table below lists models used for this benchmark, along with the source.
Model Name | N. Vertices | N. Tris | Source |
---|---|---|---|
suzanne | 7958 | 15744 | Blender |
scannet_scene0402_00 | 93834 | 177518 | Scannet |
angel | 237018 | 474048 | Large Geometric Models Archvive |
blade | 882954 | 1765388 | Large Geometric Models Archvive |
hand | 327323 | 654666 | Large Geometric Models Archvive |
horse | 48485 | 96966 | Large Geometric Models Archvive |
armadillo | 172974 | 345944 | Stanford 3D Scaning Repository |
bunny | 35947 | 69451 | Stanford 3D Scaning Repository |
dragon | 437645 | 871414 | Stanford 3D Scaning Repository |
happy_buddha | 543652 | 1087716 | Stanford 3D Scaning Repository |
lucy | 14027872 | 28055742 | Stanford 3D Scaning Repository |
xyzrgb_dragon | 3609600 | 7219045 | Stanford 3D Scaning Repository |
xyzrgb_statuette | 4999996 | 10000000 | Stanford 3D Scaning Repository |
bust_of_sappho | 140864 | 281724 | Thingiverse |
statue | 999517 | 1999038 | Sketchfab |
speeder_bike | 1473341 | 2947046 | Sketchfab |
armchair | 11558 | 23102 | Sketchfab |
bust_of_angelique_dhannetaire | 250000 | 500000 | Sketchfab |
Below is a list of libraries used in this benchmark:
Library | Author | Language | Known list size | Notes |
---|---|---|---|---|
turkply | Greg Turk | c | ❌ | Original PLY library link |
rply | Diego Nehab | c | ❌ | |
msh_ply | Maciej Halber | c | ✔️ | |
happly | Nicolas Sharp | c++ | ❌ | |
miniply | Vilya Harvey | c++ | ✔️ | Only supports reading PLY files |
micro_ply | Nick Klingensmith | c++ | ❌ | Only supports reading ASCII PLY files |
nanoply | vcglib | c++ | ❌ | |
plylib | vcglib | c++ | ❌ | PLY reading/writing used by Meshlab(?) |
tinyply | Dimitri Diakopoulos | c++ | ✔️ | This benchmark includes versions 2.1, 2.2 and 2.3 of this library. |
For the usage examples, as well as some additional comments about each of the libraries please check the tests/*_test.c(pp) files.
Below we present results for parsing PLY files storing data in both ASCII and binary format (little-endian). Times are given in milliseconds. Highlighted numbers indicate the best method in each category. As noted before, where applicable, a known list size is passed to the library.
The benchmark was compiled using MSVC 19.28.29334 with \O2 optimization flag, using AMD Ryzen 3900XT and Samsung 970 EVO PLUS.
To run the test, we run a separate program for each file that attempts to read and write the input file, and reports time taken to do so. Program for each library is run 10 times and the results are averaged. The averaged time taken for each model is used to compute the overall average time it took to process all the models.
- Disclaimer: I am the author of msh_ply library. If you see any deficiencies in code for other libraries, don't hesitate to let me know - I hope to make this benchmark as fair as possible. *
Method | ASCII | Binary |
---|---|---|
happly | 19104.671(75.3x) | 589.435(16.4x) |
micro_ply | 1131.752(4.5x) | N/A |
miniply | 253.671(1.0x) | 35.935(1.0x) |
msh_ply | 2009.957(7.9x) | 40.885(1.1x) |
nanoply | 8003.712(31.6x) | 106.312(3.0x) |
plylib | 3157.350(12.4x) | 338.514(9.4x) |
rply | 1731.580(6.8x) | 327.164(9.1x) |
tinyply21 | 11583.986(45.7x) | 1844.445(51.3x) |
tinyply22 | 7561.799(29.8x) | 318.069(8.9x) |
tinyply23 | 7500.844(29.6x) | 294.265(8.2x) |
turkply | 2086.552(8.2x) | 549.367(15.3x) |
Method | ASCII | Binary |
---|---|---|
happly | 11534.080(3.8x) | 1454.963(19.8x) |
msh_ply | 4178.405(1.4x) | 73.406(1.0x) |
nanoply | 8772.179(2.9x) | 107.735(1.5x) |
plylib | 3045.147(1.0x) | 315.647(4.3x) |
rply | 3966.512(1.3x) | 261.940(3.6x) |
tinyply21 | 9667.221(3.2x) | 1449.753(19.7x) |
tinyply22 | 9870.520(3.2x) | 526.407(7.2x) |
tinyply23 | 9653.622(3.2x) | 560.677(7.6x) |
turkply | 4017.640(1.3x) | 624.668(8.5x) |
Notes:
- miniply is the fastest library for reading the ply files. If you're only interested in reading files and use C++, it is a great choice.
- miniply and micro_ply do not support the writing of ply files.
- micro_ply does not support binary files, only ASCII format.
- In C, when you need decent read and write performance, msh_ply is a good choice ;). However, it's ASCII mode requires work, so if your models are mostly stored in ASCII, you might want to use other libraries.
- Some libraries were modified to include getter to establish whether input is binary or ASCII.
- In ASCII mode, happly is unable to convert between uint and int. Since some models (angel, bust_of_sappho, bust_of_angelique_dhannetaire ) contain vertex list specified as uint, while others use int, happly fails to parse the two aforementioned models and would need to be recompiled to support the specific type. Here, we simply omit these models when benchmarking happly.
ASCII | Read Times Table | Read Times Image | Write Times Table | Write Times Image |
Binary | Read Times Table | Read Times Image | Write Times Table | Write Times Image |
Note that the images show the read time on a log scale, since the performance of different libraries is significantly different.
Another metric we can use for deciding a library is the ease of use. Why LOC is by no means a perfect metric to measure ease of use, it does reflect how much code one needs to type to get basic PLY I/O done. Also, note that these numbers report only simple versions of reading function without any error reporting, etc.
Library | Read LOC | Write LOC |
---|---|---|
miniply | 35 | N/A |
micro_ply | 25 | N/A |
msh_ply | 29 | 23 |
nanoply | 23 | 29 |
plylib | 78 | 65 |
rply | 69 | 23 |
happly | 17 | 26 |
tinyply | 17 | 10 |
turkply | 52 | 39 |