// Copyright 2016 The Draco Authors. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // #include #include #include "compression/decode.h" #include "core/cycle_timer.h" #include "io/obj_encoder.h" #include "io/ply_encoder.h" namespace { struct Options { Options(); std::string input; std::string output; }; Options::Options() {} void Usage() { printf("Usage: draco_decoder [options] -i input\n"); printf("\n"); printf("Main options:\n"); printf(" -h | -? show help.\n"); printf(" -o output file name.\n"); } } int main(int argc, char **argv) { Options options; const int argc_check = argc - 1; for (int i = 1; i < argc; ++i) { if (!strcmp("-h", argv[i]) || !strcmp("-?", argv[i])) { Usage(); return 0; } else if (!strcmp("-i", argv[i]) && i < argc_check) { options.input = argv[++i]; } else if (!strcmp("-o", argv[i]) && i < argc_check) { options.output = argv[++i]; } } if (argc < 3 || options.input.empty()) { Usage(); return -1; } std::ifstream input_file(options.input, std::ios::binary); if (!input_file) { printf("Failed opening the input file.\n"); return -1; } // Read the file stream into a buffer. std::streampos file_size = 0; input_file.seekg(0, std::ios::end); file_size = input_file.tellg() - file_size; input_file.seekg(0, std::ios::beg); std::vector data(file_size); input_file.read(data.data(), file_size); if (data.empty()) { printf("Empty input file.\n"); return -1; } // Create a draco decoding buffer. Note that no data is copied in this step. draco::DecoderBuffer buffer; buffer.Init(data.data(), data.size()); draco::CycleTimer timer; // Decode the input data into a geometry. std::unique_ptr pc; draco::Mesh *mesh = nullptr; const draco::EncodedGeometryType geom_type = draco::GetEncodedGeometryType(&buffer); if (geom_type == draco::TRIANGULAR_MESH) { timer.Start(); std::unique_ptr in_mesh = draco::DecodeMeshFromBuffer(&buffer); timer.Stop(); if (in_mesh) { mesh = in_mesh.get(); pc = std::move(in_mesh); } } else if (geom_type == draco::POINT_CLOUD) { // Failed to decode it as mesh, so let's try to decode it as a point cloud. timer.Start(); pc = draco::DecodePointCloudFromBuffer(&buffer); timer.Stop(); } if (pc == nullptr) { printf("Failed to decode the input file.\n"); return -1; } if (options.output.empty()) { // Save the output model into a ply file. options.output = options.input + ".ply"; } // Save the decoded geometry into a file. // TODO(ostava): Currently only .ply and .obj are supported. const std::string extension = options.output.substr(options.output.size() - 4); if (extension == ".obj") { draco::ObjEncoder obj_encoder; if (mesh) { if (!obj_encoder.EncodeToFile(*mesh, options.output)) { printf("Failed to store the decoded mesh as OBJ.\n"); return -1; } } else { if (!obj_encoder.EncodeToFile(*pc.get(), options.output)) { printf("Failed to store the decoded point cloud as OBJ.\n"); return -1; } } } else if (extension == ".ply") { draco::PlyEncoder ply_encoder; if (mesh) { if (!ply_encoder.EncodeToFile(*mesh, options.output)) { printf("Failed to store the decoded mesh as PLY.\n"); return -1; } } else { if (!ply_encoder.EncodeToFile(*pc.get(), options.output)) { printf("Failed to store the decoded point cloud as PLY.\n"); return -1; } } } else { printf("Invalid extension of the output file. Use either .ply or .obj\n"); return -1; } printf("Decoded geometry saved to %s (%" PRId64 " ms to decode)\n", options.output.c_str(), timer.GetInMs()); return 0; }