SPDX-FileCopyrightText: Copyright (c) 2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
SPDX-License-Identifier: LicenseRef-NvidiaProprietary
NVIDIA CORPORATION, its affiliates and licensors retain all intellectual
property and proprietary rights in and to this material, related
documentation and any modifications thereto. Any use, reproduction,
disclosure or distribution of this material and related documentation
without an express license agreement from NVIDIA CORPORATION or
its affiliates is strictly prohibited.
name: writing-transforms description: > Writing 4x4 transform matrices to move, rotate, or scale prims. Use when user asks to move an object, set a transform, update position, animate a transform, or set a camera transform. license: LicenseRef-NvidiaProprietary version: "0.3.0" author: NVIDIA ovrtx tags: - ovrtx - transforms - attributes tools: - Read - Grep
Writing Transforms
When to Use
Use this skill when the user asks to move an object, set a transform, update position, animate a transform, or set a camera transform.
Inputs
Resolve inputs in this order: existing repository files and referenced snippets, explicit user request, then broader agent context.
- Target API surface: Python, C/C++, or both.
- Prim paths to move, desired translation/rotation/scale or full 4x4 matrices, and whether values are authored per prim.
- Transform storage name, semantic conversion, row-vector matrix convention, and sync/async behavior.
- Update cadence: one-shot movement, per-frame animation, or zero-copy GPU-computed transform updates.
- Repository source snippets referenced below. Treat these snippets as the API source of truth.
Prerequisites
- Use an ovrtx checkout that contains the referenced examples and docs tests.
- Read the relevant
> **Source:**snippet before writing or explaining API usage. - Use
attribute-bindingsfor per-frame transform writes to the same prims, andmapping-attributeswhen GPU kernels should write transforms directly into mapped ovrtx buffers.
Instructions
- Identify the target language, prim paths, transform representation, matrix convention, memory target, and sync/async requirement.
- Read the matching source snippet and copy its lifecycle pattern rather than inventing equivalent calls.
- Validate dtype, shape, semantic, and ownership rules before proposing or editing code.
- Keep bindings, mapped buffers, operation handles, and C strings alive only for the lifetimes documented by the referenced examples.
- When changing code, run the narrow docs test or example that owns the snippet whenever practical.
Output Format
- For explanations, cite the relevant API names, source snippets, and caveats.
- For code changes, summarize the files changed, snippets affected, and validation run.
Scripts
This skill has no scripts.
Limitations
- Transform semantic conversion is write-side only; read transform attributes with raw storage semantics.
- The referenced snippets remain the source of truth; update or add tested snippets before documenting new API usage.
Overview
Transforms control the position, rotation, and scale of prims in the scene. ovrtx supports three transform representations. The most common is a 4x4 matrix of doubles using the USD row-vector convention: translation is in the last row ([3][0..2]). The canonical transform attribute name is "omni:xform" (used by the C convenience helpers in ovrtx_attributes.h). The legacy name "omni:fabric:localMatrix" is also accepted.
Python
Write a 4x4 transform (single prim)
Source:
tests/docs/python/test_attribute_shapes.pysnippetdoc-shape-mat4-array
Write transforms for multiple prims
Source:
tests/docs/python/test_attribute_bindings.pysnippetdoc-bind-attribute-write
C
Write identity transform on a single prim
Source:
examples/c/vulkan-interop/src/main.cppsnippetwrite-camera-transform
Batch write (N prims)
Source:
examples/c/vulkan-interop/src/main.cppsnippetwrite-camera-transformFor multiple prims, increase
prim_list.num_pathsand provide a contiguous array of transforms.
C convenience helpers (ovrtx_attributes.h)
Instead of building DLTensor and binding descriptors manually, use the convenience helpers that handle the boilerplate:
Source:
tests/docs/c/test_transform_helpers.cppsnippetsdoc-set-xform-mat-c,doc-set-xform-pos-rot-scale-c,doc-set-xform-pos-rot3x3-c,doc-set-reset-xform-stack-c.
#include <ovrtx/ovrtx_attributes.h> provides ovrtx_set_xform_mat(renderer, paths, count, transforms), ovrtx_set_xform_pos_rot_scale(...), ovrtx_set_xform_pos_rot3x3(...), and ovrtx_set_reset_xform_stack(...).
Tensor layout
Python and C expose different transform tensor layouts:
- Python uses NumPy-style matrices: an N-prim batch has shape
(N, 4, 4)anddtype=float64. - C uses DLTensor lanes for attributes: an N-prim batch has shape
[N]andDLDataType{kDLFloat, 64, 16}. - C rendered outputs/AOVs are different from attributes; image tensors use channel-last shapes such as
[height, width, channels]withdtype.lanes=1.
Source:
tests/docs/python/test_attribute_shapes.pysnippetdoc-shape-mat4-array.Source:
tests/docs/c/test_attribute_shapes.cppsnippetdoc-shape-mat4-array-c.
Semantics are a write-side conversion hint. ovrtx_read_attribute rejects any semantic other than OVRTX_SEMANTIC_NONE with "Semantic conversion is not supported for read_attribute". When reading a transform attribute, leave the semantic as OVRTX_SEMANTIC_NONE and ask for the raw storage layout.
Key Types / Functions
| Representation | Python | C | Size per prim |
|---|---|---|---|
| 4x4 matrix (doubles) | Semantic.XFORM_MAT4x4 or dtype="float64", shape=(4, 4) |
OVRTX_SEMANTIC_XFORM_MAT4x4 with ndim=1, shape=[N], dtype={kDLFloat, 64, 16} |
128 bytes |
| pos3d + rot4f + scale3f | (C only) | OVRTX_SEMANTIC_XFORM_POS3d_ROT4f_SCALE3f |
56 bytes |
| pos3d + rot3x3f | (C only) | OVRTX_SEMANTIC_XFORM_POS3d_ROT3x3f |
64 bytes |
C convenience helpers (#include <ovrtx/ovrtx_attributes.h>):
ovrtx_set_xform_mat(renderer, paths, count, transforms)-- 4x4 matrixovrtx_set_xform_pos_rot_scale(renderer, paths, count, transforms)-- pos + quat + scaleovrtx_set_xform_pos_rot3x3(renderer, paths, count, transforms)-- pos + 3x3 rotationovrtx_set_reset_xform_stack(renderer, paths, count, values)-- setomni:resetXformStack
Python semantic: Semantic.XFORM_MAT4x4 (from from ovrtx import Semantic). For interop, the preferred approach is dtype="float64", shape=(4, 4) on bind_attribute/map_attribute, or passing an (N, 4, 4) float64 array directly to write_attribute (type is auto-inferred when the attribute already exists in the stage). The dtype parameter accepts string names ("float64"), NumPy dtypes (np.float64), or Python built-ins (float). The semantic= parameter is required when creating attributes from scratch via prim_mode=PrimMode.CREATE_NEW.
Troubleshooting
- Matrices use the USD row-vector convention (same as
GfMatrix4d). Translation is in the last row:v[12],v[13],v[14](ormatrix[3][0..2]in Python/C). - The canonical transform attribute name is
"omni:xform". The legacy name"omni:fabric:localMatrix"is also accepted. New code should prefer"omni:xform"to match the C convenience helpers (ovrtx_set_xform_mat, etc.). - Transform dtype is
float64(doubles), not float32. Using the wrong dtype will cause errors. - For repeated per-frame transform updates, use attribute bindings or mapping for better performance (see
attribute-bindingsandmapping-attributesskills). - Earlier code may use
ovrtx.math.Matrix4dfor single-prim writes; theovrtx.mathmodule and theMatrix4dclass have been removed. Pass an(N, 4, 4)float64array (NumPy, Warp, or any DLPack-compatible tensor) directly towrite_attribute, or usedtype="float64", shape=(4, 4)withbind_attribute/map_attribute.
References
- Use the
> **Source:**directives in this skill to locate tested snippets before reusing API patterns. - Keep related skills, docs, and snippets synchronized when changing the workflow.