This remove the complexity of queriying the locations at runtime and allows for more performance and upfront binding specifications. The benefit of doing everything at creation time is that we can assign binding points in a predictable order which is going to be somewhat the same for every similar shader. This also rewrite GPU_vertformat_from_shader to not use shaderface. This is to keep the shaderface simple. If it becomes necessary to not query the shader after creation (i.e: vulkan?) we could just create the vert format in advance at compilation for PyGPU shaders. Reviewed By: brecht Differential Revision: https://developer.blender.org/D7879
508 lines
17 KiB
C
508 lines
17 KiB
C
/*
|
|
* 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) 2016 by Mike Erwin.
|
|
* All rights reserved.
|
|
*/
|
|
|
|
/** \file
|
|
* \ingroup gpu
|
|
*
|
|
* GPU shader interface (C --> GLSL)
|
|
*/
|
|
|
|
#include "BKE_global.h"
|
|
|
|
#include "BLI_math_base.h"
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
#include "GPU_shader_interface.h"
|
|
|
|
#include "gpu_batch_private.h"
|
|
#include "gpu_context_private.h"
|
|
|
|
#include <stddef.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#define DEBUG_SHADER_INTERFACE 0
|
|
#define DEBUG_SHADER_UNIFORMS 0
|
|
|
|
#if DEBUG_SHADER_INTERFACE
|
|
# include <stdio.h>
|
|
#endif
|
|
|
|
static const char *BuiltinUniform_name(GPUUniformBuiltin u)
|
|
{
|
|
static const char *names[] = {
|
|
[GPU_UNIFORM_NONE] = NULL,
|
|
|
|
[GPU_UNIFORM_MODEL] = "ModelMatrix",
|
|
[GPU_UNIFORM_VIEW] = "ViewMatrix",
|
|
[GPU_UNIFORM_MODELVIEW] = "ModelViewMatrix",
|
|
[GPU_UNIFORM_PROJECTION] = "ProjectionMatrix",
|
|
[GPU_UNIFORM_VIEWPROJECTION] = "ViewProjectionMatrix",
|
|
[GPU_UNIFORM_MVP] = "ModelViewProjectionMatrix",
|
|
|
|
[GPU_UNIFORM_MODEL_INV] = "ModelMatrixInverse",
|
|
[GPU_UNIFORM_VIEW_INV] = "ViewMatrixInverse",
|
|
[GPU_UNIFORM_MODELVIEW_INV] = "ModelViewMatrixInverse",
|
|
[GPU_UNIFORM_PROJECTION_INV] = "ProjectionMatrixInverse",
|
|
[GPU_UNIFORM_VIEWPROJECTION_INV] = "ViewProjectionMatrixInverse",
|
|
|
|
[GPU_UNIFORM_NORMAL] = "NormalMatrix",
|
|
[GPU_UNIFORM_ORCO] = "OrcoTexCoFactors",
|
|
[GPU_UNIFORM_CLIPPLANES] = "WorldClipPlanes",
|
|
|
|
[GPU_UNIFORM_COLOR] = "color",
|
|
[GPU_UNIFORM_BASE_INSTANCE] = "baseInstance",
|
|
[GPU_UNIFORM_RESOURCE_CHUNK] = "resourceChunk",
|
|
[GPU_UNIFORM_RESOURCE_ID] = "resourceId",
|
|
[GPU_UNIFORM_SRGB_TRANSFORM] = "srgbTarget",
|
|
|
|
[GPU_UNIFORM_CUSTOM] = NULL,
|
|
[GPU_NUM_UNIFORMS] = NULL,
|
|
};
|
|
|
|
return names[u];
|
|
}
|
|
|
|
GPU_INLINE bool match(const char *a, const char *b)
|
|
{
|
|
return strcmp(a, b) == 0;
|
|
}
|
|
|
|
GPU_INLINE uint hash_string(const char *str)
|
|
{
|
|
uint i = 0, c;
|
|
while ((c = *str++)) {
|
|
i = i * 37 + c;
|
|
}
|
|
return i;
|
|
}
|
|
|
|
GPU_INLINE uint32_t set_input_name(GPUShaderInterface *shaderface,
|
|
GPUShaderInput *input,
|
|
char *name,
|
|
uint32_t name_len)
|
|
{
|
|
/* remove "[0]" from array name */
|
|
if (name[name_len - 1] == ']') {
|
|
name[name_len - 3] = '\0';
|
|
name_len -= 3;
|
|
}
|
|
|
|
input->name_offset = (uint32_t)(name - shaderface->name_buffer);
|
|
input->name_hash = hash_string(name);
|
|
return name_len + 1; /* include NULL terminator */
|
|
}
|
|
|
|
GPU_INLINE const GPUShaderInput *input_lookup(const GPUShaderInterface *shaderface,
|
|
const GPUShaderInput *const inputs,
|
|
const uint inputs_len,
|
|
const char *name)
|
|
{
|
|
const uint name_hash = hash_string(name);
|
|
/* Simple linear search for now. */
|
|
for (int i = inputs_len - 1; i >= 0; i--) {
|
|
if (inputs[i].name_hash == name_hash) {
|
|
if ((i > 0) && UNLIKELY(inputs[i - 1].name_hash == name_hash)) {
|
|
/* Hash colision resolve. */
|
|
for (; i >= 0 && inputs[i].name_hash == name_hash; i--) {
|
|
if (match(name, shaderface->name_buffer + inputs[i].name_offset)) {
|
|
return inputs + i; /* not found */
|
|
}
|
|
}
|
|
return NULL; /* not found */
|
|
}
|
|
else {
|
|
/* This is a bit dangerous since we could have a hash collision.
|
|
* where the asked uniform that does not exist has the same hash
|
|
* as a real uniform. */
|
|
BLI_assert(match(name, shaderface->name_buffer + inputs[i].name_offset));
|
|
return inputs + i;
|
|
}
|
|
}
|
|
}
|
|
return NULL; /* not found */
|
|
}
|
|
|
|
/* Note that this modify the src array. */
|
|
GPU_INLINE void sort_input_list(GPUShaderInput *dst, GPUShaderInput *src, const uint input_len)
|
|
{
|
|
for (uint i = 0; i < input_len; i++) {
|
|
GPUShaderInput *input_src = &src[0];
|
|
for (uint j = 1; j < input_len; j++) {
|
|
if (src[j].name_hash > input_src->name_hash) {
|
|
input_src = &src[j];
|
|
}
|
|
}
|
|
dst[i] = *input_src;
|
|
input_src->name_hash = 0;
|
|
}
|
|
}
|
|
|
|
static int block_binding(int32_t program, uint32_t block_index)
|
|
{
|
|
/* For now just assign a consecutive index. In the future, we should set it in
|
|
* the shader using layout(binding = i) and query its value. */
|
|
glUniformBlockBinding(program, block_index, block_index);
|
|
return block_index;
|
|
}
|
|
|
|
static int sampler_binding(int32_t program,
|
|
uint32_t uniform_index,
|
|
int32_t uniform_location,
|
|
int *sampler_len)
|
|
{
|
|
/* Identify sampler uniforms and asign sampler units to them. */
|
|
GLint type;
|
|
glGetActiveUniformsiv(program, 1, &uniform_index, GL_UNIFORM_TYPE, &type);
|
|
|
|
switch (type) {
|
|
case GL_SAMPLER_1D:
|
|
case GL_SAMPLER_2D:
|
|
case GL_SAMPLER_3D:
|
|
case GL_SAMPLER_CUBE:
|
|
case GL_SAMPLER_CUBE_MAP_ARRAY_ARB: /* OpenGL 4.0 */
|
|
case GL_SAMPLER_1D_SHADOW:
|
|
case GL_SAMPLER_2D_SHADOW:
|
|
case GL_SAMPLER_1D_ARRAY:
|
|
case GL_SAMPLER_2D_ARRAY:
|
|
case GL_SAMPLER_1D_ARRAY_SHADOW:
|
|
case GL_SAMPLER_2D_ARRAY_SHADOW:
|
|
case GL_SAMPLER_2D_MULTISAMPLE:
|
|
case GL_SAMPLER_2D_MULTISAMPLE_ARRAY:
|
|
case GL_SAMPLER_CUBE_SHADOW:
|
|
case GL_SAMPLER_BUFFER:
|
|
case GL_INT_SAMPLER_1D:
|
|
case GL_INT_SAMPLER_2D:
|
|
case GL_INT_SAMPLER_3D:
|
|
case GL_INT_SAMPLER_CUBE:
|
|
case GL_INT_SAMPLER_1D_ARRAY:
|
|
case GL_INT_SAMPLER_2D_ARRAY:
|
|
case GL_INT_SAMPLER_2D_MULTISAMPLE:
|
|
case GL_INT_SAMPLER_2D_MULTISAMPLE_ARRAY:
|
|
case GL_INT_SAMPLER_BUFFER:
|
|
case GL_UNSIGNED_INT_SAMPLER_1D:
|
|
case GL_UNSIGNED_INT_SAMPLER_2D:
|
|
case GL_UNSIGNED_INT_SAMPLER_3D:
|
|
case GL_UNSIGNED_INT_SAMPLER_CUBE:
|
|
case GL_UNSIGNED_INT_SAMPLER_1D_ARRAY:
|
|
case GL_UNSIGNED_INT_SAMPLER_2D_ARRAY:
|
|
case GL_UNSIGNED_INT_SAMPLER_2D_MULTISAMPLE:
|
|
case GL_UNSIGNED_INT_SAMPLER_2D_MULTISAMPLE_ARRAY:
|
|
case GL_UNSIGNED_INT_SAMPLER_BUFFER: {
|
|
/* For now just assign a consecutive index. In the future, we should set it in
|
|
* the shader using layout(binding = i) and query its value. */
|
|
int binding = *sampler_len;
|
|
glUniform1i(uniform_location, binding);
|
|
(*sampler_len)++;
|
|
return binding;
|
|
}
|
|
default:
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
GPUShaderInterface *GPU_shaderinterface_create(int32_t program)
|
|
{
|
|
GLint max_attr_name_len = 0, attr_len = 0;
|
|
glGetProgramiv(program, GL_ACTIVE_ATTRIBUTE_MAX_LENGTH, &max_attr_name_len);
|
|
glGetProgramiv(program, GL_ACTIVE_ATTRIBUTES, &attr_len);
|
|
|
|
GLint max_ubo_name_len = 0, ubo_len = 0;
|
|
glGetProgramiv(program, GL_ACTIVE_UNIFORM_BLOCK_MAX_NAME_LENGTH, &max_ubo_name_len);
|
|
glGetProgramiv(program, GL_ACTIVE_UNIFORM_BLOCKS, &ubo_len);
|
|
|
|
GLint max_uniform_name_len = 0, active_uniform_len = 0, uniform_len = 0;
|
|
glGetProgramiv(program, GL_ACTIVE_UNIFORM_MAX_LENGTH, &max_uniform_name_len);
|
|
glGetProgramiv(program, GL_ACTIVE_UNIFORMS, &active_uniform_len);
|
|
uniform_len = active_uniform_len;
|
|
|
|
/* Work around driver bug with Intel HD 4600 on Windows 7/8, where
|
|
* GL_ACTIVE_UNIFORM_BLOCK_MAX_NAME_LENGTH does not work. */
|
|
if (attr_len > 0 && max_attr_name_len == 0) {
|
|
max_attr_name_len = 256;
|
|
}
|
|
if (ubo_len > 0 && max_ubo_name_len == 0) {
|
|
max_ubo_name_len = 256;
|
|
}
|
|
if (uniform_len > 0 && max_uniform_name_len == 0) {
|
|
max_uniform_name_len = 256;
|
|
}
|
|
|
|
/* GL_ACTIVE_UNIFORMS lied to us! Remove the UBO uniforms from the total before
|
|
* allocating the uniform array. */
|
|
GLint *uniforms_block_index = MEM_mallocN(sizeof(GLint) * active_uniform_len, __func__);
|
|
if (uniform_len > 0) {
|
|
GLuint *indices = MEM_mallocN(sizeof(GLuint) * active_uniform_len, __func__);
|
|
for (uint i = 0; i < uniform_len; i++) {
|
|
indices[i] = i;
|
|
}
|
|
|
|
glGetActiveUniformsiv(
|
|
program, uniform_len, indices, GL_UNIFORM_BLOCK_INDEX, uniforms_block_index);
|
|
|
|
MEM_freeN(indices);
|
|
|
|
for (int i = 0; i < active_uniform_len; i++) {
|
|
/* If GL_UNIFORM_BLOCK_INDEX is not -1 it means the uniform belongs to a UBO. */
|
|
if (uniforms_block_index[i] != -1) {
|
|
uniform_len--;
|
|
}
|
|
}
|
|
}
|
|
|
|
uint32_t name_buffer_offset = 0;
|
|
const uint32_t name_buffer_len = attr_len * max_attr_name_len + ubo_len * max_ubo_name_len +
|
|
uniform_len * max_uniform_name_len;
|
|
|
|
int input_tot_len = attr_len + ubo_len + uniform_len;
|
|
size_t interface_size = sizeof(GPUShaderInterface) + sizeof(GPUShaderInput) * input_tot_len;
|
|
|
|
GPUShaderInterface *shaderface = MEM_callocN(interface_size, "GPUShaderInterface");
|
|
shaderface->attribute_len = attr_len;
|
|
shaderface->ubo_len = ubo_len;
|
|
shaderface->uniform_len = uniform_len;
|
|
shaderface->name_buffer = MEM_mallocN(name_buffer_len, "name_buffer");
|
|
GPUShaderInput *inputs = shaderface->inputs;
|
|
|
|
/* Temp buffer. */
|
|
int input_tmp_len = max_iii(attr_len, ubo_len, uniform_len);
|
|
GPUShaderInput *inputs_tmp = MEM_mallocN(sizeof(GPUShaderInput) * input_tmp_len, "name_buffer");
|
|
|
|
/* Attributes */
|
|
shaderface->enabled_attr_mask = 0;
|
|
for (int i = 0, idx = 0; i < attr_len; i++) {
|
|
char *name = shaderface->name_buffer + name_buffer_offset;
|
|
GLsizei remaining_buffer = name_buffer_len - name_buffer_offset;
|
|
GLsizei name_len = 0;
|
|
GLenum type;
|
|
GLint size;
|
|
|
|
glGetActiveAttrib(program, i, remaining_buffer, &name_len, &size, &type, name);
|
|
GLint location = glGetAttribLocation(program, name);
|
|
/* Ignore OpenGL names like `gl_BaseInstanceARB`, `gl_InstanceID` and `gl_VertexID`. */
|
|
if (location == -1) {
|
|
shaderface->attribute_len--;
|
|
continue;
|
|
}
|
|
|
|
GPUShaderInput *input = &inputs_tmp[idx++];
|
|
input->location = input->binding = location;
|
|
|
|
name_buffer_offset += set_input_name(shaderface, input, name, name_len);
|
|
shaderface->enabled_attr_mask |= (1 << input->location);
|
|
}
|
|
sort_input_list(inputs, inputs_tmp, shaderface->attribute_len);
|
|
inputs += shaderface->attribute_len;
|
|
|
|
/* Uniform Blocks */
|
|
for (int i = 0, idx = 0; i < ubo_len; i++) {
|
|
char *name = shaderface->name_buffer + name_buffer_offset;
|
|
GLsizei remaining_buffer = name_buffer_len - name_buffer_offset;
|
|
GLsizei name_len = 0;
|
|
|
|
glGetActiveUniformBlockName(program, i, remaining_buffer, &name_len, name);
|
|
|
|
GPUShaderInput *input = &inputs_tmp[idx++];
|
|
input->binding = input->location = block_binding(program, i);
|
|
|
|
name_buffer_offset += set_input_name(shaderface, input, name, name_len);
|
|
shaderface->enabled_ubo_mask |= (1 << input->binding);
|
|
}
|
|
sort_input_list(inputs, inputs_tmp, shaderface->ubo_len);
|
|
inputs += shaderface->ubo_len;
|
|
|
|
/* Uniforms */
|
|
for (int i = 0, idx = 0, sampler = 0; i < active_uniform_len; i++) {
|
|
/* If GL_UNIFORM_BLOCK_INDEX is not -1 it means the uniform belongs to a UBO. */
|
|
if (uniforms_block_index[i] != -1) {
|
|
continue;
|
|
}
|
|
char *name = shaderface->name_buffer + name_buffer_offset;
|
|
GLsizei remaining_buffer = name_buffer_len - name_buffer_offset;
|
|
GLsizei name_len = 0;
|
|
|
|
glGetActiveUniformName(program, i, remaining_buffer, &name_len, name);
|
|
|
|
GPUShaderInput *input = &inputs_tmp[idx++];
|
|
input->location = glGetUniformLocation(program, name);
|
|
input->binding = sampler_binding(program, i, input->location, &sampler);
|
|
|
|
name_buffer_offset += set_input_name(shaderface, input, name, name_len);
|
|
shaderface->enabled_tex_mask |= (input->binding != -1) ? (1lu << input->binding) : 0lu;
|
|
}
|
|
sort_input_list(inputs, inputs_tmp, shaderface->uniform_len);
|
|
|
|
/* Builtin Uniforms */
|
|
for (GPUUniformBuiltin u = GPU_UNIFORM_NONE + 1; u < GPU_UNIFORM_CUSTOM; u++) {
|
|
shaderface->builtins[u].location = glGetUniformLocation(program, BuiltinUniform_name(u));
|
|
shaderface->builtins[u].binding = -1;
|
|
}
|
|
|
|
/* Batches ref buffer */
|
|
shaderface->batches_len = GPU_SHADERINTERFACE_REF_ALLOC_COUNT;
|
|
shaderface->batches = MEM_callocN(shaderface->batches_len * sizeof(GPUBatch *),
|
|
"GPUShaderInterface batches");
|
|
|
|
MEM_freeN(uniforms_block_index);
|
|
MEM_freeN(inputs_tmp);
|
|
|
|
/* Resize name buffer to save some memory. */
|
|
if (name_buffer_offset < name_buffer_len) {
|
|
shaderface->name_buffer = MEM_reallocN(shaderface->name_buffer, name_buffer_offset);
|
|
}
|
|
|
|
#if DEBUG_SHADER_INTERFACE
|
|
char *name_buf = shaderface->name_buffer;
|
|
printf("--- GPUShaderInterface %p, program %d ---\n", shaderface, program);
|
|
if (shaderface->attribute_len > 0) {
|
|
printf("Attributes {\n");
|
|
for (int i = 0; i < shaderface->attribute_len; i++) {
|
|
GPUShaderInput *input = shaderface->inputs + i;
|
|
printf("\t(location = %d) %s;\n", input->location, name_buf + input->name_offset);
|
|
}
|
|
printf("};\n");
|
|
}
|
|
if (shaderface->ubo_len > 0) {
|
|
printf("Uniform Buffer Objects {\n");
|
|
for (int i = 0; i < shaderface->ubo_len; i++) {
|
|
GPUShaderInput *input = shaderface->inputs + shaderface->attribute_len + i;
|
|
printf("\t(binding = %d) %s;\n", input->binding, name_buf + input->name_offset);
|
|
}
|
|
printf("};\n");
|
|
}
|
|
if (shaderface->enabled_tex_mask > 0) {
|
|
printf("Samplers {\n");
|
|
for (int i = 0; i < shaderface->ubo_len; i++) {
|
|
GPUShaderInput *input = shaderface->inputs + shaderface->attribute_len +
|
|
shaderface->ubo_len + i;
|
|
if (input->binding != -1) {
|
|
printf("\t(location = %d, binding = %d) %s;\n",
|
|
input->location,
|
|
input->binding,
|
|
name_buf + input->name_offset);
|
|
}
|
|
}
|
|
printf("};\n");
|
|
}
|
|
if (shaderface->uniform_len > 0) {
|
|
printf("Uniforms {\n");
|
|
for (int i = 0; i < shaderface->uniform_len; i++) {
|
|
GPUShaderInput *input = shaderface->inputs + shaderface->attribute_len +
|
|
shaderface->ubo_len + i;
|
|
if (input->binding == -1) {
|
|
printf("\t(location = %d) %s;\n", input->location, name_buf + input->name_offset);
|
|
}
|
|
}
|
|
printf("};\n");
|
|
}
|
|
printf("--- GPUShaderInterface end ---\n\n");
|
|
#endif
|
|
|
|
return shaderface;
|
|
}
|
|
|
|
void GPU_shaderinterface_discard(GPUShaderInterface *shaderface)
|
|
{
|
|
/* Free memory used by name_buffer. */
|
|
MEM_freeN(shaderface->name_buffer);
|
|
/* Remove this interface from all linked Batches vao cache. */
|
|
for (int i = 0; i < shaderface->batches_len; i++) {
|
|
if (shaderface->batches[i] != NULL) {
|
|
gpu_batch_remove_interface_ref(shaderface->batches[i], shaderface);
|
|
}
|
|
}
|
|
MEM_freeN(shaderface->batches);
|
|
/* Free memory used by shader interface by its self. */
|
|
MEM_freeN(shaderface);
|
|
}
|
|
|
|
const GPUShaderInput *GPU_shaderinterface_attr(const GPUShaderInterface *shaderface,
|
|
const char *name)
|
|
{
|
|
uint ofs = 0;
|
|
return input_lookup(shaderface, shaderface->inputs + ofs, shaderface->attribute_len, name);
|
|
}
|
|
|
|
const GPUShaderInput *GPU_shaderinterface_ubo(const GPUShaderInterface *shaderface,
|
|
const char *name)
|
|
{
|
|
uint ofs = shaderface->attribute_len;
|
|
return input_lookup(shaderface, shaderface->inputs + ofs, shaderface->ubo_len, name);
|
|
}
|
|
|
|
const GPUShaderInput *GPU_shaderinterface_uniform(const GPUShaderInterface *shaderface,
|
|
const char *name)
|
|
{
|
|
uint ofs = shaderface->attribute_len + shaderface->ubo_len;
|
|
return input_lookup(shaderface, shaderface->inputs + ofs, shaderface->uniform_len, name);
|
|
}
|
|
|
|
const GPUShaderInput *GPU_shaderinterface_uniform_ensure(const GPUShaderInterface *shaderface,
|
|
const char *name)
|
|
{
|
|
const GPUShaderInput *input = GPU_shaderinterface_uniform(shaderface, name);
|
|
/* If input is not found add it so it's found next time. */
|
|
if (input == NULL) {
|
|
if ((G.debug & G_DEBUG_GPU) && (input->location == -1)) {
|
|
fprintf(stderr, "GPUShaderInterface: Warning: Uniform '%s' not found!\n", name);
|
|
}
|
|
}
|
|
return input;
|
|
}
|
|
|
|
const GPUShaderInput *GPU_shaderinterface_uniform_builtin(const GPUShaderInterface *shaderface,
|
|
GPUUniformBuiltin builtin)
|
|
{
|
|
#if TRUST_NO_ONE
|
|
assert(builtin != GPU_UNIFORM_NONE);
|
|
assert(builtin != GPU_UNIFORM_CUSTOM);
|
|
assert(builtin != GPU_NUM_UNIFORMS);
|
|
#endif
|
|
return &shaderface->builtins[builtin];
|
|
}
|
|
|
|
void GPU_shaderinterface_add_batch_ref(GPUShaderInterface *shaderface, GPUBatch *batch)
|
|
{
|
|
int i; /* find first unused slot */
|
|
for (i = 0; i < shaderface->batches_len; i++) {
|
|
if (shaderface->batches[i] == NULL) {
|
|
break;
|
|
}
|
|
}
|
|
if (i == shaderface->batches_len) {
|
|
/* Not enough place, realloc the array. */
|
|
i = shaderface->batches_len;
|
|
shaderface->batches_len += GPU_SHADERINTERFACE_REF_ALLOC_COUNT;
|
|
shaderface->batches = MEM_recallocN(shaderface->batches,
|
|
sizeof(GPUBatch *) * shaderface->batches_len);
|
|
}
|
|
shaderface->batches[i] = batch;
|
|
}
|
|
|
|
void GPU_shaderinterface_remove_batch_ref(GPUShaderInterface *shaderface, GPUBatch *batch)
|
|
{
|
|
for (int i = 0; i < shaderface->batches_len; i++) {
|
|
if (shaderface->batches[i] == batch) {
|
|
shaderface->batches[i] = NULL;
|
|
break; /* cannot have duplicates */
|
|
}
|
|
}
|
|
}
|