starpu_data.h 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560
  1. /* StarPU --- Runtime system for heterogeneous multicore architectures.
  2. *
  3. * Copyright (C) 2009-2019 Université de Bordeaux
  4. * Copyright (C) 2011-2013,2016,2017 Inria
  5. * Copyright (C) 2010-2015,2017,2019 CNRS
  6. *
  7. * StarPU is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU Lesser General Public License as published by
  9. * the Free Software Foundation; either version 2.1 of the License, or (at
  10. * your option) any later version.
  11. *
  12. * StarPU is distributed in the hope that it will be useful, but
  13. * WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  15. *
  16. * See the GNU Lesser General Public License in COPYING.LGPL for more details.
  17. */
  18. #ifndef __STARPU_DATA_H__
  19. #define __STARPU_DATA_H__
  20. /**
  21. @defgroup API_Data_Management Data Management
  22. @brief Data management facilities provided by StarPU. We show how
  23. to use existing data interfaces in \ref API_Data_Interfaces, but
  24. developers can design their own data interfaces if required.
  25. @{
  26. */
  27. #include <starpu.h>
  28. #ifdef __cplusplus
  29. extern "C"
  30. {
  31. #endif
  32. /**
  33. This macro is used when the RAM memory node is specified.
  34. */
  35. #define STARPU_MAIN_RAM 0
  36. struct _starpu_data_state;
  37. /**
  38. StarPU uses ::starpu_data_handle_t as an opaque handle to manage a
  39. piece of data. Once a piece of data has been registered to StarPU,
  40. it is associated to a ::starpu_data_handle_t which keeps track of
  41. the state of the piece of data over the entire machine, so that we
  42. can maintain data consistency and locate data replicates for
  43. instance.
  44. */
  45. typedef struct _starpu_data_state* starpu_data_handle_t;
  46. /**
  47. Describe a StarPU data access mode
  48. Note: when adding a flag here, update
  49. _starpu_detect_implicit_data_deps_with_handle
  50. Note: other STARPU_* values in include/starpu_task_util.h
  51. */
  52. enum starpu_data_access_mode
  53. {
  54. STARPU_NONE=0, /**< todo */
  55. STARPU_R=(1<<0), /**< read-only mode */
  56. STARPU_W=(1<<1), /**< write-only mode */
  57. STARPU_RW=(STARPU_R|STARPU_W), /**< read-write mode. Equivalent to ::STARPU_R|::STARPU_W */
  58. STARPU_SCRATCH=(1<<2), /**< A temporary buffer is allocated
  59. for the task, but StarPU does not
  60. enforce data consistency---i.e. each
  61. device has its own buffer,
  62. independently from each other (even
  63. for CPUs), and no data transfer is
  64. ever performed. This is useful for
  65. temporary variables to avoid
  66. allocating/freeing buffers inside
  67. each task. Currently, no behavior is
  68. defined concerning the relation with
  69. the ::STARPU_R and ::STARPU_W modes
  70. and the value provided at
  71. registration --- i.e., the value of
  72. the scratch buffer is undefined at
  73. entry of the codelet function. It
  74. is being considered for future
  75. extensions at least to define the
  76. initial value. For now, data to be
  77. used in ::STARPU_SCRATCH mode should
  78. be registered with node -1 and a
  79. <c>NULL</c> pointer, since the value
  80. of the provided buffer is simply
  81. ignored for now.
  82. */
  83. STARPU_REDUX=(1<<3), /**< todo */
  84. STARPU_COMMUTE=(1<<4), /**< ::STARPU_COMMUTE can be passed
  85. along ::STARPU_W or ::STARPU_RW to
  86. express that StarPU can let tasks
  87. commute, which is useful e.g. when
  88. bringing a contribution into some
  89. data, which can be done in any order
  90. (but still require sequential
  91. consistency against reads or
  92. non-commutative writes).
  93. */
  94. STARPU_SSEND=(1<<5), /**< used in starpu_mpi_insert_task() to
  95. specify the data has to be sent using
  96. a synchronous and non-blocking mode
  97. (see starpu_mpi_issend())
  98. */
  99. STARPU_LOCALITY=(1<<6), /**< used to tell the scheduler which
  100. data is the most important for the
  101. task, and should thus be used to
  102. try to group tasks on the same core
  103. or cache, etc. For now only the ws
  104. and lws schedulers take this flag
  105. into account, and only when rebuild
  106. with \c USE_LOCALITY flag defined in
  107. the
  108. src/sched_policies/work_stealing_policy.c
  109. source code.
  110. */
  111. STARPU_ACCESS_MODE_MAX=(1<<7) /**< todo */
  112. };
  113. /**
  114. Describe a data handle along with an access mode.
  115. */
  116. struct starpu_data_descr
  117. {
  118. starpu_data_handle_t handle; /**< data */
  119. enum starpu_data_access_mode mode; /**< access mode */
  120. };
  121. struct starpu_data_interface_ops;
  122. /** Set the name of the data, to be shown in various profiling tools. */
  123. void starpu_data_set_name(starpu_data_handle_t handle, const char *name);
  124. /**
  125. Set the coordinates of the data, to be shown in various profiling
  126. tools. \p dimensions is the size of the \p dims array. This can be
  127. for instance the tile coordinates within a big matrix.
  128. */
  129. void starpu_data_set_coordinates_array(starpu_data_handle_t handle, int dimensions, int dims[]);
  130. /**
  131. Set the coordinates of the data, to be shown in various profiling
  132. tools. \p dimensions is the number of subsequent \c int parameters.
  133. This can be for instance the tile coordinates within a big matrix.
  134. */
  135. void starpu_data_set_coordinates(starpu_data_handle_t handle, unsigned dimensions, ...);
  136. /**
  137. Unregister a data \p handle from StarPU. If the data was
  138. automatically allocated by StarPU because the home node was -1, all
  139. automatically allocated buffers are freed. Otherwise, a valid copy
  140. of the data is put back into the home node in the buffer that was
  141. initially registered. Using a data handle that has been
  142. unregistered from StarPU results in an undefined behaviour. In case
  143. we do not need to update the value of the data in the home node, we
  144. can use the function starpu_data_unregister_no_coherency() instead.
  145. */
  146. void starpu_data_unregister(starpu_data_handle_t handle);
  147. /**
  148. Similar to starpu_data_unregister(), except that StarPU does not
  149. put back a valid copy into the home node, in the buffer that was
  150. initially registered.
  151. */
  152. void starpu_data_unregister_no_coherency(starpu_data_handle_t handle);
  153. /**
  154. Destroy the data \p handle once it is no longer needed by any
  155. submitted task. No coherency is assumed.
  156. */
  157. void starpu_data_unregister_submit(starpu_data_handle_t handle);
  158. /**
  159. Destroy all replicates of the data \p handle immediately. After
  160. data invalidation, the first access to \p handle must be performed
  161. in ::STARPU_W mode. Accessing an invalidated data in ::STARPU_R
  162. mode results in undefined behaviour.
  163. */
  164. void starpu_data_invalidate(starpu_data_handle_t handle);
  165. /**
  166. Submit invalidation of the data \p handle after completion of
  167. previously submitted tasks.
  168. */
  169. void starpu_data_invalidate_submit(starpu_data_handle_t handle);
  170. /**
  171. Specify that the data \p handle can be discarded without impacting
  172. the application.
  173. */
  174. void starpu_data_advise_as_important(starpu_data_handle_t handle, unsigned is_important);
  175. /** @name Access registered data from the application
  176. * @{
  177. */
  178. /**
  179. This macro can be used to acquire data, but not require it to be
  180. available on a given node, only enforce R/W dependencies. This can
  181. for instance be used to wait for tasks which produce the data, but
  182. without requesting a fetch to the main memory.
  183. */
  184. #define STARPU_ACQUIRE_NO_NODE -1
  185. /**
  186. Similar to ::STARPU_ACQUIRE_NO_NODE, but will lock the data on all
  187. nodes, preventing them from being evicted for instance. This is
  188. mostly useful inside StarPU only.
  189. */
  190. #define STARPU_ACQUIRE_NO_NODE_LOCK_ALL -2
  191. /**
  192. The application must call this function prior to accessing
  193. registered data from main memory outside tasks. StarPU ensures that
  194. the application will get an up-to-date copy of \p handle in main
  195. memory located where the data was originally registered, and that
  196. all concurrent accesses (e.g. from tasks) will be consistent with
  197. the access mode specified with \p mode. starpu_data_release() must
  198. be called once the application no longer needs to access the piece
  199. of data. Note that implicit data dependencies are also enforced by
  200. starpu_data_acquire(), i.e. starpu_data_acquire() will wait for all
  201. tasks scheduled to work on the data, unless they have been disabled
  202. explictly by calling
  203. starpu_data_set_default_sequential_consistency_flag() or
  204. starpu_data_set_sequential_consistency_flag().
  205. starpu_data_acquire() is a blocking call, so that it cannot be
  206. called from tasks or from their callbacks (in that case,
  207. starpu_data_acquire() returns <c>-EDEADLK</c>). Upon successful
  208. completion, this function returns 0.
  209. */
  210. int starpu_data_acquire(starpu_data_handle_t handle, enum starpu_data_access_mode mode);
  211. /**
  212. Similar to starpu_data_acquire(), except that the data will be
  213. available on the given memory node instead of main memory.
  214. ::STARPU_ACQUIRE_NO_NODE and ::STARPU_ACQUIRE_NO_NODE_LOCK_ALL can
  215. be used instead of an explicit node number.
  216. */
  217. int starpu_data_acquire_on_node(starpu_data_handle_t handle, int node, enum starpu_data_access_mode mode);
  218. /**
  219. Asynchronous equivalent of starpu_data_acquire(). When the data
  220. specified in \p handle is available in the access \p mode, the \p
  221. callback function is executed. The application may access
  222. the requested data during the execution of \p callback. The \p callback
  223. function must call starpu_data_release() once the application no longer
  224. needs to access the piece of data. Note that implicit data
  225. dependencies are also enforced by starpu_data_acquire_cb() in case they
  226. are not disabled. Contrary to starpu_data_acquire(), this function is
  227. non-blocking and may be called from task callbacks. Upon successful
  228. completion, this function returns 0.
  229. */
  230. int starpu_data_acquire_cb(starpu_data_handle_t handle, enum starpu_data_access_mode mode, void (*callback)(void *), void *arg);
  231. /**
  232. Similar to starpu_data_acquire_cb(), except that the
  233. data will be available on the given memory node instead of main
  234. memory.
  235. ::STARPU_ACQUIRE_NO_NODE and ::STARPU_ACQUIRE_NO_NODE_LOCK_ALL can be
  236. used instead of an explicit node number.
  237. */
  238. int starpu_data_acquire_on_node_cb(starpu_data_handle_t handle, int node, enum starpu_data_access_mode mode, void (*callback)(void *), void *arg);
  239. /**
  240. Similar to starpu_data_acquire_cb() with the possibility of
  241. enabling or disabling data dependencies.
  242. When the data specified in \p handle is available in the access
  243. \p mode, the \p callback function is executed. The application may access
  244. the requested data during the execution of this \p callback. The \p callback
  245. function must call starpu_data_release() once the application no longer
  246. needs to access the piece of data. Note that implicit data
  247. dependencies are also enforced by starpu_data_acquire_cb_sequential_consistency() in case they
  248. are not disabled specifically for the given \p handle or by the parameter \p sequential_consistency.
  249. Similarly to starpu_data_acquire_cb(), this function is
  250. non-blocking and may be called from task callbacks. Upon successful
  251. completion, this function returns 0.
  252. */
  253. int starpu_data_acquire_cb_sequential_consistency(starpu_data_handle_t handle, enum starpu_data_access_mode mode, void (*callback)(void *), void *arg, int sequential_consistency);
  254. /**
  255. Similar to starpu_data_acquire_cb_sequential_consistency(), except that the
  256. data will be available on the given memory node instead of main
  257. memory.
  258. ::STARPU_ACQUIRE_NO_NODE and ::STARPU_ACQUIRE_NO_NODE_LOCK_ALL can be used instead of an
  259. explicit node number.
  260. */
  261. int starpu_data_acquire_on_node_cb_sequential_consistency(starpu_data_handle_t handle, int node, enum starpu_data_access_mode mode, void (*callback)(void *), void *arg, int sequential_consistency);
  262. int starpu_data_acquire_on_node_cb_sequential_consistency_quick(starpu_data_handle_t handle, int node, enum starpu_data_access_mode mode, void (*callback)(void *), void *arg, int sequential_consistency, int quick);
  263. /**
  264. Similar to starpu_data_acquire_on_node_cb_sequential_consistency(),
  265. except that the \e pre_sync_jobid and \e post_sync_jobid parameters can be used
  266. to retrieve the jobid of the synchronization tasks. \e pre_sync_jobid happens
  267. just before the acquisition, and \e post_sync_jobid happens just after the
  268. release.
  269. */
  270. int starpu_data_acquire_on_node_cb_sequential_consistency_sync_jobids(starpu_data_handle_t handle, int node, enum starpu_data_access_mode mode, void (*callback)(void *), void *arg, int sequential_consistency, int quick, long *pre_sync_jobid, long *post_sync_jobid);
  271. /**
  272. The application can call this function instead of starpu_data_acquire() so as to
  273. acquire the data like starpu_data_acquire(), but only if all
  274. previously-submitted tasks have completed, in which case starpu_data_acquire_try()
  275. returns 0. StarPU will have ensured that the application will get an up-to-date
  276. copy of \p handle in main memory located where the data was originally
  277. registered. starpu_data_release() must be called once the application no longer
  278. needs to access the piece of data.
  279. */
  280. int starpu_data_acquire_try(starpu_data_handle_t handle, enum starpu_data_access_mode mode);
  281. /**
  282. Similar to starpu_data_acquire_try(), except that the
  283. data will be available on the given memory node instead of main
  284. memory.
  285. ::STARPU_ACQUIRE_NO_NODE and ::STARPU_ACQUIRE_NO_NODE_LOCK_ALL can be used instead of an
  286. explicit node number.
  287. */
  288. int starpu_data_acquire_on_node_try(starpu_data_handle_t handle, int node, enum starpu_data_access_mode mode);
  289. #ifdef __GCC__
  290. /**
  291. STARPU_DATA_ACQUIRE_CB() is the same as starpu_data_acquire_cb(),
  292. except that the code to be executed in a callback is directly provided
  293. as a macro parameter, and the data \p handle is automatically released
  294. after it. This permits to easily execute code which depends on the
  295. value of some registered data. This is non-blocking too and may be
  296. called from task callbacks.
  297. */
  298. # define STARPU_DATA_ACQUIRE_CB(handle, mode, code) do \
  299. { \ \
  300. void callback(void *arg) \
  301. { \
  302. code; \
  303. starpu_data_release(handle); \
  304. } \
  305. starpu_data_acquire_cb(handle, mode, callback, NULL); \
  306. } \
  307. while(0)
  308. #endif
  309. /**
  310. Release the piece of data acquired by the
  311. application either by starpu_data_acquire() or by
  312. starpu_data_acquire_cb().
  313. */
  314. void starpu_data_release(starpu_data_handle_t handle);
  315. /**
  316. Similar to starpu_data_release(), except that the data
  317. will be available on the given memory \p node instead of main memory.
  318. The \p node parameter must be exactly the same as the corresponding \c
  319. starpu_data_acquire_on_node* call.
  320. */
  321. void starpu_data_release_on_node(starpu_data_handle_t handle, int node);
  322. /** @} */
  323. /**
  324. This is an arbiter, which implements an advanced but centralized
  325. management of concurrent data accesses, see \ref
  326. ConcurrentDataAccess for the details.
  327. */
  328. typedef struct starpu_arbiter *starpu_arbiter_t;
  329. /**
  330. Create a data access arbiter, see \ref ConcurrentDataAccess for the
  331. details
  332. */
  333. starpu_arbiter_t starpu_arbiter_create(void) STARPU_ATTRIBUTE_MALLOC;
  334. /**
  335. Make access to \p handle managed by \p arbiter
  336. */
  337. void starpu_data_assign_arbiter(starpu_data_handle_t handle, starpu_arbiter_t arbiter);
  338. /**
  339. Destroy the \p arbiter . This must only be called after all data
  340. assigned to it have been unregistered.
  341. */
  342. void starpu_arbiter_destroy(starpu_arbiter_t arbiter);
  343. /**
  344. Explicitly ask StarPU to allocate room for a piece of data on
  345. the specified memory \p node.
  346. */
  347. int starpu_data_request_allocation(starpu_data_handle_t handle, unsigned node);
  348. /**
  349. Issue a fetch request for the data \p handle to \p node, i.e.
  350. requests that the data be replicated to the given node as soon as possible, so that it is
  351. available there for tasks. If \p async is 0, the call will
  352. block until the transfer is achieved, else the call will return immediately,
  353. after having just queued the request. In the latter case, the request will
  354. asynchronously wait for the completion of any task writing on the
  355. data.
  356. */
  357. int starpu_data_fetch_on_node(starpu_data_handle_t handle, unsigned node, unsigned async);
  358. /**
  359. Issue a prefetch request for the data \p handle to \p node, i.e.
  360. requests that the data be replicated to \p node when there is room for it, so that it is
  361. available there for tasks. If \p async is 0, the call will
  362. block until the transfer is achieved, else the call will return immediately,
  363. after having just queued the request. In the latter case, the request will
  364. asynchronously wait for the completion of any task writing on the
  365. data.
  366. */
  367. int starpu_data_prefetch_on_node(starpu_data_handle_t handle, unsigned node, unsigned async);
  368. int starpu_data_prefetch_on_node_prio(starpu_data_handle_t handle, unsigned node, unsigned async, int prio);
  369. /**
  370. Issue an idle prefetch request for the data \p handle to \p node, i.e.
  371. requests that the data be replicated to \p node, so that it is
  372. available there for tasks, but only when the bus is really idle. If \p async is 0, the call will
  373. block until the transfer is achieved, else the call will return immediately,
  374. after having just queued the request. In the latter case, the request will
  375. asynchronously wait for the completion of any task writing on the data.
  376. */
  377. int starpu_data_idle_prefetch_on_node(starpu_data_handle_t handle, unsigned node, unsigned async);
  378. int starpu_data_idle_prefetch_on_node_prio(starpu_data_handle_t handle, unsigned node, unsigned async, int prio);
  379. /**
  380. Check whether a valid copy of \p handle is currently available on
  381. memory node \p node.
  382. */
  383. unsigned starpu_data_is_on_node(starpu_data_handle_t handle, unsigned node);
  384. /**
  385. Advise StarPU that \p handle will not be used in the close future, and is
  386. thus a good candidate for eviction from GPUs. StarPU will thus write its value
  387. back to its home node when the bus is idle, and select this data in priority
  388. for eviction when memory gets low.
  389. */
  390. void starpu_data_wont_use(starpu_data_handle_t handle);
  391. /**
  392. Set the write-through mask of the data \p handle (and
  393. its children), i.e. a bitmask of nodes where the data should be always
  394. replicated after modification. It also prevents the data from being
  395. evicted from these nodes when memory gets scarse. When the data is
  396. modified, it is automatically transfered into those memory nodes. For
  397. instance a <c>1<<0</c> write-through mask means that the CUDA workers
  398. will commit their changes in main memory (node 0).
  399. */
  400. void starpu_data_set_wt_mask(starpu_data_handle_t handle, uint32_t wt_mask);
  401. /** @name Implicit Data Dependencies
  402. In this section, we describe how StarPU makes it possible to
  403. insert implicit task dependencies in order to enforce sequential data
  404. consistency. When this data consistency is enabled on a specific data
  405. handle, any data access will appear as sequentially consistent from
  406. the application. For instance, if the application submits two tasks
  407. that access the same piece of data in read-only mode, and then a third
  408. task that access it in write mode, dependencies will be added between
  409. the two first tasks and the third one. Implicit data dependencies are
  410. also inserted in the case of data accesses from the application.
  411. @{
  412. */
  413. /**
  414. Set the data consistency mode associated to a data handle. The
  415. consistency mode set using this function has the priority over the
  416. default mode which can be set with
  417. starpu_data_set_default_sequential_consistency_flag().
  418. */
  419. void starpu_data_set_sequential_consistency_flag(starpu_data_handle_t handle, unsigned flag);
  420. /**
  421. Get the data consistency mode associated to the data handle \p handle
  422. */
  423. unsigned starpu_data_get_sequential_consistency_flag(starpu_data_handle_t handle);
  424. /**
  425. Return the default sequential consistency flag
  426. */
  427. unsigned starpu_data_get_default_sequential_consistency_flag(void);
  428. /**
  429. Set the default sequential consistency flag. If a non-zero
  430. value is passed, a sequential data consistency will be enforced for
  431. all handles registered after this function call, otherwise it is
  432. disabled. By default, StarPU enables sequential data consistency. It
  433. is also possible to select the data consistency mode of a specific
  434. data handle with the function
  435. starpu_data_set_sequential_consistency_flag().
  436. */
  437. void starpu_data_set_default_sequential_consistency_flag(unsigned flag);
  438. /** @} */
  439. /**
  440. Set whether this data should be elligible to be evicted to disk
  441. storage (1) or not (0). The default is 1.
  442. */
  443. void starpu_data_set_ooc_flag(starpu_data_handle_t handle, unsigned flag);
  444. /**
  445. Get whether this data was set to be elligible to be evicted to disk
  446. storage (1) or not (0).
  447. */
  448. unsigned starpu_data_get_ooc_flag(starpu_data_handle_t handle);
  449. /**
  450. Query the status of \p handle on the specified \p memory_node.
  451. */
  452. void starpu_data_query_status(starpu_data_handle_t handle, int memory_node, int *is_allocated, int *is_valid, int *is_requested);
  453. struct starpu_codelet;
  454. /**
  455. Set the codelets to be used for \p handle when it is accessed in the
  456. mode ::STARPU_REDUX. Per-worker buffers will be initialized with
  457. the codelet \p init_cl, and reduction between per-worker buffers will be
  458. done with the codelet \p redux_cl.
  459. */
  460. void starpu_data_set_reduction_methods(starpu_data_handle_t handle, struct starpu_codelet *redux_cl, struct starpu_codelet *init_cl);
  461. struct starpu_data_interface_ops* starpu_data_get_interface_ops(starpu_data_handle_t handle);
  462. unsigned starpu_data_test_if_allocated_on_node(starpu_data_handle_t handle, unsigned memory_node);
  463. void starpu_memchunk_tidy(unsigned memory_node);
  464. /**
  465. Set the field \c user_data for the \p handle to \p user_data . It can
  466. then be retrieved with starpu_data_get_user_data(). \p user_data can be any
  467. application-defined value, for instance a pointer to an object-oriented
  468. container for the data.
  469. */
  470. void starpu_data_set_user_data(starpu_data_handle_t handle, void* user_data);
  471. /**
  472. Retrieve the field \c user_data previously set for the \p handle.
  473. */
  474. void *starpu_data_get_user_data(starpu_data_handle_t handle);
  475. /**
  476. Copy the content of \p src_handle into \p dst_handle. The parameter \p
  477. asynchronous indicates whether the function should block or not. In
  478. the case of an asynchronous call, it is possible to synchronize with
  479. the termination of this operation either by the means of implicit
  480. dependencies (if enabled) or by calling starpu_task_wait_for_all(). If
  481. \p callback_func is not <c>NULL</c>, this callback function is executed after
  482. the handle has been copied, and it is given the pointer \p
  483. callback_arg as argument.
  484. */
  485. int starpu_data_cpy(starpu_data_handle_t dst_handle, starpu_data_handle_t src_handle, int asynchronous, void (*callback_func)(void*), void *callback_arg);
  486. #ifdef __cplusplus
  487. }
  488. #endif
  489. /** @} */
  490. #endif /* __STARPU_DATA_H__ */