gmio/tests/test_stl_io.c

464 lines
16 KiB
C
Raw Normal View History

2015-05-28 23:33:42 +08:00
/****************************************************************************
** gmio
** Copyright Fougue (2 Mar. 2015)
2015-07-13 17:42:03 +08:00
** contact@fougue.pro
2015-05-28 23:33:42 +08:00
**
** This software is a reusable library whose purpose is to provide complete
** I/O support for various CAD file formats (eg. STL)
**
** This software is governed by the CeCILL-B license under French law and
** abiding by the rules of distribution of free software. You can use,
** modify and/ or redistribute the software under the terms of the CeCILL-B
** license as circulated by CEA, CNRS and INRIA at the following URL
** "http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.html".
****************************************************************************/
#include "utest_assert.h"
2015-09-14 17:25:29 +08:00
#include "core_utils.h"
#include "stl_utils.h"
2015-05-28 23:33:42 +08:00
#include "../src/gmio_core/error.h"
#include "../src/gmio_core/internal/min_max.h"
#include "../src/gmio_core/internal/string.h"
2015-05-28 23:33:42 +08:00
#include "../src/gmio_stl/stl_error.h"
#include "../src/gmio_stl/stl_infos.h"
2015-05-28 23:33:42 +08:00
#include "../src/gmio_stl/stl_io.h"
#include "../src/gmio_stl/stl_io_options.h"
2015-05-28 23:33:42 +08:00
#include <stddef.h>
#include <stdlib.h>
static const char stl_grabcad_arm11_filepath[] =
2015-09-11 22:36:03 +08:00
"models/solid_grabcad_arm11_link0_hb.le_stlb";
struct stl_testcase_result
{
char solid_name[2048];
};
void stl_testcase_result__begin_solid(
void* cookie, const struct gmio_stl_mesh_creator_infos* infos)
{
if (infos->format == GMIO_STL_FORMAT_ASCII) {
struct stl_testcase_result* res = (struct stl_testcase_result*)cookie;
if (res != NULL) {
res->solid_name[0] = 0;
if (infos->stla_solid_name != NULL)
gmio_cstr_copy(
res->solid_name,
sizeof(res->solid_name),
infos->stla_solid_name,
strlen(infos->stla_solid_name));
}
}
}
struct stl_testcase
2015-05-28 23:33:42 +08:00
{
const char* filepath;
int errorcode;
enum gmio_stl_format format;
const char* solid_name;
2015-05-28 23:33:42 +08:00
};
const char* test_stl_read()
{
2015-12-10 01:51:03 +08:00
const struct stl_testcase expected[] = {
2016-01-26 18:50:55 +08:00
{ "models/file_empty",
GMIO_STL_ERROR_UNKNOWN_FORMAT,
GMIO_STL_FORMAT_UNKNOWN,
NULL
},
{ "models/solid_4vertex.stla",
GMIO_STL_ERROR_PARSING,
GMIO_STL_FORMAT_ASCII,
NULL
},
{ "models/solid_anonymous_empty.stla",
GMIO_ERROR_OK,
GMIO_STL_FORMAT_ASCII,
NULL
},
{ "models/solid_empty.stla",
GMIO_ERROR_OK,
GMIO_STL_FORMAT_ASCII,
"emptysolid"
},
{ "models/solid_empty.stlb",
GMIO_ERROR_OK,
GMIO_STL_FORMAT_BINARY_LE,
NULL
},
{ "models/solid_empty_name_many_words.stla",
GMIO_ERROR_OK,
GMIO_STL_FORMAT_ASCII,
"name with multiple words"
},
{ "models/solid_empty_name_many_words_single_letters.stla",
GMIO_ERROR_OK,
GMIO_STL_FORMAT_ASCII,
"a b c d e f\t\tg h"
2016-01-26 18:50:55 +08:00
},
{ "models/solid_grabcad_arm11_link0_hb.le_stlb",
GMIO_ERROR_OK,
GMIO_STL_FORMAT_BINARY_LE,
NULL
},
{ "models/solid_jburkardt_sphere.stla",
GMIO_ERROR_OK,
GMIO_STL_FORMAT_ASCII,
"sphere"
},
{ "models/solid_lack_z.stla",
GMIO_STL_ERROR_PARSING,
GMIO_STL_FORMAT_ASCII,
NULL
},
{ "models/solid_one_facet.stla",
GMIO_ERROR_OK,
GMIO_STL_FORMAT_ASCII,
NULL
},
{ "models/solid_one_facet.le_stlb",
GMIO_ERROR_OK,
GMIO_STL_FORMAT_BINARY_LE,
NULL
},
{ "models/solid_one_facet.be_stlb",
GMIO_ERROR_OK,
GMIO_STL_FORMAT_BINARY_BE,
NULL
},
{ "models/solid_one_facet_uppercase.stla",
GMIO_ERROR_OK,
GMIO_STL_FORMAT_ASCII,
NULL
}
2015-05-28 23:33:42 +08:00
};
const size_t expected_count =
2015-12-10 01:51:03 +08:00
sizeof(expected) / sizeof(struct stl_testcase);
2015-05-29 00:02:26 +08:00
size_t i; /* for loop counter */
2016-01-29 19:47:01 +08:00
struct gmio_stl_mesh_creator mesh_creator = {0};
2015-12-10 01:51:03 +08:00
struct stl_testcase_result result = {0};
2015-05-28 23:33:42 +08:00
2016-01-29 19:47:01 +08:00
mesh_creator.cookie = &result;
mesh_creator.func_begin_solid = &stl_testcase_result__begin_solid;
2016-01-29 19:47:01 +08:00
mesh_creator.func_add_triangle = &gmio_stl_nop_add_triangle;
2015-05-28 23:33:42 +08:00
2015-05-29 00:02:26 +08:00
for (i = 0; i < expected_count; ++i) {
const enum gmio_stl_format format =
gmio_stl_get_format_file(expected[i].filepath);
const int err =
gmio_stl_read_file(expected[i].filepath, &mesh_creator, NULL);
/* Check format */
if (format != expected[i].format) {
printf("\nfilepath : %s\n"
"expected format : %d\n"
"actual format : %d\n",
expected[i].filepath,
expected[i].format,
format);
}
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_UINT(expected[i].format, format);
/* Check error code */
2015-05-28 23:33:42 +08:00
if (err != expected[i].errorcode) {
printf("\nfilepath : %s\n"
"expected error : 0x%x\n"
"actual error : 0x%x\n",
expected[i].filepath,
expected[i].errorcode,
err);
}
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_UINT(expected[i].errorcode, err);
/* Check solid name */
if (expected[i].format == GMIO_STL_FORMAT_ASCII) {
const char* expected_name =
expected[i].solid_name != NULL ? expected[i].solid_name : "";
if (strcmp(result.solid_name, expected_name) != 0) {
printf("\nfilepath : %s\n"
"expected solidname : %s\n"
"actual solidname : %s\n",
expected[i].filepath,
expected_name,
result.solid_name);
}
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_CSTR(expected_name, result.solid_name);
}
2015-05-28 23:33:42 +08:00
}
return NULL;
}
const char* test_stlb_write_header()
{
const char* filepath = "temp/solid.stlb";
struct gmio_stlb_header header = {0};
const char* header_str = "temp/solid.stlb generated with gmio library";
int error = GMIO_ERROR_OK;
{
FILE* outfile = fopen(filepath, "wb");
struct gmio_stream stream = gmio_stream_stdio(outfile);
memcpy(&header,
header_str,
GMIO_MIN(GMIO_STLB_HEADER_SIZE, strlen(header_str)));
error = gmio_stlb_write_header(
&stream, GMIO_ENDIANNESS_LITTLE, &header, 0);
fclose(outfile);
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_INT(GMIO_ERROR_OK, error);
}
{
struct gmio_stl_data data = {0};
struct gmio_stl_mesh_creator creator = gmio_stl_data_mesh_creator(&data);
error = gmio_stl_read_file(filepath, &creator, NULL);
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_INT(GMIO_ERROR_OK, error);
UTEST_ASSERT(gmio_stlb_header_equal(&header, &data.header));
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_UINT(0, data.tri_array.count);
}
return NULL;
}
/* Safely closes the two files \p f1 and \p f2 */
static void fclose_2(FILE* f1, FILE* f2)
{
if (f1 != NULL)
fclose(f1);
if (f2 != NULL)
fclose(f2);
}
const char* test_stlb_write()
{
const char* model_fpath = stl_grabcad_arm11_filepath;
const char* model_fpath_out = "temp/solid.le_stlb";
const char* model_fpath_out_be = "temp/solid.be_stlb";
struct gmio_stl_data data = {0};
int error = GMIO_ERROR_OK;
/* Read input model file */
{
struct gmio_stl_mesh_creator creator = gmio_stl_data_mesh_creator(&data);
error = gmio_stl_read_file(model_fpath, &creator, NULL);
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_INT(GMIO_ERROR_OK, error);
}
/* Write back input model file
* Write also the model file in big-endian STL format
*/
{
const struct gmio_stl_mesh mesh = gmio_stl_data_mesh(&data);
2016-01-29 19:47:01 +08:00
struct gmio_stl_write_options opts = {0};
opts.stlb_header = data.header;
2015-12-17 19:31:30 +08:00
error = gmio_stl_write_file(
GMIO_STL_FORMAT_BINARY_LE, model_fpath_out, &mesh, &opts);
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_INT(GMIO_ERROR_OK, error);
/* Big-endian version */
2015-12-17 19:31:30 +08:00
error = gmio_stl_write_file(
GMIO_STL_FORMAT_BINARY_BE, model_fpath_out_be, &mesh, &opts);
}
/* Check input and output models are equal */
{
uint8_t buffer_in[2048] = {0};
uint8_t buffer_out[2048] = {0};
const size_t buff_size = 2048;
size_t bytes_read_in = 0;
size_t bytes_read_out = 0;
FILE* in = fopen(model_fpath, "rb");
FILE* out = fopen(model_fpath_out, "rb");
if (in == NULL || out == NULL) {
fclose_2(in, out);
perror("test_stlb_write()");
UTEST_FAIL("fopen() error for in/out model files");
}
do {
2016-01-29 19:47:01 +08:00
bytes_read_in = fread(buffer_in, 1, buff_size, in);
bytes_read_out = fread(buffer_out, 1, buff_size, out);
if (bytes_read_in != bytes_read_out) {
fclose_2(in, out);
UTEST_FAIL("Different byte count between in/out");
}
2016-01-29 19:47:01 +08:00
if (memcmp(buffer_in, buffer_out, buff_size) != 0) {
fclose_2(in, out);
UTEST_FAIL("Different buffer contents between in/out");
}
} while (!feof(in) && !feof(out)
&& bytes_read_in > 0 && bytes_read_out > 0);
fclose_2(in, out);
}
/* Check output LE/BE models are equal */
{
struct gmio_stl_data data_be = {0};
struct gmio_stl_mesh_creator creator = gmio_stl_data_mesh_creator(&data_be);
error = gmio_stl_read_file(model_fpath_out_be, &creator, NULL);
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_INT(GMIO_ERROR_OK, error);
UTEST_ASSERT(gmio_stlb_header_equal(&data.header, &data_be.header));
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_UINT(data.tri_array.count, data_be.tri_array.count);
UTEST_ASSERT(memcmp(data.tri_array.ptr,
data_be.tri_array.ptr,
data.tri_array.count * sizeof(struct gmio_stl_triangle))
== 0);
free(data_be.tri_array.ptr);
}
free(data.tri_array.ptr);
return NULL;
2015-05-28 23:33:42 +08:00
}
2015-09-11 22:36:03 +08:00
const char* test_stla_write()
{
const char* model_filepath = stl_grabcad_arm11_filepath;
const char* model_filepath_out = "temp/solid.stla";
struct gmio_stl_data data = {0}; /* TODO: fix memory leak on error */
2015-09-11 22:36:03 +08:00
char header_str[GMIO_STLB_HEADER_SIZE + 1] = {0};
int error = GMIO_ERROR_OK;
/* Read input model file */
{
struct gmio_stl_mesh_creator creator = gmio_stl_data_mesh_creator(&data);
error = gmio_stl_read_file(model_filepath, &creator, NULL);
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_INT(GMIO_ERROR_OK, error);
2015-09-11 22:36:03 +08:00
}
/* Write the model to STL ascii format */
{
2016-01-29 19:47:01 +08:00
struct gmio_stl_write_options opts = {0};
const struct gmio_stl_mesh mesh = gmio_stl_data_mesh(&data);
2016-01-29 19:47:01 +08:00
gmio_stlb_header_to_printable_str(&data.header, header_str, '_');
opts.stla_solid_name = header_str;
opts.stla_float32_prec = 7;
opts.stla_float32_format = GMIO_FLOAT_TEXT_FORMAT_SHORTEST_LOWERCASE;
2015-12-17 19:31:30 +08:00
error = gmio_stl_write_file(
GMIO_STL_FORMAT_ASCII, model_filepath_out, &mesh, &opts);
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_INT(GMIO_ERROR_OK, error);
2015-09-11 22:36:03 +08:00
}
/* Read the output STL ascii model */
{
char trim_header_str[sizeof(header_str)] = {0};
struct gmio_stl_data data_stla = {0};
struct gmio_stl_mesh_creator creator =
gmio_stl_data_mesh_creator(&data_stla);
2015-09-11 22:36:03 +08:00
size_t i = 0;
gmio_cstr_copy(
trim_header_str,
sizeof(trim_header_str),
header_str,
sizeof(header_str));
2015-09-11 22:36:03 +08:00
gmio_string_trim_from_end(trim_header_str, sizeof(header_str));
error = gmio_stl_read_file(model_filepath_out, &creator, NULL);
2016-01-29 19:47:01 +08:00
UTEST_COMPARE_INT(GMIO_ERROR_OK, error);
UTEST_COMPARE_UINT(data.tri_array.count, data_stla.tri_array.count);
UTEST_COMPARE_CSTR(trim_header_str, data_stla.solid_name);
2015-09-11 22:36:03 +08:00
for (i = 0; i < data.tri_array.count; ++i) {
const struct gmio_stl_triangle* lhs = &data.tri_array.ptr[i];
const struct gmio_stl_triangle* rhs = &data_stla.tri_array.ptr[i];
2016-01-27 00:03:58 +08:00
const bool tri_equal = gmio_stl_triangle_equal(lhs, rhs, 5);
2015-09-11 22:36:03 +08:00
UTEST_ASSERT(tri_equal);
}
}
return NULL;
}
const char* generic_test_stl_read_multi_solid(
const char* filepath, unsigned expected_solid_count)
{
FILE* infile = fopen(filepath, "rb");
if (infile != NULL) {
unsigned solid_count = 0;
int error = GMIO_ERROR_OK;
struct gmio_stream stream = gmio_stream_stdio(infile);
struct gmio_stl_read_options roptions = {0};
struct gmio_stl_mesh_creator null_creator = {0};
roptions.func_stla_get_streamsize = gmio_stla_infos_get_streamsize;
while (gmio_no_error(error)) {
error = gmio_stl_read(&stream, &null_creator, &roptions);
if (gmio_no_error(error))
++solid_count;
}
fclose(infile);
UTEST_COMPARE_UINT(expected_solid_count, solid_count);
}
else {
perror(NULL);
UTEST_FAIL("");
}
return NULL;
}
const char* test_stl_read_multi_solid()
{
const char* res = NULL;
res = generic_test_stl_read_multi_solid("models/solid_4meshs.stla", 4);
if (res != NULL)
return res;
res = generic_test_stl_read_multi_solid("models/solid_4meshs.le_stlb", 4);
return res;
}
2015-05-28 23:33:42 +08:00
void generate_stlb_tests_models()
{
{
FILE* outfile = fopen("models/solid_empty.stlb", "wb");
struct gmio_stream stream = gmio_stream_stdio(outfile);
2015-05-28 23:33:42 +08:00
gmio_stlb_write_header(&stream, GMIO_ENDIANNESS_LITTLE, NULL, 0);
fclose(outfile);
}
{
const char model_fpath_le[] = "models/solid_one_facet.le_stlb";
const char model_fpath_be[] = "models/solid_one_facet.be_stlb";
2015-12-10 01:51:03 +08:00
struct gmio_stl_triangle tri = {
{ 0.f, 0.f, 1.f }, /* normal */
{ 0.f, 0.f, 0.f }, /* v1 */
{ 10.f, 0.f, 0.f }, /* v2 */
{ 5.f, 10.f, 0.f }, /* v3 */
0 /* attr */
};
struct gmio_stl_data data = {0};
struct gmio_stl_mesh mesh = {0};
2015-12-10 01:51:03 +08:00
data.tri_array.ptr = &tri;
data.tri_array.count = 1;
mesh = gmio_stl_data_mesh(&data);
2015-12-10 01:51:03 +08:00
2015-12-17 19:31:30 +08:00
gmio_stl_write_file(
GMIO_STL_FORMAT_BINARY_LE, model_fpath_le, &mesh, NULL);
2015-12-17 19:31:30 +08:00
gmio_stl_write_file(
GMIO_STL_FORMAT_BINARY_BE, model_fpath_be, &mesh, NULL);
2015-05-28 23:33:42 +08:00
}
{
FILE* infile = fopen("models/solid_4meshs.stla", "rb");
FILE* outfile = fopen("models/solid_4meshs.le_stlb", "wb");
int read_error = GMIO_ERROR_OK;
struct gmio_stream istream = gmio_stream_stdio(infile);
struct gmio_stream ostream = gmio_stream_stdio(outfile);
struct gmio_stl_read_options ropts = {0};
ropts.func_stla_get_streamsize = gmio_stla_infos_get_streamsize;
while (gmio_no_error(read_error)) {
struct gmio_stl_data data = {0};
struct gmio_stl_mesh_creator creator = gmio_stl_data_mesh_creator(&data);
struct gmio_stl_mesh mesh = {0};
struct gmio_stl_write_options wopts = {0};
read_error = gmio_stla_read(&istream, &creator, &ropts);
mesh = gmio_stl_data_mesh(&data);
wopts.stlb_header = gmio_stlb_header_str(data.solid_name);
gmio_stl_write(GMIO_STL_FORMAT_BINARY_LE, &ostream, &mesh, &wopts);
gmio_stl_triangle_array_free(&data.tri_array);
}
fclose(infile);
fclose(outfile);
}
2015-05-28 23:33:42 +08:00
}