1 /*
2 * Mesa 3-D graphics library
3 *
4 * Copyright (C) 2012-2013 LunarG, Inc.
5 *
6 * Permission is hereby granted, free of charge, to any person obtaining a
7 * copy of this software and associated documentation files (the "Software"),
8 * to deal in the Software without restriction, including without limitation
9 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
10 * and/or sell copies of the Software, and to permit persons to whom the
11 * Software is furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice shall be included
14 * in all copies or substantial portions of the Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
22 * DEALINGS IN THE SOFTWARE.
23 *
24 * Authors:
25 * Chia-I Wu <olv@lunarg.com>
26 */
27
28 #include "core/ilo_builder_mi.h"
29 #include "core/intel_winsys.h"
30
31 #include "ilo_shader.h"
32 #include "ilo_cp.h"
33
34 static const struct ilo_cp_owner ilo_cp_default_owner;
35
36 static void
ilo_cp_release_owner(struct ilo_cp * cp)37 ilo_cp_release_owner(struct ilo_cp *cp)
38 {
39 if (cp->owner != &ilo_cp_default_owner) {
40 const struct ilo_cp_owner *owner = cp->owner;
41
42 cp->owner = &ilo_cp_default_owner;
43
44 assert(ilo_cp_space(cp) >= owner->reserve);
45 owner->release(cp, owner->data);
46 }
47 }
48
49 /**
50 * Set the parser owner. If this is a new owner or a new ring, the old owner
51 * is released and the new owner's own() is called. The parser may implicitly
52 * submit if there is a ring change.
53 *
54 * own() is called before \p owner owns the parser. It must make sure there
55 * is more space than \p owner->reserve when it returns. Calling
56 * ilo_cp_submit() is allowed.
57 *
58 * release() will be called after \p owner loses the parser. That may happen
59 * just before the parser submits and ilo_cp_submit() is not allowed.
60 */
61 void
ilo_cp_set_owner(struct ilo_cp * cp,enum intel_ring_type ring,const struct ilo_cp_owner * owner)62 ilo_cp_set_owner(struct ilo_cp *cp, enum intel_ring_type ring,
63 const struct ilo_cp_owner *owner)
64 {
65 if (!owner)
66 owner = &ilo_cp_default_owner;
67
68 if (cp->ring != ring) {
69 ilo_cp_submit(cp, "ring change");
70 cp->ring = ring;
71 }
72
73 if (cp->owner != owner) {
74 ilo_cp_release_owner(cp);
75
76 owner->own(cp, owner->data);
77
78 assert(ilo_cp_space(cp) >= owner->reserve);
79 cp->owner = owner;
80 }
81 }
82
83 static struct intel_bo *
ilo_cp_end_batch(struct ilo_cp * cp,unsigned * used)84 ilo_cp_end_batch(struct ilo_cp *cp, unsigned *used)
85 {
86 struct intel_bo *bo;
87
88 ilo_cp_release_owner(cp);
89
90 if (!ilo_builder_batch_used(&cp->builder)) {
91 ilo_builder_batch_discard(&cp->builder);
92 return NULL;
93 }
94
95 /* see ilo_cp_space() */
96 assert(ilo_builder_batch_space(&cp->builder) >= 2);
97 gen6_mi_batch_buffer_end(&cp->builder);
98
99 bo = ilo_builder_end(&cp->builder, used);
100
101 /* we have to assume that kernel uploads also failed */
102 if (!bo)
103 ilo_shader_cache_invalidate(cp->shader_cache);
104
105 return bo;
106 }
107
108 static bool
ilo_cp_detect_hang(struct ilo_cp * cp)109 ilo_cp_detect_hang(struct ilo_cp *cp)
110 {
111 uint32_t active_lost, pending_lost;
112 bool guilty = false;
113
114 if (likely(!(ilo_debug & ILO_DEBUG_HANG)))
115 return false;
116
117 /* wait and get reset stats */
118 if (intel_bo_wait(cp->last_submitted_bo, -1) ||
119 intel_winsys_get_reset_stats(cp->winsys, cp->render_ctx,
120 &active_lost, &pending_lost))
121 return false;
122
123 if (cp->active_lost != active_lost) {
124 ilo_err("GPU hang caused by bo %p\n", cp->last_submitted_bo);
125 cp->active_lost = active_lost;
126 guilty = true;
127 }
128
129 if (cp->pending_lost != pending_lost) {
130 ilo_err("GPU hang detected\n");
131 cp->pending_lost = pending_lost;
132 }
133
134 return guilty;
135 }
136
137 /**
138 * Flush the command parser and execute the commands. When the parser buffer
139 * is empty, the callback is not invoked.
140 */
141 void
ilo_cp_submit_internal(struct ilo_cp * cp)142 ilo_cp_submit_internal(struct ilo_cp *cp)
143 {
144 const bool do_exec = !(ilo_debug & ILO_DEBUG_NOHW);
145 struct intel_bo *bo;
146 unsigned used;
147 int err;
148
149 bo = ilo_cp_end_batch(cp, &used);
150 if (!bo)
151 return;
152
153 if (likely(do_exec)) {
154 err = intel_winsys_submit_bo(cp->winsys, cp->ring,
155 bo, used, cp->render_ctx, cp->one_off_flags);
156 }
157 else {
158 err = 0;
159 }
160
161 cp->one_off_flags = 0;
162
163 if (!err) {
164 bool guilty;
165
166 intel_bo_unref(cp->last_submitted_bo);
167 cp->last_submitted_bo = intel_bo_ref(bo);
168
169 guilty = ilo_cp_detect_hang(cp);
170
171 if (unlikely((ilo_debug & ILO_DEBUG_BATCH) || guilty)) {
172 ilo_builder_decode(&cp->builder);
173 if (guilty)
174 abort();
175 }
176
177 if (cp->submit_callback)
178 cp->submit_callback(cp, cp->submit_callback_data);
179 }
180
181 ilo_builder_begin(&cp->builder);
182 }
183
184 /**
185 * Destroy the command parser.
186 */
187 void
ilo_cp_destroy(struct ilo_cp * cp)188 ilo_cp_destroy(struct ilo_cp *cp)
189 {
190 ilo_builder_reset(&cp->builder);
191
192 intel_winsys_destroy_context(cp->winsys, cp->render_ctx);
193 FREE(cp);
194 }
195
196 /**
197 * Create a command parser.
198 */
199 struct ilo_cp *
ilo_cp_create(const struct ilo_dev * dev,struct intel_winsys * winsys,struct ilo_shader_cache * shc)200 ilo_cp_create(const struct ilo_dev *dev,
201 struct intel_winsys *winsys,
202 struct ilo_shader_cache *shc)
203 {
204 struct ilo_cp *cp;
205
206 cp = CALLOC_STRUCT(ilo_cp);
207 if (!cp)
208 return NULL;
209
210 cp->winsys = winsys;
211 cp->shader_cache = shc;
212 cp->render_ctx = intel_winsys_create_context(winsys);
213 if (!cp->render_ctx) {
214 FREE(cp);
215 return NULL;
216 }
217
218 cp->ring = INTEL_RING_RENDER;
219 cp->owner = &ilo_cp_default_owner;
220
221 ilo_builder_init(&cp->builder, dev, winsys);
222
223 if (!ilo_builder_begin(&cp->builder)) {
224 ilo_cp_destroy(cp);
225 return NULL;
226 }
227
228 return cp;
229 }
230