From 06daa9b7dd868fb2fd3ce9eb00998fc4fdb0d593 Mon Sep 17 00:00:00 2001 From: lzwdgc Date: Tue, 20 Feb 2024 15:51:12 +0300 Subject: [PATCH] Initial bms converter (aim 0.03 demo format). --- src/bms_converter/bms_converter.cpp | 155 ++++++++++++++++++++++++++++ sw.cpp | 1 + 2 files changed, 156 insertions(+) create mode 100644 src/bms_converter/bms_converter.cpp diff --git a/src/bms_converter/bms_converter.cpp b/src/bms_converter/bms_converter.cpp new file mode 100644 index 0000000..c4321c3 --- /dev/null +++ b/src/bms_converter/bms_converter.cpp @@ -0,0 +1,155 @@ +/* + * AIM bms_converter (0.03 demo binary model files) + * Copyright (C) 2024 lzwdgc + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +#include "mmap.h" + +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +#pragma pack(push, 1) +struct bms { + struct header { + uint32_t magic; + uint32_t version; + uint32_t n_blocks; // anims prob + uint32_t unk; + }; + struct unk { + uint32_t unk1; + uint32_t n_unk; + uint32_t unk2; + }; + struct desc { + uint32_t unk0; + uint32_t unk1; + uint32_t n_vertices; + uint32_t unk2; + uint32_t n_faces; + uint32_t n_unk; + }; + struct v { + float x,y,z; + }; + struct vn { + float x, y, z; + }; + struct vt { + float x, y; + }; + struct f { + uint16_t x,y; + }; + struct unk2 { + uint16_t id; + uint32_t unk1; + float unk2[2]; + uint32_t unk3; + float unk4[2]; + }; + + using charC = char[0xC]; + + int n_blocks; + char header_[0x40]; +}; +#pragma pack(pop) + +void convert_model(const path &fn) { + primitives::templates2::mmap_file f{fn}; + stream s{f}; + bms::header h = s; + auto names = s.span(h.n_blocks); + uint32_t unk1 = s; + auto unk_descs = s.span(h.n_blocks); + uint32_t n_vertices = s; + uint32_t n_faces = s; + //bms::desc d = s; + auto vcs = s.span(n_vertices); + auto vns = s.span(n_vertices); + auto vts = s.span(n_vertices); + auto vfs = s.span(n_faces); + for (auto &&d : unk_descs) { + auto unk2 = s.span(d.n_unk); + auto unk3 = s.span(d.unk2); + } + + std::ofstream ofile{path{fn} += ".obj"}; + for (auto &&v : vcs) { + ofile << std::format("v {} {} {}\n", v.x, v.y, v.z); + } + ofile << "\n"; + for (auto &&v : vns) { + ofile << std::format("vn {} {} {}\n", v.x, v.y, v.z); + } + ofile << "\n"; + for (auto &&v : vts) { + ofile << std::format("vt {} {}\n", v.x, v.y); + } + ofile << "\n"; + ofile << "g obj\n"; + for (auto &&f : std::views::chunk(vfs, 3)) { + ofile << std::format("f {} {} {}\n", f[0], f[1], f[2]); + /*ofile << std::format("f {}/{}/{} {}/{}/{} {}/{}/{}\n" + , f[0], f[0], f[0] + , f[1], f[1], f[1] + , f[2], f[2], f[2] + );*/ + } + ofile << "g\n"; +} + +int main(int argc, char *argv[]) +{ + cl::opt p(cl::Positional, cl::desc(""), cl::Required); + + cl::ParseCommandLineOptions(argc, argv); + + if (fs::is_regular_file(p)) { + convert_model(p); + } else if (fs::is_directory(p)) { + auto files = enumerate_files(p, false); + for (auto &f : FilesSorted(files.begin(), files.end())) + { + if (f.extension() != ".bms") { + continue; + } + std::cout << "processing: " << f << "\n"; + try + { + convert_model(f); + } + catch (std::exception &e) + { + std::cout << "error: " << e.what() << "\n"; + } + } + } + else + throw std::runtime_error("No such file or directory: " + to_printable_string(normalize_path(p))); + + return 0; +} diff --git a/sw.cpp b/sw.cpp index 7179f91..b8a9034 100644 --- a/sw.cpp +++ b/sw.cpp @@ -60,6 +60,7 @@ void build(Solution &s) add_exe("name_generator"); add_exe_with_common("save_loader"); add_exe_with_common("mod_converter2"); + add_exe_with_common("bms_converter"); add_exe_with_common("unpaker") += "org.sw.demo.oberhumer.lzo.lzo"_dep, "org.sw.demo.xz_utils.lzma"_dep