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 (at) 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 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 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 * 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 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 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 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 * 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