2020-02-12 12:48:44 +01:00
|
|
|
/*
|
|
|
|
* 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) 2005 Blender Foundation.
|
|
|
|
* All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/** \file
|
|
|
|
* \ingroup gpu
|
|
|
|
*
|
|
|
|
* Intermediate node graph for generating GLSL shaders.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
|
|
|
|
#include "DNA_node_types.h"
|
|
|
|
|
|
|
|
#include "BLI_utildefines.h"
|
|
|
|
#include "BLI_ghash.h"
|
|
|
|
#include "BLI_listbase.h"
|
|
|
|
#include "BLI_string.h"
|
|
|
|
|
|
|
|
#include "gpu_material_library.h"
|
|
|
|
#include "gpu_node_graph.h"
|
|
|
|
|
|
|
|
/* Node Link Functions */
|
|
|
|
|
|
|
|
static GPUNodeLink *gpu_node_link_create(void)
|
|
|
|
{
|
|
|
|
GPUNodeLink *link = MEM_callocN(sizeof(GPUNodeLink), "GPUNodeLink");
|
|
|
|
link->users++;
|
|
|
|
|
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gpu_node_link_free(GPUNodeLink *link)
|
|
|
|
{
|
|
|
|
link->users--;
|
|
|
|
|
|
|
|
if (link->users < 0) {
|
|
|
|
fprintf(stderr, "gpu_node_link_free: negative refcount\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (link->users == 0) {
|
|
|
|
if (link->output) {
|
|
|
|
link->output->link = NULL;
|
|
|
|
}
|
|
|
|
MEM_freeN(link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Node Functions */
|
|
|
|
|
|
|
|
static GPUNode *gpu_node_create(const char *name)
|
|
|
|
{
|
|
|
|
GPUNode *node = MEM_callocN(sizeof(GPUNode), "GPUNode");
|
|
|
|
|
|
|
|
node->name = name;
|
|
|
|
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gpu_node_input_link(GPUNode *node, GPUNodeLink *link, const eGPUType type)
|
|
|
|
{
|
|
|
|
GPUInput *input;
|
|
|
|
GPUNode *outnode;
|
|
|
|
const char *name;
|
|
|
|
|
|
|
|
if (link->link_type == GPU_NODE_LINK_OUTPUT) {
|
|
|
|
outnode = link->output->node;
|
|
|
|
name = outnode->name;
|
|
|
|
input = outnode->inputs.first;
|
|
|
|
|
|
|
|
if ((STR_ELEM(name, "set_value", "set_rgb", "set_rgba")) && (input->type == type)) {
|
|
|
|
input = MEM_dupallocN(outnode->inputs.first);
|
|
|
|
if (input->link) {
|
|
|
|
input->link->users++;
|
|
|
|
}
|
|
|
|
BLI_addtail(&node->inputs, input);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
input = MEM_callocN(sizeof(GPUInput), "GPUInput");
|
|
|
|
input->node = node;
|
|
|
|
input->type = type;
|
|
|
|
|
|
|
|
switch (link->link_type) {
|
|
|
|
case GPU_NODE_LINK_BUILTIN:
|
|
|
|
input->source = GPU_SOURCE_BUILTIN;
|
|
|
|
input->builtin = link->builtin;
|
|
|
|
break;
|
|
|
|
case GPU_NODE_LINK_OUTPUT:
|
|
|
|
input->source = GPU_SOURCE_OUTPUT;
|
|
|
|
input->link = link;
|
|
|
|
link->users++;
|
|
|
|
break;
|
2020-02-27 13:55:29 +01:00
|
|
|
case GPU_NODE_LINK_IMAGE:
|
|
|
|
case GPU_NODE_LINK_IMAGE_TILED:
|
2020-02-12 12:48:44 +01:00
|
|
|
case GPU_NODE_LINK_COLORBAND:
|
|
|
|
input->source = GPU_SOURCE_TEX;
|
2020-02-27 13:55:29 +01:00
|
|
|
input->texture = link->texture;
|
2020-02-12 12:48:44 +01:00
|
|
|
break;
|
2020-02-27 13:55:29 +01:00
|
|
|
case GPU_NODE_LINK_IMAGE_TILED_MAPPING:
|
|
|
|
input->source = GPU_SOURCE_TEX_TILED_MAPPING;
|
|
|
|
input->texture = link->texture;
|
2020-02-12 12:48:44 +01:00
|
|
|
break;
|
2020-03-11 14:58:19 +01:00
|
|
|
case GPU_NODE_LINK_VOLUME_GRID:
|
|
|
|
input->source = GPU_SOURCE_VOLUME_GRID;
|
|
|
|
input->volume_grid = link->volume_grid;
|
|
|
|
break;
|
2020-02-12 12:48:44 +01:00
|
|
|
case GPU_NODE_LINK_ATTR:
|
|
|
|
input->source = GPU_SOURCE_ATTR;
|
2020-02-27 13:55:29 +01:00
|
|
|
input->attr = link->attr;
|
|
|
|
input->attr->gputype = type;
|
2020-02-12 12:48:44 +01:00
|
|
|
break;
|
|
|
|
case GPU_NODE_LINK_CONSTANT:
|
|
|
|
input->source = (type == GPU_CLOSURE) ? GPU_SOURCE_STRUCT : GPU_SOURCE_CONSTANT;
|
|
|
|
break;
|
|
|
|
case GPU_NODE_LINK_UNIFORM:
|
|
|
|
input->source = GPU_SOURCE_UNIFORM;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ELEM(input->source, GPU_SOURCE_CONSTANT, GPU_SOURCE_UNIFORM)) {
|
|
|
|
memcpy(input->vec, link->data, type * sizeof(float));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (link->link_type != GPU_NODE_LINK_OUTPUT) {
|
|
|
|
MEM_freeN(link);
|
|
|
|
}
|
|
|
|
BLI_addtail(&node->inputs, input);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *gpu_uniform_set_function_from_type(eNodeSocketDatatype type)
|
|
|
|
{
|
|
|
|
switch (type) {
|
|
|
|
/* For now INT is supported as float. */
|
|
|
|
case SOCK_INT:
|
|
|
|
case SOCK_FLOAT:
|
|
|
|
return "set_value";
|
|
|
|
case SOCK_VECTOR:
|
|
|
|
return "set_rgb";
|
|
|
|
case SOCK_RGBA:
|
|
|
|
return "set_rgba";
|
|
|
|
default:
|
|
|
|
BLI_assert(!"No gpu function for non-supported eNodeSocketDatatype");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Link stack uniform buffer.
|
|
|
|
* This is called for the input/output sockets that are note connected.
|
|
|
|
*/
|
|
|
|
static GPUNodeLink *gpu_uniformbuffer_link(GPUMaterial *mat,
|
|
|
|
bNode *node,
|
|
|
|
GPUNodeStack *stack,
|
|
|
|
const int index,
|
|
|
|
const eNodeSocketInOut in_out)
|
|
|
|
{
|
|
|
|
bNodeSocket *socket;
|
|
|
|
|
|
|
|
if (in_out == SOCK_IN) {
|
|
|
|
socket = BLI_findlink(&node->inputs, index);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
socket = BLI_findlink(&node->outputs, index);
|
|
|
|
}
|
|
|
|
|
|
|
|
BLI_assert(socket != NULL);
|
|
|
|
BLI_assert(socket->in_out == in_out);
|
|
|
|
|
|
|
|
if ((socket->flag & SOCK_HIDE_VALUE) == 0) {
|
|
|
|
GPUNodeLink *link;
|
|
|
|
switch (socket->type) {
|
|
|
|
case SOCK_FLOAT: {
|
|
|
|
bNodeSocketValueFloat *socket_data = socket->default_value;
|
|
|
|
link = GPU_uniform(&socket_data->value);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SOCK_VECTOR: {
|
|
|
|
bNodeSocketValueVector *socket_data = socket->default_value;
|
|
|
|
link = GPU_uniform(socket_data->value);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SOCK_RGBA: {
|
|
|
|
bNodeSocketValueRGBA *socket_data = socket->default_value;
|
|
|
|
link = GPU_uniform(socket_data->value);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return NULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (in_out == SOCK_IN) {
|
|
|
|
GPU_link(mat, gpu_uniform_set_function_from_type(socket->type), link, &stack->link);
|
|
|
|
}
|
|
|
|
return link;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gpu_node_input_socket(
|
|
|
|
GPUMaterial *material, bNode *bnode, GPUNode *node, GPUNodeStack *sock, const int index)
|
|
|
|
{
|
|
|
|
if (sock->link) {
|
|
|
|
gpu_node_input_link(node, sock->link, sock->type);
|
|
|
|
}
|
|
|
|
else if ((material != NULL) &&
|
|
|
|
(gpu_uniformbuffer_link(material, bnode, sock, index, SOCK_IN) != NULL)) {
|
|
|
|
gpu_node_input_link(node, sock->link, sock->type);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
gpu_node_input_link(node, GPU_constant(sock->vec), sock->type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gpu_node_output(GPUNode *node, const eGPUType type, GPUNodeLink **link)
|
|
|
|
{
|
|
|
|
GPUOutput *output = MEM_callocN(sizeof(GPUOutput), "GPUOutput");
|
|
|
|
|
|
|
|
output->type = type;
|
|
|
|
output->node = node;
|
|
|
|
|
|
|
|
if (link) {
|
|
|
|
*link = output->link = gpu_node_link_create();
|
|
|
|
output->link->link_type = GPU_NODE_LINK_OUTPUT;
|
|
|
|
output->link->output = output;
|
|
|
|
|
|
|
|
/* note: the caller owns the reference to the link, GPUOutput
|
|
|
|
* merely points to it, and if the node is destroyed it will
|
|
|
|
* set that pointer to NULL */
|
|
|
|
}
|
|
|
|
|
|
|
|
BLI_addtail(&node->outputs, output);
|
|
|
|
}
|
|
|
|
|
2020-02-27 13:55:29 +01:00
|
|
|
/* Attributes and Textures */
|
2020-02-12 12:48:44 +01:00
|
|
|
|
2020-02-27 13:55:29 +01:00
|
|
|
static GPUMaterialAttribute *gpu_node_graph_add_attribute(GPUNodeGraph *graph,
|
|
|
|
CustomDataType type,
|
|
|
|
const char *name)
|
2020-02-12 12:48:44 +01:00
|
|
|
{
|
|
|
|
/* Fall back to the UV layer, which matches old behavior. */
|
|
|
|
if (type == CD_AUTO_FROM_NAME && name[0] == '\0') {
|
2020-02-27 13:55:29 +01:00
|
|
|
type = CD_MTFACE;
|
2020-02-12 12:48:44 +01:00
|
|
|
}
|
2020-02-27 13:55:29 +01:00
|
|
|
|
|
|
|
/* Find existing attribute. */
|
|
|
|
int num_attributes = 0;
|
|
|
|
GPUMaterialAttribute *attr = graph->attributes.first;
|
|
|
|
for (; attr; attr = attr->next) {
|
|
|
|
if (attr->type == type && STREQ(attr->name, name)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
num_attributes++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add new requested attribute if it's within GPU limits. */
|
|
|
|
if (attr == NULL && num_attributes < GPU_MAX_ATTR) {
|
|
|
|
attr = MEM_callocN(sizeof(*attr), __func__);
|
|
|
|
attr->type = type;
|
|
|
|
STRNCPY(attr->name, name);
|
|
|
|
attr->id = num_attributes;
|
|
|
|
BLI_addtail(&graph->attributes, attr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (attr != NULL) {
|
|
|
|
attr->users++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return attr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GPUMaterialTexture *gpu_node_graph_add_texture(GPUNodeGraph *graph,
|
|
|
|
Image *ima,
|
|
|
|
ImageUser *iuser,
|
|
|
|
struct GPUTexture **colorband,
|
|
|
|
GPUNodeLinkType link_type)
|
|
|
|
{
|
|
|
|
/* Find existing texture. */
|
|
|
|
int num_textures = 0;
|
|
|
|
GPUMaterialTexture *tex = graph->textures.first;
|
|
|
|
for (; tex; tex = tex->next) {
|
|
|
|
if (tex->ima == ima && tex->colorband == colorband) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
num_textures++;
|
2020-02-12 12:48:44 +01:00
|
|
|
}
|
2020-02-27 13:55:29 +01:00
|
|
|
|
|
|
|
/* Add new requested texture. */
|
|
|
|
if (tex == NULL) {
|
|
|
|
tex = MEM_callocN(sizeof(*tex), __func__);
|
|
|
|
tex->ima = ima;
|
|
|
|
tex->iuser = iuser;
|
|
|
|
tex->colorband = colorband;
|
|
|
|
BLI_snprintf(tex->sampler_name, sizeof(tex->sampler_name), "samp%d", num_textures);
|
|
|
|
if (ELEM(link_type, GPU_NODE_LINK_IMAGE_TILED, GPU_NODE_LINK_IMAGE_TILED_MAPPING)) {
|
|
|
|
BLI_snprintf(
|
|
|
|
tex->tiled_mapping_name, sizeof(tex->tiled_mapping_name), "tsamp%d", num_textures);
|
|
|
|
}
|
|
|
|
BLI_addtail(&graph->textures, tex);
|
|
|
|
}
|
|
|
|
|
|
|
|
tex->users++;
|
|
|
|
|
|
|
|
return tex;
|
|
|
|
}
|
|
|
|
|
2020-03-11 14:58:19 +01:00
|
|
|
static GPUMaterialVolumeGrid *gpu_node_graph_add_volume_grid(GPUNodeGraph *graph, const char *name)
|
|
|
|
{
|
|
|
|
/* Find existing volume grid. */
|
|
|
|
int num_grids = 0;
|
|
|
|
GPUMaterialVolumeGrid *grid = graph->volume_grids.first;
|
|
|
|
for (; grid; grid = grid->next) {
|
|
|
|
if (STREQ(grid->name, name)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
num_grids++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add new requested volume grid. */
|
|
|
|
if (grid == NULL) {
|
|
|
|
grid = MEM_callocN(sizeof(*grid), __func__);
|
|
|
|
grid->name = BLI_strdup(name);
|
|
|
|
BLI_snprintf(grid->sampler_name, sizeof(grid->sampler_name), "vsamp%d", num_grids);
|
|
|
|
BLI_addtail(&graph->volume_grids, grid);
|
|
|
|
}
|
|
|
|
|
|
|
|
grid->users++;
|
|
|
|
|
|
|
|
return grid;
|
|
|
|
}
|
|
|
|
|
2020-02-27 13:55:29 +01:00
|
|
|
/* Creating Inputs */
|
|
|
|
|
|
|
|
GPUNodeLink *GPU_attribute(GPUMaterial *mat, const CustomDataType type, const char *name)
|
|
|
|
{
|
|
|
|
GPUNodeGraph *graph = gpu_material_node_graph(mat);
|
|
|
|
GPUMaterialAttribute *attr = gpu_node_graph_add_attribute(graph, type, name);
|
|
|
|
|
|
|
|
if (attr == NULL) {
|
|
|
|
static const float zero_data[GPU_MAX_CONSTANT_DATA] = {0.0f};
|
|
|
|
return GPU_constant(zero_data);
|
|
|
|
}
|
|
|
|
|
|
|
|
GPUNodeLink *link = gpu_node_link_create();
|
|
|
|
link->link_type = GPU_NODE_LINK_ATTR;
|
|
|
|
link->attr = attr;
|
2020-02-12 12:48:44 +01:00
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
2020-02-27 13:55:29 +01:00
|
|
|
GPUNodeLink *GPU_constant(const float *num)
|
2020-02-12 12:48:44 +01:00
|
|
|
{
|
|
|
|
GPUNodeLink *link = gpu_node_link_create();
|
|
|
|
link->link_type = GPU_NODE_LINK_CONSTANT;
|
|
|
|
link->data = num;
|
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
2020-02-27 13:55:29 +01:00
|
|
|
GPUNodeLink *GPU_uniform(const float *num)
|
2020-02-12 12:48:44 +01:00
|
|
|
{
|
|
|
|
GPUNodeLink *link = gpu_node_link_create();
|
|
|
|
link->link_type = GPU_NODE_LINK_UNIFORM;
|
|
|
|
link->data = num;
|
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
2020-02-27 13:55:29 +01:00
|
|
|
GPUNodeLink *GPU_image(GPUMaterial *mat, Image *ima, ImageUser *iuser)
|
2020-02-12 12:48:44 +01:00
|
|
|
{
|
2020-02-27 13:55:29 +01:00
|
|
|
GPUNodeGraph *graph = gpu_material_node_graph(mat);
|
2020-02-12 12:48:44 +01:00
|
|
|
GPUNodeLink *link = gpu_node_link_create();
|
2020-02-27 13:55:29 +01:00
|
|
|
link->link_type = GPU_NODE_LINK_IMAGE;
|
|
|
|
link->texture = gpu_node_graph_add_texture(graph, ima, iuser, NULL, link->link_type);
|
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
|
|
|
GPUNodeLink *GPU_image_tiled(GPUMaterial *mat, Image *ima, ImageUser *iuser)
|
|
|
|
{
|
|
|
|
GPUNodeGraph *graph = gpu_material_node_graph(mat);
|
|
|
|
GPUNodeLink *link = gpu_node_link_create();
|
|
|
|
link->link_type = GPU_NODE_LINK_IMAGE_TILED;
|
|
|
|
link->texture = gpu_node_graph_add_texture(graph, ima, iuser, NULL, link->link_type);
|
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
|
|
|
GPUNodeLink *GPU_image_tiled_mapping(GPUMaterial *mat, Image *ima, ImageUser *iuser)
|
|
|
|
{
|
|
|
|
GPUNodeGraph *graph = gpu_material_node_graph(mat);
|
|
|
|
GPUNodeLink *link = gpu_node_link_create();
|
|
|
|
link->link_type = GPU_NODE_LINK_IMAGE_TILED_MAPPING;
|
|
|
|
link->texture = gpu_node_graph_add_texture(graph, ima, iuser, NULL, link->link_type);
|
2020-02-12 12:48:44 +01:00
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
|
|
|
GPUNodeLink *GPU_color_band(GPUMaterial *mat, int size, float *pixels, float *row)
|
|
|
|
{
|
2020-02-27 13:55:29 +01:00
|
|
|
struct GPUTexture **colorband = gpu_material_ramp_texture_row_set(mat, size, pixels, row);
|
|
|
|
MEM_freeN(pixels);
|
|
|
|
|
|
|
|
GPUNodeGraph *graph = gpu_material_node_graph(mat);
|
2020-02-12 12:48:44 +01:00
|
|
|
GPUNodeLink *link = gpu_node_link_create();
|
|
|
|
link->link_type = GPU_NODE_LINK_COLORBAND;
|
2020-02-27 13:55:29 +01:00
|
|
|
link->texture = gpu_node_graph_add_texture(graph, NULL, NULL, colorband, link->link_type);
|
2020-02-12 12:48:44 +01:00
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
2020-03-11 14:58:19 +01:00
|
|
|
GPUNodeLink *GPU_volume_grid(GPUMaterial *mat, const char *name)
|
|
|
|
{
|
|
|
|
/* NOTE: this could be optimized by automatically merging duplicate
|
|
|
|
* lookups of the same attribute. */
|
|
|
|
GPUNodeGraph *graph = gpu_material_node_graph(mat);
|
|
|
|
GPUNodeLink *link = gpu_node_link_create();
|
|
|
|
link->link_type = GPU_NODE_LINK_VOLUME_GRID;
|
|
|
|
link->volume_grid = gpu_node_graph_add_volume_grid(graph, name);
|
|
|
|
|
|
|
|
/* Two special cases, where we adjust the output values of smoke grids to
|
|
|
|
* bring the into standard range without having to modify the grid values. */
|
|
|
|
if (strcmp(name, "color") == 0) {
|
|
|
|
GPU_link(mat, "node_attribute_volume_color", link, &link);
|
|
|
|
}
|
|
|
|
else if (strcmp(name, "temperature") == 0) {
|
|
|
|
GPU_link(mat, "node_attribute_volume_temperature", link, &link);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
GPU_link(mat, "node_attribute_volume", link, &link);
|
|
|
|
}
|
|
|
|
|
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
2020-02-12 12:48:44 +01:00
|
|
|
GPUNodeLink *GPU_builtin(eGPUBuiltin builtin)
|
|
|
|
{
|
|
|
|
GPUNodeLink *link = gpu_node_link_create();
|
|
|
|
link->link_type = GPU_NODE_LINK_BUILTIN;
|
|
|
|
link->builtin = builtin;
|
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Creating Nodes */
|
|
|
|
|
|
|
|
bool GPU_link(GPUMaterial *mat, const char *name, ...)
|
|
|
|
{
|
|
|
|
GSet *used_libraries = gpu_material_used_libraries(mat);
|
|
|
|
GPUNode *node;
|
|
|
|
GPUFunction *function;
|
|
|
|
GPUNodeLink *link, **linkptr;
|
|
|
|
va_list params;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
function = gpu_material_library_use_function(used_libraries, name);
|
|
|
|
if (!function) {
|
|
|
|
fprintf(stderr, "GPU failed to find function %s\n", name);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
node = gpu_node_create(name);
|
|
|
|
|
|
|
|
va_start(params, name);
|
|
|
|
for (i = 0; i < function->totparam; i++) {
|
|
|
|
if (function->paramqual[i] != FUNCTION_QUAL_IN) {
|
|
|
|
linkptr = va_arg(params, GPUNodeLink **);
|
|
|
|
gpu_node_output(node, function->paramtype[i], linkptr);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
link = va_arg(params, GPUNodeLink *);
|
|
|
|
gpu_node_input_link(node, link, function->paramtype[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
va_end(params);
|
|
|
|
|
2020-02-27 13:55:29 +01:00
|
|
|
GPUNodeGraph *graph = gpu_material_node_graph(mat);
|
|
|
|
BLI_addtail(&graph->nodes, node);
|
2020-02-12 12:48:44 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GPU_stack_link(GPUMaterial *material,
|
|
|
|
bNode *bnode,
|
|
|
|
const char *name,
|
|
|
|
GPUNodeStack *in,
|
|
|
|
GPUNodeStack *out,
|
|
|
|
...)
|
|
|
|
{
|
|
|
|
GSet *used_libraries = gpu_material_used_libraries(material);
|
|
|
|
GPUNode *node;
|
|
|
|
GPUFunction *function;
|
|
|
|
GPUNodeLink *link, **linkptr;
|
|
|
|
va_list params;
|
|
|
|
int i, totin, totout;
|
|
|
|
|
|
|
|
function = gpu_material_library_use_function(used_libraries, name);
|
|
|
|
if (!function) {
|
|
|
|
fprintf(stderr, "GPU failed to find function %s\n", name);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
node = gpu_node_create(name);
|
|
|
|
totin = 0;
|
|
|
|
totout = 0;
|
|
|
|
|
|
|
|
if (in) {
|
|
|
|
for (i = 0; !in[i].end; i++) {
|
|
|
|
if (in[i].type != GPU_NONE) {
|
|
|
|
gpu_node_input_socket(material, bnode, node, &in[i], i);
|
|
|
|
totin++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (out) {
|
|
|
|
for (i = 0; !out[i].end; i++) {
|
|
|
|
if (out[i].type != GPU_NONE) {
|
|
|
|
gpu_node_output(node, out[i].type, &out[i].link);
|
|
|
|
totout++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
va_start(params, out);
|
|
|
|
for (i = 0; i < function->totparam; i++) {
|
|
|
|
if (function->paramqual[i] != FUNCTION_QUAL_IN) {
|
|
|
|
if (totout == 0) {
|
|
|
|
linkptr = va_arg(params, GPUNodeLink **);
|
|
|
|
gpu_node_output(node, function->paramtype[i], linkptr);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
totout--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (totin == 0) {
|
|
|
|
link = va_arg(params, GPUNodeLink *);
|
|
|
|
if (link->socket) {
|
|
|
|
gpu_node_input_socket(NULL, NULL, node, link->socket, -1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
gpu_node_input_link(node, link, function->paramtype[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
totin--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
va_end(params);
|
|
|
|
|
2020-02-27 13:55:29 +01:00
|
|
|
GPUNodeGraph *graph = gpu_material_node_graph(material);
|
|
|
|
BLI_addtail(&graph->nodes, node);
|
2020-02-12 12:48:44 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
GPUNodeLink *GPU_uniformbuffer_link_out(GPUMaterial *mat,
|
|
|
|
bNode *node,
|
|
|
|
GPUNodeStack *stack,
|
|
|
|
const int index)
|
|
|
|
{
|
|
|
|
return gpu_uniformbuffer_link(mat, node, stack, index, SOCK_OUT);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Node Graph */
|
|
|
|
|
|
|
|
static void gpu_inputs_free(ListBase *inputs)
|
|
|
|
{
|
|
|
|
GPUInput *input;
|
|
|
|
|
|
|
|
for (input = inputs->first; input; input = input->next) {
|
2020-02-27 13:55:29 +01:00
|
|
|
if (input->source == GPU_SOURCE_ATTR) {
|
|
|
|
input->attr->users--;
|
|
|
|
}
|
|
|
|
else if (ELEM(input->source, GPU_SOURCE_TEX, GPU_SOURCE_TEX_TILED_MAPPING)) {
|
|
|
|
input->texture->users--;
|
|
|
|
}
|
2020-03-11 14:58:19 +01:00
|
|
|
else if (ELEM(input->source, GPU_SOURCE_VOLUME_GRID)) {
|
|
|
|
input->volume_grid->users--;
|
|
|
|
}
|
2020-02-27 13:55:29 +01:00
|
|
|
|
2020-02-12 12:48:44 +01:00
|
|
|
if (input->link) {
|
|
|
|
gpu_node_link_free(input->link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BLI_freelistN(inputs);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gpu_node_free(GPUNode *node)
|
|
|
|
{
|
|
|
|
GPUOutput *output;
|
|
|
|
|
|
|
|
gpu_inputs_free(&node->inputs);
|
|
|
|
|
|
|
|
for (output = node->outputs.first; output; output = output->next) {
|
|
|
|
if (output->link) {
|
|
|
|
output->link->output = NULL;
|
|
|
|
gpu_node_link_free(output->link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BLI_freelistN(&node->outputs);
|
|
|
|
MEM_freeN(node);
|
|
|
|
}
|
|
|
|
|
2020-02-14 10:47:20 +01:00
|
|
|
/* Free intermediate node graph. */
|
|
|
|
void gpu_node_graph_free_nodes(GPUNodeGraph *graph)
|
2020-02-12 12:48:44 +01:00
|
|
|
{
|
|
|
|
GPUNode *node;
|
|
|
|
|
|
|
|
while ((node = BLI_pophead(&graph->nodes))) {
|
|
|
|
gpu_node_free(node);
|
|
|
|
}
|
|
|
|
|
|
|
|
graph->outlink = NULL;
|
|
|
|
}
|
|
|
|
|
2020-02-14 10:47:20 +01:00
|
|
|
/* Free both node graph and requested attributes and textures. */
|
|
|
|
void gpu_node_graph_free(GPUNodeGraph *graph)
|
2020-02-12 12:48:44 +01:00
|
|
|
{
|
2020-02-14 10:47:20 +01:00
|
|
|
gpu_node_graph_free_nodes(graph);
|
2020-03-11 14:58:19 +01:00
|
|
|
|
|
|
|
for (GPUMaterialVolumeGrid *grid = graph->volume_grids.first; grid; grid = grid->next) {
|
|
|
|
MEM_SAFE_FREE(grid->name);
|
|
|
|
}
|
|
|
|
BLI_freelistN(&graph->volume_grids);
|
2020-02-14 10:47:20 +01:00
|
|
|
BLI_freelistN(&graph->textures);
|
2020-02-27 13:55:29 +01:00
|
|
|
BLI_freelistN(&graph->attributes);
|
2020-02-12 12:48:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Prune Unused Nodes */
|
|
|
|
|
|
|
|
static void gpu_nodes_tag(GPUNodeLink *link)
|
|
|
|
{
|
|
|
|
GPUNode *node;
|
|
|
|
GPUInput *input;
|
|
|
|
|
|
|
|
if (!link->output) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
node = link->output->node;
|
|
|
|
if (node->tag) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
node->tag = true;
|
|
|
|
for (input = node->inputs.first; input; input = input->next) {
|
|
|
|
if (input->link) {
|
|
|
|
gpu_nodes_tag(input->link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpu_node_graph_prune_unused(GPUNodeGraph *graph)
|
|
|
|
{
|
2020-02-27 13:55:29 +01:00
|
|
|
for (GPUNode *node = graph->nodes.first; node; node = node->next) {
|
2020-02-12 12:48:44 +01:00
|
|
|
node->tag = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
gpu_nodes_tag(graph->outlink);
|
|
|
|
|
2020-02-27 13:55:29 +01:00
|
|
|
for (GPUNode *node = graph->nodes.first, *next = NULL; node; node = next) {
|
2020-02-12 12:48:44 +01:00
|
|
|
next = node->next;
|
|
|
|
|
|
|
|
if (!node->tag) {
|
|
|
|
BLI_remlink(&graph->nodes, node);
|
|
|
|
gpu_node_free(node);
|
|
|
|
}
|
|
|
|
}
|
2020-02-27 13:55:29 +01:00
|
|
|
|
|
|
|
for (GPUMaterialAttribute *attr = graph->attributes.first, *next = NULL; attr; attr = next) {
|
|
|
|
next = attr->next;
|
|
|
|
if (attr->users == 0) {
|
|
|
|
BLI_freelinkN(&graph->attributes, attr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (GPUMaterialTexture *tex = graph->textures.first, *next = NULL; tex; tex = next) {
|
|
|
|
next = tex->next;
|
|
|
|
if (tex->users == 0) {
|
|
|
|
BLI_freelinkN(&graph->textures, tex);
|
|
|
|
}
|
|
|
|
}
|
2020-03-11 14:58:19 +01:00
|
|
|
|
|
|
|
for (GPUMaterialVolumeGrid *grid = graph->volume_grids.first, *next = NULL; grid; grid = next) {
|
|
|
|
next = grid->next;
|
|
|
|
if (grid->users == 0) {
|
|
|
|
MEM_SAFE_FREE(grid->name);
|
|
|
|
BLI_freelinkN(&graph->volume_grids, grid);
|
|
|
|
}
|
|
|
|
}
|
2020-02-12 12:48:44 +01:00
|
|
|
}
|