task.cc 7.7 KB
Newer Older
1
/*
2
 * Copyright (c) 2014-2015, Siemens AG. All rights reserved.
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * 1. Redistributions of source code must retain the above copyright notice,
 * this list of conditions and the following disclaimer.
 *
 * 2. Redistributions in binary form must reproduce the above copyright notice,
 * this list of conditions and the following disclaimer in the documentation
 * and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */

#include <cstring>
#include <cassert>

#include <embb/base/memory_allocation.h>
#include <embb/base/exceptions.h>
#include <embb/mtapi/mtapi.h>

namespace embb {
namespace mtapi {

Task::Task() {
  handle_.id = 0;
  handle_.tag = 0;
}

Task::Task(Task const & task)
  : handle_(task.handle_) {
  // empty
}

Task::Task(
48
  Action action) {
49 50
  mtapi_status_t status;
  mtapi_task_attributes_t attr;
51
  ExecutionPolicy policy = action.GetExecutionPolicy();
52 53 54
  mtapi_taskattr_init(&attr, &status);
  assert(MTAPI_SUCCESS == status);
  mtapi_taskattr_set(&attr, MTAPI_TASK_PRIORITY,
55
    &policy.priority_, sizeof(policy.priority_), &status);
56 57
  assert(MTAPI_SUCCESS == status);
  mtapi_taskattr_set(&attr, MTAPI_TASK_AFFINITY,
58
    &policy.affinity_, sizeof(policy.affinity_), &status);
59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74
  assert(MTAPI_SUCCESS == status);
  mtapi_domain_t domain_id = mtapi_domain_id_get(&status);
  assert(MTAPI_SUCCESS == status);
  mtapi_job_hndl_t job = mtapi_job_get(MTAPI_CPP_TASK_JOB, domain_id, &status);
  assert(MTAPI_SUCCESS == status);
  Action* holder = embb::base::Allocation::New<Action>(action);
  handle_ = mtapi_task_start(MTAPI_TASK_ID_NONE, job,
    holder, sizeof(Action), MTAPI_NULL, 0, &attr, MTAPI_GROUP_NONE, &status);
  if (MTAPI_SUCCESS != status) {
    EMBB_THROW(embb::base::ErrorException,
      "mtapi::Task could not be started");
  }
}

Task::Task(
  Action action,
75
  mtapi_group_hndl_t group) {
76 77
  mtapi_status_t status;
  mtapi_task_attributes_t attr;
78
  ExecutionPolicy policy = action.GetExecutionPolicy();
79 80 81
  mtapi_taskattr_init(&attr, &status);
  assert(MTAPI_SUCCESS == status);
  mtapi_taskattr_set(&attr, MTAPI_TASK_PRIORITY,
82
    &policy.priority_, sizeof(policy.priority_), &status);
83 84
  assert(MTAPI_SUCCESS == status);
  mtapi_taskattr_set(&attr, MTAPI_TASK_AFFINITY,
85
    &policy.affinity_, sizeof(policy.affinity_), &status);
86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102
  assert(MTAPI_SUCCESS == status);
  mtapi_domain_t domain_id = mtapi_domain_id_get(&status);
  assert(MTAPI_SUCCESS == status);
  mtapi_job_hndl_t job = mtapi_job_get(MTAPI_CPP_TASK_JOB, domain_id, &status);
  assert(MTAPI_SUCCESS == status);
  Action* holder = embb::base::Allocation::New<Action>(action);
  handle_ = mtapi_task_start(MTAPI_TASK_ID_NONE, job,
    holder, sizeof(Action), MTAPI_NULL, 0, &attr, group, &status);
  if (MTAPI_SUCCESS != status) {
    EMBB_THROW(embb::base::ErrorException,
      "mtapi::Task could not be started");
  }
}

Task::Task(
  mtapi_task_id_t id,
  Action action,
103
  mtapi_group_hndl_t group) {
104 105
  mtapi_status_t status;
  mtapi_task_attributes_t attr;
106
  ExecutionPolicy policy = action.GetExecutionPolicy();
107 108 109
  mtapi_taskattr_init(&attr, &status);
  assert(MTAPI_SUCCESS == status);
  mtapi_taskattr_set(&attr, MTAPI_TASK_PRIORITY,
110
    &policy.priority_, sizeof(policy.priority_), &status);
111 112
  assert(MTAPI_SUCCESS == status);
  mtapi_taskattr_set(&attr, MTAPI_TASK_AFFINITY,
113
    &policy.affinity_, sizeof(policy.affinity_), &status);
114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131
  assert(MTAPI_SUCCESS == status);
  mtapi_domain_t domain_id = mtapi_domain_id_get(&status);
  assert(MTAPI_SUCCESS == status);
  mtapi_job_hndl_t job = mtapi_job_get(MTAPI_CPP_TASK_JOB, domain_id, &status);
  assert(MTAPI_SUCCESS == status);
  Action* holder = embb::base::Allocation::New<Action>(action);
  void * idptr = MTAPI_NULL;
  memcpy(&idptr, &id, sizeof(id));
  handle_ = mtapi_task_start(id, job,
    holder, sizeof(Action), idptr, 0, &attr, group, &status);
  if (MTAPI_SUCCESS != status) {
    EMBB_THROW(embb::base::ErrorException,
      "mtapi::Task could not be started");
  }
}

Task::Task(
  Action action,
132
  mtapi_queue_hndl_t queue) {
133 134
  mtapi_status_t status;
  mtapi_task_attributes_t attr;
135
  ExecutionPolicy policy = action.GetExecutionPolicy();
136 137 138
  mtapi_taskattr_init(&attr, &status);
  assert(MTAPI_SUCCESS == status);
  mtapi_taskattr_set(&attr, MTAPI_TASK_PRIORITY,
139
    &policy.priority_, sizeof(policy.priority_), &status);
140 141
  assert(MTAPI_SUCCESS == status);
  mtapi_taskattr_set(&attr, MTAPI_TASK_AFFINITY,
142
    &policy.affinity_, sizeof(policy.affinity_), &status);
143 144 145 146 147 148 149 150 151 152 153 154 155
  assert(MTAPI_SUCCESS == status);
  Action* holder = embb::base::Allocation::New<Action>(action);
  handle_ = mtapi_task_enqueue(MTAPI_TASK_ID_NONE, queue,
    holder, sizeof(Action), MTAPI_NULL, 0, &attr, MTAPI_GROUP_NONE, &status);
  if (MTAPI_SUCCESS != status) {
    EMBB_THROW(embb::base::ErrorException,
      "mtapi::Task could not be started");
  }
}

Task::Task(
  Action action,
  mtapi_queue_hndl_t queue,
156
  mtapi_group_hndl_t group) {
157 158
  mtapi_status_t status;
  mtapi_task_attributes_t attr;
159
  ExecutionPolicy policy = action.GetExecutionPolicy();
160 161 162
  mtapi_taskattr_init(&attr, &status);
  assert(MTAPI_SUCCESS == status);
  mtapi_taskattr_set(&attr, MTAPI_TASK_PRIORITY,
163
    &policy.priority_, sizeof(policy.priority_), &status);
164 165
  assert(MTAPI_SUCCESS == status);
  mtapi_taskattr_set(&attr, MTAPI_TASK_AFFINITY,
166
    &policy.affinity_, sizeof(policy.affinity_), &status);
167 168 169 170 171 172 173 174 175 176 177 178 179 180
  assert(MTAPI_SUCCESS == status);
  Action* holder = embb::base::Allocation::New<Action>(action);
  handle_ = mtapi_task_enqueue(MTAPI_TASK_ID_NONE, queue,
    holder, sizeof(Action), MTAPI_NULL, 0, &attr, group, &status);
  if (MTAPI_SUCCESS != status) {
    EMBB_THROW(embb::base::ErrorException,
      "mtapi::Task could not be started");
  }
}

Task::Task(
  mtapi_task_id_t id,
  Action action,
  mtapi_queue_hndl_t queue,
181
  mtapi_group_hndl_t group) {
182 183
  mtapi_status_t status;
  mtapi_task_attributes_t attr;
184
  ExecutionPolicy policy = action.GetExecutionPolicy();
185 186 187
  mtapi_taskattr_init(&attr, &status);
  assert(MTAPI_SUCCESS == status);
  mtapi_taskattr_set(&attr, MTAPI_TASK_PRIORITY,
188
    &policy.priority_, sizeof(policy.priority_), &status);
189 190
  assert(MTAPI_SUCCESS == status);
  mtapi_taskattr_set(&attr, MTAPI_TASK_AFFINITY,
191
    &policy.affinity_, sizeof(policy.affinity_), &status);
192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220
  assert(MTAPI_SUCCESS == status);
  Action* holder = embb::base::Allocation::New<Action>(action);
  void * idptr = MTAPI_NULL;
  memcpy(&idptr, &id, sizeof(id));
  handle_ = mtapi_task_enqueue(id, queue,
    holder, sizeof(Action), idptr, 0, &attr, group, &status);
  if (MTAPI_SUCCESS != status) {
    EMBB_THROW(embb::base::ErrorException,
      "mtapi::Task could not be started");
  }
}

Task::~Task() {
}

mtapi_status_t Task::Wait(mtapi_timeout_t timeout) {
  mtapi_status_t status;
  mtapi_task_wait(handle_, timeout, &status);
  return status;
}

void Task::Cancel() {
  mtapi_status_t status;
  mtapi_task_cancel(handle_, &status);
  assert(MTAPI_SUCCESS == status);
}

} // namespace mtapi
} // namespace embb