2011-10-23 17:52:20 +00:00
|
|
|
/*
|
2011-09-05 21:01:50 +00:00
|
|
|
* ***** 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) 2007 Blender Foundation.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* The Original Code is: all of this file.
|
|
|
|
*
|
|
|
|
* Contributor(s): Nathan Letwory.
|
|
|
|
*
|
|
|
|
* ***** END GPL LICENSE BLOCK *****
|
|
|
|
*/
|
|
|
|
|
|
|
|
/** \file blender/nodes/intern/node_exec.c
|
|
|
|
* \ingroup nodes
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include "DNA_node_types.h"
|
|
|
|
|
|
|
|
#include "BLI_listbase.h"
|
|
|
|
#include "BLI_math.h"
|
|
|
|
#include "BLI_utildefines.h"
|
|
|
|
|
|
|
|
#include "BKE_node.h"
|
|
|
|
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
|
|
|
|
#include "node_exec.h"
|
|
|
|
|
|
|
|
|
|
|
|
/* for a given socket, find the actual stack entry */
|
|
|
|
bNodeStack *node_get_socket_stack(bNodeStack *stack, bNodeSocket *sock)
|
|
|
|
{
|
2012-06-01 13:42:18 +00:00
|
|
|
if (stack && sock)
|
|
|
|
return stack + sock->stack_index;
|
|
|
|
return NULL;
|
2011-09-05 21:01:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void node_get_stack(bNode *node, bNodeStack *stack, bNodeStack **in, bNodeStack **out)
|
|
|
|
{
|
|
|
|
bNodeSocket *sock;
|
|
|
|
|
|
|
|
/* build pointer stack */
|
|
|
|
if (in) {
|
2012-03-24 06:38:07 +00:00
|
|
|
for (sock= node->inputs.first; sock; sock= sock->next) {
|
2011-09-05 21:01:50 +00:00
|
|
|
*(in++) = node_get_socket_stack(stack, sock);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (out) {
|
2012-03-24 06:38:07 +00:00
|
|
|
for (sock= node->outputs.first; sock; sock= sock->next) {
|
2011-09-05 21:01:50 +00:00
|
|
|
*(out++) = node_get_socket_stack(stack, sock);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-24 12:57:48 +00:00
|
|
|
static void node_init_input_index(bNodeSocket *sock, int *index)
|
2011-09-05 21:01:50 +00:00
|
|
|
{
|
|
|
|
if (sock->link && sock->link->fromsock) {
|
|
|
|
sock->stack_index = sock->link->fromsock->stack_index;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
sock->stack_index = (*index)++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-24 12:57:48 +00:00
|
|
|
static void node_init_output_index(bNodeSocket *sock, int *index, ListBase *internal_links)
|
2011-09-05 21:01:50 +00:00
|
|
|
{
|
2012-10-24 12:57:48 +00:00
|
|
|
if (internal_links) {
|
|
|
|
bNodeLink *link;
|
|
|
|
/* copy the stack index from internally connected input to skip the node */
|
|
|
|
for (link = internal_links->first; link; link = link->next) {
|
|
|
|
if (link->tosock == sock) {
|
|
|
|
sock->stack_index = link->fromsock->stack_index;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* if not internally connected, assign a new stack index anyway to avoid bad stack access */
|
|
|
|
if (!link)
|
|
|
|
sock->stack_index = (*index)++;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
sock->stack_index = (*index)++;
|
|
|
|
}
|
2011-09-05 21:01:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* basic preparation of socket stacks */
|
|
|
|
static struct bNodeStack *setup_stack(bNodeStack *stack, bNodeSocket *sock)
|
|
|
|
{
|
|
|
|
bNodeStack *ns = node_get_socket_stack(stack, sock);
|
2012-10-22 08:15:51 +00:00
|
|
|
float null_value[4] = {0.0f, 0.0f, 0.0f, 0.0f};
|
2011-09-05 21:01:50 +00:00
|
|
|
|
|
|
|
/* don't mess with remote socket stacks, these are initialized by other nodes! */
|
|
|
|
if (sock->link)
|
|
|
|
return ns;
|
|
|
|
|
|
|
|
ns->sockettype = sock->type;
|
|
|
|
|
|
|
|
if (sock->default_value) {
|
|
|
|
switch (sock->type) {
|
|
|
|
case SOCK_FLOAT:
|
2012-12-28 14:19:05 +00:00
|
|
|
ns->vec[0] = ((bNodeSocketValueFloat *)sock->default_value)->value;
|
2011-09-05 21:01:50 +00:00
|
|
|
break;
|
|
|
|
case SOCK_VECTOR:
|
2012-12-28 14:19:05 +00:00
|
|
|
copy_v3_v3(ns->vec, ((bNodeSocketValueVector *)sock->default_value)->value);
|
2011-09-05 21:01:50 +00:00
|
|
|
break;
|
|
|
|
case SOCK_RGBA:
|
2012-12-28 14:19:05 +00:00
|
|
|
copy_v4_v4(ns->vec, ((bNodeSocketValueRGBA *)sock->default_value)->value);
|
2011-09-05 21:01:50 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
switch (sock->type) {
|
|
|
|
case SOCK_FLOAT:
|
|
|
|
ns->vec[0] = 0.0f;
|
|
|
|
break;
|
|
|
|
case SOCK_VECTOR:
|
|
|
|
copy_v3_v3(ns->vec, null_value);
|
|
|
|
break;
|
|
|
|
case SOCK_RGBA:
|
|
|
|
copy_v4_v4(ns->vec, null_value);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ns;
|
|
|
|
}
|
|
|
|
|
|
|
|
bNodeTreeExec *ntree_exec_begin(bNodeTree *ntree)
|
|
|
|
{
|
|
|
|
bNodeTreeExec *exec;
|
|
|
|
bNode *node;
|
|
|
|
bNodeExec *nodeexec;
|
|
|
|
bNodeSocket *sock, *gsock;
|
|
|
|
bNodeStack *ns;
|
2012-10-24 12:57:48 +00:00
|
|
|
int index;
|
2011-09-05 21:01:50 +00:00
|
|
|
bNode **nodelist;
|
|
|
|
int totnodes, n;
|
|
|
|
|
2012-03-24 06:38:07 +00:00
|
|
|
if ((ntree->init & NTREE_TYPE_INIT)==0)
|
2011-09-05 21:01:50 +00:00
|
|
|
ntreeInitTypes(ntree);
|
|
|
|
|
|
|
|
/* get a dependency-sorted list of nodes */
|
|
|
|
ntreeGetDependencyList(ntree, &nodelist, &totnodes);
|
|
|
|
|
|
|
|
/* XXX could let callbacks do this for specialized data */
|
|
|
|
exec = MEM_callocN(sizeof(bNodeTreeExec), "node tree execution data");
|
|
|
|
/* backpointer to node tree */
|
|
|
|
exec->nodetree = ntree;
|
|
|
|
|
2012-10-24 12:57:48 +00:00
|
|
|
/* set stack indices */
|
|
|
|
index = 0;
|
2011-09-05 21:01:50 +00:00
|
|
|
/* group inputs essentially work as outputs */
|
2012-03-24 06:38:07 +00:00
|
|
|
for (gsock=ntree->inputs.first; gsock; gsock = gsock->next)
|
2012-10-24 12:57:48 +00:00
|
|
|
node_init_output_index(gsock, &index, NULL);
|
2012-03-24 06:38:07 +00:00
|
|
|
for (n=0; n < totnodes; ++n) {
|
2011-09-05 21:01:50 +00:00
|
|
|
node = nodelist[n];
|
|
|
|
|
|
|
|
node->stack_index = index;
|
|
|
|
|
|
|
|
/* init node socket stack indexes */
|
|
|
|
for (sock=node->inputs.first; sock; sock=sock->next)
|
|
|
|
node_init_input_index(sock, &index);
|
2012-10-24 12:57:48 +00:00
|
|
|
|
2012-10-25 16:49:06 +00:00
|
|
|
if (node->flag & NODE_MUTED || node->type == NODE_REROUTE) {
|
2012-10-24 12:57:48 +00:00
|
|
|
for (sock=node->outputs.first; sock; sock=sock->next)
|
2012-10-25 16:49:06 +00:00
|
|
|
node_init_output_index(sock, &index, &node->internal_links);
|
2012-10-24 12:57:48 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
for (sock=node->outputs.first; sock; sock=sock->next)
|
|
|
|
node_init_output_index(sock, &index, NULL);
|
|
|
|
}
|
2011-09-05 21:01:50 +00:00
|
|
|
}
|
|
|
|
/* group outputs essentially work as inputs */
|
2012-03-24 06:38:07 +00:00
|
|
|
for (gsock=ntree->outputs.first; gsock; gsock = gsock->next)
|
2011-09-05 21:01:50 +00:00
|
|
|
node_init_input_index(gsock, &index);
|
|
|
|
|
|
|
|
/* allocated exec data pointers for nodes */
|
|
|
|
exec->totnodes = totnodes;
|
|
|
|
exec->nodeexec = MEM_callocN(exec->totnodes * sizeof(bNodeExec), "node execution data");
|
|
|
|
/* allocate data pointer for node stack */
|
|
|
|
exec->stacksize = index;
|
|
|
|
exec->stack = MEM_callocN(exec->stacksize * sizeof(bNodeStack), "bNodeStack");
|
|
|
|
|
2011-10-12 12:55:32 +00:00
|
|
|
/* all non-const results are considered inputs */
|
|
|
|
for (n=0; n < exec->stacksize; ++n)
|
|
|
|
exec->stack[n].hasinput = 1;
|
|
|
|
|
2011-09-05 21:01:50 +00:00
|
|
|
/* prepare group tree inputs */
|
|
|
|
for (sock=ntree->inputs.first; sock; sock=sock->next) {
|
2011-10-15 07:19:34 +00:00
|
|
|
/* ns = */ setup_stack(exec->stack, sock);
|
2011-09-05 21:01:50 +00:00
|
|
|
}
|
|
|
|
/* prepare all internal nodes for execution */
|
2012-03-24 06:38:07 +00:00
|
|
|
for (n=0, nodeexec= exec->nodeexec; n < totnodes; ++n, ++nodeexec) {
|
2011-09-05 21:01:50 +00:00
|
|
|
node = nodeexec->node = nodelist[n];
|
|
|
|
|
|
|
|
/* tag inputs */
|
|
|
|
for (sock=node->inputs.first; sock; sock=sock->next) {
|
|
|
|
/* disable the node if an input link is invalid */
|
2012-03-24 06:38:07 +00:00
|
|
|
if (sock->link && !(sock->link->flag & NODE_LINK_VALID))
|
2011-09-05 21:01:50 +00:00
|
|
|
node->need_exec= 0;
|
|
|
|
|
|
|
|
ns = setup_stack(exec->stack, sock);
|
2011-10-12 12:55:32 +00:00
|
|
|
ns->hasoutput = 1;
|
2011-09-05 21:01:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* tag all outputs */
|
|
|
|
for (sock=node->outputs.first; sock; sock=sock->next) {
|
2011-10-15 07:19:34 +00:00
|
|
|
/* ns = */ setup_stack(exec->stack, sock);
|
2011-09-05 21:01:50 +00:00
|
|
|
}
|
|
|
|
|
2012-03-24 06:38:07 +00:00
|
|
|
if (node->typeinfo->initexecfunc)
|
2011-09-05 21:01:50 +00:00
|
|
|
nodeexec->data = node->typeinfo->initexecfunc(node);
|
|
|
|
}
|
|
|
|
/* prepare group tree outputs */
|
|
|
|
for (sock=ntree->outputs.first; sock; sock=sock->next) {
|
|
|
|
ns = setup_stack(exec->stack, sock);
|
|
|
|
ns->hasoutput = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nodelist)
|
|
|
|
MEM_freeN(nodelist);
|
|
|
|
|
|
|
|
return exec;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ntree_exec_end(bNodeTreeExec *exec)
|
|
|
|
{
|
|
|
|
bNodeExec *nodeexec;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
if (exec->stack)
|
|
|
|
MEM_freeN(exec->stack);
|
|
|
|
|
2012-03-24 06:38:07 +00:00
|
|
|
for (n=0, nodeexec= exec->nodeexec; n < exec->totnodes; ++n, ++nodeexec) {
|
2011-09-05 21:01:50 +00:00
|
|
|
if (nodeexec->node->typeinfo->freeexecfunc)
|
|
|
|
nodeexec->node->typeinfo->freeexecfunc(nodeexec->node, nodeexec->data);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (exec->nodeexec)
|
|
|
|
MEM_freeN(exec->nodeexec);
|
|
|
|
|
|
|
|
MEM_freeN(exec);
|
|
|
|
}
|
|
|
|
|
2013-01-15 12:16:44 +00:00
|
|
|
/**** Material/Texture trees ****/
|
2011-09-05 21:01:50 +00:00
|
|
|
|
|
|
|
bNodeThreadStack *ntreeGetThreadStack(bNodeTreeExec *exec, int thread)
|
|
|
|
{
|
|
|
|
ListBase *lb= &exec->threadstack[thread];
|
|
|
|
bNodeThreadStack *nts;
|
|
|
|
|
2012-03-24 06:38:07 +00:00
|
|
|
for (nts=lb->first; nts; nts=nts->next) {
|
|
|
|
if (!nts->used) {
|
2012-05-19 13:28:19 +00:00
|
|
|
nts->used = TRUE;
|
2011-09-05 21:01:50 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!nts) {
|
|
|
|
nts= MEM_callocN(sizeof(bNodeThreadStack), "bNodeThreadStack");
|
|
|
|
nts->stack= MEM_dupallocN(exec->stack);
|
2012-05-19 13:28:19 +00:00
|
|
|
nts->used = TRUE;
|
2011-09-05 21:01:50 +00:00
|
|
|
BLI_addtail(lb, nts);
|
|
|
|
}
|
|
|
|
|
|
|
|
return nts;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ntreeReleaseThreadStack(bNodeThreadStack *nts)
|
|
|
|
{
|
|
|
|
nts->used = 0;
|
|
|
|
}
|
|
|
|
|
2013-01-24 16:11:07 +00:00
|
|
|
int ntreeExecThreadNodes(bNodeTreeExec *exec, bNodeThreadStack *nts, void *callerdata, int thread)
|
2011-09-05 21:01:50 +00:00
|
|
|
{
|
|
|
|
bNodeStack *nsin[MAX_SOCKET]; /* arbitrary... watch this */
|
|
|
|
bNodeStack *nsout[MAX_SOCKET]; /* arbitrary... watch this */
|
|
|
|
bNodeExec *nodeexec;
|
|
|
|
bNode *node;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
/* nodes are presorted, so exec is in order of list */
|
|
|
|
|
2012-03-24 06:38:07 +00:00
|
|
|
for (n=0, nodeexec= exec->nodeexec; n < exec->totnodes; ++n, ++nodeexec) {
|
2011-09-05 21:01:50 +00:00
|
|
|
node = nodeexec->node;
|
2012-03-24 06:38:07 +00:00
|
|
|
if (node->need_exec) {
|
2011-09-05 21:01:50 +00:00
|
|
|
node_get_stack(node, nts->stack, nsin, nsout);
|
2011-11-20 16:38:23 +00:00
|
|
|
/* Handle muted nodes...
|
|
|
|
* If the mute func is not set, assume the node should never be muted,
|
|
|
|
* and hence execute it!
|
|
|
|
*/
|
2013-01-24 19:31:44 +00:00
|
|
|
// if (node->typeinfo->compatibility == NODE_NEW_SHADING)
|
|
|
|
// return 0;
|
2012-03-24 06:38:07 +00:00
|
|
|
if (node->typeinfo->execfunc)
|
2011-09-05 21:01:50 +00:00
|
|
|
node->typeinfo->execfunc(callerdata, node, nsin, nsout);
|
|
|
|
else if (node->typeinfo->newexecfunc)
|
|
|
|
node->typeinfo->newexecfunc(callerdata, thread, node, nodeexec->data, nsin, nsout);
|
|
|
|
}
|
|
|
|
}
|
2013-01-24 16:11:07 +00:00
|
|
|
|
2013-01-24 18:56:04 +00:00
|
|
|
/* signal to that all went OK, for render */
|
2013-01-24 16:11:07 +00:00
|
|
|
return 1;
|
2011-09-05 21:01:50 +00:00
|
|
|
}
|