Add Custom nGraph Operations

Inference Engine Extension API allows to register operation sets (opsets) with custom nGraph operations, it allows to support Networks with unknown operations.

Operation Class

To add your custom nGraph operation, create a new class that extends ngraph::Op, which is in turn derived from ngraph::Node, the base class for all graph operations in nGraph. Follow the steps below:

  1. Define a NodeTypeInfo object that identifies the type of the operation to the graph users and helps with dynamic type resolution. The type info of an nGraph operation currently consists of a string identifier and a version number, but this may change in the future.
  2. Implement constructors that can optionally take the operation inputs and attributes as parameters.
  3. Override the shape inference method validate_and_infer_types. This method is called multiple times during graph manipulations to determine the shapes and element types of the outputs of the operations. You can access the input shapes through the get_input_partial_shape() method and input element types through the get_input_element_type() method of ngraph::Node. Set the inferred shape and element type of the output using set_output_type.
  4. Override the copy_with_new_args method, which allows graph manipulation routines to create copies of this operation and connect it to different nodes during optimization.
  5. Override the visit_attributes method, which allows serialization and deserialization of attributes. An AttributeVisitor is passed to the method, and the implementation is expected to walk over all the attributes in the op using the type-aware on_attribute helper. Helpers are already implemented for standard C++ types like int64_t, float, bool, vector and for existing nGraph defined types.

Based on that, declaration of a operation class can look as follows:

namespace TemplateExtension {
class Operation : public ngraph::op::Op {
public:
static constexpr ngraph::NodeTypeInfo type_info{"Template", 0};
const ngraph::NodeTypeInfo& get_type_info() const override { return type_info; }
Operation() = default;
Operation(const ngraph::Output<ngraph::Node>& arg, int64_t add);
void validate_and_infer_types() override;
std::shared_ptr<ngraph::Node> copy_with_new_args(const ngraph::NodeVector& new_args) const override;
bool visit_attributes(ngraph::AttributeVisitor& visitor) override;
int64_t getAddAttr() { return add; }
private:
int64_t add;
};

Class Fields

The provided implementation has several fields:

Operation Constructors

nGraph operation contains two constructors: a default constructor, which allows to create operation without attributes and a constructor that creates and validates operation with specified inputs and attributes.

Operation::Operation(const ngraph::Output<ngraph::Node> &arg, int64_t add) : Op({arg}), add(add) {
constructor_validate_and_infer_types();
}

validate_and_infer_types()

ngraph::Node::validate_and_infer_types method validates operation attributes and calculates output shapes using attributes of operation.

void Operation::validate_and_infer_types() {
// Operation doesn't change shapes end element type
set_output_type(0, get_input_element_type(0), get_input_partial_shape(0));
}

copy_with_new_args()

ngraph::Node::copy_with_new_args method creates a copy of nGraph operation with new inputs.

std::shared_ptr<ngraph::Node> Operation::copy_with_new_args(const ngraph::NodeVector &new_args) const {
if (new_args.size() != 1) {
throw ngraph::ngraph_error("Incorrect number of new arguments");
}
return std::make_shared<Operation>(new_args.at(0), add);
}

visit_attributes()

ngraph::Node::visit_attributes method allows to visit all operation attributes.

bool Operation::visit_attributes(ngraph::AttributeVisitor &visitor) {
visitor.on_attribute("add", add);
return true;
}

Register Custom Operations in Extension Class

To add custom operations to the Extension class, create an operation set with custom operations and implement the InferenceEngine::IExtension::getOpSets method:

std::map<std::string, ngraph::OpSet> Extension::getOpSets() {
std::map<std::string, ngraph::OpSet> opsets;
ngraph::OpSet opset;
opset.insert<Operation>();
opsets["custom_opset"] = opset;
return opsets;
}

This method returns a map of opsets that exist in the extension library.

nGraph provides opsets mechanism for operation versioning. Different opsets distinguish between different versions of one operation.

When specifying opset names, follow the rules below:

Use a custom opset to create a new operation or extend functionality of an existing operation from another opset.