C++ API Reference
identityNode/identityNode.cpp
//-
// ==========================================================================
// Copyright 2015 Autodesk, Inc. All rights reserved.
//
// Use of this software is subject to the terms of the Autodesk
// license agreement provided at the time of installation or download,
// or which otherwise accompanies this software in either electronic
// or hard copy form.
// ==========================================================================
//+
//
// File: identityNode.cpp
//
// Description:
// Empty implementation of a deformer. This node
// performs no deformation and is basically an empty
// shell that can be used to create actual deformers.
//
//
// Use this script to create a simple example with the identity node.
//
// loadPlugin identityNode;
//
// polyTorus -r 1 -sr 0.5 -tw 0 -sx 50 -sy 50 -ax 0 1 0 -cuv 1 -ch 1;
// deformer -type "identity";
// setKeyframe -v 0 -at weightList[0].weights[0] -t 1 identity1;
// setKeyframe -v 1 -at weightList[0].weights[0] -t 60 identity1;
// select -cl;
#include <maya/MFnPlugin.h>
#include <maya/MTypeId.h>
#include <maya/MStringArray.h>
#include <maya/MPxDeformerNode.h>
#include <maya/MItGeometry.h>
#include <maya/MPoint.h>
#include <maya/MPxGPUDeformer.h>
#include <maya/MGPUDeformerRegistry.h>
#include <maya/MOpenCLInfo.h>
#include <clew/clew.h>
class identityNode : public MPxDeformerNode
{
public:
static void* creator();
static MStatus initialize();
// Deformation function
//
MItGeometry& iter,
const MMatrix& mat,
unsigned int multiIndex) override;
static const MTypeId id;
// path from where the plugin was loaded
static MString pluginPath;
};
const MTypeId identityNode::id( 0x8000d );
MString identityNode::pluginPath;
void* identityNode::creator()
{
return new identityNode();
}
MStatus identityNode::initialize()
{
}
identityNode::deform( MDataBlock& block,
MItGeometry& iter,
const MMatrix& /*m*/,
unsigned int multiIndex)
{
MStatus returnStatus;
// Iterate through each point in the geometry.
//
for ( ; !iter.isDone(); iter.next()) {
MPoint pt = iter.position();
// Perform some calculation on pt.
// ...
// Set the final position.
iter.setPosition(pt);
}
return returnStatus;
}
// The GPU override implementation of the identityNode
//
class identityGPUDeformer : public MPxGPUDeformer
{
public:
static MGPUDeformerRegistrationInfo* getGPUDeformerInfo();
static bool validateNodeInGraph(MDataBlock& block, const MEvaluationNode&, const MPlug& plug, MStringArray* messages);
static bool validateNodeValues(MDataBlock& block, const MEvaluationNode&, const MPlug& plug, MStringArray* messages);
// Virtual methods from MPxGPUDeformer
identityGPUDeformer();
~identityGPUDeformer() override;
// Implementation of MPxGPUDeformer.
MPxGPUDeformer::DeformerStatus evaluate(MDataBlock& block, const MEvaluationNode& evaluationNode, const MPlug& outputPlug, const MPlugArray& inputPlugs, const MGPUDeformerData& inputData, MGPUDeformerData& outputData) override;
void terminate() override;
private:
// Kernel
MAutoCLKernel fKernel;
size_t fLocalWorkSize;
size_t fGlobalWorkSize;
};
// registration information for the identity GPU deformer.
class identityGPUDeformerInfo : public MGPUDeformerRegistrationInfo
{
public:
identityGPUDeformerInfo(){}
~identityGPUDeformerInfo() override{}
{
return new identityGPUDeformer();
}
bool validateNodeInGraph(MDataBlock& block, const MEvaluationNode& evaluationNode, const MPlug& plug, MStringArray* messages) override
{
return identityGPUDeformer::validateNodeInGraph(block, evaluationNode, plug, messages);
}
bool validateNodeValues(MDataBlock& block, const MEvaluationNode& evaluationNode, const MPlug& plug, MStringArray* messages) override
{
return identityGPUDeformer::validateNodeValues(block, evaluationNode, plug, messages);
}
};
MGPUDeformerRegistrationInfo* identityGPUDeformer::getGPUDeformerInfo()
{
static identityGPUDeformerInfo theOne;
return &theOne;
}
identityGPUDeformer::identityGPUDeformer()
{
}
identityGPUDeformer::~identityGPUDeformer()
{
terminate();
}
bool identityGPUDeformer::validateNodeInGraph(MDataBlock& block, const MEvaluationNode& evaluationNode, const MPlug& plug, MStringArray* messages)
{
// Support everything.
return true;
}
bool identityGPUDeformer::validateNodeValues(MDataBlock& block, const MEvaluationNode& evaluationNode, const MPlug& plug, MStringArray* messages)
{
// Support everything.
return true;
}
MPxGPUDeformer::DeformerStatus identityGPUDeformer::evaluate(
MDataBlock& block,
const MEvaluationNode& evaluationNode,
const MPlug& plug,
const MPlugArray& inputPlugs,
const MGPUDeformerData& inputData,
MGPUDeformerData& outputData
)
{
// identityNode only supports a single mesh input plug
if (inputPlugs.length() != 1)
return MPxGPUDeformer::kDeformerFailure;
const MPlug& inputPlug = inputPlugs[0];
const MGPUDeformerBuffer inputPositions = inputData.getBuffer(MPxGPUDeformer::sPositionsName(), inputPlug);
MGPUDeformerBuffer outputPositions = createOutputBuffer(inputPositions);
if (!inputPositions.isValid() || !outputPositions.isValid()) return MPxGPUDeformer::kDeformerFailure;
unsigned int numElements = inputPositions.elementCount();
cl_int err = CL_SUCCESS;
// Setup OpenCL kernel.
if ( !fKernel.get() )
{
// Get and compile the kernel.
MString openCLKernelFile = identityNode::pluginPath + "/identity.cl";
MString openCLKernelName("identity");
MAutoCLKernel kernel = MOpenCLInfo::getOpenCLKernel( openCLKernelFile, openCLKernelName );
if ( kernel.isNull() )
{
return MPxGPUDeformer::kDeformerFailure;
}
fKernel = kernel;
// Figure out a good work group size for our kernel.
fLocalWorkSize = 0;
fGlobalWorkSize = 0;
size_t retSize = 0;
err = clGetKernelWorkGroupInfo(
fKernel.get(),
CL_KERNEL_WORK_GROUP_SIZE,
sizeof(size_t),
&fLocalWorkSize,
&retSize
);
if ( err != CL_SUCCESS || retSize == 0 || fLocalWorkSize == 0)
{
return MPxGPUDeformer::kDeformerFailure;
}
// Global work size must be a multiple of local work size.
const size_t remain = numElements % fLocalWorkSize;
if ( remain )
{
fGlobalWorkSize = numElements + ( fLocalWorkSize - remain );
}
else
{
fGlobalWorkSize = numElements;
}
}
// Set all of our kernel parameters. Input buffer and output buffer may be changing every frame
// so always set them.
unsigned int parameterId = 0;
err = clSetKernelArg(fKernel.get(), parameterId++, sizeof(cl_mem), (void*)outputPositions.buffer().getReadOnlyRef());
MOpenCLInfo::checkCLErrorStatus(err);
err = clSetKernelArg(fKernel.get(), parameterId++, sizeof(cl_mem), (void*)inputPositions.buffer().getReadOnlyRef());
MOpenCLInfo::checkCLErrorStatus(err);
err = clSetKernelArg(fKernel.get(), parameterId++, sizeof(cl_uint), (void*)&numElements);
MOpenCLInfo::checkCLErrorStatus(err);
// Set up our input events. The input event could be NULL, in that case we need to pass
// slightly different parameters into clEnqueueNDRangeKernel.
cl_event events[ 1 ] = { 0 };
cl_uint eventCount = 0;
if (inputPositions.bufferReadyEvent().get() )
{
events[ eventCount++ ] = inputPositions.bufferReadyEvent().get();
}
// Run the kernel
MAutoCLEvent kernelFinishedEvent;
err = clEnqueueNDRangeKernel(
fKernel.get() ,
1 ,
NULL ,
&fGlobalWorkSize ,
&fLocalWorkSize ,
eventCount ,
eventCount ? events : NULL ,
kernelFinishedEvent.getReferenceForAssignment()
);
outputPositions.setBufferReadyEvent(kernelFinishedEvent);
if ( err != CL_SUCCESS )
{
return MPxGPUDeformer::kDeformerFailure;
}
outputData.setBuffer(outputPositions);
return MPxGPUDeformer::kDeformerSuccess;
}
void identityGPUDeformer::terminate()
{
fKernel.reset();
}
// standard initialization procedures
//
MStatus initializePlugin( MObject obj )
{
MStatus result;
MFnPlugin plugin( obj, PLUGIN_COMPANY, "3.0", "Any");
result = plugin.registerNode(
"identity" ,
identityNode::id ,
&identityNode::creator ,
&identityNode::initialize ,
);
MString nodeClassName("identity");
MString registrantId("mayaPluginExample");
nodeClassName,
registrantId,
identityGPUDeformer::getGPUDeformerInfo()
);
identityNode::pluginPath = plugin.loadPath();
return result;
}
MStatus uninitializePlugin( MObject obj )
{
MStatus result;
MString nodeClassName("identity");
MString registrantId("mayaPluginExample");
nodeClassName,
registrantId
);
MFnPlugin plugin( obj );
result = plugin.deregisterNode( identityNode::id );
return result;
}