From dd7f5acde255aa509149cc1bb688e5373e26e47e Mon Sep 17 00:00:00 2001 From: rdeioris Date: Sat, 2 Nov 2024 10:04:58 +0100 Subject: [PATCH 1/2] fixed sublayers compositionarc --- src/composition.cc | 132 +++++++++++++++++++-------------------------- 1 file changed, 56 insertions(+), 76 deletions(-) diff --git a/src/composition.cc b/src/composition.cc index ec5e4934..bcb710e6 100644 --- a/src/composition.cc +++ b/src/composition.cc @@ -408,6 +408,46 @@ bool LoadAsset(AssetResolutionResolver &resolver, return true; } +bool CombinePrimSpecRec(uint32_t depth, PrimSpec &dst, const PrimSpec &src, std::string *warn, + std::string *err) { + (void)warn; + + if (depth > (1024 * 1024 * 128)) { + PUSH_ERROR_AND_RETURN("PrimSpec tree too deep."); + } + + // Combine metadataum + dst.metas().update_from(src.metas(), false); + + // Combine properties + for (const auto &prop : src.props()) { + // add if not existent + if (dst.props().count(prop.first) == 0) { + dst.props()[prop.first] = prop.second; + } + } + + // Combine child primspecs. + for (auto &child : src.children()) { + auto dst_it = std::find_if( + dst.children().begin(), dst.children().end(), + [&child](const PrimSpec &ps) { return ps.name() == child.name(); }); + + // if exists, combine properties and children + if (dst_it != dst.children().end()) { + if (!CombinePrimSpecRec(depth + 1, (*dst_it), child, warn, err)) { + return false; + } + } + // otherwise add it + else { + dst.children().push_back(child); + } + } + + return true; +} + bool CompositeSublayersRec(AssetResolutionResolver &resolver, const Layer &in_layer, std::vector> layer_names_stack, @@ -424,10 +464,21 @@ bool CompositeSublayersRec(AssetResolutionResolver &resolver, layer_names_stack.emplace_back(std::set()); std::set &curr_layer_names = layer_names_stack.back(); + for (auto const &prim : in_layer.primspecs()) { + if (composited_layer->has_primspec(prim.first)) + { + if (!CombinePrimSpecRec(0, composited_layer->primspecs().at(prim.first), prim.second, warn, err)) { + return false; + } + } + else { + composited_layer->add_primspec(prim.first, prim.second); + } + } + for (const auto &layer : in_layer.metas().subLayers) { // TODO: subLayerOffset std::string sublayer_asset_path = layer.assetPath.GetAssetPath(); - DCOUT("Load subLayer " << sublayer_asset_path); // Do cyclic referencing check. // TODO: Use resolved name? @@ -458,46 +509,11 @@ bool CompositeSublayersRec(AssetResolutionResolver &resolver, curr_layer_names.insert(sublayer_asset_path); - Layer composited_sublayer; - // Recursively load subLayer if (!CompositeSublayersRec(resolver, sublayer, layer_names_stack, - &composited_sublayer, warn, err, options)) { + composited_layer, warn, err, options)) { return false; } - - { - // 1/2. merge sublayer's sublayers - - // NOTE: `over` specifier is ignored when merging Prims among different - // subLayers - for (auto &prim : composited_sublayer.primspecs()) { - if (composited_layer->has_primspec(prim.first)) { - // Skip - } else { - if (!composited_layer->emplace_primspec(prim.first, std::move(prim.second))) { - PUSH_ERROR_AND_RETURN( - fmt::format("Compositing PrimSpec {} in {} failed.", prim.first, - layer_filepath)); - } - DCOUT("add primspec: " << prim.first); - } - } - - // 2/2. merge sublayer - for (auto &prim : sublayer.primspecs()) { - if (composited_layer->has_primspec(prim.first)) { - // Skip - } else { - if (!composited_layer->emplace_primspec(prim.first, std::move(prim.second))) { - PUSH_ERROR_AND_RETURN( - fmt::format("Compositing PrimSpec {} in {} failed.", prim.first, - layer_filepath)); - } - DCOUT("add primspec: " << prim.first); - } - } - } } layer_names_stack.pop_back(); @@ -517,51 +533,15 @@ bool CompositeSublayers(AssetResolutionResolver &resolver, std::vector> layer_names_stack; + // keep metas from the root layer + composited_layer->metas() = in_layer.metas(); + DCOUT("Resolve subLayers.."); if (!CompositeSublayersRec(resolver, in_layer, layer_names_stack, composited_layer, warn, err, options)) { PUSH_ERROR_AND_RETURN("Composite subLayers failed."); } - // merge Prims in root layer. - // NOTE: local Prims(prims in root layer) wins against subLayer's Prim - DCOUT("in_layer # of primspecs: " << in_layer.primspecs().size()); - for (auto &prim : in_layer.primspecs()) { - DCOUT("in_layer.prim: " << prim.first); - if (composited_layer->has_primspec(prim.first)) { - // over - if (prim.second.specifier() == Specifier::Class) { - // TODO: Simply ignore? - DCOUT("TODO: `class` Prim"); - } else if (prim.second.specifier() == Specifier::Over) { - PrimSpec &dst = composited_layer->primspecs()[prim.first]; - if (!OverridePrimSpec(dst, prim.second, warn, err)) { - return false; - } - } else if (prim.second.specifier() == Specifier::Def) { - DCOUT("overewrite prim: " << prim.first); - // overwrite - if (!composited_layer->replace_primspec(prim.first, prim.second)) { - PUSH_ERROR_AND_RETURN( - fmt::format("Failed to replace PrimSpec: {}", prim.first)); - } - } else { - /// ??? - PUSH_ERROR_AND_RETURN(fmt::format("Prim {} has invalid Prim specifier.", - prim.second.name())); - } - } else { - if (!composited_layer->add_primspec(prim.first, prim.second)) { - PUSH_ERROR_AND_RETURN( - fmt::format("Compositing PrimSpec {} in {} failed.", prim.first, - in_layer.name())); - } - DCOUT("added primspec: " << prim.first); - } - } - - composited_layer->metas() = in_layer.metas(); - // Remove subLayers metadatum composited_layer->metas().subLayers.clear(); DCOUT("Composite subLayers ok."); From 7c85e00ae00dfd880633a93bdcb63133ce7df82f Mon Sep 17 00:00:00 2001 From: Roberto De Ioris Date: Fri, 22 Nov 2024 11:50:07 +0100 Subject: [PATCH 2/2] added sublayers example usda files --- tests/usda/composition/sublayers-207-000.usda | 47 +++++++++++++++++++ tests/usda/composition/sublayers-207-001.usda | 14 ++++++ tests/usda/composition/sublayers-207-002.usda | 19 ++++++++ tests/usda/composition/sublayers-207.usda | 24 ++++++++++ 4 files changed, 104 insertions(+) create mode 100644 tests/usda/composition/sublayers-207-000.usda create mode 100644 tests/usda/composition/sublayers-207-001.usda create mode 100644 tests/usda/composition/sublayers-207-002.usda create mode 100644 tests/usda/composition/sublayers-207.usda diff --git a/tests/usda/composition/sublayers-207-000.usda b/tests/usda/composition/sublayers-207-000.usda new file mode 100644 index 00000000..3d9ebedf --- /dev/null +++ b/tests/usda/composition/sublayers-207-000.usda @@ -0,0 +1,47 @@ +#usda 1.0 +( + "Hello World" + doc = "Ops" + upAxis = "Z" + + subLayers = [ + @sublayers-207-001.usda@, + @sublayers-207-002.usda@ + ] +) + +def "One" +{ + def "Two" + { + string before = "before" + def Test "Three" + { + def Test "Four" + { + string value = "xyz" + } + } + string after = "after" + } +} + +over "Root" +{ + over "Broken" + { + string value = "broken" + } + + def "Child" + { + def "SubChild" + { + float value = 17 + } + + string another = "one" + } + + string additional = "HEY!" +} diff --git a/tests/usda/composition/sublayers-207-001.usda b/tests/usda/composition/sublayers-207-001.usda new file mode 100644 index 00000000..9e257946 --- /dev/null +++ b/tests/usda/composition/sublayers-207-001.usda @@ -0,0 +1,14 @@ +#usda 1.0 + +def "Rooter" +{ + def Scope "FirstLayer" + { + + } + + def Scope "FirstLayerCheck" + { + + } +} diff --git a/tests/usda/composition/sublayers-207-002.usda b/tests/usda/composition/sublayers-207-002.usda new file mode 100644 index 00000000..771c07f0 --- /dev/null +++ b/tests/usda/composition/sublayers-207-002.usda @@ -0,0 +1,19 @@ +#usda 1.0 + +def Scope "SecondLayer" +{ + +} + +def Scope "SecondLayerCheck" +{ + +} + +def "Root" +{ + def "AdditionalScope" + { + + } +} diff --git a/tests/usda/composition/sublayers-207.usda b/tests/usda/composition/sublayers-207.usda new file mode 100644 index 00000000..e8803249 --- /dev/null +++ b/tests/usda/composition/sublayers-207.usda @@ -0,0 +1,24 @@ +#usda 1.0 + +( + subLayers = [ + @sublayers-207-000.usda@ + ] +""" +repro scene for https://github.com/lighttransport/tinyusdz/issues/207 +Issue 207: sublayers recursive management and properties combination +""" +) + +def "Root" +{ + def "Hello" + { + string value = "Hello World" + } + + def "Child" + { + string here = "now" + } +}