[Bf-blender-cvs] [be77eeae46e] blender2.8: Depsgraph: First draft of graph filtering API implementation

Joshua Leung noreply at git.blender.org
Thu Aug 23 07:11:06 CEST 2018


Commit: be77eeae46e17c1b6ffb0e4d4834a1bd28aedf64
Author: Joshua Leung
Date:   Thu Aug 23 01:24:48 2018 +1200
Branches: blender2.8
https://developer.blender.org/rBbe77eeae46e17c1b6ffb0e4d4834a1bd28aedf64

Depsgraph: First draft of graph filtering API implementation

When this works correctly, we should be able to feed in an existing
depsgraph instance, and get out a "filtered" copy of it that contains
only the subset of nodes needed to evaluate what we're interested in.

The current implementation only filters on ID blocks/nodes,
and starts by building a full new depsgraph instance first.
I'd originally intended to do it per operation instead, copying
over individual nodes as appropriate to have the smallest and least
memory intensive graph possible. However, I ended up running into
into problems with function binding + COW arguments, hence the
current slow solution.

===================================================================

M	source/blender/depsgraph/CMakeLists.txt
M	source/blender/depsgraph/DEG_depsgraph.h
M	source/blender/depsgraph/DEG_depsgraph_query.h
M	source/blender/depsgraph/intern/depsgraph_query.cc
A	source/blender/depsgraph/intern/depsgraph_query_filter.cc

===================================================================

