/* * Copyright (C) 2012 by Jan Boon (Kaetemi) * * This file is part of RYZOM CORE PIPELINE. * RYZOM CORE PIPELINE is free software: you can redistribute it * and/or modify it under the terms of the GNU Affero General Public * License as published by the Free Software Foundation, either * version 3 of the License, or (at your option) any later version. * * RYZOM CORE PIPELINE 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 Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public * License along with RYZOM CORE PIPELINE. If not, see * . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "nel/misc/dynloadlib.h" #include "nel/misc/debug.h" #include "nel/misc/path.h" #include "nel/misc/algo.h" #include "nel/misc/file.h" #include "nel/misc/mem_stream.h" #include "../max/storage_stream.h" #include "../max/storage_object.h" #include "../max/dll_directory.h" #include "../max/class_directory_3.h" #include "../max/class_data.h" #include "../max/config.h" #include "../max/scene.h" #include "../max/scene_class_registry.h" // Testing #include "../max/builtin/builtin.h" #include "../max/update1/update1.h" #include "../max/epoly/epoly.h" #include "../max/builtin/storage/app_data.h" #include "../max/builtin/storage/geom_buffers.h" #include "../max/builtin/scene_impl.h" #include "../max/builtin/i_node.h" #include "../max/update1/editable_mesh.h" #include "../max/epoly/editable_poly.h" #include using namespace std; using namespace boost::algorithm; using namespace PIPELINE::MAX; using namespace PIPELINE::MAX::BUILTIN; using namespace PIPELINE::MAX::BUILTIN::STORAGE; using namespace PIPELINE::MAX::UPDATE1; using namespace PIPELINE::MAX::EPOLY; CSceneClassRegistry SceneClassRegistry; // Never enable this bool DebugParser = false; bool DisplayStream = false; bool DisplayReplaces = false; bool ReplacePaths = true; bool ReplaceMapExt = true; bool WriteModified = true; bool WriteDummy = false; bool HaltOnIssue = false; const char *DatabaseDirectory = "w:\\database\\"; const char *LinuxDatabaseDirectory = "/mnt/tsurugi/ryzom-assets/database/"; bool RunningLinux = true; //const char *SrcDirectoryRecursive = "w:\\database\\interfaces\\"; //const char *SrcDirectoryRecursive = "w:\\database\\"; //const char *SrcDirectoryRecursive = "w:\\database\\stuff\\fyros\\city\\newpositionville\\"; const char *SrcDirectoryRecursiveInit = "w:\\database\\"; //const char *SrcDirectoryRecursiveHandle = "w:\\database\\stuff\\generique\\agents\\accessories\\"; const char *SrcDirectoryRecursiveHandle = "w:\\database\\landscape\\ligo\\primes_racines\\max\\"; bool UseFallbackTga = false; const char *FallbackTga = "w:\\database\\stuff\\lod_actors\\texture_lod\\trame.png"; std::set MissingFiles; std::map KnownFileCache; std::string nativeDatabasePath(const std::string &standardizedPath) { if (RunningLinux) { std::string result = standardizedPath; NLMISC::strFindReplace(result, DatabaseDirectory, LinuxDatabaseDirectory); while (NLMISC::strFindReplace(result, "\\", "/")) { } return result; } return standardizedPath; } std::string unnativeDatabasePath(const std::string &nativizedPath) { if (RunningLinux) { std::string result = nativizedPath; NLMISC::strFindReplace(result, LinuxDatabaseDirectory, DatabaseDirectory); while (NLMISC::strFindReplace(result, "/", "\\")) { } return result; } return nativizedPath; } // COPY FROM PIPELINE_SERVICE.CPP WITH BACKSLASHES INSTEAD OF FORWARD SLASHES std::string standardizePath(const std::string &path, bool addFinalSlash) { // check empty path if (path.empty()) return ""; std::string newPath; newPath.resize(path.size() + 1); std::string::size_type j = 0; for (std::string::size_type i = 0; i < path.size(); ++i) { if (path[i] == '\\' || path[i] == '/') { if (j <= 1 && path[i] == '\\') { // for windows network newPath[j] = '\\'; ++j; } else if (j == 0 || newPath[j - 1] != '\\') { newPath[j] = '\\'; ++j; } } else { newPath[j] = path[i]; ++j; } } newPath[j] = 0; newPath.resize(j); // add terminal slash if (addFinalSlash && newPath[newPath.size()-1] != '\\') newPath += '\\'; return newPath; } inline bool isCharacter(char c0) { uint8 c = *(uint8 *)(void *)(&c0); return (32 <= c /*&& c <= 127) || (161 <= c*/ /*&& c <= 255*/); } inline char stripFrenchLocale(char c0) { uint8 c = *(uint8 *)(void *)(&c0); if (192 <= c && c <= 197) return 'a'; if (200 <= c && c <= 203) return 'e'; if (204 <= c && c <= 207) return 'i'; if (210 <= c && c <= 214) return 'o'; if (217 <= c && c <= 220) return 'u'; if (c == 221) return 'y'; if (224 <= c && c <= 229) return 'a'; if (232 <= c && c <= 235) return 'e'; if (236 <= c && c <= 239) return 'i'; if (242 <= c && c <= 246) return 'o'; if (249 <= c && c <= 252) return 'u'; if (c == 253 || c == 255) return 'y'; return c0; } std::string rewritePath(const std::string &path, const std::string &databaseDirectory) { static std::set fileNameCache; std::string stdPath = standardizePath(path, false); for (std::string::size_type i = 0; i < stdPath.size(); ++i) stdPath[i] = stripFrenchLocale(stdPath[i]); stdPath = NLMISC::toLower(stdPath); // TODO: remove ./stuff/caravan/agents/_textures/actors/trame.png NLMISC::strFindReplace(stdPath, "w:\\database\\", databaseDirectory); NLMISC::strFindReplace(stdPath, "\\\\amiga\\3d\\database\\", databaseDirectory); NLMISC::strFindReplace(stdPath, "ma_hom_armor_01", "ma_hom_armor01"); NLMISC::strFindReplace(stdPath, "ma_hof_armor_01", "ma_hof_armor01"); NLMISC::strFindReplace(stdPath, "ma_hom_armor_00", "ma_hom_armor00"); NLMISC::strFindReplace(stdPath, "ma_hof_armor_00", "ma_hof_armor00"); NLMISC::strFindReplace(stdPath, "zo_hom_armor_00", "zo_hom_armor00"); NLMISC::strFindReplace(stdPath, "zo_hof_armor_00", "zo_hof_armor00"); NLMISC::strFindReplace(stdPath, "fy_hof_cheveux_shave01", "fy_hof_cheveux_shave"); NLMISC::strFindReplace(stdPath, "matis\\agents\\_textures\\actors\\ma_hom_underwear_avtbras", "tryker\\agents\\_textures\\actors\\tr_hom_underwear_avtbras"); NLMISC::strFindReplace(stdPath, "matis\\agents\\_textures\\actors\\ma_hom_underwear_epaule", "tryker\\agents\\_textures\\actors\\tr_hom_underwear_epaule"); NLMISC::strFindReplace(stdPath, "matis\\agents\\_textures\\actors\\ma_hom_underwear_hand-downside", "tryker\\agents\\_textures\\actors\\tr_hom_underwear_hand-downside"); NLMISC::strFindReplace(stdPath, "matis\\agents\\_textures\\actors\\ma_hom_underwear_hand-upside", "tryker\\agents\\_textures\\actors\\tr_hom_underwear_hand-upside"); NLMISC::strFindReplace(stdPath, "matis\\agents\\_textures\\actors\\ma_hom_underwear_molet", "tryker\\agents\\_textures\\actors\\tr_hom_underwear_molet"); NLMISC::strFindReplace(stdPath, "matis\\agents\\_textures\\actors\\ma_hof_underwear_hand-downside", "tryker\\agents\\_textures\\actors\\tr_hof_underwear_hand-downside"); NLMISC::strFindReplace(stdPath, "matis\\agents\\_textures\\actors\\ma_hof_underwear_hand-upside", "tryker\\agents\\_textures\\actors\\tr_hof_underwear_hand-upside"); NLMISC::strFindReplace(stdPath, "tr_hof_underwear_torso", "tr_hof_underwear_torse"); NLMISC::strFindReplace(stdPath, "tr_hof_underwear_hand-", "tr_hof_underwear_hand_"); NLMISC::strFindReplace(stdPath, "fy_hom_armor00_avbras.", "fy_hom_armor00_avbras_c1."); NLMISC::strFindReplace(stdPath, "fy_hom_armor00_bottes.", "fy_hom_armor00_bottes_c1."); NLMISC::strFindReplace(stdPath, "fy_hom_armor00_bras.", "fy_hom_armor00_bras_c1."); NLMISC::strFindReplace(stdPath, "fy_hom_armor00_cuissear.", "fy_hom_armor00_cuissear_c1."); NLMISC::strFindReplace(stdPath, "fy_hom_armor00_cuisseav.", "fy_hom_armor00_cuisseav_c1."); NLMISC::strFindReplace(stdPath, "fy_hom_armor00_dos.", "fy_hom_armor00_dos_c1."); NLMISC::strFindReplace(stdPath, "fy_hom_armor00_hand-downside.", "fy_hom_armor00_hand-downside_c1."); NLMISC::strFindReplace(stdPath, "fy_hom_armor00_hand-upside.", "fy_hom_armor00_hand-upside_c1."); NLMISC::strFindReplace(stdPath, "fy_hom_armor00_pieds.", "fy_hom_armor00_pieds_c1."); NLMISC::strFindReplace(stdPath, "fy_hom_armor00_torse.", "fy_hom_armor00_torse_c1."); NLMISC::strFindReplace(stdPath, "fy-hof-gilet02-dos", "fy_hof_gilet02_arriere_n"); NLMISC::strFindReplace(stdPath, "fy-hof-pantalon02-arriere", "fy_hof_pantalon02_arriere_n"); NLMISC::strFindReplace(stdPath, "fy-hof-pantalon02-avant", "fy_hof_pantalon02_avant_n"); NLMISC::strFindReplace(stdPath, "fy-hom-blouson02-avantbras", "fy_hom_blouson02_avbras_n"); NLMISC::strFindReplace(stdPath, "fy-hom-blouson02-bras", "fy_hom_blouson02_bras_n"); NLMISC::strFindReplace(stdPath, "fy-hom-blouson02-dos", "fy_hom_blouson02_arriere_n"); NLMISC::strFindReplace(stdPath, "fy-hom-blouson02-torse", "fy_hom_blouson02_avant_n"); NLMISC::strFindReplace(stdPath, "fy-hom-pantalon02-mollet", "fy_hom_pantalon02_mollet_n"); NLMISC::strFindReplace(stdPath, "fy-hom-shoes02", "fy_hom_shoes02_shoes_n"); NLMISC::strFindReplace(stdPath, "fyhof-gilet02-torse", "fy_hof_gilet02_torse_n"); NLMISC::strFindReplace(stdPath, "ge_hom_armor02_armpad", "ge_hom_armor02_avbras"); NLMISC::strFindReplace(stdPath, "ge_hom_armor02_bottes", "ge_hom_armor02_pied"); NLMISC::strFindReplace(stdPath, "ge_hom_armor02_cuisse-arr", "ge_hom_armor02_cuisse_arr"); NLMISC::strFindReplace(stdPath, "ge_hom_armor02_cuisse-avt", "ge_hom_armor02_cuisse_avt"); NLMISC::strFindReplace(stdPath, "ge_hom_armor02_hand-down", "ge_hom_armor02_hand_down"); NLMISC::strFindReplace(stdPath, "ge_hom_armor02_hand-up", "ge_hom_armor02_hand_up"); NLMISC::strFindReplace(stdPath, "ge_hom_armor02_torse", "ge_hom_armor02_torso"); NLMISC::strFindReplace(stdPath, "fy_lovejail.", "fy_lovejail_su."); NLMISC::strFindReplace(stdPath, "\\caissebonusaction.", "\\g_caissebonus."); NLMISC::strFindReplace(stdPath, "\\sactoile.", "\\g_sactoile."); NLMISC::strFindReplace(stdPath, "\\ge_mission_stele_kami0.", "\\ge_mission_stele_kami."); NLMISC::strFindReplace(stdPath, "\\ge_mission_reward_karavan_bigshield.", "\\ge_mission_reward_karavan_bigshield_c1."); NLMISC::strFindReplace(stdPath, "\\tr_hom_underwear_cuisse_arr.", "\\tr_hom_underwear_cuisse-arr."); NLMISC::strFindReplace(stdPath, "\\tr_hom_underwear_cuisse_avt.", "\\tr_hom_underwear_cuisse-avt."); NLMISC::strFindReplace(stdPath, "interfaces_visage", "visage_interface"); if (stdPath.find("\\trame.") != std::string::npos) stdPath = standardizePath(databaseDirectory + "/stuff/lod_actors/texture_lod/trame.png", false); if (stdPath.find("\\tr_hof_visage_c1.") != std::string::npos) stdPath = standardizePath(databaseDirectory + "/stuff/tryker/agents/_textures/actors/tr_hof_visage_c1.png", false); if (stdPath.find("\\tr_hof_visage_c2.") != std::string::npos) stdPath = standardizePath(databaseDirectory + "/stuff/tryker/agents/_textures/actors/tr_hof_visage_c2.png", false); if (stdPath.find("\\tr_hof_visage_c3.") != std::string::npos) stdPath = standardizePath(databaseDirectory + "/stuff/tryker/agents/_textures/actors/tr_hof_visage_c3.png", false); if (stdPath.find("\\ma_hof_cheveux_medium02.") != std::string::npos) stdPath = standardizePath(databaseDirectory + "/stuff/matis/agents/_textures/actors/ma_hof_cheveux_medium02.png", false); /*if (stdPath.size() > path.size()) { nlwarning("Path size becomes too large: '%s' -> '%s'", path.c_str(), stdPath.c_str()); return path; }*/ if (NLMISC::CFile::getFilename(stdPath) == stdPath) { breakable { if (fileNameCache.find(stdPath) == fileNameCache.end() || KnownFileCache.find(stdPath) == KnownFileCache.end()) { if (stdPath[stdPath.size() - 3] == 't' && stdPath[stdPath.size() - 2] == 'g' && stdPath[stdPath.size() - 1] == 'a') { stdPath[stdPath.size() - 3] = 'p'; stdPath[stdPath.size() - 2] = 'n'; stdPath[stdPath.size() - 1] = 'g'; if (fileNameCache.find(stdPath) != fileNameCache.end()) break; } //#nlwarning("File name not known: '%s' ('%s')", path.c_str(), stdPath.c_str()); // MissingFiles.insert(path); return stdPath; } } return stdPath; } else { if (stdPath.find(databaseDirectory) == std::string::npos) { if (stdPath[1] == ':' || (stdPath[0] == '\\' && stdPath[1] == '\\')) { if (KnownFileCache.find(NLMISC::CFile::getFilename(stdPath)) != KnownFileCache.end()) return KnownFileCache[NLMISC::CFile::getFilename(stdPath)]; if (stdPath[stdPath.size() - 3] == 't' && stdPath[stdPath.size() - 2] == 'g' && stdPath[stdPath.size() - 1] == 'a') { stdPath[stdPath.size() - 3] = 'p'; stdPath[stdPath.size() - 2] = 'n'; stdPath[stdPath.size() - 1] = 'g'; } if (KnownFileCache.find(NLMISC::CFile::getFilename(stdPath)) != KnownFileCache.end()) return KnownFileCache[NLMISC::CFile::getFilename(stdPath)]; if (stdPath[stdPath.size() - 3] == 'p' && stdPath[stdPath.size() - 2] == 'n' && stdPath[stdPath.size() - 1] == 'g') { stdPath[stdPath.size() - 3] = 't'; stdPath[stdPath.size() - 2] = 'g'; stdPath[stdPath.size() - 1] = 'a'; } if (KnownFileCache.find(NLMISC::CFile::getFilename(stdPath)) != KnownFileCache.end()) return KnownFileCache[NLMISC::CFile::getFilename(stdPath)]; //#nlwarning("Path not in database: '%s' ('%s')", path.c_str(), stdPath.c_str()); MissingFiles.insert(path); return stdPath; } else { // invalid path, don't care too much return stdPath; } } breakable { if (!NLMISC::CFile::fileExists(nativeDatabasePath(stdPath))) { if (stdPath[stdPath.size() - 3] == 't' && stdPath[stdPath.size() - 2] == 'g' && stdPath[stdPath.size() - 1] == 'a') { stdPath[stdPath.size() - 3] = 'p'; stdPath[stdPath.size() - 2] = 'n'; stdPath[stdPath.size() - 1] = 'g'; if (NLMISC::CFile::fileExists(nativeDatabasePath(stdPath))) break; } { std::string stdPathVv2 = standardizePath(NLMISC::CFile::getPath(stdPath) + "/vv2/" + NLMISC::CFile::getFilename(stdPath), false); bool vv2works = false; if (NLMISC::CFile::fileExists(nativeDatabasePath(stdPathVv2))) { vv2works = true; } else { if (stdPathVv2[stdPathVv2.size() - 3] == 'p' && stdPathVv2[stdPathVv2.size() - 2] == 'n' && stdPathVv2[stdPathVv2.size() - 1] == 'g') { stdPathVv2[stdPathVv2.size() - 3] = 't'; stdPathVv2[stdPathVv2.size() - 2] = 'g'; stdPathVv2[stdPathVv2.size() - 1] = 'a'; if (NLMISC::CFile::fileExists(nativeDatabasePath(stdPathVv2))) { vv2works = true; } } } if (vv2works) { // try with vv2 /*if (stdPathVv2.size() > path.size()) { nlwarning("Path with vv2 size becomes too large: '%s' -> '%s'", path.c_str(), stdPathVv2.c_str()); return stdPath; } else {*/ stdPath = stdPathVv2; break; //} } } // try find if (KnownFileCache.find(NLMISC::CFile::getFilename(stdPath)) != KnownFileCache.end()) return KnownFileCache[NLMISC::CFile::getFilename(stdPath)]; if (stdPath[stdPath.size() - 3] == 't' && stdPath[stdPath.size() - 2] == 'g' && stdPath[stdPath.size() - 1] == 'a') { stdPath[stdPath.size() - 3] = 'p'; stdPath[stdPath.size() - 2] = 'n'; stdPath[stdPath.size() - 1] = 'g'; } if (KnownFileCache.find(NLMISC::CFile::getFilename(stdPath)) != KnownFileCache.end()) return KnownFileCache[NLMISC::CFile::getFilename(stdPath)]; if (stdPath[stdPath.size() - 3] == 'p' && stdPath[stdPath.size() - 2] == 'n' && stdPath[stdPath.size() - 1] == 'g') { stdPath[stdPath.size() - 3] = 't'; stdPath[stdPath.size() - 2] = 'g'; stdPath[stdPath.size() - 1] = 'a'; } if (KnownFileCache.find(NLMISC::CFile::getFilename(stdPath)) != KnownFileCache.end()) return KnownFileCache[NLMISC::CFile::getFilename(stdPath)]; //#nlwarning("Path file does not exist: '%s' ('%s')", path.c_str(), stdPath.c_str()); MissingFiles.insert(path); return stdPath; } } } fileNameCache.insert(NLMISC::CFile::getFilename(stdPath)); return stdPath; } void doFileInitialize(const std::string &filePath) { // nldebug("File: '%s'", filePath.c_str()); // nldebug("Native: '%s'", nativeDatabasePath(filePath).c_str()); KnownFileCache[NLMISC::CFile::getFilename(filePath)] = standardizePath(filePath, false); } // maxRewritePaths W:/database/interfaces/anims_max void doDirectoryInitialize(const std::string &directoryPath) { nldebug("Directory: '%s'", directoryPath.c_str()); std::string dirPath = standardizePath(directoryPath, true); std::vector dirContents; NLMISC::CPath::getPathContent(nativeDatabasePath(dirPath), false, true, true, dirContents); // nldebug("Native: '%s'", nativeDatabasePath(dirPath).c_str()); // nldebug("Contents: %i", dirContents.size()); for (std::vector::iterator it = dirContents.begin(), end = dirContents.end(); it != end; ++it) { std::string subPath = standardizePath(unnativeDatabasePath(*it), false); // nldebug("Path: '%s'", subPath.c_str()); // nldebug("Native: '%s'", nativeDatabasePath(subPath + " ").c_str()); if (NLMISC::CFile::isDirectory(nativeDatabasePath(subPath))) { if (subPath.find("\\.") == std::string::npos) doDirectoryInitialize(subPath); } else doFileInitialize(subPath); } } void runInitialize() { nlinfo("DatabaseDirectory: '%s'", DatabaseDirectory); doDirectoryInitialize(std::string(SrcDirectoryRecursiveInit)); } // Scary stuff void doFileScanner(const std::string &filePath) { if (filePath[filePath.size() - 3] == 'm' && filePath[filePath.size() - 2] == 'a' && filePath[filePath.size() - 1] == 'x') { nldebug("File: '%s'", filePath.c_str()); std::vector buffer; buffer.resize(NLMISC::CFile::getFileSize(nativeDatabasePath(filePath))); // read { NLMISC::CIFile ifile; ifile.open(nativeDatabasePath(filePath), false); ifile.serialBuffer((uint8 *)(&buffer[0]), buffer.size()); ifile.close(); } // find paths for (std::vector::size_type i = 256; i < buffer.size(); ++i) // skip the first 256 lol :) { if (((NLMISC::toLower(buffer[i - 1]) == 'x' && NLMISC::toLower(buffer [i - 2]) == 'a' && NLMISC::toLower(buffer[i - 3]) == 'm') || (NLMISC::toLower(buffer[i - 1]) == 'a' && NLMISC::toLower(buffer [i - 2]) == 'g' && NLMISC::toLower(buffer[i - 3]) == 't') || (NLMISC::toLower(buffer[i - 1]) == 'g' && NLMISC::toLower(buffer [i - 2]) == 'n' && NLMISC::toLower(buffer[i - 3]) == 'p')) && (NLMISC::toLower(buffer[i - 4]) == '.')) { // buffer[i] is the character after the path! :) std::vector::size_type beginPath = 0; for (std::vector::size_type j = i - 4; j > 0; --j) { if (!isCharacter(buffer[j])) { if ((buffer[j + 1] == '\\' && buffer[j + 2] == '\\') || buffer[j] == 0) { beginPath = j + 1; break; } // nlwarning("Invalid characters '%i' in path at %i, len %i!", (uint32)buffer[j], (uint32)j, (uint32)(i - j)); // beginPath = j + 1; // test break; } if (buffer[j] == ':') { beginPath = j - 1; break; } } if (beginPath != 0) { std::vector::size_type sizePath = i - beginPath; std::string foundPath = std::string(&buffer[beginPath], sizePath); //std::string(buffer.at(beginPath), buffer.at(i)); //nldebug("Found path: '%s' from %i to %i", foundPath.c_str(), (uint32)beginPath, (uint32)i); std::string fixedPath = rewritePath(foundPath, DatabaseDirectory); //nldebug("Rewrite to: '%s'", fixedPath.c_str()); } } } //NLMISC::CFile::re // ... } } void doDirectoryScanner(const std::string &directoryPath) { nldebug("Directory: '%s'", directoryPath.c_str()); std::string dirPath = standardizePath(directoryPath, true); std::vector dirContents; NLMISC::CPath::getPathContent(nativeDatabasePath(dirPath), false, true, true, dirContents); for (std::vector::iterator it = dirContents.begin(), end = dirContents.end(); it != end; ++it) { std::string subPath = standardizePath(unnativeDatabasePath(*it), false); if (NLMISC::CFile::isDirectory(nativeDatabasePath(subPath))) { if (subPath.find("\\.") == std::string::npos) doDirectoryScanner(subPath); } else doFileScanner(subPath); } } void runScanner() { nlinfo("DatabaseDirectory: '%s'", DatabaseDirectory); doDirectoryScanner(SrcDirectoryRecursiveHandle); for (std::set::iterator it = MissingFiles.begin(), end = MissingFiles.end(); it != end; ++it) nlinfo("Missing: '%s'", (*it).c_str()); } void handleFile(const std::string &path); void doFileHandler(const std::string &filePath) { if (filePath[filePath.size() - 3] == 'm' && filePath[filePath.size() - 2] == 'a' && filePath[filePath.size() - 1] == 'x') { nldebug("File: '%s'", filePath.c_str()); handleFile(nativeDatabasePath(filePath)); } } void doDirectoryHandler(const std::string &directoryPath) { nldebug("Directory: '%s'", directoryPath.c_str()); std::string dirPath = standardizePath(directoryPath, true); std::vector dirContents; NLMISC::CPath::getPathContent(nativeDatabasePath(dirPath), false, true, true, dirContents); for (std::vector::iterator it = dirContents.begin(), end = dirContents.end(); it != end; ++it) { std::string subPath = standardizePath(unnativeDatabasePath(*it), false); if (NLMISC::CFile::isDirectory(nativeDatabasePath(subPath))) { if (subPath.find("\\.") == std::string::npos) doDirectoryHandler(subPath); } else doFileHandler(subPath); } } void runHandler() { nlinfo("DatabaseDirectory: '%s'", DatabaseDirectory); doDirectoryHandler(SrcDirectoryRecursiveHandle); for (std::set::iterator it = MissingFiles.begin(), end = MissingFiles.end(); it != end; ++it) nlinfo("Missing: '%s'", (*it).c_str()); } void serializeStorageContainer(PIPELINE::MAX::CStorageContainer *storageContainer, GsfInfile *infile, const char *streamName) { GsfInput *input = gsf_infile_child_by_name(infile, streamName); if (!input) { nlerror("GSF Could not read stream %s", streamName); return; } { PIPELINE::MAX::CStorageStream instream(input); storageContainer->serial(instream); } g_object_unref(input); } void serializeStorageContainer(PIPELINE::MAX::CStorageContainer *storageContainer, GsfOutfile *outfile, const char *streamName) { //nldebug("write"); GsfOutput *output = GSF_OUTPUT(gsf_outfile_new_child(outfile, streamName, false)); if (!output) { nlerror("GSF Could not write stream %s", streamName); return; } { PIPELINE::MAX::CStorageStream outstream(output); storageContainer->serial(outstream); } gsf_output_close(output); g_object_unref(G_OBJECT(output)); } void serializeRaw(std::vector &rawOutput, GsfInfile *infile, const char *streamName) { GsfInput *input = gsf_infile_child_by_name(infile, streamName); if (!input) { nlerror("GSF Could not read stream %s", streamName); return; } { PIPELINE::MAX::CStorageStream instream(input); rawOutput.resize(instream.size()); instream.serialBuffer(&rawOutput[0], rawOutput.size()); } g_object_unref(input); } void serializeRaw(std::vector &rawOutput, GsfOutfile *outfile, const char *streamName) { GsfOutput *output = GSF_OUTPUT(gsf_outfile_new_child(outfile, streamName, false)); if (!output) { nlerror("GSF Could not write stream %s", streamName); return; } { PIPELINE::MAX::CStorageStream outstream(output); outstream.serialBuffer(&rawOutput[0], rawOutput.size()); } gsf_output_close(output); g_object_unref(G_OBJECT(output)); } std::string cleanString(const std::string &str) { std::string res = str; trim(res); // \\Amiga\3D (10 chars) if (res.size() > 10) { if (res.substr(res.size() - 10) == "\\\\Amiga\\3D") res = res.substr(0, res.size() - 10); } if (res.size() > 1) { if (res.substr(res.size() - 1) == "0") res = res.substr(0, res.size() - 1); } if (res.size() > 4 && res[res.size() - 1] == '.') { if (res.substr(res.size() - 4) == "max.") // fix a stupid typo res = res.substr(0, res.size() - 1); } return res; } std::string rewritePathFinal(const std::string &str) { std::string strtrimmed = cleanString(str); std::string result = rewritePath(strtrimmed, DatabaseDirectory); if (NLMISC::CFile::getFilename(result) != result && !NLMISC::CFile::fileExists(nativeDatabasePath(result)) && ((result[result.size() - 3] == 't' && result[result.size() - 2] == 'g' && result[result.size() - 1] == 'a') || (result[result.size() - 3] == 'p' && result[result.size() - 2] == 'n' && result[result.size() - 1] == 'g')) ) { // nlwarning("Replacing missing '%s' with '%s'", result.c_str(), FallbackTga); if (UseFallbackTga) { return FallbackTga; } else { return NLMISC::CFile::getFilename(result); } } if (DisplayReplaces) { nldebug("Replacing '%s' with '%s'", str.c_str(), result.c_str()); } return result; } bool isImportantFilePath(const std::string &str) { std::string strtrimmed = cleanString(str); if (strtrimmed.size() >= 4) { std::string strlw = NLMISC::toLower(strtrimmed); return (strlw[strlw.size() - 3] == 'm' && strlw[strlw.size() - 2] == 'a' && strlw[strlw.size() - 1] == 'x') || (strlw[strlw.size() - 3] == 't' && strlw[strlw.size() - 2] == 'g' && strlw[strlw.size() - 1] == 'a') || (strlw[strlw.size() - 3] == 'p' && strlw[strlw.size() - 2] == 'n' && strlw[strlw.size() - 1] == 'g'); } return false; } bool hasImportantFilePath(CStorageRaw *raw) { if (raw->Value.size() >= 4) { // Find any occurences of .max, .png or .tga in ascii or utf16 for (uint i = 0; i < raw->Value.size() - 3; ++i) { if (NLMISC::toLower(((char *)&raw->Value[0])[i]) == '.' && NLMISC::toLower(((char *)&raw->Value[0])[i + 1]) == 'm' && NLMISC::toLower(((char *)&raw->Value[0])[i + 2]) == 'a' && NLMISC::toLower(((char *)&raw->Value[0])[i + 3]) == 'x') return true; if (NLMISC::toLower(((char *)&raw->Value[0])[i]) == '.' && NLMISC::toLower(((char *)&raw->Value[0])[i + 1]) == 't' && NLMISC::toLower(((char *)&raw->Value[0])[i + 2]) == 'g' && NLMISC::toLower(((char *)&raw->Value[0])[i + 3]) == 'a') return true; if (NLMISC::toLower(((char *)&raw->Value[0])[i]) == '.' && NLMISC::toLower(((char *)&raw->Value[0])[i + 1]) == 'p' && NLMISC::toLower(((char *)&raw->Value[0])[i + 2]) == 'n' && NLMISC::toLower(((char *)&raw->Value[0])[i + 3]) == 'g') return true; } } if (raw->Value.size() >= 6) { for (uint i = 0; i < raw->Value.size() - 6; ++i) { if (NLMISC::toLower(((char *)&raw->Value[0])[i]) == '.' && NLMISC::toLower(((char *)&raw->Value[0])[i + 2]) == 'm' && NLMISC::toLower(((char *)&raw->Value[0])[i + 4]) == 'a' && NLMISC::toLower(((char *)&raw->Value[0])[i + 6]) == 'x') return true; if (NLMISC::toLower(((char *)&raw->Value[0])[i]) == '.' && NLMISC::toLower(((char *)&raw->Value[0])[i + 2]) == 't' && NLMISC::toLower(((char *)&raw->Value[0])[i + 4]) == 'g' && NLMISC::toLower(((char *)&raw->Value[0])[i + 6]) == 'a') return true; if (NLMISC::toLower(((char *)&raw->Value[0])[i]) == '.' && NLMISC::toLower(((char *)&raw->Value[0])[i + 2]) == 'p' && NLMISC::toLower(((char *)&raw->Value[0])[i + 4]) == 'n' && NLMISC::toLower(((char *)&raw->Value[0])[i + 6]) == 'g') return true; } } return false; } void fixChunk(uint16 id, IStorageObject *chunk) { CStorageValue *asString = dynamic_cast *>(chunk); if (asString) { // nldebug("String: %s", asString->Value.c_str()); if (isImportantFilePath(asString->Value)) asString->Value = rewritePathFinal(asString->Value); return; } CStorageValue *asUCString = dynamic_cast *>(chunk); if (asUCString) { // nldebug("UCString: %s", asUCString->Value.toUtf8().c_str()); if (isImportantFilePath(asUCString->Value.toString())) asUCString->Value.fromUtf8(rewritePathFinal(asUCString->Value.toString())); return; } CStorageRaw *asRaw = dynamic_cast(chunk); if (asRaw) { switch (id) { case 256: case 4656: case 16385: case 288: // .max xref in classdata if (hasImportantFilePath(asRaw)) { // generic ucstring really nlassert(asRaw->Value.size() % 2 == 0); ucstring str; str.resize(asRaw->Value.size() / 2); memcpy(&str[0], &asRaw->Value[0], asRaw->Value.size()); // nldebug("[%s]", str.toString().c_str()); nlassert(isImportantFilePath(str.toString())); str.fromUtf8(rewritePathFinal(str.toString())); asRaw->Value.resize(str.size() * 2); memcpy(&asRaw->Value[0], &str[0], asRaw->Value.size()); break; } case 10: if (hasImportantFilePath(asRaw)) { // 10 00 08 00 00 00 02 00 80 00 40 // 11 bytes O_O // 4d 00 00 00 // 57 3a 5c 44 61 74 61 62 61 73 // 65 5c 73 74 75 66 66 5c 74 72 // 79 6b 65 72 5c 61 67 65 6e 74 // 73 5c 5f 74 65 78 74 75 72 65 // 73 5c 61 63 74 6f 72 73 5c 54 // 52 5f 48 4f 46 5f 63 69 76 69 // 6c 30 31 5f 74 6f 72 73 6f 5f // 43 31 2e 74 67 61 00 bool overrideFF = true; // Patch for some ligo max files if (overrideFF && asRaw->Value.size() > 4 && asRaw->Value[asRaw->Value.size() - 4] == 0xFF) { nlwarning("0xFFFFFFFF"); } else if (!(asRaw->Value[asRaw->Value.size() - 1] == 0)) { nlinfo("Id: %i, size: %i", (uint32)id, asRaw->Value.size()); asRaw->toString(std::cout); nldebug("x"); nlwarning("not null term"); if (HaltOnIssue) { std::string x; std::cin >> x; } break; } uint32 size = ((uint32 *)&asRaw->Value[11])[0]; if (!(asRaw->Value.size() == size + 4 + 11)) { nlinfo("Id: %i, size: %i", (uint32)id, asRaw->Value.size()); asRaw->toString(std::cout); nldebug("x"); nlwarning("size '%i' does not match '%i', use different algo :)", size, asRaw->Value.size() - 4 - 11); uint8 nonsense[11]; uint32 counter; std::vector strings; { NLMISC::CMemStream mem; asRaw->serial(mem); mem.invert(); mem.serialBuffer(nonsense, 11); mem.serial(counter); uint i = 0; while ((sint)mem.getPos() != (sint)mem.size()) { //nldebug("pos %i", mem.getPos()); //nldebug("size %i", mem.size()); char funny; mem.serial(funny); nlassert(funny == '@'); sint32 size; mem.serial(size); //nldebug("size %i", size); if (!overrideFF && size == -1) { nldebug("size %i", size); nlwarning("bad size"); if (HaltOnIssue) { std::string x; std::cin >> x; } return; } std::string v; if (overrideFF && size == -1) { v.resize(1); *(uint8 *)&v[0] = 0; } else { v.resize(size); mem.serialBuffer((uint8 *)&v[0], size); } if (!(v[v.size() - 1] == 0)) { nlinfo("Id: %i, size: %i", (uint32)id, asRaw->Value.size()); asRaw->toString(std::cout); nldebug("x"); nlwarning("not null term inside array stuff %i '%s'", i, v.c_str()); if (HaltOnIssue) { std::string x; std::cin >> x; } return; } v.resize(v.size() - 1); // nldebug("%s", v.c_str()); strings.push_back(v); ++i; // nldebug("ok"); } nlassert(strings.size() == counter); asRaw->Value.resize(0); } bool foundone = false; for (uint i = 0; i < strings.size(); ++i) { if (isImportantFilePath(strings[i])) { foundone = true; strings[i] = rewritePathFinal(strings[i]); } } nlassert(foundone); { //nldebug("go"); NLMISC::CMemStream mem; mem.serialBuffer(nonsense, 11); mem.serial(counter); for (uint i = 0; i < strings.size(); ++i) { //nldebug("one"); char funny = '@'; mem.serial(funny); strings[i].resize(strings[i].size() + 1); strings[i][strings[i].size() - 1] = 0; uint32 size = strings[i].size(); mem.serial(size); mem.serialBuffer((uint8 *)&strings[i][0], size); } asRaw->setSize(mem.getPos()); mem.invert(); asRaw->serial(mem); } //std::string x; //std::cin >> x; return; } std::string str; str.resize(size - 1); memcpy(&str[0], &asRaw->Value[15], str.size()); // nldebug("test '%s'", str.c_str()); // asRaw->toString(std::cout); if (!isImportantFilePath(str)) { nlinfo("Id: %i, size: %i", (uint32)id, asRaw->Value.size()); asRaw->toString(std::cout); nldebug("x"); nlwarning("not important"); if (HaltOnIssue) { std::string x; std::cin >> x; } break; } str = rewritePathFinal(str); asRaw->Value.resize(str.size() + 11 + 4 + 1); memcpy(&asRaw->Value[15], &str[0], str.size()); ((uint32 *)&asRaw->Value[11])[0] = str.size() + 1; asRaw->Value[asRaw->Value.size() - 1] = 0; break; } case 304: if (hasImportantFilePath(asRaw)) { // null term c string nlassert(asRaw->Value[asRaw->Value.size() - 1] == 0); std::string str; str.resize(asRaw->Value.size() - 1); memcpy(&str[0], &asRaw->Value[0], str.size()); if (!isImportantFilePath(str)) { nlinfo("Id: %i", (uint32)id); asRaw->toString(std::cout); nlerror("not important"); } str = rewritePathFinal(str); asRaw->Value.resize(str.size() + 1); memcpy(&asRaw->Value[0], &str[0], str.size()); asRaw->Value[asRaw->Value.size() - 1] = 0; break; } case 9730: if (asRaw->Value.size() > 0 && asRaw->Value[0] == 'I') { // ignore Init.max break; } default: if (hasImportantFilePath(asRaw)) { nlinfo("Id: %i", (uint32)id); asRaw->toString(std::cout); nlwarning("Found important file path"); if (HaltOnIssue) { std::string x; std::cin >> x; } return; } break; } } } void fixChunks(CStorageContainer *container) { for (CStorageContainer::TStorageObjectConstIt it = container->chunks().begin(), end = container->chunks().end(); it != end; ++it) { if (it->second->isContainer()) { fixChunks(static_cast(it->second)); } else { fixChunk(it->first, it->second); } } } void handleFile(const std::string &path) { GError *err = NULL; GsfDocMetaData *metadata = gsf_doc_meta_data_new(); nlassert(metadata); GsfInput *src = gsf_input_stdio_new(path.c_str(), &err); if (err) { nlerror("GSF Failed to open %s", path.c_str()); return; } GsfInfile *infile = gsf_infile_msole_new(src, NULL); if (!infile) { nlerror("GSF Failed to recognize %s", path.c_str()); return; } g_object_unref(src); uint8 classId[16]; if (!gsf_infile_msole_get_class_id((GsfInfileMSOle *)infile, classId)) nlerror("GSF Did not find classId"); PIPELINE::MAX::CStorageContainer videoPostQueue; serializeStorageContainer(&videoPostQueue, infile, "VideoPostQueue"); PIPELINE::MAX::CStorageContainer config; serializeStorageContainer(&config, infile, "Config"); PIPELINE::MAX::CStorageContainer classData; serializeStorageContainer(&classData, infile, "ClassData"); std::vector summaryInformation; serializeRaw(summaryInformation, infile, "\05SummaryInformation"); std::vector documentSummaryInformation; serializeRaw(documentSummaryInformation, infile, "\05DocumentSummaryInformation"); // Can't read this, don't care. PIPELINE::MAX::CDllDirectory dllDirectory; serializeStorageContainer(&dllDirectory, infile, "DllDirectory"); dllDirectory.parse(VersionUnknown); PIPELINE::MAX::CClassDirectory3 classDirectory3(&dllDirectory); serializeStorageContainer(&classDirectory3, infile, "ClassDirectory3"); classDirectory3.parse(VersionUnknown); PIPELINE::MAX::CScene scene(&SceneClassRegistry, &dllDirectory, &classDirectory3); serializeStorageContainer(&scene, infile, "Scene"); if (DebugParser) { // Not parsing the scene for this function. scene.parse(VersionUnknown); scene.clean(); } /* PIPELINE::MAX::CStorageContainer dllDirectory; serializeStorageContainer(&dllDirectory, infile, "DllDirectory"); PIPELINE::MAX::CStorageContainer classDirectory3; serializeStorageContainer(&classDirectory3, infile, "ClassDirectory3"); PIPELINE::MAX::CStorageContainer scene; serializeStorageContainer(&scene, infile, "Scene"); */ if (DisplayStream) { videoPostQueue.toString(std::cout); config.toString(std::cout); classData.toString(std::cout); dllDirectory.toString(std::cout); classDirectory3.toString(std::cout); scene.toString(std::cout); } if (DebugParser) { // Not parsing the scene for this function. scene.build(VersionUnknown); scene.disown(); } g_object_unref(infile); if (ReplacePaths) { fixChunks(&classData); fixChunks(&scene); } if (ReplaceMapExt) { // Parse the scene scene.parse(VersionUnknown); NLMISC::CClassId editMeshClassId = NLMISC::CClassId(0x00000050, 0x00000000); NLMISC::CClassId editableMeshClassId = NLMISC::CClassId(0xe44f10b3, 0x00000000); // from: (0x2ec82081, 0x045a6271) NLMISC::CClassId fromClassId = NLMISC::CClassId(0x2ec82081, 0x045a6271); // Find all object space modifier derived containing map extender for (CStorageContainer::TStorageObjectConstIt it = scene.container()->chunks().begin(), end = scene.container()->chunks().end(); it != end; ++it) { if (it->first == 0x2032) // Derived Object { // nldebug("Found derived object"); CReferenceMaker *derivedObject = dynamic_cast(it->second); nlassert(derivedObject); // Find map extender in the modifier stack uint mapExtenderIndex = 0; CReferenceMaker *mapExtender = NULL; for (uint i = 0; i < derivedObject->nbReferences(); ++i) { if (derivedObject->getReference(i) && derivedObject->getReference(i)->classDesc()->classId() == fromClassId) { nldebug("Found map extender at '%i' / '%i'!", i, derivedObject->nbReferences()); mapExtenderIndex = i; mapExtender = derivedObject->getReference(i); bool deleteDerivedGeom = false; CStorageContainer *derivedData = dynamic_cast(derivedObject->findStorageObject(0x2500, mapExtenderIndex)); nlassert(derivedData); CStorageContainer *derivedGeom = dynamic_cast(derivedData->findStorageObject(0x2512)); if (!derivedGeom) { CStorageRaw *derivedGeomRaw = dynamic_cast(derivedData->findStorageObject(0x2512)); if (derivedGeomRaw) { nlwarning("Derived geometry raw instead of as container"); NLMISC::CMemStream memGeom; derivedGeomRaw->serial(memGeom); uint size = memGeom.getPos(); memGeom.invert(); derivedGeom = new CStorageContainer(); deleteDerivedGeom = true; derivedGeom->serial(memGeom, size); } else { derivedData->toString(std::cout); nlwarning("derived geometry missing!!!"); if (HaltOnIssue) { std::string x; std::cin >> x; } continue; } } CStorageRaw *derivedVertices = dynamic_cast(derivedGeom->findStorageObject(0x03e9)); if (!derivedVertices) { derivedGeom->toString(std::cout); nlwarning("derived vertices missing!!!"); if (HaltOnIssue) { std::string x; std::cin >> x; } continue; } CStorageRaw *derivedIndices = dynamic_cast(derivedGeom->findStorageObject(0x03eb)); if (!derivedIndices) { derivedGeom->toString(std::cout); nlwarning("derived indices missing!!!"); if (HaltOnIssue) { std::string x; std::cin >> x; } continue; } CStorageContainer::TStorageObjectContainer &mapChunks = const_cast(mapExtender->chunks()); // CStorageContainer::TStorageObjectWithId nlassert(derivedVertices->Value.size() % 12 == 0); nlassert(derivedIndices->Value.size() % 12 == 0); uint32 nbVertices = derivedVertices->Value.size() / 12; uint32 nbFaces = derivedIndices->Value.size() / 12; nldebug("Vertices: %i", nbVertices); nldebug("Faces: %i", nbFaces); // Write vertex chunks CStorageValue *chunkVertCount = new CStorageValue(); chunkVertCount->Value = nbVertices; mapChunks.push_back(CStorageContainer::TStorageObjectWithId(0x0100, chunkVertCount)); CStorageRaw *chunkVerts = new CStorageRaw(); NLMISC::CMemStream memVerts; derivedVertices->serial(memVerts); chunkVerts->setSize(memVerts.getPos()); memVerts.invert(); chunkVerts->serial(memVerts); mapChunks.push_back(CStorageContainer::TStorageObjectWithId(0x0110, chunkVerts)); // Write face chunks CStorageValue *chunkFaceCount = new CStorageValue(); chunkFaceCount->Value = nbFaces; mapChunks.push_back(CStorageContainer::TStorageObjectWithId(0x0230, chunkFaceCount)); CStorageRaw *chunkFaces = new CStorageRaw(); NLMISC::CMemStream memFaces; derivedIndices->serial(memFaces); chunkFaces->setSize(memFaces.getPos()); memFaces.invert(); chunkFaces->serial(memFaces); mapChunks.push_back(CStorageContainer::TStorageObjectWithId(0x0240, chunkFaces)); // Write geom points CStorageValue *chunkGeomPointCount = new CStorageValue(); chunkGeomPointCount->Value = nbVertices; // derived vertices? mapChunks.push_back(CStorageContainer::TStorageObjectWithId(0x0320, chunkGeomPointCount)); CStorageRaw *chunkGeomPoints = new CStorageRaw(); NLMISC::CMemStream memGeomPoints; derivedVertices->serial(memGeomPoints); // derived vertices? chunkGeomPoints->setSize(memGeomPoints.getPos()); memGeomPoints.invert(); chunkGeomPoints->serial(memGeomPoints); mapChunks.push_back(CStorageContainer::TStorageObjectWithId(0x0330, chunkGeomPoints)); if (deleteDerivedGeom) delete derivedGeom; // /*break;*/ nldebug("Converted!"); } } if (mapExtender) { uint editMeshIndex; CReferenceMaker *editMesh = NULL; for (uint i = 0; i < derivedObject->nbReferences(); ++i) { if (derivedObject->getReference(i) && derivedObject->getReference(i)->classDesc()->classId() == editMeshClassId) { nldebug("Found edit mesh at '%i' / '%i'!", i, derivedObject->nbReferences()); editMeshIndex = i; editMesh = derivedObject->getReference(i); // Ensure no selection CStorageContainer::TStorageObjectContainer &meshChunks = const_cast(editMesh->chunks()); // Ensure no selection for (CStorageContainer::TStorageObjectIterator it = meshChunks.begin(), end = meshChunks.end(); it != end; ) { CStorageContainer::TStorageObjectIterator next = it; ++next; switch (it->first) { case 0x2800: nldebug("Override selection"); { delete it->second; CStorageValue *v = new CStorageValue(); v->Value = 0; it->second = v; } break; } it = next; } // /*break;*/ } } uint editableMeshIndex; CReferenceMaker *editableMesh = NULL; for (uint i = 0; i < derivedObject->nbReferences(); ++i) { if (derivedObject->getReference(i) && derivedObject->getReference(i)->classDesc()->classId() == editableMeshClassId) { nldebug("Found editable mesh at '%i' / '%i'!", i, derivedObject->nbReferences()); editableMeshIndex = i; editableMesh = derivedObject->getReference(i); CStorageContainer::TStorageObjectContainer &meshChunks = const_cast(editableMesh->chunks()); // Ensure no selection for (CStorageContainer::TStorageObjectIterator it = meshChunks.begin(), end = meshChunks.end(); it != end; ) { CStorageContainer::TStorageObjectIterator next = it; ++next; switch (it->first) { case 0x2845: case 0x2846: case 0x2847: case 0x2849: case 0x2850: case 0x3001: case 0x3003: case 0x3004: nldebug("Erase selection"); meshChunks.erase(it); break; case 0x4038: nldebug("Override selection"); { delete it->second; CStorageValue *v = new CStorageValue(); v->Value = 0; it->second = v; } break; } it = next; } // /*break;*/ } } if (editableMesh && editableMeshIndex + 1 != derivedObject->nbReferences()) { nlwarning("Editable mesh not at bottom of the stack, this is unpossible!"); std::string x; std::cin >> x; continue; } /* if (editableMesh == NULL || editMesh == NULL) { derivedObject->toString(std::cout); nlwarning("editable mesh or edit mesh not found"); std::string x; std::cin >> x; continue; } // //derivedObject->toString(std::cout); //editableMesh->toString(std::cout); // derivedObject -> 0x2500 -> 0x2512 -> 0x03e9 / 0x03eb // editableMesh -> 0x08fe (geom) -> 0x0916 / 0x0918 CStorageContainer *editableMeshGeometry = dynamic_cast(editableMesh->findStorageObject(0x08fe)); // nlassert(editableMeshGeometry); if (!editableMeshGeometry) { nlwarning("broken"); std::string x; std::cin >> x; continue; } CStorageRaw *editableVertices = dynamic_cast(editableMeshGeometry->findStorageObject(0x0916)); nlassert(editableVertices); CStorageRaw *editableIndices = dynamic_cast(editableMeshGeometry->findStorageObject(0x0918)); nlassert(editableIndices); //return new CStorageArraySizePre(); if (derivedIndices->Value.size() != editableIndices->Value.size()) { editableMesh->toString(std::cout); derivedObject->toString(std::cout); nlwarning("size mismatch"); std::string x; std::cin >> x; continue; } editableVertices->Value.clear(); editableIndices->Value.clear(); nlassert(derivedVertices->Value.size() % 12 == 0); uint32 nbVertices = derivedVertices->Value.size() / 12; NLMISC::CMemStream memVertices; memVertices.serial(nbVertices); derivedVertices->serial(memVertices); memVertices.invert(); editableVertices->setSize(derivedVertices->Value.size() + 4); editableVertices->serial(memVertices); NLMISC::CMemStream memIndices; derivedIndices->serial(memIndices); memIndices.invert(); editableIndices->setSize(derivedIndices->Value.size()); editableIndices->serial(memIndices); nldebug("ok!");*/ } } } // to: (0x02df2e3a, 0x72ba4e1f) NLMISC::CClassId toClassId = NLMISC::CClassId(0x02df2e3a, 0x72ba4e1f); // uvw unwrap // NLMISC::CClassId toClassId = NLMISC::CClassId(0x5c5b50f7,0x60397ca1); // converttomesh :| // NLMISC::CClassId toClassId = NLMISC::CClassId(0x31f9c666, 0x03b4a577); // uvw mapping add // dllname to: uvwunwrap.dlm for (CStorageContainer::TStorageObjectConstIt it = classDirectory3.chunks().begin(), end = classDirectory3.chunks().end(); it != end; ++it) { CClassEntry *classEntry = dynamic_cast(it->second); if (classEntry) { //nldebug("class entry %s", classEntry->classId().toString().c_str()); if (classEntry->classId() == fromClassId) { nldebug("Found class id to replace!"); classEntry->overrideClassId(toClassId); CDllEntry *dllEntry = const_cast(dllDirectory.get(classEntry->dllIndex())); dllEntry->overrideDllFilename(ucstring("uvwunwrap.dlm")); } } //else nldebug("not class entry"); } // Disown the scene scene.disown(); } dllDirectory.disown(); classDirectory3.disown(); if (WriteModified) { const char *outpath = (WriteDummy ? "testdummy.max" : path.c_str()); GsfOutput *output; GsfOutfile *outfile; output = gsf_output_stdio_new(outpath, &err); if (err) { nlerror("GSF Failed to create %s", outpath); return; } outfile = gsf_outfile_msole_new(output); g_object_unref(G_OBJECT(output)); serializeStorageContainer(&videoPostQueue, outfile, "VideoPostQueue"); serializeStorageContainer(&scene, outfile, "Scene"); serializeStorageContainer(&dllDirectory, outfile, "DllDirectory"); serializeStorageContainer(&config, outfile, "Config"); serializeStorageContainer(&classDirectory3, outfile, "ClassDirectory3"); serializeStorageContainer(&classData, outfile, "ClassData"); serializeRaw(summaryInformation, outfile, "\05SummaryInformation"); serializeRaw(documentSummaryInformation, outfile, "\05DocumentSummaryInformation"); if (!gsf_outfile_msole_set_class_id((GsfOutfileMSOle *)outfile, classId)) nlerror("GSF Cannot write class id"); gsf_output_close(GSF_OUTPUT(outfile)); g_object_unref(G_OBJECT(outfile)); if (WriteDummy) { nlinfo("Dummy written, press key for next"); std::string x; std::cin >> x; } } g_object_unref(metadata); } // int __stdcall WinMain(void *, void *, void *, int) int main(int argc, char **argv) { // Initialise gsf printf("Pipeline Max Rewrite Assets\n"); char const *me = (argv[0] ? argv[0] : "pipeline_max_rewrite_assets"); g_set_prgname(me); gsf_init(); // Register all plugin classes CBuiltin::registerClasses(&SceneClassRegistry); CUpdate1::registerClasses(&SceneClassRegistry); CEPoly::registerClasses(&SceneClassRegistry); //handleFile("/srv/work/database/interfaces/anims_max/cp_fy_hof_species.max"); runInitialize(); runHandler(); //handleFile(nativeDatabasePath("w:\\database\\landscape\\ligo\\desert\\max\\zonematerial-converted-brandon.max")); // overrideFF //handleFile(nativeDatabasePath("w:\\database\\landscape\\ligo\\desert\\max\\zonematerial-converted-154_dz.max")); // overrideFF //handleFile(nativeDatabasePath("w:\\database\\landscape\\ligo\\lacustre\\max\\zonematerial-converted-village_a.max")); //handleFile(nativeDatabasePath("w:\\database\\stuff\\generique\\agents\\accessories\\ge_mission_reward_karavan_bigshield.max")); //handleFile(nativeDatabasePath("w:\\database\\stuff\\generique\\agents\\accessories\\ge_acc_pick_o.max")); //handleFile(nativeDatabasePath("w:\\database\\stuff\\generique\\agents\\accessories\\ge_zo_wea_trib_masse1m.max")); //handleFile(nativeDatabasePath("w:\\database\\stuff\\generique\\agents\\accessories\\ge_fy_wea_trib_grand_bouclier.max")); //handleFile(nativeDatabasePath("w:\\database\\stuff\\generique\\agents\\accessories\\ge_mission_entrepot.max")); //handleFile(nativeDatabasePath("w:\\database\\stuff\\generique\\agents\\accessories\\mesh_wip\\all_trib_weapons.max")); //handleFile("/srv/work/database/landscape/ligo/jungle/max/zonematerial-foret-ruine_boss.max"); //handleFile("/srv/work/database/stuff/fyros/agents/actors/male/animation/anims_non_utilisees/fy_hom_assis_boire_verre.max"); //handleFile("/home/kaetemi/3dsMax/scenes/test_clear_add_uvw.max"); //runScanner(); gsf_shutdown(); return 0; }