Dynamic Load Policy#
The dynamic selection API is an experimental feature in the oneAPI DPC++ Library (oneDPL) that selects an execution resource based on a chosen selection policy. There are several policies provided as part of the API. Policies encapsulate the logic and any associated state needed to make a selection.
The dynamic load policy tracks the number of submissions currently submitted but not yet completed on each
resource and selects the resource that has the fewest unfinished submissions.
dynamic_load_policy is useful for offloading kernels of varying cost to devices
of varying performance. A load-based assignment may achieve a good load balancing
by submitting tasks to a resource that completes work faster.
namespace oneapi::dpl::experimental {
template <typename ResourceType = sycl::queue, typename ResourceAdapter = oneapi::dpl::identity,
typename Backend = default_backend<ResourceType, ResourceAdapter>>
class dynamic_load_policy
: public policy_base<dynamic_load_policy<ResourceType, ResourceAdapter, Backend>,
ResourceAdapter, Backend, execution_info::task_submission_t,
execution_info::task_completion_t>
{
public:
using resource_type = ResourceType;
using backend_type = Backend;
dynamic_load_policy(deferred_initialization_t);
dynamic_load_policy();
dynamic_load_policy(const std::vector<ResourceType>& u, ResourceAdapter adapter = {});
// deferred initializer
void initialize();
void initialize(const std::vector<resource_type>& u);
// other implementation defined functions...
};
}
This policy can be used with all the dynamic selection free functions, as well as with policy traits.
Example#
The following example demonstrates a simple approach to send work to more than
one queue concurrently using dynamic_load_policy. The policy selects the
resource with the fewest number of unfinished submissions.
#include <oneapi/dpl/dynamic_selection>
#include <sycl/sycl.hpp>
#include <iostream>
const std::size_t N = 10000;
namespace ex = oneapi::dpl::experimental;
void f(sycl::handler& h, float* v);
void do_cpu_work();
int dynamic_load_example(std::vector<sycl::queue>& devices,
std::vector<float *>& usm_data) {
ex::dynamic_load_policy p{devices}; // (1)
auto num_devices = p.get_resources().size();
auto num_arrays = usm_data.size();
// (2)
auto submission_group_size = num_arrays;
std::cout << "Running with " << num_devices << " queues\n"
<< " " << num_arrays << " usm arrays\n"
<< "Will perform " << submission_group_size << " concurrent offloads\n";
for (int i = 0; i < 100; i+=submission_group_size) { // (3)
for (int j = 0; j < submission_group_size; ++j) { // (4)
ex::submit(p, [&](sycl::queue q) { // (5)
float *data = usm_data[j];
return q.submit([=](sycl::handler &h) { // (6)
f(h, data);
});
});
do_cpu_work(); // (7)
}
ex::wait(p.get_submission_group()); // (8)
}
return 0;
}
The key points in this example are:
A
dynamic_load_policyis constructed that selects from queues in thedevicesvector.The total number of concurrent offloads,
submission_group_size, will be limited to the number of USM arrays. In this example, we allow multiple simultaneous offloads to the same queue. The only limitation is that there should be enough available vectors to support the concurrent executions.The outer
i-loop iterates from 0 to 99, stepping by thesubmission_group_size. This number of submissions will be offloaded concurrently.The inner
j-loop iterates oversubmission_group_sizesubmissions.submitis used to select a queue and pass it to the user’s function, but does not block until the event returned by that function completes. This provides the opportunity for concurrency across the submissions.The queue is used in a function to perform an asynchronous offload. The SYCL event returned from the call to
submitis returned. Returning an event is required for functions passed tosubmitandsubmit_and_wait.Some additional work is done between calls to
submit.dynamic_load_policyis most useful when there is time for work to complete on some devices before the next assignment is made. If all submissions are performed simultaneously, all devices will appear equally loaded, since the fast devices would have had no time to complete their work.waitis called to block for all the concurrentsubmission_group_sizesubmissions to complete.
Selection Algorithm#
The selection algorithm for dynamic_load_policy chooses the resource
that has the fewest number of unfinished offloads. The number of unfinished
offloads is the difference between the number of reported task submissions
and the number of reported task completions. This value is tracked for each
available resource.
Simplified, expository implementation of the selection algorithm:
//not a public function, for exposition purposes only
template<typename... Args>
selection_type dynamic_load_policy::select(Args&& ...) {
if (initialized_) {
auto least_loaded_resource = find_least_loaded(resources_);
return selection_type{dynamic_load_policy<Backend>(*this), least_loaded};
} else {
throw std::logic_error("select called before initialization");
}
}
where resources_ is a container of resources, such as
std::vector of sycl::queue. The function find_least_loaded
iterates through the resources available to the policy and returns the
resource with the fewest number of unfinished offloads.
Constructors#
dynamic_load_policy provides three constructors.
Signature |
Description |
|---|---|
|
Defers initialization. An |
|
Initialized to use the default set of resources. |
dynamic_load_policy(const std::vector<ResourceType>& u,ResourceAdapter adapter = {}); |
Overrides the default set of resources with an optional resource adapter. |
Deferred Initialization#
A dynamic_load_policy that was constructed with deferred initialization must be
initialized by calling one of its initialize member functions before it can be used
to select or submit.
Signature |
Description |
|---|---|
|
Initialize to use the default set of resources. |
|
Overrides the default set of resources. |
Queries#
A dynamic_load_policy has get_resources and get_submission_group
member functions.
Signature |
Description |
|---|---|
|
Returns the set of resources the policy is selecting from. |
|
Returns an object that can be used to wait for all active submissions. |
Reporting Requirements#
A dynamic_load_policy requires the task_submission and task_completion
reporting requirements. See the Execution Information
section for more information about reporting requirements.