blob: 03c97bcf4da0669bd6f94add97e5550ef19ed6ac [file] [log] [blame]
Francisco Jerezc6db1b32012-04-20 16:56:19 +02001//
2// Copyright 2012 Francisco Jerez
3//
4// Permission is hereby granted, free of charge, to any person obtaining a
5// copy of this software and associated documentation files (the "Software"),
6// to deal in the Software without restriction, including without limitation
7// the rights to use, copy, modify, merge, publish, distribute, sublicense,
8// and/or sell copies of the Software, and to permit persons to whom the
9// Software is furnished to do so, subject to the following conditions:
10//
11// The above copyright notice and this permission notice shall be included in
12// all copies or substantial portions of the Software.
13//
14// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
Kenneth Graunkef0cb66b2013-04-21 13:52:08 -070017// THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
18// OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19// ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20// OTHER DEALINGS IN THE SOFTWARE.
Francisco Jerezc6db1b32012-04-20 16:56:19 +020021//
22
Francisco Jerez099d2812013-09-15 15:29:34 -070023#ifndef CLOVER_CORE_EVENT_HPP
24#define CLOVER_CORE_EVENT_HPP
Francisco Jerezc6db1b32012-04-20 16:56:19 +020025
Tom Stellard9c4dc982015-03-26 19:33:24 +000026#include <condition_variable>
Francisco Jerezc6db1b32012-04-20 16:56:19 +020027#include <functional>
28
Francisco Jerezbff60c82013-10-06 13:51:01 -070029#include "core/object.hpp"
Francisco Jerezc6db1b32012-04-20 16:56:19 +020030#include "core/queue.hpp"
Francisco Jerezebfdce02013-10-06 13:48:23 -070031#include "core/timestamp.hpp"
32#include "util/lazy.hpp"
Francisco Jerezc6db1b32012-04-20 16:56:19 +020033
34namespace clover {
Francisco Jerez257781f2013-10-01 11:54:07 -070035 ///
36 /// Class that represents a task that might be executed
37 /// asynchronously at some point in the future.
38 ///
39 /// An event consists of a list of dependencies, a boolean
40 /// signalled() flag, and an associated task. An event is
41 /// considered signalled as soon as all its dependencies (if any)
42 /// are signalled as well, and the trigger() method is called; at
43 /// that point the associated task will be started through the
44 /// specified \a action_ok. If the abort() method is called
45 /// instead, the specified \a action_fail is executed and the
46 /// associated task will never be started. Dependent events will
47 /// be aborted recursively.
48 ///
49 /// The execution status of the associated task can be queried
50 /// using the status() method, and it can be waited for completion
51 /// using the wait() method.
52 ///
53 class event : public ref_counter, public _cl_event {
54 public:
55 typedef std::function<void (event &)> action;
Francisco Jerezc6db1b32012-04-20 16:56:19 +020056
Francisco Jerezc4578d22014-02-18 15:07:11 +010057 event(clover::context &ctx, const ref_vector<event> &deps,
Francisco Jerez257781f2013-10-01 11:54:07 -070058 action action_ok, action action_fail);
59 virtual ~event();
Francisco Jerezc6db1b32012-04-20 16:56:19 +020060
Francisco Jerez5226eac2013-09-16 21:44:36 -070061 event(const event &ev) = delete;
62 event &
63 operator=(const event &ev) = delete;
64
Francisco Jerez257781f2013-10-01 11:54:07 -070065 void trigger();
66 void abort(cl_int status);
67 bool signalled() const;
Francisco Jerezc6db1b32012-04-20 16:56:19 +020068
Francisco Jerez4022a462015-05-09 16:22:33 +030069 virtual cl_int status() const;
Francisco Jerez257781f2013-10-01 11:54:07 -070070 virtual command_queue *queue() const = 0;
71 virtual cl_command_type command() const = 0;
Francisco Jerezbc4000e2015-06-09 22:59:43 +030072 void wait_signalled() const;
Tom Stellard9c4dc982015-03-26 19:33:24 +000073 virtual void wait() const;
Francisco Jerezc6db1b32012-04-20 16:56:19 +020074
Marek Olšák952b5e82015-04-10 18:42:42 +020075 virtual struct pipe_fence_handle *fence() const {
76 return NULL;
77 }
78
Francisco Jerezc4578d22014-02-18 15:07:11 +010079 const intrusive_ref<clover::context> context;
Francisco Jerezc6db1b32012-04-20 16:56:19 +020080
Francisco Jerez257781f2013-10-01 11:54:07 -070081 protected:
Francisco Jerezc4578d22014-02-18 15:07:11 +010082 void chain(event &ev);
Francisco Jerezc6db1b32012-04-20 16:56:19 +020083
Francisco Jerezc4578d22014-02-18 15:07:11 +010084 std::vector<intrusive_ref<event>> deps;
Francisco Jerezc6db1b32012-04-20 16:56:19 +020085
Francisco Jerez257781f2013-10-01 11:54:07 -070086 private:
Francisco Jerez2232b922015-05-09 14:47:38 +030087 std::vector<intrusive_ref<event>> trigger_self();
88 std::vector<intrusive_ref<event>> abort_self(cl_int status);
Francisco Jerez02f8ac62015-06-09 22:52:25 +030089 unsigned wait_count() const;
Francisco Jerez2232b922015-05-09 14:47:38 +030090
Francisco Jerez02f8ac62015-06-09 22:52:25 +030091 unsigned _wait_count;
Francisco Jerez4022a462015-05-09 16:22:33 +030092 cl_int _status;
Francisco Jerez257781f2013-10-01 11:54:07 -070093 action action_ok;
94 action action_fail;
Francisco Jerezc4578d22014-02-18 15:07:11 +010095 std::vector<intrusive_ref<event>> _chain;
Tom Stellard9c4dc982015-03-26 19:33:24 +000096 mutable std::condition_variable cv;
97 mutable std::mutex mutex;
Francisco Jerez257781f2013-10-01 11:54:07 -070098 };
Francisco Jerezebfdce02013-10-06 13:48:23 -070099
Francisco Jerezc6db1b32012-04-20 16:56:19 +0200100 ///
101 /// Class that represents a task executed by a command queue.
102 ///
103 /// Similar to a normal clover::event. In addition it's associated
104 /// with a given command queue \a q and a given OpenCL \a command.
105 /// hard_event instances created for the same queue are implicitly
106 /// ordered with respect to each other, and they are implicitly
107 /// triggered on construction.
108 ///
109 /// A hard_event is considered complete when the associated
110 /// hardware task finishes execution.
111 ///
112 class hard_event : public event {
113 public:
Francisco Jerez257781f2013-10-01 11:54:07 -0700114 hard_event(command_queue &q, cl_command_type command,
115 const ref_vector<event> &deps,
Francisco Jerezc6db1b32012-04-20 16:56:19 +0200116 action action = [](event &){});
117 ~hard_event();
118
119 virtual cl_int status() const;
Francisco Jerez257781f2013-10-01 11:54:07 -0700120 virtual command_queue *queue() const;
Francisco Jerezc6db1b32012-04-20 16:56:19 +0200121 virtual cl_command_type command() const;
122 virtual void wait() const;
123
Francisco Jerezebfdce02013-10-06 13:48:23 -0700124 const lazy<cl_ulong> &time_queued() const;
125 const lazy<cl_ulong> &time_submit() const;
126 const lazy<cl_ulong> &time_start() const;
127 const lazy<cl_ulong> &time_end() const;
Niels Ole Salscheider4a3505d2013-08-09 11:59:25 +0200128
Francisco Jerez9968d9d2013-10-01 11:57:32 -0700129 friend class command_queue;
Francisco Jerezc6db1b32012-04-20 16:56:19 +0200130
Marek Olšák952b5e82015-04-10 18:42:42 +0200131 virtual struct pipe_fence_handle *fence() const {
132 return _fence;
133 }
134
Francisco Jerezc6db1b32012-04-20 16:56:19 +0200135 private:
136 virtual void fence(pipe_fence_handle *fence);
Francisco Jerezebfdce02013-10-06 13:48:23 -0700137 action profile(command_queue &q, const action &action) const;
Francisco Jerezc6db1b32012-04-20 16:56:19 +0200138
Francisco Jerezc4578d22014-02-18 15:07:11 +0100139 const intrusive_ref<command_queue> _queue;
Francisco Jerez8e14b822013-09-17 23:13:48 -0700140 cl_command_type _command;
141 pipe_fence_handle *_fence;
Francisco Jerezebfdce02013-10-06 13:48:23 -0700142 lazy<cl_ulong> _time_queued, _time_submit, _time_start, _time_end;
Francisco Jerezc6db1b32012-04-20 16:56:19 +0200143 };
144
145 ///
146 /// Class that represents a software event.
147 ///
148 /// A soft_event is not associated with any specific hardware task
149 /// or command queue. It's considered complete as soon as all its
150 /// dependencies finish execution.
151 ///
152 class soft_event : public event {
153 public:
Francisco Jerezc4578d22014-02-18 15:07:11 +0100154 soft_event(clover::context &ctx, const ref_vector<event> &deps,
Francisco Jerezc6db1b32012-04-20 16:56:19 +0200155 bool trigger, action action = [](event &){});
156
Francisco Jerezc6db1b32012-04-20 16:56:19 +0200157 virtual cl_int status() const;
Francisco Jerez257781f2013-10-01 11:54:07 -0700158 virtual command_queue *queue() const;
Francisco Jerezc6db1b32012-04-20 16:56:19 +0200159 virtual cl_command_type command() const;
160 virtual void wait() const;
161 };
162}
163
164#endif