/* * ***** 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. * * Contributor(s): Blender Foundation (2014). * * ***** END GPL LICENSE BLOCK ***** */ /** \file blender/makesrna/intern/rna_depsgraph.c * \ingroup RNA */ #include #include "BLI_utildefines.h" #include "BLI_path_util.h" #include "RNA_define.h" #include "RNA_enum_types.h" #include "rna_internal.h" #include "DEG_depsgraph.h" #include "DNA_object_types.h" #ifdef RNA_RUNTIME #include "BLI_iterator.h" #include "BKE_report.h" #include "DEG_depsgraph_build.h" #include "DEG_depsgraph_debug.h" #include "DEG_depsgraph_query.h" #include "MEM_guardedalloc.h" /* **************** Depsgraph **************** */ static PointerRNA rna_DepsgraphIter_object_get(PointerRNA *ptr) { BLI_Iterator *iterator = ptr->data; return rna_pointer_inherit_refine(ptr, &RNA_Object, iterator->current); } static PointerRNA rna_DepsgraphIter_instance_object_get(PointerRNA *ptr) { BLI_Iterator *iterator = ptr->data; DEGObjectsIteratorData *deg_iter = (DEGObjectsIteratorData *)iterator->data; Object *instance_object = NULL; if (deg_iter->dupli_object_current != NULL) { instance_object = deg_iter->dupli_object_current->ob; } return rna_pointer_inherit_refine(ptr, &RNA_Object, instance_object); } static PointerRNA rna_DepsgraphIter_parent_get(PointerRNA *ptr) { BLI_Iterator *iterator = ptr->data; DEGObjectsIteratorData *deg_iter = (DEGObjectsIteratorData *)iterator->data; Object *dupli_parent = NULL; if (deg_iter->dupli_object_current != NULL) { dupli_parent = deg_iter->dupli_parent; } return rna_pointer_inherit_refine(ptr, &RNA_Object, dupli_parent); } static void rna_DepsgraphIter_persistent_id_get(PointerRNA *ptr, int *persistent_id) { BLI_Iterator *iterator = ptr->data; DEGObjectsIteratorData *deg_iter = (DEGObjectsIteratorData *)iterator->data; memcpy(persistent_id, deg_iter->dupli_object_current->persistent_id, sizeof(deg_iter->dupli_object_current->persistent_id)); } static void rna_DepsgraphIter_orco_get(PointerRNA *ptr, float *orco) { BLI_Iterator *iterator = ptr->data; DEGObjectsIteratorData *deg_iter = (DEGObjectsIteratorData *)iterator->data; memcpy(orco, deg_iter->dupli_object_current->orco, sizeof(deg_iter->dupli_object_current->orco)); } static unsigned int rna_DepsgraphIter_random_id_get(PointerRNA *ptr) { BLI_Iterator *iterator = ptr->data; DEGObjectsIteratorData *deg_iter = (DEGObjectsIteratorData *)iterator->data; return deg_iter->dupli_object_current->random_id; } static void rna_DepsgraphIter_uv_get(PointerRNA *ptr, float *uv) { BLI_Iterator *iterator = ptr->data; DEGObjectsIteratorData *deg_iter = (DEGObjectsIteratorData *)iterator->data; memcpy(uv, deg_iter->dupli_object_current->uv, sizeof(deg_iter->dupli_object_current->uv)); } static int rna_DepsgraphIter_is_instance_get(PointerRNA *ptr) { BLI_Iterator *iterator = ptr->data; DEGObjectsIteratorData *deg_iter = (DEGObjectsIteratorData *)iterator->data; return (deg_iter->dupli_object_current != NULL); } /* **************** Depsgraph **************** */ static void rna_Depsgraph_debug_graphviz(Depsgraph *graph, const char *filename) { FILE *f = fopen(filename, "w"); if (f == NULL) return; DEG_debug_graphviz(graph, f, "Depsgraph", false); fclose(f); } static void rna_Depsgraph_debug_rebuild(Depsgraph *UNUSED(graph), bContext *C) { Main *bmain = CTX_data_main(C); EvaluationContext eval_ctx; Scene *sce; CTX_data_eval_ctx(C, &eval_ctx); DEG_relations_tag_update(bmain); for (sce = bmain->scene.first; sce; sce = sce->id.next) { DEG_scene_relations_rebuild(bmain, sce); DEG_graph_on_visible_update(bmain, sce); } } static void rna_Depsgraph_debug_stats(Depsgraph *graph, ReportList *reports) { size_t outer, ops, rels; DEG_stats_simple(graph, &outer, &ops, &rels); // XXX: report doesn't seem to work printf("Approx %lu Operations, %lu Relations, %lu Outer Nodes\n", ops, rels, outer); BKE_reportf(reports, RPT_WARNING, "Approx. %lu Operations, %lu Relations, %lu Outer Nodes", ops, rels, outer); } /* Iteration over objects, simple version */ static void rna_Depsgraph_objects_begin(CollectionPropertyIterator *iter, PointerRNA *ptr) { iter->internal.custom = MEM_callocN(sizeof(BLI_Iterator), __func__); DEGObjectsIteratorData *data = MEM_callocN(sizeof(DEGObjectsIteratorData), __func__); data->graph = (Depsgraph *)ptr->data; data->flag = DEG_OBJECT_ITER_FLAG_SET; DEG_objects_iterator_begin(iter->internal.custom, data); iter->valid = ((BLI_Iterator *)iter->internal.custom)->valid; } static void rna_Depsgraph_objects_next(CollectionPropertyIterator *iter) { DEG_objects_iterator_next(iter->internal.custom); iter->valid = ((BLI_Iterator *)iter->internal.custom)->valid; } static void rna_Depsgraph_objects_end(CollectionPropertyIterator *iter) { DEG_objects_iterator_end(iter->internal.custom); MEM_freeN(((BLI_Iterator *)iter->internal.custom)->data); MEM_freeN(iter->internal.custom); } static PointerRNA rna_Depsgraph_objects_get(CollectionPropertyIterator *iter) { Object *ob = ((BLI_Iterator *)iter->internal.custom)->current; return rna_pointer_inherit_refine(&iter->parent, &RNA_Object, ob); } /* Iteration over objects, extended version * * Contains extra information about duplicator and persistent ID. */ static void rna_Depsgraph_duplis_begin(CollectionPropertyIterator *iter, PointerRNA *ptr) { iter->internal.custom = MEM_callocN(sizeof(BLI_Iterator), __func__); DEGObjectsIteratorData *data = MEM_callocN(sizeof(DEGObjectsIteratorData), __func__); data->graph = (Depsgraph *)ptr->data; data->flag = DEG_OBJECT_ITER_FLAG_ALL; DEG_objects_iterator_begin(iter->internal.custom, data); iter->valid = ((BLI_Iterator *)iter->internal.custom)->valid; } static void rna_Depsgraph_duplis_next(CollectionPropertyIterator *iter) { DEG_objects_iterator_next(iter->internal.custom); iter->valid = ((BLI_Iterator *)iter->internal.custom)->valid; } static void rna_Depsgraph_duplis_end(CollectionPropertyIterator *iter) { DEG_objects_iterator_end(iter->internal.custom); MEM_freeN(((BLI_Iterator *)iter->internal.custom)->data); MEM_freeN(iter->internal.custom); } static PointerRNA rna_Depsgraph_duplis_get(CollectionPropertyIterator *iter) { BLI_Iterator *iterator = (BLI_Iterator *)iter->internal.custom; return rna_pointer_inherit_refine(&iter->parent, &RNA_DepsgraphIter, iterator); } static ID *rna_Depsgraph_evaluated_id_get(Depsgraph *depsgraph, ID *id_orig) { return DEG_get_evaluated_id(depsgraph, id_orig); } #else static void rna_def_depsgraph_iter(BlenderRNA *brna) { StructRNA *srna; PropertyRNA *prop; srna = RNA_def_struct(brna, "DepsgraphIter", NULL); RNA_def_struct_ui_text(srna, "Dependency Graph Iterator", "Extended information about dependency graph object iterator"); prop = RNA_def_property(srna, "object", PROP_POINTER, PROP_NONE); RNA_def_property_struct_type(prop, "Object"); RNA_def_property_ui_text(prop, "Object", "Object the iterator points to"); RNA_def_property_clear_flag(prop, PROP_ANIMATABLE | PROP_EDITABLE); RNA_def_property_pointer_funcs(prop, "rna_DepsgraphIter_object_get", NULL, NULL, NULL); prop = RNA_def_property(srna, "instance_object", PROP_POINTER, PROP_NONE); RNA_def_property_struct_type(prop, "Object"); RNA_def_property_ui_text(prop, "Instance Object", "Object which is being instanced by this iterator"); RNA_def_property_clear_flag(prop, PROP_ANIMATABLE | PROP_EDITABLE); RNA_def_property_pointer_funcs(prop, "rna_DepsgraphIter_instance_object_get", NULL, NULL, NULL); prop = RNA_def_property(srna, "parent", PROP_POINTER, PROP_NONE); RNA_def_property_struct_type(prop, "Object"); RNA_def_property_ui_text(prop, "Parent", "Parent of the duplication list"); RNA_def_property_clear_flag(prop, PROP_ANIMATABLE | PROP_EDITABLE); RNA_def_property_pointer_funcs(prop, "rna_DepsgraphIter_parent_get", NULL, NULL, NULL); prop = RNA_def_property(srna, "persistent_id", PROP_INT, PROP_NONE); RNA_def_property_ui_text(prop, "Persistent ID", "Persistent identifier for inter-frame matching of objects with motion blur"); RNA_def_property_array(prop, 2 * MAX_DUPLI_RECUR); RNA_def_property_clear_flag(prop, PROP_ANIMATABLE | PROP_EDITABLE); RNA_def_property_int_funcs(prop, "rna_DepsgraphIter_persistent_id_get", NULL, NULL); prop = RNA_def_property(srna, "orco", PROP_FLOAT, PROP_TRANSLATION); RNA_def_property_clear_flag(prop, PROP_ANIMATABLE | PROP_EDITABLE); /* Seems system is not smart enough to figure that getter function should return * array for PROP_TRANSLATION. */ RNA_def_property_array(prop, 3); RNA_def_property_ui_text(prop, "Generated Coordinates", "Generated coordinates in parent object space"); RNA_def_property_float_funcs(prop, "rna_DepsgraphIter_orco_get", NULL, NULL); prop = RNA_def_property(srna, "random_id", PROP_INT, PROP_UNSIGNED); RNA_def_property_clear_flag(prop, PROP_ANIMATABLE | PROP_EDITABLE); RNA_def_property_ui_text(prop, "Dupli random id", "Random id for this dupli object"); RNA_def_property_int_funcs(prop, "rna_DepsgraphIter_random_id_get", NULL, NULL); prop = RNA_def_property(srna, "uv", PROP_FLOAT, PROP_NONE); RNA_def_property_ui_text(prop, "UV Coordinates", "UV coordinates in parent object space"); RNA_def_property_array(prop, 2); RNA_def_property_clear_flag(prop, PROP_ANIMATABLE | PROP_EDITABLE); RNA_def_property_float_funcs(prop, "rna_DepsgraphIter_uv_get", NULL, NULL); prop = RNA_def_property(srna, "is_instance", PROP_BOOLEAN, PROP_NONE); RNA_def_property_clear_flag(prop, PROP_ANIMATABLE | PROP_EDITABLE); RNA_def_property_ui_text(prop, "Is Instance", "Denotes whether the object is ocming from dupli-list"); RNA_def_property_boolean_funcs(prop, "rna_DepsgraphIter_is_instance_get", NULL); } static void rna_def_depsgraph(BlenderRNA *brna) { StructRNA *srna; FunctionRNA *func; PropertyRNA *parm; PropertyRNA *prop; srna = RNA_def_struct(brna, "Depsgraph", NULL); RNA_def_struct_ui_text(srna, "Dependency Graph", ""); func = RNA_def_function(srna, "debug_graphviz", "rna_Depsgraph_debug_graphviz"); parm = RNA_def_string_file_path(func, "filename", NULL, FILE_MAX, "File Name", "File in which to store graphviz debug output"); RNA_def_parameter_flags(parm, 0, PARM_REQUIRED); func = RNA_def_function(srna, "debug_rebuild", "rna_Depsgraph_debug_rebuild"); RNA_def_function_flag(func, FUNC_USE_CONTEXT); func = RNA_def_function(srna, "debug_stats", "rna_Depsgraph_debug_stats"); RNA_def_function_ui_description(func, "Report the number of elements in the Dependency Graph"); RNA_def_function_flag(func, FUNC_USE_REPORTS); prop = RNA_def_property(srna, "objects", PROP_COLLECTION, PROP_NONE); RNA_def_property_struct_type(prop, "Object"); RNA_def_property_collection_funcs(prop, "rna_Depsgraph_objects_begin", "rna_Depsgraph_objects_next", "rna_Depsgraph_objects_end", "rna_Depsgraph_objects_get", NULL, NULL, NULL, NULL); func = RNA_def_function(srna, "evaluated_id_get", "rna_Depsgraph_evaluated_id_get"); parm = RNA_def_pointer(func, "id", "ID", "", "Original ID to get evaluated complementary part for"); RNA_def_parameter_flags(parm, 0, PARM_REQUIRED); parm = RNA_def_pointer(func, "evaluated_id", "ID", "", "Evaluated ID for the given original one"); RNA_def_function_return(func, parm); /* TODO(sergey): Find a better name. */ prop = RNA_def_property(srna, "duplis", PROP_COLLECTION, PROP_NONE); RNA_def_property_struct_type(prop, "DepsgraphIter"); RNA_def_property_collection_funcs(prop, "rna_Depsgraph_duplis_begin", "rna_Depsgraph_duplis_next", "rna_Depsgraph_duplis_end", "rna_Depsgraph_duplis_get", NULL, NULL, NULL, NULL); } void RNA_def_depsgraph(BlenderRNA *brna) { rna_def_depsgraph_iter(brna); rna_def_depsgraph(brna); } #endif