/** * Copyright (c) 2014-present, Facebook, Inc. * All rights reserved. * * This source code is licensed under the BSD-style license found in the * LICENSE file in the root directory of this source tree. An additional grant * of patent rights can be found in the PATENTS file in the same directory. */ #include #include #include #include #include "FBX2glTF.h" #include "utils/String_Utils.h" #include "utils/Image_Utils.h" #include #include "RawModel.h" #include "Raw2Gltf.h" #include "glTF/AccessorData.h" #include "glTF/AnimationData.h" #include "glTF/BufferData.h" #include "glTF/BufferViewData.h" #include "glTF/CameraData.h" #include "glTF/ImageData.h" #include "glTF/MaterialData.h" #include "glTF/MeshData.h" #include "glTF/NodeData.h" #include "glTF/PrimitiveData.h" #include "glTF/SamplerData.h" #include "glTF/SceneData.h" #include "glTF/SkinData.h" #include "glTF/TextureData.h" typedef unsigned short TriangleIndex; extern bool verboseOutput; const static std::string defaultSceneName = "Root Scene"; /** * glTF 2.0 is based on the idea that data structs within a file are referenced by index; an accessor will * point to the n:th buffer view, and so on. The Holder class takes a freshly instantiated class, and then * creates, stored, and returns a shared_ptr for it. * * The idea is that every glTF resource in the file will live as long as the Holder does, and the Holders * are all kept in the GLTFData struct. Clients may certainly cnhoose to perpetuate the full shared_ptr * reference counting type, but generally speaking we pass around simple T& and T* types because the GLTFData * struct will, by design, outlive all other activity that takes place during in a single conversion run. */ template struct Holder { std::vector> ptrs; std::shared_ptr hold(T *ptr) { ptr->ix = ptrs.size(); ptrs.emplace_back(ptr); return ptrs.back(); } }; struct GLTFData { explicit GLTFData(bool _isGlb) : binary(new std::vector), isGlb(_isGlb) { } std::shared_ptr GetAlignedBufferView(BufferData &buffer, const BufferViewData::GL_ArrayType target) { unsigned long bufferSize = this->binary->size(); if ((bufferSize % 4) > 0) { bufferSize += (4 - (bufferSize % 4)); this->binary->resize(bufferSize); } return this->bufferViews.hold(new BufferViewData(buffer, bufferSize, target)); } // add a bufferview on the fly and copy data into it std::shared_ptr AddRawBufferView(BufferData &buffer, const char *source, uint32_t bytes) { auto bufferView = GetAlignedBufferView(buffer, BufferViewData::GL_ARRAY_NONE); bufferView->byteLength = bytes; // make space for the new bytes (possibly moving the underlying data) unsigned long bufferSize = this->binary->size(); this->binary->resize(bufferSize + bytes); // and copy them into place memcpy(&(*this->binary)[bufferSize], source, bytes); return bufferView; } std::shared_ptr AddBufferViewForFile(BufferData &buffer, const std::string &filename) { // see if we've already created a BufferViewData for this precise file auto iter = filenameToBufferView.find(filename); if (iter != filenameToBufferView.end()) { return iter->second; } std::shared_ptr result; std::ifstream file(filename, std::ios::binary | std::ios::ate); if (file) { std::streamsize size = file.tellg(); file.seekg(0, std::ios::beg); std::vector fileBuffer(size); if (file.read(fileBuffer.data(), size)) { result = AddRawBufferView(buffer, fileBuffer.data(), size); } else { fmt::printf("Warning: Couldn't read %lu bytes from %s, skipping file.\n", size, filename); } } else { fmt::printf("Warning: Couldn't open file %s, skipping file.\n", filename); } // note that we persist here not only success, but also failure, as nullptr filenameToBufferView[filename] = result; return result; } template std::shared_ptr AddAccessorWithView( BufferViewData &bufferView, const GLType &type, const std::vector &source) { auto accessor = accessors.hold(new AccessorData(bufferView, type)); accessor->appendAsBinaryArray(source, *binary); bufferView.byteLength = accessor->byteLength(); return accessor; } template std::shared_ptr AddAccessorAndView( BufferData &buffer, const GLType &type, const std::vector &source) { auto bufferView = GetAlignedBufferView(buffer, BufferViewData::GL_ARRAY_NONE); return AddAccessorWithView(*bufferView, type, source); } template std::shared_ptr AddAttributeToPrimitive( BufferData &buffer, const RawModel &surfaceModel, PrimitiveData &primitive, const AttributeDefinition &attrDef) { // copy attribute data into vector std::vector attribArr; surfaceModel.GetAttributeArray(attribArr, attrDef.rawAttributeIx); std::shared_ptr accessor; if (attrDef.dracoComponentType != draco::DT_INVALID && primitive.dracoMesh != nullptr) { primitive.AddDracoAttrib(attrDef, attribArr); accessor = accessors.hold(new AccessorData(attrDef.glType)); accessor->count = attribArr.size(); } else { auto bufferView = GetAlignedBufferView(buffer, BufferViewData::GL_ARRAY_BUFFER); accessor = AddAccessorWithView(*bufferView, attrDef.glType, attribArr); } primitive.AddAttrib(attrDef.gltfName, *accessor); return accessor; }; template void serializeHolder(json &glTFJson, std::string key, const Holder holder) { if (!holder.ptrs.empty()) { std::vector bits; for (const auto &ptr : holder.ptrs) { bits.push_back(ptr->serialize()); } glTFJson[key] = bits; } } void serializeHolders(json &glTFJson) { serializeHolder(glTFJson, "buffers", buffers); serializeHolder(glTFJson, "bufferViews", bufferViews); serializeHolder(glTFJson, "scenes", scenes); serializeHolder(glTFJson, "accessors", accessors); serializeHolder(glTFJson, "images", images); serializeHolder(glTFJson, "samplers", samplers); serializeHolder(glTFJson, "textures", textures); serializeHolder(glTFJson, "materials", materials); serializeHolder(glTFJson, "meshes", meshes); serializeHolder(glTFJson, "skins", skins); serializeHolder(glTFJson, "animations", animations); serializeHolder(glTFJson, "cameras", cameras); serializeHolder(glTFJson, "nodes", nodes); } const bool isGlb; // cache BufferViewData instances that've already been created from a given filename std::map> filenameToBufferView; std::shared_ptr > binary; Holder buffers; Holder bufferViews; Holder accessors; Holder images; Holder samplers; Holder textures; Holder materials; Holder meshes; Holder skins; Holder animations; Holder cameras; Holder nodes; Holder scenes; }; /** * This method sanity-checks existance and then returns a *reference* to the *Data instance * registered under that name. This is safe in the context of this tool, where all such data * classes are guaranteed to stick around for the duration of the process. */ template T &require(std::map> map, std::string key) { auto iter = map.find(key); assert(iter != map.end()); T &result = *iter->second; return result; } static const std::vector getIndexArray(const RawModel &raw) { std::vector result; for (int i = 0; i < raw.GetTriangleCount(); i++) { result.push_back((TriangleIndex) raw.GetTriangle(i).verts[0]); result.push_back((TriangleIndex) raw.GetTriangle(i).verts[1]); result.push_back((TriangleIndex) raw.GetTriangle(i).verts[2]); } return result; } // TODO: replace with a proper MaterialHasher class static const std::string materialHash(const RawMaterial &m) { return m.name + "_" + std::to_string(m.type); } ModelData *Raw2Gltf( std::ofstream &gltfOutStream, const std::string &outputFolder, const RawModel &raw, const GltfOptions &options ) { if (verboseOutput) { fmt::printf("Building render model...\n"); for (int i = 0; i < raw.GetMaterialCount(); i++) { fmt::printf( "Material %d: %s [shading: %s]\n", i, raw.GetMaterial(i).name.c_str(), raw.GetMaterial(i).shadingModel.c_str()); } if (raw.GetVertexCount() > 2 * raw.GetTriangleCount()) { fmt::printf( "Warning: High vertex count. Make sure there are no unnecessary vertex attributes. (see -keepAttribute cmd-line option)"); } } std::vector materialModels; raw.CreateMaterialModels(materialModels, (1 << (sizeof(TriangleIndex) * 8)), options.keepAttribs, true); if (verboseOutput) { fmt::printf("%7d vertices\n", raw.GetVertexCount()); fmt::printf("%7d triangles\n", raw.GetTriangleCount()); fmt::printf("%7d textures\n", raw.GetTextureCount()); fmt::printf("%7d nodes\n", raw.GetNodeCount()); fmt::printf("%7d surfaces\n", (int) materialModels.size()); fmt::printf("%7d animations\n", raw.GetAnimationCount()); } std::unique_ptr gltf(new GLTFData(options.outputBinary)); std::map> nodesByName; std::map> materialsByName; std::map> meshByNodeName; // for now, we only have one buffer; data->binary points to the same vector as that BufferData does. BufferData &buffer = *gltf->buffers.hold( options.outputBinary ? new BufferData(gltf->binary) : new BufferData(extBufferFilename, gltf->binary, options.embedResources)); { // // nodes // for (int i = 0; i < raw.GetNodeCount(); i++) { // assumption: RawNode index == NodeData index const RawNode &node = raw.GetNode(i); auto nodeData = gltf->nodes.hold( new NodeData(node.name, node.translation, node.rotation, node.scale, node.isJoint)); for (const auto &childName : node.childNames) { int childIx = raw.GetNodeByName(childName.c_str()); assert(childIx >= 0); nodeData->AddChildNode(childIx); } assert(nodesByName.find(nodeData->name) == nodesByName.end()); nodesByName.insert(std::make_pair(nodeData->name, nodeData)); } // // animations // for (int i = 0; i < raw.GetAnimationCount(); i++) { const RawAnimation &animation = raw.GetAnimation(i); if (animation.channels.size() == 0) { fmt::printf("Warning: animation '%s has zero channels. Skipping.\n", animation.name.c_str()); continue; } auto accessor = gltf->AddAccessorAndView(buffer, GLT_FLOAT, animation.times); accessor->min = { *std::min_element(std::begin(animation.times), std::end(animation.times)) }; accessor->max = { *std::max_element(std::begin(animation.times), std::end(animation.times)) }; AnimationData &aDat = *gltf->animations.hold(new AnimationData(animation.name, *accessor)); if (verboseOutput) { fmt::printf("Animation '%s' has %lu channels:\n", animation.name.c_str(), animation.channels.size()); } for (size_t j = 0; j < animation.channels.size(); j++) { const RawChannel &channel = animation.channels[j]; const RawNode &node = raw.GetNode(channel.nodeIndex); if (verboseOutput) { fmt::printf( " Channel %lu (%s) has translations/rotations/scales: [%lu, %lu, %lu]\n", j, node.name.c_str(), channel.translations.size(), channel.rotations.size(), channel.scales.size()); } NodeData &nDat = require(nodesByName, node.name); if (!channel.translations.empty()) { aDat.AddNodeChannel(nDat, *gltf->AddAccessorAndView(buffer, GLT_VEC3F, channel.translations), "translation"); } if (!channel.rotations.empty()) { aDat.AddNodeChannel(nDat, *gltf->AddAccessorAndView(buffer, GLT_QUATF, channel.rotations), "rotation"); } if (!channel.scales.empty()) { aDat.AddNodeChannel(nDat, *gltf->AddAccessorAndView(buffer, GLT_VEC3F, channel.scales), "scale"); } } } // // samplers // SamplerData &defaultSampler = *gltf->samplers.hold(new SamplerData()); // // textures // for (int textureIndex = 0; textureIndex < raw.GetTextureCount(); textureIndex++) { const RawTexture &texture = raw.GetTexture(textureIndex); const std::string textureName = Gltf::StringUtils::GetFileBaseString(texture.name); const std::string relativeFilename = Gltf::StringUtils::GetFileNameString(texture.fileLocation); ImageData *source = nullptr; if (options.outputBinary) { auto bufferView = gltf->AddBufferViewForFile(buffer, texture.fileLocation); if (bufferView) { std::string suffix = Gltf::StringUtils::GetFileSuffixString(texture.fileLocation); source = new ImageData(relativeFilename, *bufferView, suffixToMimeType(suffix)); } } else if (!relativeFilename.empty()) { source = new ImageData(relativeFilename, relativeFilename); std::string outputPath = outputFolder + relativeFilename; if (FileUtils::CopyFile(texture.fileLocation, outputPath)) { if (verboseOutput) { fmt::printf("Copied texture '%s' to output folder: %s\n", textureName, outputPath); } } else { // no point commenting further on read/write error; CopyFile() does enough of that, and we // certainly want to to add an image struct to the glTF JSON, with the correct relative path // reference, even if the copy failed. } } if (!source) { // fallback is tiny transparent gif source = new ImageData(textureName, ""); } const TextureData &texDat = *gltf->textures.hold( new TextureData(textureName, defaultSampler, *gltf->images.hold(source))); assert(texDat.ix == textureIndex); } // // materials // for (int materialIndex = 0; materialIndex < raw.GetMaterialCount(); materialIndex++) { const RawMaterial &material = raw.GetMaterial(materialIndex); const bool isTransparent = material.type == RAW_MATERIAL_TYPE_VERTEX_COLORED || material.type == RAW_MATERIAL_TYPE_SKINNED_VERTEX_COLORED || material.type == RAW_MATERIAL_TYPE_TRANSPARENT || material.type == RAW_MATERIAL_TYPE_SKINNED_TRANSPARENT; // find a texture by usage and return it as a TextureData*, or nullptr if none exists. auto getTex = [&](RawTextureUsage usage) { // note that we depend on TextureData.ix == rawTexture's index return (material.textures[usage] >= 0) ? gltf->textures.ptrs[material.textures[usage]].get() : nullptr; }; std::shared_ptr pbrMetRough; if (options.usePBRMetRough) { pbrMetRough.reset(new PBRMetallicRoughness(getTex(RAW_TEXTURE_USAGE_DIFFUSE), material.diffuseFactor)); } std::shared_ptr pbrSpecGloss; if (options.usePBRSpecGloss) { pbrSpecGloss.reset( new PBRSpecularGlossiness( getTex(RAW_TEXTURE_USAGE_DIFFUSE), material.diffuseFactor, getTex(RAW_TEXTURE_USAGE_SPECULAR), material.specularFactor, material.shininess)); } std::shared_ptr khrComMat; if (options.useKHRMatCom) { auto type = KHRCommonMats::MaterialType::Constant; if (material.shadingModel == "Lambert") { type = KHRCommonMats::MaterialType::Lambert; } else if (material.shadingModel == "Blinn") { type = KHRCommonMats::MaterialType::Blinn; } else if (material.shadingModel == "Phong") { type = KHRCommonMats::MaterialType::Phong; } khrComMat.reset( new KHRCommonMats( type, getTex(RAW_TEXTURE_USAGE_SHININESS), material.shininess, getTex(RAW_TEXTURE_USAGE_AMBIENT), material.ambientFactor, getTex(RAW_TEXTURE_USAGE_DIFFUSE), material.diffuseFactor, getTex(RAW_TEXTURE_USAGE_SPECULAR), material.specularFactor)); } std::shared_ptr mData = gltf->materials.hold( new MaterialData( material.name, isTransparent, getTex(RAW_TEXTURE_USAGE_NORMAL), getTex(RAW_TEXTURE_USAGE_EMISSIVE), material.emissiveFactor, khrComMat, pbrMetRough, pbrSpecGloss)); materialsByName[materialHash(material)] = mData; } // // surfaces // // in GLTF 2.0, the structural limitation is that a node can // only belong to a single mesh. A mesh can however contain any // number of primitives, which are essentially little meshes. // // so each RawSurface turns into a primitive, and we sort them // by root node using this map; one mesh per node. for (size_t surfaceIndex = 0; surfaceIndex < materialModels.size(); surfaceIndex++) { const RawModel &surfaceModel = materialModels[surfaceIndex]; assert(surfaceModel.GetSurfaceCount() == 1); const RawSurface &rawSurface = surfaceModel.GetSurface(0); const std::string surfaceName = std::to_string(surfaceIndex) + "_" + rawSurface.name; const RawMaterial &rawMaterial = surfaceModel.GetMaterial(surfaceModel.GetTriangle(0).materialIndex); if (rawMaterial.textures[RAW_TEXTURE_USAGE_DIFFUSE] < 0 && (surfaceModel.GetVertexAttributes() & RAW_VERTEX_ATTRIBUTE_COLOR) == 0) { if (verboseOutput) { fmt::printf("Warning: surface %s has neither texture nor vertex colors.\n", surfaceName.c_str()); } } const MaterialData &mData = require(materialsByName, materialHash(rawMaterial)); std::string nodeName = rawSurface.nodeName; NodeData &meshNode = require(nodesByName, nodeName); MeshData *mesh = nullptr; auto meshIter = meshByNodeName.find(nodeName); if (meshIter != meshByNodeName.end()) { mesh = meshIter->second.get(); } else { auto meshPtr = gltf->meshes.hold(new MeshData(rawSurface.name)); meshByNodeName[nodeName] = meshPtr; meshNode.SetMesh(meshPtr->ix); mesh = meshPtr.get(); } // // surface skin // if (!rawSurface.jointNames.empty()) { if (meshNode.skin == -1) { // glTF uses column-major matrices std::vector inverseBindMatrices; for (const auto &inverseBindMatrice : rawSurface.inverseBindMatrices) { inverseBindMatrices.push_back(inverseBindMatrice.Transpose()); } std::vector jointIndexes; for (const auto &jointName : rawSurface.jointNames) { jointIndexes.push_back(require(nodesByName, jointName).ix); } // Write out inverseBindMatrices auto accIBM = gltf->AddAccessorAndView(buffer, GLT_MAT4F, inverseBindMatrices); auto skeletonRoot = require(nodesByName, rawSurface.skeletonRootName); auto skin = *gltf->skins.hold(new SkinData(jointIndexes, *accIBM, skeletonRoot)); meshNode.SetSkin(skin.ix); } } std::shared_ptr primitive; if (options.useDraco) { int triangleCount = surfaceModel.GetTriangleCount(); // initialize Draco mesh with vertex index information auto dracoMesh { std::make_shared() }; dracoMesh->SetNumFaces(static_cast(triangleCount)); for (uint32_t ii = 0; ii < triangleCount; ii++) { draco::Mesh::Face face; face[0] = surfaceModel.GetTriangle(ii).verts[0]; face[1] = surfaceModel.GetTriangle(ii).verts[1]; face[2] = surfaceModel.GetTriangle(ii).verts[2]; dracoMesh->SetFace(draco::FaceIndex(ii), face); } AccessorData &indexes = *gltf->accessors.hold(new AccessorData(GLT_USHORT)); indexes.count = 3 * triangleCount; primitive.reset(new PrimitiveData(indexes, mData, dracoMesh)); } else { const AccessorData &indexes = *gltf->AddAccessorWithView( *gltf->GetAlignedBufferView(buffer, BufferViewData::GL_ELEMENT_ARRAY_BUFFER), GLT_USHORT, getIndexArray(surfaceModel)); primitive.reset(new PrimitiveData(indexes, mData)); }; // // surface vertices // { if ((surfaceModel.GetVertexAttributes() & RAW_VERTEX_ATTRIBUTE_POSITION) != 0) { const AttributeDefinition ATTR_POSITION("POSITION", &RawVertex::position, GLT_VEC3F, draco::GeometryAttribute::POSITION, draco::DT_FLOAT32); auto accessor = gltf->AddAttributeToPrimitive( buffer, surfaceModel, *primitive, ATTR_POSITION); accessor->min = toStdVec(rawSurface.bounds.min); accessor->max = toStdVec(rawSurface.bounds.max); } if ((surfaceModel.GetVertexAttributes() & RAW_VERTEX_ATTRIBUTE_NORMAL) != 0) { const AttributeDefinition ATTR_NORMAL("NORMAL", &RawVertex::normal, GLT_VEC3F, draco::GeometryAttribute::NORMAL, draco::DT_FLOAT32); gltf->AddAttributeToPrimitive(buffer, surfaceModel, *primitive, ATTR_NORMAL); } if ((surfaceModel.GetVertexAttributes() & RAW_VERTEX_ATTRIBUTE_TANGENT) != 0) { const AttributeDefinition ATTR_TANGENT("TANGENT", &RawVertex::tangent, GLT_VEC4F); gltf->AddAttributeToPrimitive(buffer, surfaceModel, *primitive, ATTR_TANGENT); } if ((surfaceModel.GetVertexAttributes() & RAW_VERTEX_ATTRIBUTE_COLOR) != 0) { const AttributeDefinition ATTR_COLOR("COLOR_0", &RawVertex::color, GLT_VEC4F, draco::GeometryAttribute::COLOR, draco::DT_FLOAT32); gltf->AddAttributeToPrimitive(buffer, surfaceModel, *primitive, ATTR_COLOR); } if ((surfaceModel.GetVertexAttributes() & RAW_VERTEX_ATTRIBUTE_UV0) != 0) { const AttributeDefinition ATTR_TEXCOORD_0("TEXCOORD_0", &RawVertex::uv0, GLT_VEC2F, draco::GeometryAttribute::TEX_COORD, draco::DT_FLOAT32); gltf->AddAttributeToPrimitive(buffer, surfaceModel, *primitive, ATTR_TEXCOORD_0); } if ((surfaceModel.GetVertexAttributes() & RAW_VERTEX_ATTRIBUTE_UV1) != 0) { const AttributeDefinition ATTR_TEXCOORD_1("TEXCOORD_1", &RawVertex::uv1, GLT_VEC2F, draco::GeometryAttribute::TEX_COORD, draco::DT_FLOAT32); gltf->AddAttributeToPrimitive(buffer, surfaceModel, *primitive, ATTR_TEXCOORD_1); } if ((surfaceModel.GetVertexAttributes() & RAW_VERTEX_ATTRIBUTE_JOINT_INDICES) != 0) { const AttributeDefinition ATTR_JOINTS("JOINTS_0", &RawVertex::jointIndices, GLT_VEC4I, draco::GeometryAttribute::GENERIC, draco::DT_UINT16); gltf->AddAttributeToPrimitive(buffer, surfaceModel, *primitive, ATTR_JOINTS); } if ((surfaceModel.GetVertexAttributes() & RAW_VERTEX_ATTRIBUTE_JOINT_WEIGHTS) != 0) { const AttributeDefinition ATTR_WEIGHTS("WEIGHTS_0", &RawVertex::jointWeights, GLT_VEC4F, draco::GeometryAttribute::GENERIC, draco::DT_FLOAT32); gltf->AddAttributeToPrimitive(buffer, surfaceModel, *primitive, ATTR_WEIGHTS); } } if (options.useDraco) { // Set up the encoder. draco::Encoder encoder; // TODO: generalize / allow configuration encoder.SetSpeedOptions(5, 5); encoder.SetAttributeQuantization(draco::GeometryAttribute::POSITION, 14); encoder.SetAttributeQuantization(draco::GeometryAttribute::TEX_COORD, 10); encoder.SetAttributeQuantization(draco::GeometryAttribute::NORMAL, 10); encoder.SetAttributeQuantization(draco::GeometryAttribute::COLOR, 8); encoder.SetAttributeQuantization(draco::GeometryAttribute::GENERIC, 8); encoder.SetEncodingMethod(draco::MeshEncoderMethod::MESH_EDGEBREAKER_ENCODING); draco::EncoderBuffer dracoBuffer; draco::Status status = encoder.EncodeMeshToBuffer(*primitive->dracoMesh, &dracoBuffer); assert(status.code() == draco::Status::OK); auto view = gltf->AddRawBufferView(buffer, dracoBuffer.data(), dracoBuffer.size()); primitive->NoteDracoBuffer(*view); } mesh->AddPrimitive(primitive); } // // cameras // for (int i = 0; i < raw.GetCameraCount(); i++) { const RawCamera &cam = raw.GetCamera(i); CameraData &camera = *gltf->cameras.hold(new CameraData()); camera.name = cam.name; if (cam.mode == RawCamera::CAMERA_MODE_PERSPECTIVE) { camera.type = "perspective"; camera.aspectRatio = cam.perspective.aspectRatio; camera.yfov = cam.perspective.fovDegreesY * ((float) M_PI / 180.0f); camera.znear = cam.perspective.nearZ; camera.zfar = cam.perspective.farZ; } else { camera.type = "orthographic"; camera.xmag = cam.orthographic.magX; camera.ymag = cam.orthographic.magY; camera.znear = cam.orthographic.nearZ; camera.zfar = cam.orthographic.farZ; } // Add the camera to the node hierarchy. auto iter = nodesByName.find(cam.nodeName); if (iter == nodesByName.end()) { fmt::printf("Warning: Camera node name %s does not exist.\n", cam.nodeName); continue; } iter->second->AddCamera(cam.name); } } NodeData &rootNode = require(nodesByName, "RootNode"); const SceneData &rootScene = *gltf->scenes.hold(new SceneData(defaultSceneName, rootNode)); if (options.outputBinary) { // note: glTF binary is little-endian const char glbHeader[] = { 'g', 'l', 'T', 'F', // magic 0x02, 0x00, 0x00, 0x00, // version 0x00, 0x00, 0x00, 0x00, // total length: written in later }; gltfOutStream.write(glbHeader, 12); // binary glTF 2.0 has a sub-header for each of the JSON and BIN chunks const char glb2JsonHeader[] = { 0x00, 0x00, 0x00, 0x00, // chunk length: written in later 'J', 'S', 'O', 'N', // chunk type: 0x4E4F534A aka JSON }; gltfOutStream.write(glb2JsonHeader, 8); } { std::vector extensionsUsed, extensionsRequired; if (options.useKHRMatCom) { extensionsUsed.push_back(KHR_MATERIALS_COMMON); if (!options.usePBRSpecGloss && !options.usePBRMetRough) { extensionsRequired.push_back(KHR_MATERIALS_COMMON); } } if (options.usePBRSpecGloss) { extensionsUsed.push_back(KHR_MATERIALS_PBR_SPECULAR_GLOSSINESS); if (!options.useKHRMatCom && !options.usePBRMetRough) { extensionsRequired.push_back(KHR_MATERIALS_PBR_SPECULAR_GLOSSINESS); } } if (options.useDraco) { extensionsUsed.push_back(KHR_DRACO_MESH_COMPRESSION); extensionsRequired.push_back(KHR_DRACO_MESH_COMPRESSION); } json glTFJson { { "asset", { { "generator", "FBX2glTF" }, { "version", "2.0" }}}, { "scene", rootScene.ix } }; if (!extensionsUsed.empty()) { glTFJson["extensionsUsed"] = extensionsUsed; } if (!extensionsRequired.empty()) { glTFJson["extensionsRequired"] = extensionsRequired; } gltf->serializeHolders(glTFJson); gltfOutStream << glTFJson.dump(options.outputBinary ? 0 : 4); } if (options.outputBinary) { uint32_t jsonLength = (uint32_t) gltfOutStream.tellp() - 20; // the binary body must begin on a 4-aligned address, so pad json with spaces if necessary while ((jsonLength % 4) != 0) { gltfOutStream.put(' '); jsonLength++; } uint32_t binHeader = (uint32_t) gltfOutStream.tellp(); // binary glTF 2.0 has a sub-header for each of the JSON and BIN chunks const char glb2BinaryHeader[] = { 0x00, 0x00, 0x00, 0x00, // chunk length: written in later 'B', 'I', 'N', 0x00, // chunk type: 0x004E4942 aka BIN }; gltfOutStream.write(glb2BinaryHeader, 8); // append binary buffer directly to .glb file uint32_t binaryLength = gltf->binary->size(); gltfOutStream.write((const char *) &(*gltf->binary)[0], binaryLength); while ((binaryLength % 4) != 0) { gltfOutStream.put('\0'); binaryLength++; } uint32_t totalLength = (uint32_t) gltfOutStream.tellp(); // seek back to sub-header for json chunk gltfOutStream.seekp(8); // write total length, little-endian gltfOutStream.put((totalLength >> 0) & 0xFF); gltfOutStream.put((totalLength >> 8) & 0xFF); gltfOutStream.put((totalLength >> 16) & 0xFF); gltfOutStream.put((totalLength >> 24) & 0xFF); // write JSON length, little-endian gltfOutStream.put((jsonLength >> 0) & 0xFF); gltfOutStream.put((jsonLength >> 8) & 0xFF); gltfOutStream.put((jsonLength >> 16) & 0xFF); gltfOutStream.put((jsonLength >> 24) & 0xFF); // seek back to the gltf 2.0 binary chunk header gltfOutStream.seekp(binHeader); // write total length, little-endian gltfOutStream.put((binaryLength >> 0) & 0xFF); gltfOutStream.put((binaryLength >> 8) & 0xFF); gltfOutStream.put((binaryLength >> 16) & 0xFF); gltfOutStream.put((binaryLength >> 24) & 0xFF); // be tidy and return write pointer to end-of-file gltfOutStream.seekp(0, std::ios::end); } return new ModelData(gltf->binary); }