235 lines
7.6 KiB
C++
235 lines
7.6 KiB
C++
/*
|
|
* Copyright 2011, Blender Foundation.
|
|
*
|
|
* 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:
|
|
* Jeroen Bakker
|
|
* Monique Dewanchand
|
|
*/
|
|
|
|
#include "COM_NodeOperation.h"
|
|
#include <typeinfo>
|
|
#include "COM_InputSocket.h"
|
|
#include "COM_SocketConnection.h"
|
|
#include "COM_defines.h"
|
|
#include "stdio.h"
|
|
|
|
NodeOperation::NodeOperation()
|
|
{
|
|
this->resolutionInputSocketIndex = 0;
|
|
this->complex = false;
|
|
this->width = 0;
|
|
this->height = 0;
|
|
this->openCL = false;
|
|
}
|
|
|
|
void NodeOperation::determineResolution(unsigned int resolution[], unsigned int preferredResolution[])
|
|
{
|
|
unsigned int temp[2];
|
|
unsigned int temp2[2];
|
|
vector<InputSocket*> &inputsockets = this->getInputSockets();
|
|
|
|
for (unsigned int index = 0 ; index < inputsockets.size();index++) {
|
|
InputSocket *inputSocket = inputsockets[index];
|
|
if (inputSocket->isConnected()) {
|
|
if (index == this->resolutionInputSocketIndex) {
|
|
inputSocket->determineResolution(resolution, preferredResolution);
|
|
temp2[0] = resolution[0];
|
|
temp2[1] = resolution[1];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
for (unsigned int index = 0 ; index < inputsockets.size();index++) {
|
|
InputSocket *inputSocket = inputsockets[index];
|
|
if (inputSocket->isConnected()) {
|
|
if (index != resolutionInputSocketIndex) {
|
|
inputSocket->determineResolution(temp, temp2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
void NodeOperation::setResolutionInputSocketIndex(unsigned int index)
|
|
{
|
|
this->resolutionInputSocketIndex = index;
|
|
}
|
|
void NodeOperation::initExecution()
|
|
{
|
|
}
|
|
|
|
void NodeOperation::initMutex()
|
|
{
|
|
BLI_mutex_init(&mutex);
|
|
}
|
|
void NodeOperation::deinitMutex()
|
|
{
|
|
BLI_mutex_end(&mutex);
|
|
}
|
|
void NodeOperation::deinitExecution()
|
|
{
|
|
}
|
|
SocketReader *NodeOperation::getInputSocketReader(unsigned int inputSocketIndex)
|
|
{
|
|
return this->getInputSocket(inputSocketIndex)->getReader();
|
|
}
|
|
NodeOperation *NodeOperation::getInputOperation(unsigned int inputSocketIndex)
|
|
{
|
|
return this->getInputSocket(inputSocketIndex)->getOperation();
|
|
}
|
|
|
|
void NodeOperation::getConnectedInputSockets(vector<InputSocket*> *sockets)
|
|
{
|
|
vector<InputSocket*> &inputsockets = this->getInputSockets();
|
|
for (vector<InputSocket*>::iterator iterator = inputsockets.begin() ; iterator!= inputsockets.end() ; iterator++) {
|
|
InputSocket *socket = *iterator;
|
|
if (socket->isConnected()) {
|
|
sockets->push_back(socket);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool NodeOperation::determineDependingAreaOfInterest(rcti * input, ReadBufferOperation *readOperation, rcti *output)
|
|
{
|
|
if (this->isInputNode()) {
|
|
BLI_init_rcti(output, input->xmin, input->xmax, input->ymin, input->ymax);
|
|
return false;
|
|
}
|
|
else {
|
|
unsigned int index;
|
|
vector<InputSocket*> &inputsockets = this->getInputSockets();
|
|
|
|
for (index = 0 ; index < inputsockets.size() ; index++) {
|
|
InputSocket *inputsocket = inputsockets[index];
|
|
if (inputsocket->isConnected()) {
|
|
NodeOperation *inputoperation = (NodeOperation*)inputsocket->getConnection()->getFromNode();
|
|
bool result = inputoperation->determineDependingAreaOfInterest(input, readOperation, output);
|
|
if (result) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
cl_mem NodeOperation::COM_clAttachMemoryBufferToKernelParameter(cl_context context, cl_kernel kernel, int parameterIndex, int offsetIndex, list<cl_mem> *cleanup, MemoryBuffer **inputMemoryBuffers, SocketReader* reader)
|
|
{
|
|
cl_int error;
|
|
MemoryBuffer* result = (MemoryBuffer*)reader->initializeTileData(NULL, inputMemoryBuffers);
|
|
|
|
const cl_image_format imageFormat = {
|
|
CL_RGBA,
|
|
CL_FLOAT
|
|
};
|
|
|
|
cl_mem clBuffer = clCreateImage2D(context, CL_MEM_READ_ONLY|CL_MEM_USE_HOST_PTR, &imageFormat, result->getWidth(),
|
|
result->getHeight(), 0, result->getBuffer(), &error);
|
|
|
|
if (error != CL_SUCCESS) { printf("CLERROR[%d]: %s\n", error, clewErrorString(error)); }
|
|
if (error == CL_SUCCESS) cleanup->push_back(clBuffer);
|
|
|
|
error = clSetKernelArg(kernel, parameterIndex, sizeof(cl_mem), &clBuffer);
|
|
if (error != CL_SUCCESS) { printf("CLERROR[%d]: %s\n", error, clewErrorString(error)); }
|
|
|
|
COM_clAttachMemoryBufferOffsetToKernelParameter(kernel, offsetIndex, result);
|
|
return clBuffer;
|
|
}
|
|
|
|
void NodeOperation::COM_clAttachMemoryBufferOffsetToKernelParameter(cl_kernel kernel, int offsetIndex, MemoryBuffer *memoryBuffer)
|
|
{
|
|
if (offsetIndex != -1) {
|
|
cl_int error;
|
|
rcti* rect = memoryBuffer->getRect();
|
|
cl_int2 offset = {rect->xmin, rect->ymin};
|
|
|
|
error = clSetKernelArg(kernel, offsetIndex, sizeof(cl_int2), &offset);
|
|
if (error != CL_SUCCESS) { printf("CLERROR[%d]: %s\n", error, clewErrorString(error)); }
|
|
}
|
|
}
|
|
|
|
void NodeOperation::COM_clAttachSizeToKernelParameter(cl_kernel kernel, int offsetIndex)
|
|
{
|
|
if (offsetIndex != -1) {
|
|
cl_int error;
|
|
cl_int2 offset = {this->getWidth(), this->getHeight()};
|
|
|
|
error = clSetKernelArg(kernel, offsetIndex, sizeof(cl_int2), &offset);
|
|
if (error != CL_SUCCESS) { printf("CLERROR[%d]: %s\n", error, clewErrorString(error)); }
|
|
}
|
|
}
|
|
|
|
void NodeOperation::COM_clAttachOutputMemoryBufferToKernelParameter(cl_kernel kernel, int parameterIndex, cl_mem clOutputMemoryBuffer)
|
|
{
|
|
cl_int error;
|
|
error = clSetKernelArg(kernel, parameterIndex, sizeof(cl_mem), &clOutputMemoryBuffer);
|
|
if (error != CL_SUCCESS) { printf("CLERROR[%d]: %s\n", error, clewErrorString(error)); }
|
|
}
|
|
|
|
void NodeOperation::COM_clEnqueueRange(cl_command_queue queue, cl_kernel kernel, MemoryBuffer *outputMemoryBuffer) {
|
|
cl_int error;
|
|
const size_t size[] = {outputMemoryBuffer->getWidth(),outputMemoryBuffer->getHeight()};
|
|
|
|
error = clEnqueueNDRangeKernel(queue, kernel, 2, NULL, size, 0, 0, 0, NULL);
|
|
if (error != CL_SUCCESS) { printf("CLERROR[%d]: %s\n", error, clewErrorString(error)); }
|
|
}
|
|
|
|
void NodeOperation::COM_clEnqueueRange(cl_command_queue queue, cl_kernel kernel, MemoryBuffer *outputMemoryBuffer, int offsetIndex) {
|
|
cl_int error;
|
|
const int width = outputMemoryBuffer->getWidth();
|
|
const int height = outputMemoryBuffer->getHeight();
|
|
int offsetx;
|
|
int offsety;
|
|
const int localSize = 32;
|
|
size_t size[2];
|
|
cl_int2 offset;
|
|
|
|
for (offsety = 0 ; offsety < height; offsety+=localSize) {
|
|
offset[1] = offsety;
|
|
if (offsety+localSize < height) {
|
|
size[1] = localSize;
|
|
} else {
|
|
size[1] = height - offsety;
|
|
}
|
|
for (offsetx = 0 ; offsetx < width ; offsetx+=localSize) {
|
|
if (offsetx+localSize < width) {
|
|
size[0] = localSize;
|
|
} else {
|
|
size[0] = width - offsetx;
|
|
}
|
|
offset[0] = offsetx;
|
|
|
|
error = clSetKernelArg(kernel, offsetIndex, sizeof(cl_int2), &offset);
|
|
if (error != CL_SUCCESS) { printf("CLERROR[%d]: %s\n", error, clewErrorString(error)); }
|
|
error = clEnqueueNDRangeKernel(queue, kernel, 2, NULL, size, 0, 0, 0, NULL);
|
|
if (error != CL_SUCCESS) { printf("CLERROR[%d]: %s\n", error, clewErrorString(error)); }
|
|
clFlush(queue);
|
|
}
|
|
}
|
|
}
|
|
|
|
cl_kernel NodeOperation::COM_clCreateKernel(cl_program program, const char *kernelname, list<cl_kernel> *clKernelsToCleanUp)
|
|
{
|
|
cl_int error;
|
|
cl_kernel kernel = clCreateKernel(program, kernelname, &error) ;
|
|
if (error != CL_SUCCESS) { printf("CLERROR[%d]: %s\n", error, clewErrorString(error));
|
|
}
|
|
else {
|
|
if (clKernelsToCleanUp) clKernelsToCleanUp->push_back(kernel);
|
|
}
|
|
return kernel;
|
|
|
|
}
|