From b1fcc8e6ba25d5429cc868615c3b514e360ef912 Mon Sep 17 00:00:00 2001 From: Jacques Lucke Date: Wed, 21 Apr 2021 16:21:54 +0200 Subject: Fix T87659: crash when creating reroute loop in geometry nodes This was a regression introduced in rB0ff3f96a1b0193d69dae5ca69e7c93d6cdf34558. --- source/blender/nodes/intern/node_tree_ref.cc | 38 +++++++++++++++++++++------- 1 file changed, 29 insertions(+), 9 deletions(-) (limited to 'source/blender/nodes/intern') diff --git a/source/blender/nodes/intern/node_tree_ref.cc b/source/blender/nodes/intern/node_tree_ref.cc index d9ae594a860..e42572b9cb7 100644 --- a/source/blender/nodes/intern/node_tree_ref.cc +++ b/source/blender/nodes/intern/node_tree_ref.cc @@ -173,9 +173,12 @@ void NodeTreeRef::create_linked_socket_caches() /* Find logically linked sockets. */ Vector logically_linked_sockets; Vector logically_linked_skipped_sockets; + Vector handled_sockets; socket->foreach_logical_origin( [&](const OutputSocketRef &origin) { logically_linked_sockets.append(&origin); }, - [&](const SocketRef &socket) { logically_linked_skipped_sockets.append(&socket); }); + [&](const SocketRef &socket) { logically_linked_skipped_sockets.append(&socket); }, + false, + handled_sockets); if (logically_linked_sockets == directly_linked_sockets) { socket->logically_linked_sockets_ = socket->directly_linked_sockets_; } @@ -199,9 +202,11 @@ void NodeTreeRef::create_linked_socket_caches() /* Find logically linked sockets. */ Vector logically_linked_sockets; Vector logically_linked_skipped_sockets; + Vector handled_sockets; socket->foreach_logical_target( [&](const InputSocketRef &target) { logically_linked_sockets.append(&target); }, - [&](const SocketRef &socket) { logically_linked_skipped_sockets.append(&socket); }); + [&](const SocketRef &socket) { logically_linked_skipped_sockets.append(&socket); }, + handled_sockets); if (logically_linked_sockets == directly_linked_sockets) { socket->logically_linked_sockets_ = socket->directly_linked_sockets_; } @@ -216,8 +221,15 @@ void NodeTreeRef::create_linked_socket_caches() void InputSocketRef::foreach_logical_origin(FunctionRef origin_fn, FunctionRef skipped_fn, - bool only_follow_first_input_link) const + bool only_follow_first_input_link, + Vector &handled_sockets) const { + /* Protect against loops. */ + if (handled_sockets.contains(this)) { + return; + } + handled_sockets.append(this); + Span links_to_check = this->directly_linked_links(); if (only_follow_first_input_link) { links_to_check = links_to_check.take_front(1); @@ -233,7 +245,7 @@ void InputSocketRef::foreach_logical_origin(FunctionReffrom(); skipped_fn.call_safe(origin); skipped_fn.call_safe(mute_input); - mute_input.foreach_logical_origin(origin_fn, skipped_fn, true); + mute_input.foreach_logical_origin(origin_fn, skipped_fn, true, handled_sockets); break; } } @@ -252,9 +264,17 @@ void InputSocketRef::foreach_logical_origin(FunctionRef target_fn, - FunctionRef skipped_fn) const +void OutputSocketRef::foreach_logical_target( + FunctionRef target_fn, + FunctionRef skipped_fn, + Vector &handled_sockets) const { + /* Protect against loops. */ + if (handled_sockets.contains(this)) { + return; + } + handled_sockets.append(this); + for (const LinkRef *link : this->directly_linked_links()) { if (link->is_muted()) { continue; @@ -265,7 +285,7 @@ void OutputSocketRef::foreach_logical_target(FunctionRefto(); skipped_fn.call_safe(target); skipped_fn.call_safe(mute_output); - mute_output.foreach_logical_target(target_fn, skipped_fn); + mute_output.foreach_logical_target(target_fn, skipped_fn, handled_sockets); } } } -- cgit v1.2.3