diff --git a/source/blender/depsgraph/CMakeLists.txt b/source/blender/depsgraph/CMakeLists.txt
index c39ce65a337..884f1d272a5 100644
--- a/source/blender/depsgraph/CMakeLists.txt
+++ b/source/blender/depsgraph/CMakeLists.txt
@@ -70,6 +70,7 @@ set(SRC
 	intern/depsgraph_eval.cc
 	intern/depsgraph_physics.cc
 	intern/depsgraph_query.cc
+	intern/depsgraph_query_filter.cc
 	intern/depsgraph_query_foreach.cc
 	intern/depsgraph_query_iter.cc
 	intern/depsgraph_tag.cc
diff --git a/source/blender/depsgraph/DEG_depsgraph.h b/source/blender/depsgraph/DEG_depsgraph.h
index 7e26987f936..d3900bc6350 100644
--- a/source/blender/depsgraph/DEG_depsgraph.h
+++ b/source/blender/depsgraph/DEG_depsgraph.h
@@ -71,6 +71,7 @@ struct ViewLayer;
 typedef enum eEvaluationMode {
 	DAG_EVAL_VIEWPORT       = 0,    /* evaluate for OpenGL viewport */
 	DAG_EVAL_RENDER         = 1,    /* evaluate for render purposes */
+	DAG_EVAL_BACKGROUND     = 2,    /* evaluate in background for baking/caching */
 } eEvaluationMode;
 
 /* DagNode->eval_flags */
diff --git a/source/blender/depsgraph/DEG_depsgraph_query.h b/source/blender/depsgraph/DEG_depsgraph_query.h
index 41650769fcf..b7ecd0544e4 100644
--- a/source/blender/depsgraph/DEG_depsgraph_query.h
+++ b/source/blender/depsgraph/DEG_depsgraph_query.h
@@ -34,6 +34,7 @@
 #define __DEG_DEPSGRAPH_QUERY_H__
 
 #include "DEG_depsgraph.h"
+#include "DEG_depsgraph_build.h"
 
 struct ID;
 
@@ -207,6 +208,34 @@ void DEG_foreach_dependent_ID(const Depsgraph *depsgraph,
 void DEG_foreach_ID(const Depsgraph *depsgraph,
                     DEGForeachIDCallback callback, void *user_data);
 
+/* ********************* DEG graph filtering ****************** */
+
+/* ComponentKey for nodes we want to be able to evaluate in the filtered graph */
+typedef struct DEG_FilterTarget {
+	struct DEG_FilterTarget *next, *prev;
+
+	struct ID *id;
+	/* TODO: component identifiers - Component Type, Subdata/Component Name */
+} DEG_FilterTarget;
+
+typedef enum eDEG_FilterQuery_Granularity {
+	DEG_FILTER_NODES_ALL           = 0,
+	DEG_FILTER_NODES_NO_OPS        = 1,
+	DEG_FILTER_NODES_ID_ONLY       = 2,
+} eDEG_FilterQuery_Granularity;
+
+
+typedef struct DEG_FilterQuery {
+	/* List of DEG_FilterTarget's */
+	struct ListBase targets;
+
+	/* Level of detail in the resulting graph */
+	eDEG_FilterQuery_Granularity detail_level;
+} DEG_FilterQuery;
+
+/* Obtain a new graph instance that only contains the subset of desired nodes */
+Depsgraph *DEG_graph_filter(const Depsgraph *depsgraph, struct Main *bmain, DEG_FilterQuery *query);
+
 
 #ifdef __cplusplus
 } /* extern "C" */
diff --git a/source/blender/depsgraph/intern/depsgraph_query.cc b/source/blender/depsgraph/intern/depsgraph_query.cc
index a6c930196ef..631669babb2 100644
--- a/source/blender/depsgraph/intern/depsgraph_query.cc
+++ b/source/blender/depsgraph/intern/depsgraph_query.cc
@@ -27,7 +27,7 @@
 /** \file blender/depsgraph/intern/depsgraph_query.cc
  *  \ingroup depsgraph
  *
- * Implementation of Querying and Filtering API's
+ * Implementation of Querying API
  */
 
 #include "MEM_guardedalloc.h"
diff --git a/source/blender/depsgraph/intern/depsgraph_query_filter.cc b/source/blender/depsgraph/intern/depsgraph_query_filter.cc
new file mode 100644
index 00000000000..89becda3ee3
--- /dev/null
+++ b/source/blender/depsgraph/intern/depsgraph_query_filter.cc
@@ -0,0 +1,272 @@
+/*
+ * ***** BEGIN GPL LICENSE BLOCK *****
+ *
+ * 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 2
+ * 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, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ * The Original Code is Copyright (C) 2018 Blender Foundation.
+ * All rights reserved.
+ *
+ * Original Author: Joshua Leung
+ * Contributor(s): None Yet
+ *
+ * ***** END GPL LICENSE BLOCK *****
+ */
+
+/** \file blender/depsgraph/intern/depsgraph_query_filter.cc
+ *  \ingroup depsgraph
+ *
+ * Implementation of Graph Filtering API
+ */
+
+#include "MEM_guardedalloc.h"
+
+extern "C" {
+#include <string.h> // XXX: memcpy
+
+#include "BLI_utildefines.h"
+#include "BKE_idcode.h"
+#include "BKE_main.h"
+#include "BLI_listbase.h"
+#include "BLI_ghash.h"
+
+#include "BKE_action.h" // XXX: BKE_pose_channel_from_name
+} /* extern "C" */
+
+#include "DNA_object_types.h"
+#include "DNA_scene_types.h"
+
+#include "RNA_access.h"
+
+#include "DEG_depsgraph.h"
+#include "DEG_depsgraph_query.h"
+
+#include "util/deg_util_foreach.h"
+
+#include "intern/eval/deg_eval_copy_on_write.h"
+
+#include "intern/depsgraph.h"
+#include "intern/depsgraph_types.h"
+#include "intern/depsgraph_intern.h"
+
+#include "intern/nodes/deg_node.h"
+#include "intern/nodes/deg_node_component.h"
+#include "intern/nodes/deg_node_id.h"
+#include "intern/nodes/deg_node_operation.h"
+
+
+/* *************************************************** */
+/* Graph Filtering Internals */
+
+namespace DEG {
+
+/* UserData for deg_add_retained_id_cb */
+struct RetainedIdUserData {
+	DEG_FilterQuery *query;
+	GSet *set;
+};
+
+/* Helper for DEG_foreach_ancestor_id()
+ * Keep track of all ID's encountered in a set
+ */
+void deg_add_retained_id_cb(ID *id, void *user_data)
+{
+	RetainedIdUserData *data = (RetainedIdUserData *)user_data;
+	BLI_gset_add(data->set, (void *)id);
+}
+
+/* ------------------------------------------- */
+
+/* Remove relations pointing to the given OperationDepsNode */
+/* TODO: Make this part of OperationDepsNode? */
+void deg_unlink_opnode(OperationDepsNode *op_node)
+{
+	/* Delete inlinks to this operation */
+	for (DepsNode::Relations::const_iterator it_rel = op_node->inlinks.begin();
+	     it_rel != op_node->inlinks.end();
+	     )
+	{
+		DepsRelation *rel = *it_rel;
+		rel->unlink();
+		OBJECT_GUARDED_DELETE(rel, DepsRelation);
+	}
+	
+	/* Delete outlinks from this operation */
+	for (DepsNode::Relations::const_iterator it_rel = op_node->outlinks.begin();
+	     it_rel != op_node->outlinks.end();
+	     )
+	{
+		DepsRelation *rel = *it_rel;
+		rel->unlink();
+		OBJECT_GUARDED_DELETE(rel, DepsRelation);
+	}
+}
+
+/* Remove and free given ID Node */
+// XXX: Use id_cow or id_orig?
+bool deg_filter_free_idnode(Depsgraph *graph, IDDepsNode *id_node,
+                            const std::function <bool (ID_Type id_type)>& filter)
+{
+	if (id_node->done == 0) {
+		/* This node has not been marked for deletion */
+		return false;
+	}
+	else if (id_node->id_cow == NULL) {
+		/* This means builder "stole" ownership of the copy-on-written
+		 * datablock for her own dirty needs.
+		 */
+		return false;
+	}
+	else if (!deg_copy_on_write_is_expanded(id_node->id_cow)) {
+		return false;
+	}
+	else {
+		const ID_Type id_type = GS(id_node->id_cow->name);
+		if (filter(id_type)) {
+			id_node->destroy();
+			return true;
+		}
+		else {
+			return false;
+		}
+	}
+}
+
+/* Remove and free ID Nodes of a particular type from the graph
+ *
+ * See Depsgraph::clear_id_nodes() and Depsgraph::clear_id_nodes_conditional()
+ * for more details about why we need these type filters
+ */
+void deg_filter_clear_ids_conditional(
+        Depsgraph *graph,
+        const std::function <bool (ID_Type id_type)>& filter)
+{
+	/* Based on Depsgraph::clear_id_nodes_conditional()... */
+	for (Depsgraph::IDDepsNodes::const_iterator it = graph->id_nodes.begin();
+	     it != graph->id_nodes.end();
+	     )
+	{
+		IDDepsNode *id_node = *it;
+		ID *id = id_node->id_orig;
+		
+		if (deg_filter_free_idnode(graph, id_node, filter)) {
+			/* Node data got destroyed. Remove from collections, and free */
+			BLI_ghash_remove(graph->id_hash, id, NULL, NULL);
+			OBJECT_GUARDED_DELETE(id_node, IDDepsNode);
+			it = graph->id_nodes.erase(it);
+		}
+		else {
+			/* Node wasn't freed. Increment iterator */
+			++it;
+		}
+	}
+}
+
+/* Remove every ID Node (and its associated subnodes, COW data) */
+void deg_filter_remove_unwanted_ids(Depsgraph *graph, GSet *retained_ids)
+{
+	/* 1) First pass over ID nodes + their operations
+	 * - Identify and tag ID's (via "done = 1") to be removed
+	 * - Remove all links to/from operations that will be removed
+	 */
+	foreach (IDDepsNode *id_node, graph->id_nodes) {
+		id_node->done = !BLI_gset_haskey(retained_ids, (void *)id_node->id_orig);
+		if (id_node->done) {
+			GHASH_FOREACH_BEGIN(ComponentDepsNode *, comp_node, id_node->components)
+			{
+				foreach (OperationDepsNode *op_node, comp_node->operations) {
+					deg_unlink_opnode(op_node);
+				}
+			}
+			GHASH_FOREACH_END();
+		}
+	}
+	
+	/* 2) Remove unwanted operations from graph->operations */
+	for (Depsgraph::OperationNodes::const_iterator it_opnode = graph->operations.begin();
+	     it_opnode != graph->operations.end();
+	     )
+	{
+		OperationDepsNode *op_node = *it_opnode;
+		IDDepsNode *id_node = op_node->owner->owner;
+		if (id_node->done) {
+			it_opnode = graph->operations.erase(it_opnode);
+		}
+		else {
+			++it_opnode;
+		}
+	}
+	
+	/* Free ID nodes that are no longer wanted
+	 * NOTE: See clear_id_nodes() for more details about what's happening here
+	 *       (e.g. regarding the lambdas used for freeing order hacks)
+	 */
+	deg_filter_clear_ids_conditional(graph,  [](ID_Type id_type) { return id_type == ID_SCE; });
+	deg_filter_clear_ids_conditional(graph,  [](ID_Type id_type) { return id_type != ID_PA; });
+}
+
+} //namespace DEG
+
+/* *************************************************** */
+/* Graph Filtering API */
+
+/* Obtain a new graph instance that only contains the nodes needed */
+Depsgraph *DEG_graph_filter(const Depsgraph *graph_src, Main *bmain, DEG_FilterQuery *query)
+{
+	const DEG::Depsgraph *deg_graph_src = reinterpret_cast<const DEG::Depsgraph *>(graph_src);
+	if (deg_graph_src == NULL) {
+		return NULL;
+	}
+	
+	/

@@ Diff output truncated at 10240 characters. @@



More information about the Bf-blender-cvs mailing list