/**************************************************************************** ** gmio benchmarks ** Copyright Fougue (2 Mar. 2015) ** contact@fougue.pro ** ** This software provides performance benchmarks for the gmio library ** (https://github.com/fougue/gmio) ** ** 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 #include #include #include #include #include #include #include "../commons/benchmark_tools.h" #include struct gmio_stl_triangle; struct my_igeom { uint32_t facet_count; }; static void dummy_process_triangle( void* cookie, uint32_t triangle_id, const struct gmio_stl_triangle* triangle) { struct my_igeom* my_igeom = (struct my_igeom*)(cookie); GMIO_UNUSED(triangle_id); GMIO_UNUSED(triangle); if (my_igeom != NULL) ++(my_igeom->facet_count); } static void bmk_gmio_stl_read(const void* filepath) { struct my_igeom cookie = {0}; struct gmio_stl_mesh_creator mesh_creator = {0}; int error = GMIO_ERROR_OK; mesh_creator.cookie = &cookie; mesh_creator.func_add_triangle = dummy_process_triangle; error = gmio_stl_read_file(filepath, mesh_creator, NULL); if (error != GMIO_ERROR_OK) printf("gmio error: 0x%X\n", error); } static enum gmio_endianness to_byte_order(enum gmio_stl_format format) { switch (format) { case GMIO_STL_FORMAT_BINARY_LE: return GMIO_ENDIANNESS_LITTLE; case GMIO_STL_FORMAT_BINARY_BE: return GMIO_ENDIANNESS_BIG; case GMIO_STL_FORMAT_ASCII: case GMIO_STL_FORMAT_UNKNOWN: return GMIO_ENDIANNESS_HOST; } return GMIO_ENDIANNESS_UNKNOWN; } enum { STL_TRIANGLE_ARRAY_SIZE = 512 }; struct stl_readwrite_conv { struct gmio_stream stream; struct gmio_streampos out_stream_pos_begin; enum gmio_stl_format in_format; enum gmio_stl_format out_format; struct gmio_stl_triangle triangle_array[STL_TRIANGLE_ARRAY_SIZE]; unsigned triangle_pos; uint32_t total_triangle_count; }; static void readwrite_begin_solid( void* cookie, const struct gmio_stl_mesh_creator_infos* infos) { struct stl_readwrite_conv* rw_conv = (struct stl_readwrite_conv*)cookie; struct gmio_stream* stream = &rw_conv->stream; if (infos->format == GMIO_STL_FORMAT_ASCII) { if (rw_conv->out_format == GMIO_STL_FORMAT_ASCII) { stream->func_write(stream->cookie, "solid ", 1, 6); stream->func_write( stream->cookie, infos->stla_solid_name, sizeof(char), strlen(infos->stla_solid_name)); stream->func_write(stream->cookie, "\n", 1, 1); } else { /* For binary STL, facet count <- 0 because it cannot be known at this * point. Header will be correctly rewritten at the end of the read * procedure (in gmio_stl_mesh_creator::func_end_solid() callback) */ const enum gmio_endianness byte_order = to_byte_order(rw_conv->out_format); gmio_stlb_write_header(stream, byte_order, NULL, 0); } } else { if (rw_conv->out_format == GMIO_STL_FORMAT_ASCII) { stream->func_write(stream->cookie, "solid\n", 1, 6); } else { gmio_stlb_write_header( stream, to_byte_order(rw_conv->out_format), infos->stlb_header, infos->stlb_triangle_count); } } } static void readwrite_get_triangle( const void* cookie, uint32_t tri_id, struct gmio_stl_triangle* triangle) { const struct gmio_stl_triangle* tri_array = (const struct gmio_stl_triangle*)cookie; *triangle = tri_array[tri_id]; } static void stl_readwrite_flush_triangles(struct stl_readwrite_conv* rw_conv) { struct gmio_stl_mesh mesh = {0}; struct gmio_stl_write_options options = {0}; mesh.cookie = &rw_conv->triangle_array[0]; mesh.triangle_count = rw_conv->triangle_pos; mesh.func_get_triangle = &readwrite_get_triangle; options.stl_write_triangles_only = true; options.stla_float32_format = GMIO_FLOAT_TEXT_FORMAT_SCIENTIFIC_LOWERCASE; options.stla_float32_prec = 6; gmio_stl_write(rw_conv->out_format, rw_conv->stream, mesh, &options); rw_conv->triangle_pos = 0; } static void readwrite_add_triangle( void* cookie, uint32_t tri_id, const struct gmio_stl_triangle* triangle) { struct stl_readwrite_conv* rw_conv = (struct stl_readwrite_conv*)cookie; if (rw_conv->triangle_pos >= STL_TRIANGLE_ARRAY_SIZE) stl_readwrite_flush_triangles(rw_conv); rw_conv->triangle_array[rw_conv->triangle_pos] = *triangle; ++(rw_conv->triangle_pos); rw_conv->total_triangle_count = tri_id + 1; } static void readwrite_end_solid(void* cookie) { struct stl_readwrite_conv* rw_conv = (struct stl_readwrite_conv*)cookie; struct gmio_stream* stream = &rw_conv->stream; if (rw_conv->triangle_pos != 0) stl_readwrite_flush_triangles(rw_conv); if (rw_conv->out_format == GMIO_STL_FORMAT_ASCII) { stream->func_write(stream->cookie, "endsolid", 1, 8); } else if (rw_conv->in_format == GMIO_STL_FORMAT_ASCII) { const enum gmio_endianness byte_order = to_byte_order(rw_conv->out_format); /* The total facet count has to be written because it wasn't known at * the beginning of the read procedure */ stream->func_set_pos(stream->cookie, &rw_conv->out_stream_pos_begin); gmio_stlb_write_header( stream, byte_order, NULL, rw_conv->total_triangle_count); } } static void bmk_gmio_stl_readwrite_conv(const void* filepath) { FILE* infile = fopen(filepath, "rb"); FILE* outfile = fopen("_readwrite_conv.stl", "wb"); struct gmio_stream instream = {0}; struct stl_readwrite_conv rw_conv = {0}; struct gmio_stl_mesh_creator mesh_creator = {0}; int error = GMIO_ERROR_OK; /* rw_conv.out_format = GMIO_STL_FORMAT_BINARY_LE; */ rw_conv.out_format = GMIO_STL_FORMAT_ASCII; if (infile != NULL) { instream = gmio_stream_stdio(infile); rw_conv.in_format = gmio_stl_get_format(&instream); } if (outfile != NULL) { rw_conv.stream = gmio_stream_stdio(outfile); rw_conv.stream.func_get_pos( rw_conv.stream.cookie, &rw_conv.out_stream_pos_begin); } mesh_creator.cookie = &rw_conv; mesh_creator.func_begin_solid = &readwrite_begin_solid; mesh_creator.func_add_triangle = &readwrite_add_triangle; mesh_creator.func_end_solid = &readwrite_end_solid; error = gmio_stl_read(instream, mesh_creator, NULL); if (error != GMIO_ERROR_OK) printf("gmio error: 0x%X\n", error); fclose(infile); fclose(outfile); } void bmk_gmio_stl_infos_get_all(const void* filepath) { static bool already_exec = false; FILE* file = fopen(filepath, "rb"); if (file != NULL) { const struct gmio_stream stream = gmio_stream_stdio(file); struct gmio_stl_infos infos = {0}; gmio_stl_infos_get(&infos, stream, GMIO_STL_INFO_FLAG_ALL, NULL); if (!already_exec) { printf("stl_infos_get(ALL)\n" " File: %s\n" " Size: %uKo\n" " Facets: %u\n", (const char*)filepath, infos.size / 1024, infos.facet_count); if (infos.format == GMIO_STL_FORMAT_ASCII) { printf(" [STLA]Solid name: %s\n", infos.stla_solidname); } else if (infos.format == GMIO_STL_FORMAT_BINARY_LE || infos.format == GMIO_STL_FORMAT_BINARY_BE) { printf(" [STLB]Header: %80.80s\n", infos.stlb_header.data); } } already_exec = true; } fclose(file); } void bmk_gmio_stl_infos_get_size(const void* filepath) { static bool already_exec = false; FILE* file = fopen(filepath, "rb"); if (file != NULL) { const struct gmio_stream stream = gmio_stream_stdio(file); struct gmio_stl_infos infos = {0}; gmio_stl_infos_get(&infos, stream, GMIO_STL_INFO_FLAG_SIZE, NULL); if (!already_exec) { printf("stl_infos_get(SIZE)\n" " File: %s\n" " Size: %uKo\n", (const char*)filepath, infos.size / 1024); } already_exec = true; } fclose(file); } int main(int argc, char** argv) { if (argc > 1) { const char* filepath = argv[1]; /* Declare benchmarks */ struct benchmark_cmp_arg cmp_args[] = { { "read_file()", bmk_gmio_stl_read, NULL, NULL, NULL }, { "readwrite_conv()", bmk_gmio_stl_readwrite_conv, NULL, NULL, NULL }, { "stl_infos_get(ALL)", bmk_gmio_stl_infos_get_all, NULL, NULL, NULL }, { "stl_infos_get(size)", bmk_gmio_stl_infos_get_size, NULL, NULL, NULL }, {0} }; const size_t cmp_count = GMIO_ARRAY_SIZE(cmp_args) - 1; struct benchmark_cmp_result cmp_res[GMIO_ARRAY_SIZE(cmp_args)] = {0}; struct benchmark_cmp_result_array res_array = {0}; const struct benchmark_cmp_result_header header = { "gmio", NULL }; { size_t i = 0; for (i = 0; i < cmp_count; ++i) cmp_args[i].func1_arg = filepath; } res_array.ptr = &cmp_res[0]; res_array.count = cmp_count; /* Execute benchmarks */ benchmark_cmp_batch(5, cmp_args, cmp_res, NULL, NULL); /* Print results */ benchmark_print_results( BENCHMARK_PRINT_FORMAT_MARKDOWN, header, res_array); } return 0; }