:orphan: .. _ZE_experimental_mutable_command_list: ================================ Mutable Command List Extension ================================ API ---- * Enumerations * :ref:`ze-mutable-command-list-exp-version-t` * :ref:`ze-mutable-command-exp-flags-t` * :ref:`ze-mutable-command-list-exp-flags-t` * Structures * :ref:`ze-mutable-command-id-exp-desc-t` * :ref:`ze-mutable-command-list-exp-properties-t` * :ref:`ze-mutable-command-list-exp-desc-t` * :ref:`ze-mutable-commands-exp-desc-t` * :ref:`ze-mutable-kernel-argument-exp-desc-t` * :ref:`ze-mutable-group-count-exp-desc-t` * :ref:`ze-mutable-group-size-exp-desc-t` * :ref:`ze-mutable-global-offset-exp-desc-t` * :ref:`ze-mutable-graph-argument-exp-desc-t` * Functions * :ref:`zeCommandListGetNextCommandIdExp` * :ref:`zeCommandListGetNextCommandIdWithKernelsExp` * :ref:`zeCommandListUpdateMutableCommandsExp` * :ref:`zeCommandListUpdateMutableCommandSignalEventExp` * :ref:`zeCommandListUpdateMutableCommandWaitEventsExp` * :ref:`zeCommandListUpdateMutableCommandKernelsExp` ====================== Mutable Command List ====================== - A mutable command list is created by supplying a :ref:`ze-mutable-command-list-exp-desc-t` object via the `pNext` member of :ref:`ze-command-list-desc-t`\. - Mutable command lists support mutation to **identified** commands *after* being closed with :ref:`zeCommandListClose`\. - Implementation support for mutable commands may be discovered by providing a :ref:`ze-mutable-command-list-exp-properties-t` object in the `pNext` member of :ref:`ze-device-properties-t` in a call to :ref:`zeDeviceGetProperties`\. .. parsed-literal:: // Discover mutable command list properties :ref:`ze-mutable-command-list-exp-properties-t` mutCmdListProps = { :ref:`ZE_STRUCTURE_TYPE_MUTABLE_COMMAND_LIST_EXP_PROPERTIES `\, // stype nullptr, // pNext 0, // mutableCommandListFlags 0 // mutableCommandFlags }; :ref:`ze-device-properties-t` deviceProps = { :ref:`ZE_STRUCTURE_TYPE_DEVICE_PROPERTIES ` }; deviceProps.pNext = &mutCmdListProps; :ref:`zeDeviceGetProperties`\(hDevice, &deviceProps); // ... // Create a mutable command list :ref:`ze-mutable-command-list-exp-desc-t` mutCmdListDesc = { :ref:`ZE_STRUCTURE_TYPE_MUTABLE_COMMAND_LIST_EXP_DESC `\, nullptr, 0 // flags }; :ref:`ze-command-list-desc-t` commandListDesc = { :ref:`ZE_STRUCTURE_TYPE_COMMAND_LIST_DESC `\, &mutCmdListDesc, 0, 0 // flags }; ze_command_list_handle_t hCommandList = nullptr; :ref:`zeCommandListCreate`\(hContext, hDevice, &commandListDesc, &hCommandList); // [ ...create fence, signal event and wait event objects... ] // This example assumes hFence, hSignalEvent and hWaitEvent have been created. // Create kernel from module :ref:`ze-kernel-desc-t` kernelDesc = { :ref:`ZE_STRUCTURE_TYPE_KERNEL_DESC `\, 0, "example" }; ze_kernel_handle_t hKernel = nullptr; :ref:`ze-result-t` result = :ref:`zeKernelCreate`\(hModule, &kernelDesc, &hKernel); // Set the kernel arguments :ref:`ze-group-count-t` groupSize = {}; :ref:`zeKernelSuggestGroupSize`\(hKernel, 1024, 1024, 1, &groupSize.groupCountX, &groupSize.groupCountY, &groupSize.groupCountZ); int defaultValue = 0; :ref:`zeKernelSetArgumentValue`\(hKernel, 0, sizeof(int), &defaultValue); // Get next command identifier :ref:`ze-mutable-command-id-exp-desc-t` cmdIdDesc = { :ref:`ZE_STRUCTURE_TYPE_MUTABLE_COMMAND_ID_EXP_DESC `\, // stype nullptr, // pNext 0 // flags }; uint64_t commandId = 0; :ref:`zeCommandListGetNextCommandIdExp`\(hCommandList, &cmdIdDesc, &commandId); // Encode command into command list :ref:`zeCommandListAppendLaunchKernel`\(hCommandList, hKernel, &groupSize, hSignalEvent, 1, &hWaitEvent); // Close the command list :ref:`zeCommandListClose`\(hCommandList); // ... // Execute the command list zeCommandQueueExecuteCommandLists(hCommandQueue, 1, &hCommandList, hFence); // ... The application may subsequently mutate specific commands, as follows: .. parsed-literal:: // Prepare to modify group count :ref:`ze-group-count-t` groupCount = { 256, // groupCountX 256, // groupCountY 1 // groupCountZ }; :ref:`ze-mutable-group-count-exp-desc-t` groupCountDesc = { :ref:`ZE_STRUCTURE_TYPE_MUTABLE_GROUP_COUNT_EXP_DESC `\, // stype nullptr, // pNext commandId, // commandId &groupCount // pGroupCount }; // Prepare to modify Kernel Argument int argValue = 1; :ref:`ze-mutable-kernel-argument-exp-desc-t` krnlArgDesc = { :ref:`ZE_STRUCTURE_TYPE_MUTABLE_KERNEL_ARGUMENT_EXP_DESC `\, // stype &groupCountDesc, // pNext commandId, // commandId 0, // argIndex sizeof(int), // argSize &argValue // pArgValue }; // Prepare to update mutable commands :ref:`ze-mutable-commands-exp-desc-t` desc = { :ref:`ZE_STRUCTURE_TYPE_MUTABLE_COMMANDS_EXP_DESC `\, // stype &krnlArgDesc, // pNext 0 // flags }; // Synchronize command list execution :ref:`zeFenceHostSynchronize`\(hFence, UINT64_MAX); // Update mutable commands :ref:`zeCommandListUpdateMutableCommandsExp`\(hCommandList, &desc); // Update signal event for the launch kernel command :ref:`zeCommandListUpdateMutableCommandSignalEventExp`\(hCommandList, commandId, hNewLaunchKernelSignalEvent); // Update the wait events for the launch kernel command :ref:`zeCommandListUpdateMutableCommandWaitEventsExp`\(hCommandList, commandId, 1, &hNewLaunchKernelWaitEvent); // Close the command list :ref:`zeCommandListClose`\(hCommandList); // ... Note, the command list must be explicitly closed after updating mutable commands and events. This informs the implementation that the application has finished with updates and is ready to submit the command list. In preparation for kernel mutation user must provide all possible kernels for the command. .. parsed-literal:: // define all possible kernels ze_kernel_handle_t addKernel; ze_kernel_handle_t mulKernel; ze_kernel_handle_t kernels[] = {addKernel, mulKernel}; // when users want kernel mutation, they need to explicitly state this, as 0 does not include kernel instruction mutation by default :ref:`ze-mutable-command-exp-flags-t` mutationFlags = :ref:`ZE_MUTABLE_COMMAND_EXP_FLAG_KERNEL_ARGUMENTS ` | :ref:`ZE_MUTABLE_COMMAND_EXP_FLAG_GROUP_COUNT ` | :ref:`ZE_MUTABLE_COMMAND_EXP_FLAG_GROUP_SIZE ` | :ref:`ZE_MUTABLE_COMMAND_EXP_FLAG_KERNEL_INSTRUCTION `\; // Get next command identifier :ref:`ze-mutable-command-id-exp-desc-t` cmdIdDesc = { :ref:`ZE_STRUCTURE_TYPE_MUTABLE_COMMAND_ID_EXP_DESC `\, // stype nullptr, // pNext mutationFlags // flags }; // retrieve id for the append operation and provide all possible kernels for this command uint64_t mutableKernelCommandId = 0; :ref:`zeCommandListGetNextCommandIdWithKernelsExp`\(hCommandList, &cmdIdDesc, &mutableKernelCommandId, 2, kernels); // Encode command into command list :ref:`zeCommandListAppendLaunchKernel`\(hCommandList, addKernel, &groupSize, nullptr, 0, nullptr); // Close the command list :ref:`zeCommandListClose`\(hCommandList); Mutation of kernels must obey two rules: - kernel handle mutation function must be called as first for a given command id - kernel mutation invalidates all kernel arguments and dispatch parameters, these must be provided for the new kernel .. parsed-literal:: // Update mutable kernel for the command, switch from `addKernel` to `mulKernel` :ref:`zeCommandListUpdateMutableCommandKernelsExp`\(hCommandList, 1, &mutableKernelCommandId, &mulKernel); // modify group count :ref:`ze-group-count-t` groupCount = { 32, // groupCountX 1, // groupCountY 1 // groupCountZ }; :ref:`ze-mutable-group-count-exp-desc-t` groupCountDesc = { :ref:`ZE_STRUCTURE_TYPE_MUTABLE_GROUP_COUNT_EXP_DESC `\, // stype nullptr, // pNext mutableKernelCommandId, // commandId &groupCount // pGroupCount }; :ref:`ze-mutable-group-size-exp-desc-t` groupSizeDesc = { :ref:`ZE_STRUCTURE_TYPE_MUTABLE_GROUP_SIZE_EXP_DESC `\, // stype &groupCountDesc, // pNext mutableKernelCommandId, // commandId 32, // groupSizeX 1, // groupSizeY 1, // groupSizeZ }; // Prepare to modify Kernel Argument int argValue = 1; void *usmPointer; :ref:`ze-mutable-kernel-argument-exp-desc-t` krnlArgMemoryDesc = { :ref:`ZE_STRUCTURE_TYPE_MUTABLE_KERNEL_ARGUMENT_EXP_DESC `\, // stype &groupSizeDesc, // pNext mutableKernelCommandId, // commandId 0, // argIndex sizeof(void *), // argSize &usmPointer // pArgValue }; :ref:`ze-mutable-kernel-argument-exp-desc-t` krnlArgScalarDesc = { :ref:`ZE_STRUCTURE_TYPE_MUTABLE_KERNEL_ARGUMENT_EXP_DESC `\, // stype &krnlArgMemoryDesc, // pNext mutableKernelCommandId, // commandId 1, // argIndex sizeof(int), // argSize &argValue // pArgValue }; // Prepare to update mutable commands :ref:`ze-mutable-commands-exp-desc-t` desc = { :ref:`ZE_STRUCTURE_TYPE_MUTABLE_COMMANDS_EXP_DESC `\, // stype &krnlArgScalarDesc, // pNext 0 // flags }; // Update mutable kernel arguments and dispatch parameters for the command :ref:`zeCommandListUpdateMutableCommandsExp`\(hCommandList, &desc); // Close the command list :ref:`zeCommandListClose`\(hCommandList); The command list must be explicitly closed after updating mutable commands.