blob: 1760a076113048e7c8c8cca6204317fd124ca627 [file] [log] [blame]
Eric Anholt673a3942008-07-30 12:06:12 -07001/*
Daniel Vetterbe6a0372015-03-18 10:46:04 +01002 * Copyright © 2008-2015 Intel Corporation
Eric Anholt673a3942008-07-30 12:06:12 -07003 *
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 (including the next
12 * paragraph) shall be included in all copies or substantial portions of the
13 * Software.
14 *
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
21 * IN THE SOFTWARE.
22 *
23 * Authors:
24 * Eric Anholt <eric@anholt.net>
25 *
26 */
27
David Herrmann0de23972013-07-24 21:07:52 +020028#include <drm/drm_vma_manager.h>
David Howells760285e2012-10-02 18:01:07 +010029#include <drm/i915_drm.h>
Chris Wilson6b5e90f2016-11-14 20:41:05 +000030#include <linux/dma-fence-array.h>
Chris Wilsonfe3288b2017-02-12 17:20:01 +000031#include <linux/kthread.h>
Chris Wilsonc13d87e2016-07-20 09:21:15 +010032#include <linux/reservation.h>
Hugh Dickins5949eac2011-06-27 16:18:18 -070033#include <linux/shmem_fs.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090034#include <linux/slab.h>
Chris Wilson20e49332016-11-22 14:41:21 +000035#include <linux/stop_machine.h>
Eric Anholt673a3942008-07-30 12:06:12 -070036#include <linux/swap.h>
Jesse Barnes79e53942008-11-07 14:24:08 -080037#include <linux/pci.h>
Daniel Vetter1286ff72012-05-10 15:25:09 +020038#include <linux/dma-buf.h>
Daniel Vetterfcd70cd2019-01-17 22:03:34 +010039#include <linux/mman.h>
Eric Anholt673a3942008-07-30 12:06:12 -070040
Jani Nikuladf0566a2019-06-13 11:44:16 +030041#include "display/intel_display.h"
42#include "display/intel_frontbuffer.h"
43
Chris Wilson10be98a2019-05-28 10:29:49 +010044#include "gem/i915_gem_clflush.h"
45#include "gem/i915_gem_context.h"
Chris Wilsonafa13082019-05-28 10:29:43 +010046#include "gem/i915_gem_ioctls.h"
Chris Wilson10be98a2019-05-28 10:29:49 +010047#include "gem/i915_gem_pm.h"
48#include "gem/i915_gemfs.h"
Chris Wilson79ffac852019-04-24 21:07:17 +010049#include "gt/intel_engine_pm.h"
50#include "gt/intel_gt_pm.h"
Chris Wilson112ed2d2019-04-24 18:48:39 +010051#include "gt/intel_mocs.h"
52#include "gt/intel_reset.h"
53#include "gt/intel_workarounds.h"
54
Chris Wilson9f588922019-01-16 15:33:04 +000055#include "i915_drv.h"
Chris Wilson37d63f82019-05-28 10:29:50 +010056#include "i915_scatterlist.h"
Chris Wilson9f588922019-01-16 15:33:04 +000057#include "i915_trace.h"
58#include "i915_vgpu.h"
59
60#include "intel_drv.h"
Jani Nikula696173b2019-04-05 14:00:15 +030061#include "intel_pm.h"
Chris Wilson9f588922019-01-16 15:33:04 +000062
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +053063static int
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +010064insert_mappable_node(struct i915_ggtt *ggtt,
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +053065 struct drm_mm_node *node, u32 size)
66{
67 memset(node, 0, sizeof(*node));
Chris Wilson82ad6442018-06-05 16:37:58 +010068 return drm_mm_insert_node_in_range(&ggtt->vm.mm, node,
Chris Wilson4e64e552017-02-02 21:04:38 +000069 size, 0, I915_COLOR_UNEVICTABLE,
70 0, ggtt->mappable_end,
71 DRM_MM_INSERT_LOW);
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +053072}
73
74static void
75remove_mappable_node(struct drm_mm_node *node)
76{
77 drm_mm_remove_node(node);
78}
79
Eric Anholt673a3942008-07-30 12:06:12 -070080int
Eric Anholt5a125c32008-10-22 21:40:13 -070081i915_gem_get_aperture_ioctl(struct drm_device *dev, void *data,
Chris Wilson05394f32010-11-08 19:18:58 +000082 struct drm_file *file)
Eric Anholt5a125c32008-10-22 21:40:13 -070083{
Chris Wilson09d7e462019-01-28 10:23:53 +000084 struct i915_ggtt *ggtt = &to_i915(dev)->ggtt;
Joonas Lahtinen72e96d62016-03-30 16:57:10 +030085 struct drm_i915_gem_get_aperture *args = data;
Tvrtko Ursulinca1543b2015-07-01 11:51:10 +010086 struct i915_vma *vma;
Weinan Liff8f7972017-05-31 10:35:52 +080087 u64 pinned;
Eric Anholt5a125c32008-10-22 21:40:13 -070088
Chris Wilson09d7e462019-01-28 10:23:53 +000089 mutex_lock(&ggtt->vm.mutex);
90
Chris Wilson82ad6442018-06-05 16:37:58 +010091 pinned = ggtt->vm.reserved;
Chris Wilson499197d2019-01-28 10:23:52 +000092 list_for_each_entry(vma, &ggtt->vm.bound_list, vm_link)
Chris Wilson20dfbde2016-08-04 16:32:30 +010093 if (i915_vma_is_pinned(vma))
Tvrtko Ursulinca1543b2015-07-01 11:51:10 +010094 pinned += vma->node.size;
Chris Wilson09d7e462019-01-28 10:23:53 +000095
96 mutex_unlock(&ggtt->vm.mutex);
Eric Anholt5a125c32008-10-22 21:40:13 -070097
Chris Wilson82ad6442018-06-05 16:37:58 +010098 args->aper_size = ggtt->vm.total;
Akshay Joshi0206e352011-08-16 15:34:10 -040099 args->aper_available_size = args->aper_size - pinned;
Chris Wilson6299f992010-11-24 12:23:44 +0000100
Eric Anholt5a125c32008-10-22 21:40:13 -0700101 return 0;
102}
103
Chris Wilson35a96112016-08-14 18:44:40 +0100104int i915_gem_object_unbind(struct drm_i915_gem_object *obj)
Chris Wilsonaa653a62016-08-04 07:52:27 +0100105{
106 struct i915_vma *vma;
107 LIST_HEAD(still_in_list);
Chris Wilson6951e582019-05-28 10:29:51 +0100108 int ret = 0;
Chris Wilsonaa653a62016-08-04 07:52:27 +0100109
Chris Wilson02bef8f2016-08-14 18:44:41 +0100110 lockdep_assert_held(&obj->base.dev->struct_mutex);
111
Chris Wilson528cbd12019-01-28 10:23:54 +0000112 spin_lock(&obj->vma.lock);
113 while (!ret && (vma = list_first_entry_or_null(&obj->vma.list,
114 struct i915_vma,
115 obj_link))) {
Chris Wilsonaa653a62016-08-04 07:52:27 +0100116 list_move_tail(&vma->obj_link, &still_in_list);
Chris Wilson528cbd12019-01-28 10:23:54 +0000117 spin_unlock(&obj->vma.lock);
118
Chris Wilsonaa653a62016-08-04 07:52:27 +0100119 ret = i915_vma_unbind(vma);
Chris Wilson528cbd12019-01-28 10:23:54 +0000120
121 spin_lock(&obj->vma.lock);
Chris Wilsonaa653a62016-08-04 07:52:27 +0100122 }
Chris Wilson528cbd12019-01-28 10:23:54 +0000123 list_splice(&still_in_list, &obj->vma.list);
124 spin_unlock(&obj->vma.lock);
Chris Wilsonaa653a62016-08-04 07:52:27 +0100125
126 return ret;
127}
128
Chris Wilson00731152014-05-21 12:42:56 +0100129static int
130i915_gem_phys_pwrite(struct drm_i915_gem_object *obj,
131 struct drm_i915_gem_pwrite *args,
Chris Wilson03ac84f2016-10-28 13:58:36 +0100132 struct drm_file *file)
Chris Wilson00731152014-05-21 12:42:56 +0100133{
Chris Wilson00731152014-05-21 12:42:56 +0100134 void *vaddr = obj->phys_handle->vaddr + args->offset;
Gustavo Padovan3ed605b2016-04-26 12:32:27 -0300135 char __user *user_data = u64_to_user_ptr(args->data_ptr);
Chris Wilson6a2c4232014-11-04 04:51:40 -0800136
137 /* We manually control the domain here and pretend that it
138 * remains coherent i.e. in the GTT domain, like shmem_pwrite.
139 */
Rodrigo Vivi77a0d1c2015-06-18 11:43:24 -0700140 intel_fb_obj_invalidate(obj, ORIGIN_CPU);
Chris Wilson10466d22017-01-06 15:22:38 +0000141 if (copy_from_user(vaddr, user_data, args->size))
142 return -EFAULT;
Chris Wilson00731152014-05-21 12:42:56 +0100143
Chris Wilson6a2c4232014-11-04 04:51:40 -0800144 drm_clflush_virt_range(vaddr, args->size);
Chris Wilson10466d22017-01-06 15:22:38 +0000145 i915_gem_chipset_flush(to_i915(obj->base.dev));
Paulo Zanoni063e4e62015-02-13 17:23:45 -0200146
Chris Wilsond59b21e2017-02-22 11:40:49 +0000147 intel_fb_obj_flush(obj, ORIGIN_CPU);
Chris Wilson10466d22017-01-06 15:22:38 +0000148 return 0;
Chris Wilson00731152014-05-21 12:42:56 +0100149}
150
Dave Airlieff72145b2011-02-07 12:16:14 +1000151static int
152i915_gem_create(struct drm_file *file,
Tvrtko Ursulin12d79d72016-12-01 14:16:37 +0000153 struct drm_i915_private *dev_priv,
Michał Winiarskie1634842019-03-26 18:02:18 +0100154 u64 *size_p,
Jani Nikula739f3ab2019-01-16 11:15:19 +0200155 u32 *handle_p)
Eric Anholt673a3942008-07-30 12:06:12 -0700156{
Chris Wilson05394f32010-11-08 19:18:58 +0000157 struct drm_i915_gem_object *obj;
Pekka Paalanena1a2d1d2009-08-23 12:40:55 +0300158 u32 handle;
Michał Winiarskie1634842019-03-26 18:02:18 +0100159 u64 size;
160 int ret;
Eric Anholt673a3942008-07-30 12:06:12 -0700161
Michał Winiarskie1634842019-03-26 18:02:18 +0100162 size = round_up(*size_p, PAGE_SIZE);
Chris Wilson8ffc0242011-09-14 14:14:28 +0200163 if (size == 0)
164 return -EINVAL;
Eric Anholt673a3942008-07-30 12:06:12 -0700165
166 /* Allocate the new object */
Chris Wilson84753552019-05-28 10:29:45 +0100167 obj = i915_gem_object_create_shmem(dev_priv, size);
Chris Wilsonfe3db792016-04-25 13:32:13 +0100168 if (IS_ERR(obj))
169 return PTR_ERR(obj);
Eric Anholt673a3942008-07-30 12:06:12 -0700170
Chris Wilson05394f32010-11-08 19:18:58 +0000171 ret = drm_gem_handle_create(file, &obj->base, &handle);
Chris Wilson202f2fe2010-10-14 13:20:40 +0100172 /* drop reference from allocate - handle holds it now */
Chris Wilsonf0cd5182016-10-28 13:58:43 +0100173 i915_gem_object_put(obj);
Daniel Vetterd861e332013-07-24 23:25:03 +0200174 if (ret)
175 return ret;
Chris Wilson202f2fe2010-10-14 13:20:40 +0100176
Dave Airlieff72145b2011-02-07 12:16:14 +1000177 *handle_p = handle;
Chris Wilson99534022019-04-17 14:25:07 +0100178 *size_p = size;
Eric Anholt673a3942008-07-30 12:06:12 -0700179 return 0;
180}
181
Dave Airlieff72145b2011-02-07 12:16:14 +1000182int
183i915_gem_dumb_create(struct drm_file *file,
184 struct drm_device *dev,
185 struct drm_mode_create_dumb *args)
186{
Ville Syrjäläaa5ca8b2019-05-09 15:21:57 +0300187 int cpp = DIV_ROUND_UP(args->bpp, 8);
188 u32 format;
189
190 switch (cpp) {
191 case 1:
192 format = DRM_FORMAT_C8;
193 break;
194 case 2:
195 format = DRM_FORMAT_RGB565;
196 break;
197 case 4:
198 format = DRM_FORMAT_XRGB8888;
199 break;
200 default:
201 return -EINVAL;
202 }
203
Dave Airlieff72145b2011-02-07 12:16:14 +1000204 /* have to work out size/pitch and return them */
Ville Syrjäläaa5ca8b2019-05-09 15:21:57 +0300205 args->pitch = ALIGN(args->width * cpp, 64);
206
207 /* align stride to page size so that we can remap */
208 if (args->pitch > intel_plane_fb_max_stride(to_i915(dev), format,
209 DRM_FORMAT_MOD_LINEAR))
210 args->pitch = ALIGN(args->pitch, 4096);
211
Dave Airlieff72145b2011-02-07 12:16:14 +1000212 args->size = args->pitch * args->height;
Tvrtko Ursulin12d79d72016-12-01 14:16:37 +0000213 return i915_gem_create(file, to_i915(dev),
Michał Winiarskie1634842019-03-26 18:02:18 +0100214 &args->size, &args->handle);
Dave Airlieff72145b2011-02-07 12:16:14 +1000215}
216
Dave Airlieff72145b2011-02-07 12:16:14 +1000217/**
218 * Creates a new mm object and returns a handle to it.
Tvrtko Ursulin14bb2c12016-06-03 14:02:17 +0100219 * @dev: drm device pointer
220 * @data: ioctl data blob
221 * @file: drm file pointer
Dave Airlieff72145b2011-02-07 12:16:14 +1000222 */
223int
224i915_gem_create_ioctl(struct drm_device *dev, void *data,
225 struct drm_file *file)
226{
Tvrtko Ursulin12d79d72016-12-01 14:16:37 +0000227 struct drm_i915_private *dev_priv = to_i915(dev);
Dave Airlieff72145b2011-02-07 12:16:14 +1000228 struct drm_i915_gem_create *args = data;
Daniel Vetter63ed2cb2012-04-23 16:50:50 +0200229
Tvrtko Ursulin12d79d72016-12-01 14:16:37 +0000230 i915_gem_flush_free_objects(dev_priv);
Chris Wilsonfbbd37b2016-10-28 13:58:42 +0100231
Tvrtko Ursulin12d79d72016-12-01 14:16:37 +0000232 return i915_gem_create(file, dev_priv,
Michał Winiarskie1634842019-03-26 18:02:18 +0100233 &args->size, &args->handle);
Dave Airlieff72145b2011-02-07 12:16:14 +1000234}
235
Chris Wilson7125397b2017-12-06 12:49:14 +0000236void i915_gem_flush_ggtt_writes(struct drm_i915_private *dev_priv)
Chris Wilsonef749212017-04-12 12:01:10 +0100237{
Chris Wilson538ef962019-01-14 14:21:18 +0000238 intel_wakeref_t wakeref;
239
Chris Wilson7125397b2017-12-06 12:49:14 +0000240 /*
241 * No actual flushing is required for the GTT write domain for reads
242 * from the GTT domain. Writes to it "immediately" go to main memory
243 * as far as we know, so there's no chipset flush. It also doesn't
244 * land in the GPU render cache.
Chris Wilsonef749212017-04-12 12:01:10 +0100245 *
246 * However, we do have to enforce the order so that all writes through
247 * the GTT land before any writes to the device, such as updates to
248 * the GATT itself.
249 *
250 * We also have to wait a bit for the writes to land from the GTT.
251 * An uncached read (i.e. mmio) seems to be ideal for the round-trip
252 * timing. This issue has only been observed when switching quickly
253 * between GTT writes and CPU reads from inside the kernel on recent hw,
254 * and it appears to only affect discrete GTT blocks (i.e. on LLC
Chris Wilson7125397b2017-12-06 12:49:14 +0000255 * system agents we cannot reproduce this behaviour, until Cannonlake
256 * that was!).
Chris Wilsonef749212017-04-12 12:01:10 +0100257 */
Chris Wilson7125397b2017-12-06 12:49:14 +0000258
Chris Wilson900ccf32018-07-20 11:19:10 +0100259 wmb();
260
261 if (INTEL_INFO(dev_priv)->has_coherent_ggtt)
262 return;
263
Chris Wilsona8bd3b82018-07-17 10:26:55 +0100264 i915_gem_chipset_flush(dev_priv);
Chris Wilsonef749212017-04-12 12:01:10 +0100265
Daniele Ceraolo Spurioc447ff72019-06-13 16:21:55 -0700266 with_intel_runtime_pm(&dev_priv->runtime_pm, wakeref) {
Tvrtko Ursuline33a4be2019-06-11 11:45:44 +0100267 struct intel_uncore *uncore = &dev_priv->uncore;
Chris Wilson7125397b2017-12-06 12:49:14 +0000268
Tvrtko Ursuline33a4be2019-06-11 11:45:44 +0100269 spin_lock_irq(&uncore->lock);
270 intel_uncore_posting_read_fw(uncore,
271 RING_HEAD(RENDER_RING_BASE));
272 spin_unlock_irq(&uncore->lock);
Chris Wilsond4225a52019-01-14 14:21:23 +0000273 }
Chris Wilson7125397b2017-12-06 12:49:14 +0000274}
275
Daniel Vetterd174bd62012-03-25 19:47:40 +0200276static int
Chris Wilsonb9d126e2019-01-05 12:07:58 +0000277shmem_pread(struct page *page, int offset, int len, char __user *user_data,
278 bool needs_clflush)
Daniel Vetterd174bd62012-03-25 19:47:40 +0200279{
280 char *vaddr;
281 int ret;
282
283 vaddr = kmap(page);
Daniel Vetterd174bd62012-03-25 19:47:40 +0200284
Chris Wilsonb9d126e2019-01-05 12:07:58 +0000285 if (needs_clflush)
286 drm_clflush_virt_range(vaddr + offset, len);
287
288 ret = __copy_to_user(user_data, vaddr + offset, len);
289
Daniel Vetterd174bd62012-03-25 19:47:40 +0200290 kunmap(page);
291
Chris Wilsonb9d126e2019-01-05 12:07:58 +0000292 return ret ? -EFAULT : 0;
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100293}
294
295static int
296i915_gem_shmem_pread(struct drm_i915_gem_object *obj,
297 struct drm_i915_gem_pread *args)
298{
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100299 unsigned int needs_clflush;
300 unsigned int idx, offset;
Chris Wilson6951e582019-05-28 10:29:51 +0100301 struct dma_fence *fence;
302 char __user *user_data;
303 u64 remain;
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100304 int ret;
305
Chris Wilson6951e582019-05-28 10:29:51 +0100306 ret = i915_gem_object_prepare_read(obj, &needs_clflush);
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100307 if (ret)
308 return ret;
309
Chris Wilson6951e582019-05-28 10:29:51 +0100310 fence = i915_gem_object_lock_fence(obj);
311 i915_gem_object_finish_access(obj);
312 if (!fence)
313 return -ENOMEM;
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100314
315 remain = args->size;
316 user_data = u64_to_user_ptr(args->data_ptr);
317 offset = offset_in_page(args->offset);
318 for (idx = args->offset >> PAGE_SHIFT; remain; idx++) {
319 struct page *page = i915_gem_object_get_page(obj, idx);
Chris Wilsona5e856a52018-10-12 15:02:28 +0100320 unsigned int length = min_t(u64, remain, PAGE_SIZE - offset);
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100321
322 ret = shmem_pread(page, offset, length, user_data,
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100323 needs_clflush);
324 if (ret)
325 break;
326
327 remain -= length;
328 user_data += length;
329 offset = 0;
330 }
331
Chris Wilson6951e582019-05-28 10:29:51 +0100332 i915_gem_object_unlock_fence(obj, fence);
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100333 return ret;
334}
335
336static inline bool
337gtt_user_read(struct io_mapping *mapping,
338 loff_t base, int offset,
339 char __user *user_data, int length)
340{
Ville Syrjäläafe722b2017-09-01 20:12:52 +0300341 void __iomem *vaddr;
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100342 unsigned long unwritten;
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530343
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530344 /* We can use the cpu mem copy function because this is X86. */
Ville Syrjäläafe722b2017-09-01 20:12:52 +0300345 vaddr = io_mapping_map_atomic_wc(mapping, base);
346 unwritten = __copy_to_user_inatomic(user_data,
347 (void __force *)vaddr + offset,
348 length);
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100349 io_mapping_unmap_atomic(vaddr);
350 if (unwritten) {
Ville Syrjäläafe722b2017-09-01 20:12:52 +0300351 vaddr = io_mapping_map_wc(mapping, base, PAGE_SIZE);
352 unwritten = copy_to_user(user_data,
353 (void __force *)vaddr + offset,
354 length);
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100355 io_mapping_unmap(vaddr);
356 }
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530357 return unwritten;
358}
359
360static int
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100361i915_gem_gtt_pread(struct drm_i915_gem_object *obj,
362 const struct drm_i915_gem_pread *args)
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530363{
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100364 struct drm_i915_private *i915 = to_i915(obj->base.dev);
365 struct i915_ggtt *ggtt = &i915->ggtt;
Chris Wilson538ef962019-01-14 14:21:18 +0000366 intel_wakeref_t wakeref;
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530367 struct drm_mm_node node;
Chris Wilson6951e582019-05-28 10:29:51 +0100368 struct dma_fence *fence;
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100369 void __user *user_data;
Chris Wilson6951e582019-05-28 10:29:51 +0100370 struct i915_vma *vma;
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100371 u64 remain, offset;
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530372 int ret;
373
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100374 ret = mutex_lock_interruptible(&i915->drm.struct_mutex);
375 if (ret)
376 return ret;
377
Daniele Ceraolo Spuriod858d562019-06-13 16:21:54 -0700378 wakeref = intel_runtime_pm_get(&i915->runtime_pm);
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100379 vma = i915_gem_object_ggtt_pin(obj, NULL, 0, 0,
Chris Wilsona3259ca2017-10-09 09:44:00 +0100380 PIN_MAPPABLE |
381 PIN_NONFAULT |
382 PIN_NONBLOCK);
Chris Wilson18034582016-08-18 17:16:45 +0100383 if (!IS_ERR(vma)) {
384 node.start = i915_ggtt_offset(vma);
385 node.allocated = false;
Chris Wilson49ef5292016-08-18 17:17:00 +0100386 ret = i915_vma_put_fence(vma);
Chris Wilson18034582016-08-18 17:16:45 +0100387 if (ret) {
388 i915_vma_unpin(vma);
389 vma = ERR_PTR(ret);
390 }
391 }
Chris Wilson058d88c2016-08-15 10:49:06 +0100392 if (IS_ERR(vma)) {
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100393 ret = insert_mappable_node(ggtt, &node, PAGE_SIZE);
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530394 if (ret)
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100395 goto out_unlock;
396 GEM_BUG_ON(!node.allocated);
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530397 }
398
Chris Wilson6951e582019-05-28 10:29:51 +0100399 mutex_unlock(&i915->drm.struct_mutex);
400
401 ret = i915_gem_object_lock_interruptible(obj);
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530402 if (ret)
403 goto out_unpin;
404
Chris Wilson6951e582019-05-28 10:29:51 +0100405 ret = i915_gem_object_set_to_gtt_domain(obj, false);
406 if (ret) {
407 i915_gem_object_unlock(obj);
408 goto out_unpin;
409 }
410
411 fence = i915_gem_object_lock_fence(obj);
412 i915_gem_object_unlock(obj);
413 if (!fence) {
414 ret = -ENOMEM;
415 goto out_unpin;
416 }
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530417
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100418 user_data = u64_to_user_ptr(args->data_ptr);
419 remain = args->size;
420 offset = args->offset;
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530421
422 while (remain > 0) {
423 /* Operation in this page
424 *
425 * page_base = page offset within aperture
426 * page_offset = offset within page
427 * page_length = bytes to copy for this page
428 */
429 u32 page_base = node.start;
430 unsigned page_offset = offset_in_page(offset);
431 unsigned page_length = PAGE_SIZE - page_offset;
432 page_length = remain < page_length ? remain : page_length;
433 if (node.allocated) {
434 wmb();
Chris Wilson82ad6442018-06-05 16:37:58 +0100435 ggtt->vm.insert_page(&ggtt->vm,
436 i915_gem_object_get_dma_address(obj, offset >> PAGE_SHIFT),
437 node.start, I915_CACHE_NONE, 0);
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530438 wmb();
439 } else {
440 page_base += offset & PAGE_MASK;
441 }
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100442
Matthew Auld73ebd502017-12-11 15:18:20 +0000443 if (gtt_user_read(&ggtt->iomap, page_base, page_offset,
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100444 user_data, page_length)) {
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530445 ret = -EFAULT;
446 break;
447 }
448
449 remain -= page_length;
450 user_data += page_length;
451 offset += page_length;
452 }
453
Chris Wilson6951e582019-05-28 10:29:51 +0100454 i915_gem_object_unlock_fence(obj, fence);
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530455out_unpin:
Chris Wilson6951e582019-05-28 10:29:51 +0100456 mutex_lock(&i915->drm.struct_mutex);
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530457 if (node.allocated) {
458 wmb();
Chris Wilson82ad6442018-06-05 16:37:58 +0100459 ggtt->vm.clear_range(&ggtt->vm, node.start, node.size);
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530460 remove_mappable_node(&node);
461 } else {
Chris Wilson058d88c2016-08-15 10:49:06 +0100462 i915_vma_unpin(vma);
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530463 }
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100464out_unlock:
Daniele Ceraolo Spuriod858d562019-06-13 16:21:54 -0700465 intel_runtime_pm_put(&i915->runtime_pm, wakeref);
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100466 mutex_unlock(&i915->drm.struct_mutex);
Chris Wilsonf60d7f02012-09-04 21:02:56 +0100467
Eric Anholteb014592009-03-10 11:44:52 -0700468 return ret;
469}
470
Eric Anholt673a3942008-07-30 12:06:12 -0700471/**
472 * Reads data from the object referenced by handle.
Tvrtko Ursulin14bb2c12016-06-03 14:02:17 +0100473 * @dev: drm device pointer
474 * @data: ioctl data blob
475 * @file: drm file pointer
Eric Anholt673a3942008-07-30 12:06:12 -0700476 *
477 * On error, the contents of *data are undefined.
478 */
479int
480i915_gem_pread_ioctl(struct drm_device *dev, void *data,
Chris Wilson05394f32010-11-08 19:18:58 +0000481 struct drm_file *file)
Eric Anholt673a3942008-07-30 12:06:12 -0700482{
483 struct drm_i915_gem_pread *args = data;
Chris Wilson05394f32010-11-08 19:18:58 +0000484 struct drm_i915_gem_object *obj;
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100485 int ret;
Eric Anholt673a3942008-07-30 12:06:12 -0700486
Chris Wilson51311d02010-11-17 09:10:42 +0000487 if (args->size == 0)
488 return 0;
489
Linus Torvalds96d4f262019-01-03 18:57:57 -0800490 if (!access_ok(u64_to_user_ptr(args->data_ptr),
Chris Wilson51311d02010-11-17 09:10:42 +0000491 args->size))
492 return -EFAULT;
493
Chris Wilson03ac0642016-07-20 13:31:51 +0100494 obj = i915_gem_object_lookup(file, args->handle);
Chris Wilson258a5ed2016-08-05 10:14:16 +0100495 if (!obj)
496 return -ENOENT;
Eric Anholt673a3942008-07-30 12:06:12 -0700497
Chris Wilson7dcd2492010-09-26 20:21:44 +0100498 /* Bounds check source. */
Matthew Auld966d5bf2016-12-13 20:32:22 +0000499 if (range_overflows_t(u64, args->offset, args->size, obj->base.size)) {
Chris Wilsonce9d4192010-09-26 20:50:05 +0100500 ret = -EINVAL;
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100501 goto out;
Chris Wilsonce9d4192010-09-26 20:50:05 +0100502 }
503
Chris Wilsondb53a302011-02-03 11:57:46 +0000504 trace_i915_gem_object_pread(obj, args->offset, args->size);
505
Chris Wilsone95433c2016-10-28 13:58:27 +0100506 ret = i915_gem_object_wait(obj,
507 I915_WAIT_INTERRUPTIBLE,
Chris Wilson62eb3c22019-02-13 09:25:04 +0000508 MAX_SCHEDULE_TIMEOUT);
Chris Wilson258a5ed2016-08-05 10:14:16 +0100509 if (ret)
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100510 goto out;
Chris Wilson258a5ed2016-08-05 10:14:16 +0100511
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100512 ret = i915_gem_object_pin_pages(obj);
Chris Wilson258a5ed2016-08-05 10:14:16 +0100513 if (ret)
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100514 goto out;
Chris Wilson258a5ed2016-08-05 10:14:16 +0100515
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100516 ret = i915_gem_shmem_pread(obj, args);
Chris Wilson9c870d02016-10-24 13:42:15 +0100517 if (ret == -EFAULT || ret == -ENODEV)
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100518 ret = i915_gem_gtt_pread(obj, args);
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530519
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100520 i915_gem_object_unpin_pages(obj);
521out:
Chris Wilsonf0cd5182016-10-28 13:58:43 +0100522 i915_gem_object_put(obj);
Eric Anholteb014592009-03-10 11:44:52 -0700523 return ret;
Eric Anholt673a3942008-07-30 12:06:12 -0700524}
525
Keith Packard0839ccb2008-10-30 19:38:48 -0700526/* This is the fast write path which cannot handle
527 * page faults in the source data
Linus Torvalds9b7530cc2008-10-20 14:16:43 -0700528 */
Linus Torvalds9b7530cc2008-10-20 14:16:43 -0700529
Chris Wilsonfe115622016-10-28 13:58:40 +0100530static inline bool
531ggtt_write(struct io_mapping *mapping,
532 loff_t base, int offset,
533 char __user *user_data, int length)
Keith Packard0839ccb2008-10-30 19:38:48 -0700534{
Ville Syrjäläafe722b2017-09-01 20:12:52 +0300535 void __iomem *vaddr;
Keith Packard0839ccb2008-10-30 19:38:48 -0700536 unsigned long unwritten;
537
Ben Widawsky4f0c7cf2012-04-16 14:07:47 -0700538 /* We can use the cpu mem copy function because this is X86. */
Ville Syrjäläafe722b2017-09-01 20:12:52 +0300539 vaddr = io_mapping_map_atomic_wc(mapping, base);
540 unwritten = __copy_from_user_inatomic_nocache((void __force *)vaddr + offset,
Keith Packard0839ccb2008-10-30 19:38:48 -0700541 user_data, length);
Chris Wilsonfe115622016-10-28 13:58:40 +0100542 io_mapping_unmap_atomic(vaddr);
543 if (unwritten) {
Ville Syrjäläafe722b2017-09-01 20:12:52 +0300544 vaddr = io_mapping_map_wc(mapping, base, PAGE_SIZE);
545 unwritten = copy_from_user((void __force *)vaddr + offset,
546 user_data, length);
Chris Wilsonfe115622016-10-28 13:58:40 +0100547 io_mapping_unmap(vaddr);
548 }
Keith Packard0839ccb2008-10-30 19:38:48 -0700549
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100550 return unwritten;
551}
552
Eric Anholt3de09aa2009-03-09 09:42:23 -0700553/**
554 * This is the fast pwrite path, where we copy the data directly from the
555 * user into the GTT, uncached.
Chris Wilsonfe115622016-10-28 13:58:40 +0100556 * @obj: i915 GEM object
Tvrtko Ursulin14bb2c12016-06-03 14:02:17 +0100557 * @args: pwrite arguments structure
Eric Anholt3de09aa2009-03-09 09:42:23 -0700558 */
Eric Anholt673a3942008-07-30 12:06:12 -0700559static int
Chris Wilsonfe115622016-10-28 13:58:40 +0100560i915_gem_gtt_pwrite_fast(struct drm_i915_gem_object *obj,
561 const struct drm_i915_gem_pwrite *args)
Eric Anholt673a3942008-07-30 12:06:12 -0700562{
Chris Wilsonfe115622016-10-28 13:58:40 +0100563 struct drm_i915_private *i915 = to_i915(obj->base.dev);
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +0530564 struct i915_ggtt *ggtt = &i915->ggtt;
Daniele Ceraolo Spuriod858d562019-06-13 16:21:54 -0700565 struct intel_runtime_pm *rpm = &i915->runtime_pm;
Chris Wilson538ef962019-01-14 14:21:18 +0000566 intel_wakeref_t wakeref;
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +0530567 struct drm_mm_node node;
Chris Wilson6951e582019-05-28 10:29:51 +0100568 struct dma_fence *fence;
Chris Wilsonfe115622016-10-28 13:58:40 +0100569 struct i915_vma *vma;
570 u64 remain, offset;
571 void __user *user_data;
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +0530572 int ret;
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530573
Chris Wilsonfe115622016-10-28 13:58:40 +0100574 ret = mutex_lock_interruptible(&i915->drm.struct_mutex);
575 if (ret)
576 return ret;
Daniel Vetter935aaa62012-03-25 19:47:35 +0200577
Chris Wilson8bd818152017-10-19 07:37:33 +0100578 if (i915_gem_object_has_struct_page(obj)) {
579 /*
580 * Avoid waking the device up if we can fallback, as
581 * waking/resuming is very slow (worst-case 10-100 ms
582 * depending on PCI sleeps and our own resume time).
583 * This easily dwarfs any performance advantage from
584 * using the cache bypass of indirect GGTT access.
585 */
Daniele Ceraolo Spuriod858d562019-06-13 16:21:54 -0700586 wakeref = intel_runtime_pm_get_if_in_use(rpm);
Chris Wilson538ef962019-01-14 14:21:18 +0000587 if (!wakeref) {
Chris Wilson8bd818152017-10-19 07:37:33 +0100588 ret = -EFAULT;
589 goto out_unlock;
590 }
591 } else {
592 /* No backing pages, no fallback, we must force GGTT access */
Daniele Ceraolo Spuriod858d562019-06-13 16:21:54 -0700593 wakeref = intel_runtime_pm_get(rpm);
Chris Wilson8bd818152017-10-19 07:37:33 +0100594 }
595
Chris Wilson058d88c2016-08-15 10:49:06 +0100596 vma = i915_gem_object_ggtt_pin(obj, NULL, 0, 0,
Chris Wilsona3259ca2017-10-09 09:44:00 +0100597 PIN_MAPPABLE |
598 PIN_NONFAULT |
599 PIN_NONBLOCK);
Chris Wilson18034582016-08-18 17:16:45 +0100600 if (!IS_ERR(vma)) {
601 node.start = i915_ggtt_offset(vma);
602 node.allocated = false;
Chris Wilson49ef5292016-08-18 17:17:00 +0100603 ret = i915_vma_put_fence(vma);
Chris Wilson18034582016-08-18 17:16:45 +0100604 if (ret) {
605 i915_vma_unpin(vma);
606 vma = ERR_PTR(ret);
607 }
608 }
Chris Wilson058d88c2016-08-15 10:49:06 +0100609 if (IS_ERR(vma)) {
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100610 ret = insert_mappable_node(ggtt, &node, PAGE_SIZE);
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +0530611 if (ret)
Chris Wilson8bd818152017-10-19 07:37:33 +0100612 goto out_rpm;
Chris Wilsonfe115622016-10-28 13:58:40 +0100613 GEM_BUG_ON(!node.allocated);
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +0530614 }
Daniel Vetter935aaa62012-03-25 19:47:35 +0200615
Chris Wilson6951e582019-05-28 10:29:51 +0100616 mutex_unlock(&i915->drm.struct_mutex);
617
618 ret = i915_gem_object_lock_interruptible(obj);
Daniel Vetter935aaa62012-03-25 19:47:35 +0200619 if (ret)
620 goto out_unpin;
621
Chris Wilson6951e582019-05-28 10:29:51 +0100622 ret = i915_gem_object_set_to_gtt_domain(obj, true);
623 if (ret) {
624 i915_gem_object_unlock(obj);
625 goto out_unpin;
626 }
627
628 fence = i915_gem_object_lock_fence(obj);
629 i915_gem_object_unlock(obj);
630 if (!fence) {
631 ret = -ENOMEM;
632 goto out_unpin;
633 }
Chris Wilsonfe115622016-10-28 13:58:40 +0100634
Chris Wilsonb19482d2016-08-18 17:16:43 +0100635 intel_fb_obj_invalidate(obj, ORIGIN_CPU);
Paulo Zanoni063e4e62015-02-13 17:23:45 -0200636
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +0530637 user_data = u64_to_user_ptr(args->data_ptr);
638 offset = args->offset;
639 remain = args->size;
640 while (remain) {
Eric Anholt673a3942008-07-30 12:06:12 -0700641 /* Operation in this page
642 *
Keith Packard0839ccb2008-10-30 19:38:48 -0700643 * page_base = page offset within aperture
644 * page_offset = offset within page
645 * page_length = bytes to copy for this page
Eric Anholt673a3942008-07-30 12:06:12 -0700646 */
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +0530647 u32 page_base = node.start;
Chris Wilsonbb6dc8d2016-10-28 13:58:39 +0100648 unsigned int page_offset = offset_in_page(offset);
649 unsigned int page_length = PAGE_SIZE - page_offset;
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +0530650 page_length = remain < page_length ? remain : page_length;
651 if (node.allocated) {
652 wmb(); /* flush the write before we modify the GGTT */
Chris Wilson82ad6442018-06-05 16:37:58 +0100653 ggtt->vm.insert_page(&ggtt->vm,
654 i915_gem_object_get_dma_address(obj, offset >> PAGE_SHIFT),
655 node.start, I915_CACHE_NONE, 0);
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +0530656 wmb(); /* flush modifications to the GGTT (insert_page) */
657 } else {
658 page_base += offset & PAGE_MASK;
659 }
Keith Packard0839ccb2008-10-30 19:38:48 -0700660 /* If we get a fault while copying data, then (presumably) our
Eric Anholt3de09aa2009-03-09 09:42:23 -0700661 * source page isn't available. Return the error and we'll
662 * retry in the slow path.
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530663 * If the object is non-shmem backed, we retry again with the
664 * path that handles page fault.
Keith Packard0839ccb2008-10-30 19:38:48 -0700665 */
Matthew Auld73ebd502017-12-11 15:18:20 +0000666 if (ggtt_write(&ggtt->iomap, page_base, page_offset,
Chris Wilsonfe115622016-10-28 13:58:40 +0100667 user_data, page_length)) {
668 ret = -EFAULT;
669 break;
Daniel Vetter935aaa62012-03-25 19:47:35 +0200670 }
Eric Anholt673a3942008-07-30 12:06:12 -0700671
Keith Packard0839ccb2008-10-30 19:38:48 -0700672 remain -= page_length;
673 user_data += page_length;
674 offset += page_length;
Eric Anholt673a3942008-07-30 12:06:12 -0700675 }
Chris Wilsond59b21e2017-02-22 11:40:49 +0000676 intel_fb_obj_flush(obj, ORIGIN_CPU);
Chris Wilsonfe115622016-10-28 13:58:40 +0100677
Chris Wilson6951e582019-05-28 10:29:51 +0100678 i915_gem_object_unlock_fence(obj, fence);
Daniel Vetter935aaa62012-03-25 19:47:35 +0200679out_unpin:
Chris Wilson6951e582019-05-28 10:29:51 +0100680 mutex_lock(&i915->drm.struct_mutex);
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +0530681 if (node.allocated) {
682 wmb();
Chris Wilson82ad6442018-06-05 16:37:58 +0100683 ggtt->vm.clear_range(&ggtt->vm, node.start, node.size);
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +0530684 remove_mappable_node(&node);
685 } else {
Chris Wilson058d88c2016-08-15 10:49:06 +0100686 i915_vma_unpin(vma);
Ankitprasad Sharma4f1959e2016-06-10 14:23:01 +0530687 }
Chris Wilson8bd818152017-10-19 07:37:33 +0100688out_rpm:
Daniele Ceraolo Spuriod858d562019-06-13 16:21:54 -0700689 intel_runtime_pm_put(rpm, wakeref);
Chris Wilson8bd818152017-10-19 07:37:33 +0100690out_unlock:
Chris Wilsonfe115622016-10-28 13:58:40 +0100691 mutex_unlock(&i915->drm.struct_mutex);
Eric Anholt3de09aa2009-03-09 09:42:23 -0700692 return ret;
Eric Anholt673a3942008-07-30 12:06:12 -0700693}
694
Chris Wilsonfe115622016-10-28 13:58:40 +0100695/* Per-page copy function for the shmem pwrite fastpath.
696 * Flushes invalid cachelines before writing to the target if
697 * needs_clflush_before is set and flushes out any written cachelines after
698 * writing if needs_clflush is set.
699 */
Eric Anholt40123c12009-03-09 13:42:30 -0700700static int
Chris Wilsonfe115622016-10-28 13:58:40 +0100701shmem_pwrite(struct page *page, int offset, int len, char __user *user_data,
Chris Wilsonfe115622016-10-28 13:58:40 +0100702 bool needs_clflush_before,
703 bool needs_clflush_after)
Eric Anholt40123c12009-03-09 13:42:30 -0700704{
Chris Wilsonb9d126e2019-01-05 12:07:58 +0000705 char *vaddr;
Chris Wilsonfe115622016-10-28 13:58:40 +0100706 int ret;
Eric Anholt40123c12009-03-09 13:42:30 -0700707
Chris Wilsonb9d126e2019-01-05 12:07:58 +0000708 vaddr = kmap(page);
Chris Wilsonfe115622016-10-28 13:58:40 +0100709
Chris Wilsonb9d126e2019-01-05 12:07:58 +0000710 if (needs_clflush_before)
711 drm_clflush_virt_range(vaddr + offset, len);
Chris Wilsonfe115622016-10-28 13:58:40 +0100712
Chris Wilsonb9d126e2019-01-05 12:07:58 +0000713 ret = __copy_from_user(vaddr + offset, user_data, len);
714 if (!ret && needs_clflush_after)
715 drm_clflush_virt_range(vaddr + offset, len);
Chris Wilsonfe115622016-10-28 13:58:40 +0100716
Chris Wilsonb9d126e2019-01-05 12:07:58 +0000717 kunmap(page);
718
719 return ret ? -EFAULT : 0;
Chris Wilsonfe115622016-10-28 13:58:40 +0100720}
721
722static int
723i915_gem_shmem_pwrite(struct drm_i915_gem_object *obj,
724 const struct drm_i915_gem_pwrite *args)
725{
Chris Wilsonfe115622016-10-28 13:58:40 +0100726 unsigned int partial_cacheline_write;
727 unsigned int needs_clflush;
728 unsigned int offset, idx;
Chris Wilson6951e582019-05-28 10:29:51 +0100729 struct dma_fence *fence;
730 void __user *user_data;
731 u64 remain;
Chris Wilsonfe115622016-10-28 13:58:40 +0100732 int ret;
733
Chris Wilson6951e582019-05-28 10:29:51 +0100734 ret = i915_gem_object_prepare_write(obj, &needs_clflush);
Chris Wilson43394c72016-08-18 17:16:47 +0100735 if (ret)
736 return ret;
Eric Anholt40123c12009-03-09 13:42:30 -0700737
Chris Wilson6951e582019-05-28 10:29:51 +0100738 fence = i915_gem_object_lock_fence(obj);
739 i915_gem_object_finish_access(obj);
740 if (!fence)
741 return -ENOMEM;
Chris Wilsonfe115622016-10-28 13:58:40 +0100742
Chris Wilsonfe115622016-10-28 13:58:40 +0100743 /* If we don't overwrite a cacheline completely we need to be
744 * careful to have up-to-date data by first clflushing. Don't
745 * overcomplicate things and flush the entire patch.
746 */
747 partial_cacheline_write = 0;
748 if (needs_clflush & CLFLUSH_BEFORE)
749 partial_cacheline_write = boot_cpu_data.x86_clflush_size - 1;
750
Chris Wilson43394c72016-08-18 17:16:47 +0100751 user_data = u64_to_user_ptr(args->data_ptr);
Chris Wilson43394c72016-08-18 17:16:47 +0100752 remain = args->size;
Chris Wilsonfe115622016-10-28 13:58:40 +0100753 offset = offset_in_page(args->offset);
754 for (idx = args->offset >> PAGE_SHIFT; remain; idx++) {
755 struct page *page = i915_gem_object_get_page(obj, idx);
Chris Wilsona5e856a52018-10-12 15:02:28 +0100756 unsigned int length = min_t(u64, remain, PAGE_SIZE - offset);
Chris Wilsone5281cc2010-10-28 13:45:36 +0100757
Chris Wilsonfe115622016-10-28 13:58:40 +0100758 ret = shmem_pwrite(page, offset, length, user_data,
Chris Wilsonfe115622016-10-28 13:58:40 +0100759 (offset | length) & partial_cacheline_write,
760 needs_clflush & CLFLUSH_AFTER);
761 if (ret)
Chris Wilson9da3da62012-06-01 15:20:22 +0100762 break;
763
Chris Wilsonfe115622016-10-28 13:58:40 +0100764 remain -= length;
765 user_data += length;
766 offset = 0;
Eric Anholt40123c12009-03-09 13:42:30 -0700767 }
768
Chris Wilsond59b21e2017-02-22 11:40:49 +0000769 intel_fb_obj_flush(obj, ORIGIN_CPU);
Chris Wilson6951e582019-05-28 10:29:51 +0100770 i915_gem_object_unlock_fence(obj, fence);
771
Eric Anholt40123c12009-03-09 13:42:30 -0700772 return ret;
Eric Anholt673a3942008-07-30 12:06:12 -0700773}
774
775/**
776 * Writes data to the object referenced by handle.
Tvrtko Ursulin14bb2c12016-06-03 14:02:17 +0100777 * @dev: drm device
778 * @data: ioctl data blob
779 * @file: drm file
Eric Anholt673a3942008-07-30 12:06:12 -0700780 *
781 * On error, the contents of the buffer that were to be modified are undefined.
782 */
783int
784i915_gem_pwrite_ioctl(struct drm_device *dev, void *data,
Chris Wilsonfbd5a262010-10-14 15:03:58 +0100785 struct drm_file *file)
Eric Anholt673a3942008-07-30 12:06:12 -0700786{
787 struct drm_i915_gem_pwrite *args = data;
Chris Wilson05394f32010-11-08 19:18:58 +0000788 struct drm_i915_gem_object *obj;
Chris Wilson51311d02010-11-17 09:10:42 +0000789 int ret;
790
791 if (args->size == 0)
792 return 0;
793
Linus Torvalds96d4f262019-01-03 18:57:57 -0800794 if (!access_ok(u64_to_user_ptr(args->data_ptr), args->size))
Chris Wilson51311d02010-11-17 09:10:42 +0000795 return -EFAULT;
796
Chris Wilson03ac0642016-07-20 13:31:51 +0100797 obj = i915_gem_object_lookup(file, args->handle);
Chris Wilson258a5ed2016-08-05 10:14:16 +0100798 if (!obj)
799 return -ENOENT;
Eric Anholt673a3942008-07-30 12:06:12 -0700800
Chris Wilson7dcd2492010-09-26 20:21:44 +0100801 /* Bounds check destination. */
Matthew Auld966d5bf2016-12-13 20:32:22 +0000802 if (range_overflows_t(u64, args->offset, args->size, obj->base.size)) {
Chris Wilsonce9d4192010-09-26 20:50:05 +0100803 ret = -EINVAL;
Chris Wilson258a5ed2016-08-05 10:14:16 +0100804 goto err;
Chris Wilsonce9d4192010-09-26 20:50:05 +0100805 }
806
Chris Wilsonf8c1cce2018-07-12 19:53:14 +0100807 /* Writes not allowed into this read-only object */
808 if (i915_gem_object_is_readonly(obj)) {
809 ret = -EINVAL;
810 goto err;
811 }
812
Chris Wilsondb53a302011-02-03 11:57:46 +0000813 trace_i915_gem_object_pwrite(obj, args->offset, args->size);
814
Chris Wilson7c55e2c2017-03-07 12:03:38 +0000815 ret = -ENODEV;
816 if (obj->ops->pwrite)
817 ret = obj->ops->pwrite(obj, args);
818 if (ret != -ENODEV)
819 goto err;
820
Chris Wilsone95433c2016-10-28 13:58:27 +0100821 ret = i915_gem_object_wait(obj,
822 I915_WAIT_INTERRUPTIBLE |
823 I915_WAIT_ALL,
Chris Wilson62eb3c22019-02-13 09:25:04 +0000824 MAX_SCHEDULE_TIMEOUT);
Chris Wilson258a5ed2016-08-05 10:14:16 +0100825 if (ret)
826 goto err;
827
Chris Wilsonfe115622016-10-28 13:58:40 +0100828 ret = i915_gem_object_pin_pages(obj);
Chris Wilson258a5ed2016-08-05 10:14:16 +0100829 if (ret)
Chris Wilsonfe115622016-10-28 13:58:40 +0100830 goto err;
Chris Wilson258a5ed2016-08-05 10:14:16 +0100831
Daniel Vetter935aaa62012-03-25 19:47:35 +0200832 ret = -EFAULT;
Eric Anholt673a3942008-07-30 12:06:12 -0700833 /* We can only do the GTT pwrite on untiled buffers, as otherwise
834 * it would end up going through the fenced access, and we'll get
835 * different detiling behavior between reading and writing.
836 * pread/pwrite currently are reading and writing from the CPU
837 * perspective, requiring manual detiling by the client.
838 */
Chris Wilson6eae0052016-06-20 15:05:52 +0100839 if (!i915_gem_object_has_struct_page(obj) ||
Chris Wilson9c870d02016-10-24 13:42:15 +0100840 cpu_write_needs_clflush(obj))
Daniel Vetter935aaa62012-03-25 19:47:35 +0200841 /* Note that the gtt paths might fail with non-page-backed user
842 * pointers (e.g. gtt mappings when moving data between
Chris Wilson9c870d02016-10-24 13:42:15 +0100843 * textures). Fallback to the shmem path in that case.
844 */
Chris Wilsonfe115622016-10-28 13:58:40 +0100845 ret = i915_gem_gtt_pwrite_fast(obj, args);
Eric Anholt673a3942008-07-30 12:06:12 -0700846
Chris Wilsond1054ee2016-07-16 18:42:36 +0100847 if (ret == -EFAULT || ret == -ENOSPC) {
Chris Wilson6a2c4232014-11-04 04:51:40 -0800848 if (obj->phys_handle)
849 ret = i915_gem_phys_pwrite(obj, args, file);
Ankitprasad Sharmab50a5372016-06-10 14:23:03 +0530850 else
Chris Wilsonfe115622016-10-28 13:58:40 +0100851 ret = i915_gem_shmem_pwrite(obj, args);
Chris Wilson6a2c4232014-11-04 04:51:40 -0800852 }
Daniel Vetter5c0480f2011-12-14 13:57:30 +0100853
Chris Wilsonfe115622016-10-28 13:58:40 +0100854 i915_gem_object_unpin_pages(obj);
Chris Wilson258a5ed2016-08-05 10:14:16 +0100855err:
Chris Wilsonf0cd5182016-10-28 13:58:43 +0100856 i915_gem_object_put(obj);
Chris Wilson258a5ed2016-08-05 10:14:16 +0100857 return ret;
Eric Anholt673a3942008-07-30 12:06:12 -0700858}
859
Eric Anholt673a3942008-07-30 12:06:12 -0700860/**
861 * Called when user space has done writes to this buffer
Tvrtko Ursulin14bb2c12016-06-03 14:02:17 +0100862 * @dev: drm device
863 * @data: ioctl data blob
864 * @file: drm file
Eric Anholt673a3942008-07-30 12:06:12 -0700865 */
866int
867i915_gem_sw_finish_ioctl(struct drm_device *dev, void *data,
Chris Wilson05394f32010-11-08 19:18:58 +0000868 struct drm_file *file)
Eric Anholt673a3942008-07-30 12:06:12 -0700869{
870 struct drm_i915_gem_sw_finish *args = data;
Chris Wilson05394f32010-11-08 19:18:58 +0000871 struct drm_i915_gem_object *obj;
Chris Wilson1d7cfea2010-10-17 09:45:41 +0100872
Chris Wilson03ac0642016-07-20 13:31:51 +0100873 obj = i915_gem_object_lookup(file, args->handle);
Chris Wilsonc21724c2016-08-05 10:14:19 +0100874 if (!obj)
875 return -ENOENT;
Eric Anholt673a3942008-07-30 12:06:12 -0700876
Tina Zhanga03f3952017-11-14 10:25:13 +0000877 /*
878 * Proxy objects are barred from CPU access, so there is no
879 * need to ban sw_finish as it is a nop.
880 */
881
Eric Anholt673a3942008-07-30 12:06:12 -0700882 /* Pinned buffers may be scanout, so flush the cache */
Chris Wilson5a97bcc2017-02-22 11:40:46 +0000883 i915_gem_object_flush_if_display(obj);
Chris Wilsonf0cd5182016-10-28 13:58:43 +0100884 i915_gem_object_put(obj);
Chris Wilson5a97bcc2017-02-22 11:40:46 +0000885
886 return 0;
Eric Anholt673a3942008-07-30 12:06:12 -0700887}
888
Chris Wilson0cf289b2019-06-13 08:32:54 +0100889void i915_gem_runtime_suspend(struct drm_i915_private *i915)
Chris Wilsoneedd10f2014-06-16 08:57:44 +0100890{
Chris Wilson3594a3e2016-10-24 13:42:16 +0100891 struct drm_i915_gem_object *obj, *on;
Chris Wilson7c108fd2016-10-24 13:42:18 +0100892 int i;
Chris Wilsoneedd10f2014-06-16 08:57:44 +0100893
Chris Wilson3594a3e2016-10-24 13:42:16 +0100894 /*
895 * Only called during RPM suspend. All users of the userfault_list
896 * must be holding an RPM wakeref to ensure that this can not
897 * run concurrently with themselves (and use the struct_mutex for
898 * protection between themselves).
899 */
900
901 list_for_each_entry_safe(obj, on,
Chris Wilson0cf289b2019-06-13 08:32:54 +0100902 &i915->ggtt.userfault_list, userfault_link)
Chris Wilsona65adaf2017-10-09 09:43:57 +0100903 __i915_gem_object_release_mmap(obj);
Chris Wilson7c108fd2016-10-24 13:42:18 +0100904
Chris Wilson0cf289b2019-06-13 08:32:54 +0100905 /*
906 * The fence will be lost when the device powers down. If any were
Chris Wilson7c108fd2016-10-24 13:42:18 +0100907 * in use by hardware (i.e. they are pinned), we should not be powering
908 * down! All other fences will be reacquired by the user upon waking.
909 */
Chris Wilson0cf289b2019-06-13 08:32:54 +0100910 for (i = 0; i < i915->ggtt.num_fences; i++) {
911 struct i915_fence_reg *reg = &i915->ggtt.fence_regs[i];
Chris Wilson7c108fd2016-10-24 13:42:18 +0100912
Chris Wilson0cf289b2019-06-13 08:32:54 +0100913 /*
914 * Ideally we want to assert that the fence register is not
Chris Wilsone0ec3ec2017-02-03 12:57:17 +0000915 * live at this point (i.e. that no piece of code will be
916 * trying to write through fence + GTT, as that both violates
917 * our tracking of activity and associated locking/barriers,
918 * but also is illegal given that the hw is powered down).
919 *
920 * Previously we used reg->pin_count as a "liveness" indicator.
921 * That is not sufficient, and we need a more fine-grained
922 * tool if we want to have a sanity check here.
923 */
Chris Wilson7c108fd2016-10-24 13:42:18 +0100924
925 if (!reg->vma)
926 continue;
927
Chris Wilsona65adaf2017-10-09 09:43:57 +0100928 GEM_BUG_ON(i915_vma_has_userfault(reg->vma));
Chris Wilson7c108fd2016-10-24 13:42:18 +0100929 reg->dirty = true;
930 }
Chris Wilsoneedd10f2014-06-16 08:57:44 +0100931}
932
Chris Wilson25112b62017-03-30 15:50:39 +0100933static int wait_for_engines(struct drm_i915_private *i915)
934{
Chris Wilsonee42c002017-12-11 19:41:34 +0000935 if (wait_for(intel_engines_are_idle(i915), I915_IDLE_ENGINES_TIMEOUT)) {
Chris Wilson59e4b192017-12-11 19:41:35 +0000936 dev_err(i915->drm.dev,
937 "Failed to idle engines, declaring wedged!\n");
Chris Wilson629820f2018-03-09 10:11:14 +0000938 GEM_TRACE_DUMP();
Chris Wilsoncad99462017-08-26 12:09:33 +0100939 i915_gem_set_wedged(i915);
940 return -EIO;
Chris Wilson25112b62017-03-30 15:50:39 +0100941 }
942
943 return 0;
944}
945
Chris Wilson1e345562019-01-28 10:23:56 +0000946static long
947wait_for_timelines(struct drm_i915_private *i915,
948 unsigned int flags, long timeout)
949{
950 struct i915_gt_timelines *gt = &i915->gt.timelines;
951 struct i915_timeline *tl;
952
Chris Wilson1e345562019-01-28 10:23:56 +0000953 mutex_lock(&gt->mutex);
Chris Wilson9407d3b2019-01-28 18:18:12 +0000954 list_for_each_entry(tl, &gt->active_list, link) {
Chris Wilson1e345562019-01-28 10:23:56 +0000955 struct i915_request *rq;
956
Chris Wilson21950ee2019-02-05 13:00:05 +0000957 rq = i915_active_request_get_unlocked(&tl->last_request);
Chris Wilson1e345562019-01-28 10:23:56 +0000958 if (!rq)
959 continue;
960
961 mutex_unlock(&gt->mutex);
962
963 /*
964 * "Race-to-idle".
965 *
966 * Switching to the kernel context is often used a synchronous
967 * step prior to idling, e.g. in suspend for flushing all
968 * current operations to memory before sleeping. These we
969 * want to complete as quickly as possible to avoid prolonged
970 * stalls, so allow the gpu to boost to maximum clocks.
971 */
972 if (flags & I915_WAIT_FOR_IDLE_BOOST)
Chris Wilson62eb3c22019-02-13 09:25:04 +0000973 gen6_rps_boost(rq);
Chris Wilson1e345562019-01-28 10:23:56 +0000974
975 timeout = i915_request_wait(rq, flags, timeout);
976 i915_request_put(rq);
977 if (timeout < 0)
978 return timeout;
979
980 /* restart after reacquiring the lock */
981 mutex_lock(&gt->mutex);
Chris Wilson9407d3b2019-01-28 18:18:12 +0000982 tl = list_entry(&gt->active_list, typeof(*tl), link);
Chris Wilson1e345562019-01-28 10:23:56 +0000983 }
984 mutex_unlock(&gt->mutex);
985
986 return timeout;
987}
988
Chris Wilsonec625fb2018-07-09 13:20:42 +0100989int i915_gem_wait_for_idle(struct drm_i915_private *i915,
990 unsigned int flags, long timeout)
Chris Wilson73cb9702016-10-28 13:58:46 +0100991{
Chris Wilson79ffac852019-04-24 21:07:17 +0100992 GEM_TRACE("flags=%x (%s), timeout=%ld%s, awake?=%s\n",
Chris Wilsonec625fb2018-07-09 13:20:42 +0100993 flags, flags & I915_WAIT_LOCKED ? "locked" : "unlocked",
Chris Wilson79ffac852019-04-24 21:07:17 +0100994 timeout, timeout == MAX_SCHEDULE_TIMEOUT ? " (forever)" : "",
995 yesno(i915->gt.awake));
Chris Wilson09a4c022018-05-24 09:11:35 +0100996
Chris Wilson863e9fd2017-05-30 13:13:32 +0100997 /* If the device is asleep, we have no requests outstanding */
998 if (!READ_ONCE(i915->gt.awake))
999 return 0;
1000
Chris Wilson1e345562019-01-28 10:23:56 +00001001 timeout = wait_for_timelines(i915, flags, timeout);
1002 if (timeout < 0)
1003 return timeout;
1004
Chris Wilson9caa34a2016-11-11 14:58:08 +00001005 if (flags & I915_WAIT_LOCKED) {
Chris Wilsona89d1f92018-05-02 17:38:39 +01001006 int err;
Chris Wilson9caa34a2016-11-11 14:58:08 +00001007
1008 lockdep_assert_held(&i915->drm.struct_mutex);
1009
Chris Wilsona61b47f2018-06-27 12:53:34 +01001010 err = wait_for_engines(i915);
1011 if (err)
1012 return err;
1013
Chris Wilsone61e0f52018-02-21 09:56:36 +00001014 i915_retire_requests(i915);
Chris Wilsona89d1f92018-05-02 17:38:39 +01001015 }
Chris Wilsona61b47f2018-06-27 12:53:34 +01001016
1017 return 0;
Daniel Vetter4df2faf2010-02-19 11:52:00 +01001018}
1019
Chris Wilson058d88c2016-08-15 10:49:06 +01001020struct i915_vma *
Joonas Lahtinenec7adb62015-03-16 14:11:13 +02001021i915_gem_object_ggtt_pin(struct drm_i915_gem_object *obj,
1022 const struct i915_ggtt_view *view,
Chris Wilson91b2db62016-08-04 16:32:23 +01001023 u64 size,
Chris Wilson2ffffd02016-08-04 16:32:22 +01001024 u64 alignment,
1025 u64 flags)
Joonas Lahtinenec7adb62015-03-16 14:11:13 +02001026{
Chris Wilsonad16d2e2016-10-13 09:55:04 +01001027 struct drm_i915_private *dev_priv = to_i915(obj->base.dev);
Chris Wilson82ad6442018-06-05 16:37:58 +01001028 struct i915_address_space *vm = &dev_priv->ggtt.vm;
Chris Wilson59bfa122016-08-04 16:32:31 +01001029 struct i915_vma *vma;
1030 int ret;
Joonas Lahtinen72e96d62016-03-30 16:57:10 +03001031
Chris Wilson4c7d62c2016-10-28 13:58:32 +01001032 lockdep_assert_held(&obj->base.dev->struct_mutex);
1033
Chris Wilsonac87a6fd2018-02-20 13:42:05 +00001034 if (flags & PIN_MAPPABLE &&
1035 (!view || view->type == I915_GGTT_VIEW_NORMAL)) {
Chris Wilson43ae70d92017-10-09 09:44:01 +01001036 /* If the required space is larger than the available
1037 * aperture, we will not able to find a slot for the
1038 * object and unbinding the object now will be in
1039 * vain. Worse, doing so may cause us to ping-pong
1040 * the object in and out of the Global GTT and
1041 * waste a lot of cycles under the mutex.
1042 */
1043 if (obj->base.size > dev_priv->ggtt.mappable_end)
1044 return ERR_PTR(-E2BIG);
1045
1046 /* If NONBLOCK is set the caller is optimistically
1047 * trying to cache the full object within the mappable
1048 * aperture, and *must* have a fallback in place for
1049 * situations where we cannot bind the object. We
1050 * can be a little more lax here and use the fallback
1051 * more often to avoid costly migrations of ourselves
1052 * and other objects within the aperture.
1053 *
1054 * Half-the-aperture is used as a simple heuristic.
1055 * More interesting would to do search for a free
1056 * block prior to making the commitment to unbind.
1057 * That caters for the self-harm case, and with a
1058 * little more heuristics (e.g. NOFAULT, NOEVICT)
1059 * we could try to minimise harm to others.
1060 */
1061 if (flags & PIN_NONBLOCK &&
1062 obj->base.size > dev_priv->ggtt.mappable_end / 2)
1063 return ERR_PTR(-ENOSPC);
1064 }
1065
Chris Wilson718659a2017-01-16 15:21:28 +00001066 vma = i915_vma_instance(obj, vm, view);
Chengguang Xu772b5402019-02-21 10:08:19 +08001067 if (IS_ERR(vma))
Chris Wilson058d88c2016-08-15 10:49:06 +01001068 return vma;
Chris Wilson59bfa122016-08-04 16:32:31 +01001069
1070 if (i915_vma_misplaced(vma, size, alignment, flags)) {
Chris Wilson43ae70d92017-10-09 09:44:01 +01001071 if (flags & PIN_NONBLOCK) {
1072 if (i915_vma_is_pinned(vma) || i915_vma_is_active(vma))
1073 return ERR_PTR(-ENOSPC);
Chris Wilson59bfa122016-08-04 16:32:31 +01001074
Chris Wilson43ae70d92017-10-09 09:44:01 +01001075 if (flags & PIN_MAPPABLE &&
Chris Wilson944397f2017-01-09 16:16:11 +00001076 vma->fence_size > dev_priv->ggtt.mappable_end / 2)
Chris Wilsonad16d2e2016-10-13 09:55:04 +01001077 return ERR_PTR(-ENOSPC);
1078 }
1079
Chris Wilson59bfa122016-08-04 16:32:31 +01001080 WARN(i915_vma_is_pinned(vma),
1081 "bo is already pinned in ggtt with incorrect alignment:"
Chris Wilson05a20d02016-08-18 17:16:55 +01001082 " offset=%08x, req.alignment=%llx,"
1083 " req.map_and_fenceable=%d, vma->map_and_fenceable=%d\n",
1084 i915_ggtt_offset(vma), alignment,
Chris Wilson59bfa122016-08-04 16:32:31 +01001085 !!(flags & PIN_MAPPABLE),
Chris Wilson05a20d02016-08-18 17:16:55 +01001086 i915_vma_is_map_and_fenceable(vma));
Chris Wilson59bfa122016-08-04 16:32:31 +01001087 ret = i915_vma_unbind(vma);
1088 if (ret)
Chris Wilson058d88c2016-08-15 10:49:06 +01001089 return ERR_PTR(ret);
Chris Wilson59bfa122016-08-04 16:32:31 +01001090 }
1091
Chris Wilson058d88c2016-08-15 10:49:06 +01001092 ret = i915_vma_pin(vma, size, alignment, flags | PIN_GLOBAL);
1093 if (ret)
1094 return ERR_PTR(ret);
Joonas Lahtinenec7adb62015-03-16 14:11:13 +02001095
Chris Wilson058d88c2016-08-15 10:49:06 +01001096 return vma;
Eric Anholt673a3942008-07-30 12:06:12 -07001097}
1098
Eric Anholt673a3942008-07-30 12:06:12 -07001099int
Chris Wilson3ef94da2009-09-14 16:50:29 +01001100i915_gem_madvise_ioctl(struct drm_device *dev, void *data,
1101 struct drm_file *file_priv)
1102{
Chris Wilson3b4fa962019-05-30 21:34:59 +01001103 struct drm_i915_private *i915 = to_i915(dev);
Chris Wilson3ef94da2009-09-14 16:50:29 +01001104 struct drm_i915_gem_madvise *args = data;
Chris Wilson05394f32010-11-08 19:18:58 +00001105 struct drm_i915_gem_object *obj;
Chris Wilson1233e2d2016-10-28 13:58:37 +01001106 int err;
Chris Wilson3ef94da2009-09-14 16:50:29 +01001107
1108 switch (args->madv) {
1109 case I915_MADV_DONTNEED:
1110 case I915_MADV_WILLNEED:
1111 break;
1112 default:
1113 return -EINVAL;
1114 }
1115
Chris Wilson03ac0642016-07-20 13:31:51 +01001116 obj = i915_gem_object_lookup(file_priv, args->handle);
Chris Wilson1233e2d2016-10-28 13:58:37 +01001117 if (!obj)
1118 return -ENOENT;
1119
1120 err = mutex_lock_interruptible(&obj->mm.lock);
1121 if (err)
1122 goto out;
Chris Wilson3ef94da2009-09-14 16:50:29 +01001123
Chris Wilsonf1fa4f42017-10-13 21:26:13 +01001124 if (i915_gem_object_has_pages(obj) &&
Chris Wilson3e510a82016-08-05 10:14:23 +01001125 i915_gem_object_is_tiled(obj) &&
Chris Wilson3b4fa962019-05-30 21:34:59 +01001126 i915->quirks & QUIRK_PIN_SWIZZLED_PAGES) {
Chris Wilsonbc0629a2016-11-01 10:03:17 +00001127 if (obj->mm.madv == I915_MADV_WILLNEED) {
1128 GEM_BUG_ON(!obj->mm.quirked);
Chris Wilsona4f5ea62016-10-28 13:58:35 +01001129 __i915_gem_object_unpin_pages(obj);
Chris Wilsonbc0629a2016-11-01 10:03:17 +00001130 obj->mm.quirked = false;
1131 }
1132 if (args->madv == I915_MADV_WILLNEED) {
Chris Wilson2c3a3f42016-11-04 10:30:01 +00001133 GEM_BUG_ON(obj->mm.quirked);
Chris Wilsona4f5ea62016-10-28 13:58:35 +01001134 __i915_gem_object_pin_pages(obj);
Chris Wilsonbc0629a2016-11-01 10:03:17 +00001135 obj->mm.quirked = true;
1136 }
Daniel Vetter656bfa32014-11-20 09:26:30 +01001137 }
1138
Chris Wilsona4f5ea62016-10-28 13:58:35 +01001139 if (obj->mm.madv != __I915_MADV_PURGED)
1140 obj->mm.madv = args->madv;
Chris Wilson3ef94da2009-09-14 16:50:29 +01001141
Chris Wilson3b4fa962019-05-30 21:34:59 +01001142 if (i915_gem_object_has_pages(obj)) {
1143 struct list_head *list;
1144
Chris Wilsond82b4b22019-05-30 21:35:00 +01001145 if (i915_gem_object_is_shrinkable(obj)) {
Chris Wilsona8cff4c82019-06-10 15:54:30 +01001146 unsigned long flags;
1147
1148 spin_lock_irqsave(&i915->mm.obj_lock, flags);
1149
Chris Wilsond82b4b22019-05-30 21:35:00 +01001150 if (obj->mm.madv != I915_MADV_WILLNEED)
1151 list = &i915->mm.purge_list;
Chris Wilsond82b4b22019-05-30 21:35:00 +01001152 else
Chris Wilsonecab9be2019-06-12 11:57:20 +01001153 list = &i915->mm.shrink_list;
Chris Wilsond82b4b22019-05-30 21:35:00 +01001154 list_move_tail(&obj->mm.link, list);
Chris Wilsona8cff4c82019-06-10 15:54:30 +01001155
1156 spin_unlock_irqrestore(&i915->mm.obj_lock, flags);
Chris Wilsond82b4b22019-05-30 21:35:00 +01001157 }
Chris Wilson3b4fa962019-05-30 21:34:59 +01001158 }
1159
Chris Wilson6c085a72012-08-20 11:40:46 +02001160 /* if the object is no longer attached, discard its backing storage */
Chris Wilsonf1fa4f42017-10-13 21:26:13 +01001161 if (obj->mm.madv == I915_MADV_DONTNEED &&
1162 !i915_gem_object_has_pages(obj))
Chris Wilsonf0334282019-05-28 10:29:46 +01001163 i915_gem_object_truncate(obj);
Chris Wilson2d7ef392009-09-20 23:13:10 +01001164
Chris Wilsona4f5ea62016-10-28 13:58:35 +01001165 args->retained = obj->mm.madv != __I915_MADV_PURGED;
Chris Wilson1233e2d2016-10-28 13:58:37 +01001166 mutex_unlock(&obj->mm.lock);
Chris Wilsonbb6baf72009-09-22 14:24:13 +01001167
Chris Wilson1233e2d2016-10-28 13:58:37 +01001168out:
Chris Wilsonf8c417c2016-07-20 13:31:53 +01001169 i915_gem_object_put(obj);
Chris Wilson1233e2d2016-10-28 13:58:37 +01001170 return err;
Chris Wilson3ef94da2009-09-14 16:50:29 +01001171}
1172
Chris Wilson24145512017-01-24 11:01:35 +00001173void i915_gem_sanitize(struct drm_i915_private *i915)
1174{
Chris Wilson538ef962019-01-14 14:21:18 +00001175 intel_wakeref_t wakeref;
1176
Chris Wilsonc3160da2018-05-31 09:22:45 +01001177 GEM_TRACE("\n");
1178
Daniele Ceraolo Spuriod858d562019-06-13 16:21:54 -07001179 wakeref = intel_runtime_pm_get(&i915->runtime_pm);
Daniele Ceraolo Spurio3ceea6a2019-03-19 11:35:36 -07001180 intel_uncore_forcewake_get(&i915->uncore, FORCEWAKE_ALL);
Chris Wilsonc3160da2018-05-31 09:22:45 +01001181
1182 /*
1183 * As we have just resumed the machine and woken the device up from
1184 * deep PCI sleep (presumably D3_cold), assume the HW has been reset
1185 * back to defaults, recovering from whatever wedged state we left it
1186 * in and so worth trying to use the device once more.
1187 */
Chris Wilsonc41166f2019-02-20 14:56:37 +00001188 if (i915_terminally_wedged(i915))
Chris Wilsonf36325f2017-08-26 12:09:34 +01001189 i915_gem_unset_wedged(i915);
Chris Wilsonf36325f2017-08-26 12:09:34 +01001190
Chris Wilson24145512017-01-24 11:01:35 +00001191 /*
1192 * If we inherit context state from the BIOS or earlier occupants
1193 * of the GPU, the GPU may be in an inconsistent state when we
1194 * try to take over. The only way to remove the earlier state
1195 * is by resetting. However, resetting on earlier gen is tricky as
1196 * it may impact the display and we are uncertain about the stability
Joonas Lahtinenea117b82017-04-28 10:53:38 +03001197 * of the reset, so this could be applied to even earlier gen.
Chris Wilson24145512017-01-24 11:01:35 +00001198 */
Chris Wilson79ffac852019-04-24 21:07:17 +01001199 intel_gt_sanitize(i915, false);
Chris Wilsonc3160da2018-05-31 09:22:45 +01001200
Daniele Ceraolo Spurio3ceea6a2019-03-19 11:35:36 -07001201 intel_uncore_forcewake_put(&i915->uncore, FORCEWAKE_ALL);
Daniele Ceraolo Spuriod858d562019-06-13 16:21:54 -07001202 intel_runtime_pm_put(&i915->runtime_pm, wakeref);
Chris Wilson24145512017-01-24 11:01:35 +00001203}
1204
Tvrtko Ursulincf6844b2019-06-21 08:07:47 +01001205static void init_unused_ring(struct intel_gt *gt, u32 base)
Ville Syrjälä81e7f202014-08-15 01:21:55 +03001206{
Tvrtko Ursulincf6844b2019-06-21 08:07:47 +01001207 struct intel_uncore *uncore = gt->uncore;
1208
1209 intel_uncore_write(uncore, RING_CTL(base), 0);
1210 intel_uncore_write(uncore, RING_HEAD(base), 0);
1211 intel_uncore_write(uncore, RING_TAIL(base), 0);
1212 intel_uncore_write(uncore, RING_START(base), 0);
Ville Syrjälä81e7f202014-08-15 01:21:55 +03001213}
1214
Tvrtko Ursulincf6844b2019-06-21 08:07:47 +01001215static void init_unused_rings(struct intel_gt *gt)
Ville Syrjälä81e7f202014-08-15 01:21:55 +03001216{
Tvrtko Ursulincf6844b2019-06-21 08:07:47 +01001217 struct drm_i915_private *i915 = gt->i915;
1218
1219 if (IS_I830(i915)) {
1220 init_unused_ring(gt, PRB1_BASE);
1221 init_unused_ring(gt, SRB0_BASE);
1222 init_unused_ring(gt, SRB1_BASE);
1223 init_unused_ring(gt, SRB2_BASE);
1224 init_unused_ring(gt, SRB3_BASE);
1225 } else if (IS_GEN(i915, 2)) {
1226 init_unused_ring(gt, SRB0_BASE);
1227 init_unused_ring(gt, SRB1_BASE);
1228 } else if (IS_GEN(i915, 3)) {
1229 init_unused_ring(gt, PRB1_BASE);
1230 init_unused_ring(gt, PRB2_BASE);
Ville Syrjälä81e7f202014-08-15 01:21:55 +03001231 }
1232}
1233
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001234static int init_hw(struct intel_gt *gt)
Chris Wilson20a8a742017-02-08 14:30:31 +00001235{
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001236 struct drm_i915_private *i915 = gt->i915;
1237 struct intel_uncore *uncore = gt->uncore;
Chris Wilsond200cda2016-04-28 09:56:44 +01001238 int ret;
Ben Widawsky4fc7c972013-02-08 11:49:24 -08001239
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001240 gt->last_init_time = ktime_get();
Chris Wilsonde867c22016-10-25 13:16:02 +01001241
Chris Wilson5e4f5182015-02-13 14:35:59 +00001242 /* Double layer security blanket, see i915_gem_init() */
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001243 intel_uncore_forcewake_get(uncore, FORCEWAKE_ALL);
Chris Wilson5e4f5182015-02-13 14:35:59 +00001244
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001245 if (HAS_EDRAM(i915) && INTEL_GEN(i915) < 9)
1246 intel_uncore_rmw(uncore, HSW_IDICR, 0, IDIHASHMSK(0xf));
Ben Widawsky4fc7c972013-02-08 11:49:24 -08001247
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001248 if (IS_HASWELL(i915))
1249 intel_uncore_write(uncore,
1250 MI_PREDICATE_RESULT_2,
1251 IS_HSW_GT3(i915) ?
1252 LOWER_SLICE_ENABLED : LOWER_SLICE_DISABLED);
Rodrigo Vivi94353732013-08-28 16:45:46 -03001253
Tvrtko Ursulin094304b2018-12-03 12:50:10 +00001254 /* Apply the GT workarounds... */
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001255 intel_gt_apply_workarounds(gt);
Tvrtko Ursulin094304b2018-12-03 12:50:10 +00001256 /* ...and determine whether they are sticking. */
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001257 intel_gt_verify_workarounds(gt, "init");
Oscar Mateo59b449d2018-04-10 09:12:47 -07001258
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001259 intel_gt_init_swizzling(gt);
Ben Widawsky4fc7c972013-02-08 11:49:24 -08001260
Daniel Vetterd5abdfd2014-11-20 09:45:19 +01001261 /*
1262 * At least 830 can leave some of the unused rings
1263 * "active" (ie. head != tail) after resume which
1264 * will prevent c3 entry. Makes sure all unused rings
1265 * are totally idle.
1266 */
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001267 init_unused_rings(gt);
Daniel Vetterd5abdfd2014-11-20 09:45:19 +01001268
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001269 ret = i915_ppgtt_init_hw(gt);
John Harrison4ad2fd82015-06-18 13:11:20 +01001270 if (ret) {
Chris Wilson8177e112018-02-07 11:15:45 +00001271 DRM_ERROR("Enabling PPGTT failed (%d)\n", ret);
John Harrison4ad2fd82015-06-18 13:11:20 +01001272 goto out;
1273 }
1274
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001275 ret = intel_wopcm_init_hw(&i915->wopcm);
Jackie Lif08e2032018-03-13 17:32:53 -07001276 if (ret) {
1277 DRM_ERROR("Enabling WOPCM failed (%d)\n", ret);
1278 goto out;
1279 }
1280
Michał Winiarski9bdc3572017-10-25 18:25:19 +01001281 /* We can't enable contexts until all firmware is loaded */
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001282 ret = intel_uc_init_hw(i915);
Chris Wilson8177e112018-02-07 11:15:45 +00001283 if (ret) {
1284 DRM_ERROR("Enabling uc failed (%d)\n", ret);
Michał Winiarski9bdc3572017-10-25 18:25:19 +01001285 goto out;
Chris Wilson8177e112018-02-07 11:15:45 +00001286 }
Michał Winiarski9bdc3572017-10-25 18:25:19 +01001287
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001288 intel_mocs_init_l3cc_table(gt);
Peter Antoine0ccdacf2016-04-13 15:03:25 +01001289
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001290 intel_uncore_forcewake_put(uncore, FORCEWAKE_ALL);
Michał Winiarski60c0a662018-07-12 14:48:10 +02001291
1292 return 0;
Michal Wajdeczkob96f6eb2018-06-05 12:24:43 +00001293
Michał Winiarski60c0a662018-07-12 14:48:10 +02001294out:
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001295 intel_uncore_forcewake_put(uncore, FORCEWAKE_ALL);
1296
1297 return ret;
1298}
1299
1300int i915_gem_init_hw(struct drm_i915_private *i915)
1301{
Tvrtko Ursulin86491872019-06-21 08:07:54 +01001302 struct intel_uncore *uncore = &i915->uncore;
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001303 int ret;
1304
1305 BUG_ON(!i915->kernel_context);
1306 ret = i915_terminally_wedged(i915);
1307 if (ret)
1308 return ret;
1309
Tvrtko Ursulin86491872019-06-21 08:07:54 +01001310 /* Double layer security blanket, see i915_gem_init() */
1311 intel_uncore_forcewake_get(uncore, FORCEWAKE_ALL);
1312
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001313 ret = init_hw(&i915->gt);
Tvrtko Ursulin86491872019-06-21 08:07:54 +01001314 if (ret)
1315 goto err_init;
1316
1317 /* Only when the HW is re-initialised, can we replay the requests */
1318 ret = intel_engines_resume(i915);
1319 if (ret)
1320 goto err_engines;
1321
1322 intel_uncore_forcewake_put(uncore, FORCEWAKE_ALL);
1323
1324 intel_engines_set_scheduler_caps(i915);
1325
1326 return 0;
1327
1328err_engines:
1329 intel_uc_fini_hw(i915);
1330err_init:
1331 intel_uncore_forcewake_put(uncore, FORCEWAKE_ALL);
Tvrtko Ursulinabc584f2019-06-21 08:07:53 +01001332
1333 intel_engines_set_scheduler_caps(i915);
Michał Winiarski60c0a662018-07-12 14:48:10 +02001334
1335 return ret;
Zou Nan hai8187a2b2010-05-21 09:08:55 +08001336}
1337
Chris Wilsond2b4b972017-11-10 14:26:33 +00001338static int __intel_engines_record_defaults(struct drm_i915_private *i915)
1339{
Chris Wilsond2b4b972017-11-10 14:26:33 +00001340 struct intel_engine_cs *engine;
Chris Wilson5e2a0412019-04-26 17:33:34 +01001341 struct i915_gem_context *ctx;
1342 struct i915_gem_engines *e;
Chris Wilsond2b4b972017-11-10 14:26:33 +00001343 enum intel_engine_id id;
Chris Wilson604c37d2019-03-08 09:36:55 +00001344 int err = 0;
Chris Wilsond2b4b972017-11-10 14:26:33 +00001345
1346 /*
1347 * As we reset the gpu during very early sanitisation, the current
1348 * register state on the GPU should reflect its defaults values.
1349 * We load a context onto the hw (with restore-inhibit), then switch
1350 * over to a second context to save that default register state. We
1351 * can then prime every new context with that state so they all start
1352 * from the same default HW values.
1353 */
1354
1355 ctx = i915_gem_context_create_kernel(i915, 0);
1356 if (IS_ERR(ctx))
1357 return PTR_ERR(ctx);
1358
Chris Wilson5e2a0412019-04-26 17:33:34 +01001359 e = i915_gem_context_lock_engines(ctx);
1360
Chris Wilsond2b4b972017-11-10 14:26:33 +00001361 for_each_engine(engine, i915, id) {
Chris Wilson5e2a0412019-04-26 17:33:34 +01001362 struct intel_context *ce = e->engines[id];
Chris Wilsone61e0f52018-02-21 09:56:36 +00001363 struct i915_request *rq;
Chris Wilsond2b4b972017-11-10 14:26:33 +00001364
Chris Wilson5e2a0412019-04-26 17:33:34 +01001365 rq = intel_context_create_request(ce);
Chris Wilsond2b4b972017-11-10 14:26:33 +00001366 if (IS_ERR(rq)) {
1367 err = PTR_ERR(rq);
Chris Wilson5e2a0412019-04-26 17:33:34 +01001368 goto err_active;
Chris Wilsond2b4b972017-11-10 14:26:33 +00001369 }
1370
Chris Wilson3fef5cd2017-11-20 10:20:02 +00001371 err = 0;
Chris Wilson5e2a0412019-04-26 17:33:34 +01001372 if (rq->engine->init_context)
1373 err = rq->engine->init_context(rq);
Chris Wilsond2b4b972017-11-10 14:26:33 +00001374
Chris Wilson697b9a82018-06-12 11:51:35 +01001375 i915_request_add(rq);
Chris Wilsond2b4b972017-11-10 14:26:33 +00001376 if (err)
1377 goto err_active;
1378 }
1379
Chris Wilson604c37d2019-03-08 09:36:55 +00001380 /* Flush the default context image to memory, and enable powersaving. */
Chris Wilson23c3c3d2019-04-24 21:07:14 +01001381 if (!i915_gem_load_power_context(i915)) {
Chris Wilson604c37d2019-03-08 09:36:55 +00001382 err = -EIO;
Chris Wilsond2b4b972017-11-10 14:26:33 +00001383 goto err_active;
Chris Wilson2621cef2018-07-09 13:20:43 +01001384 }
Chris Wilsond2b4b972017-11-10 14:26:33 +00001385
Chris Wilsond2b4b972017-11-10 14:26:33 +00001386 for_each_engine(engine, i915, id) {
Chris Wilson5e2a0412019-04-26 17:33:34 +01001387 struct intel_context *ce = e->engines[id];
1388 struct i915_vma *state = ce->state;
Chris Wilson37d7c9c2018-09-14 13:35:03 +01001389 void *vaddr;
Chris Wilsond2b4b972017-11-10 14:26:33 +00001390
Chris Wilsond2b4b972017-11-10 14:26:33 +00001391 if (!state)
1392 continue;
1393
Chris Wilson08819542019-03-08 13:25:22 +00001394 GEM_BUG_ON(intel_context_is_pinned(ce));
Chris Wilsonc4d52fe2019-03-08 13:25:19 +00001395
Chris Wilsond2b4b972017-11-10 14:26:33 +00001396 /*
1397 * As we will hold a reference to the logical state, it will
1398 * not be torn down with the context, and importantly the
1399 * object will hold onto its vma (making it possible for a
1400 * stray GTT write to corrupt our defaults). Unmap the vma
1401 * from the GTT to prevent such accidents and reclaim the
1402 * space.
1403 */
1404 err = i915_vma_unbind(state);
1405 if (err)
1406 goto err_active;
1407
Chris Wilson6951e582019-05-28 10:29:51 +01001408 i915_gem_object_lock(state->obj);
Chris Wilsond2b4b972017-11-10 14:26:33 +00001409 err = i915_gem_object_set_to_cpu_domain(state->obj, false);
Chris Wilson6951e582019-05-28 10:29:51 +01001410 i915_gem_object_unlock(state->obj);
Chris Wilsond2b4b972017-11-10 14:26:33 +00001411 if (err)
1412 goto err_active;
1413
1414 engine->default_state = i915_gem_object_get(state->obj);
Chris Wilsona679f582019-03-21 16:19:07 +00001415 i915_gem_object_set_cache_coherency(engine->default_state,
1416 I915_CACHE_LLC);
Chris Wilson37d7c9c2018-09-14 13:35:03 +01001417
1418 /* Check we can acquire the image of the context state */
1419 vaddr = i915_gem_object_pin_map(engine->default_state,
Chris Wilson666424a2018-09-14 13:35:04 +01001420 I915_MAP_FORCE_WB);
Chris Wilson37d7c9c2018-09-14 13:35:03 +01001421 if (IS_ERR(vaddr)) {
1422 err = PTR_ERR(vaddr);
1423 goto err_active;
1424 }
1425
1426 i915_gem_object_unpin_map(engine->default_state);
Chris Wilsond2b4b972017-11-10 14:26:33 +00001427 }
1428
1429 if (IS_ENABLED(CONFIG_DRM_I915_DEBUG_GEM)) {
1430 unsigned int found = intel_engines_has_context_isolation(i915);
1431
1432 /*
1433 * Make sure that classes with multiple engine instances all
1434 * share the same basic configuration.
1435 */
1436 for_each_engine(engine, i915, id) {
1437 unsigned int bit = BIT(engine->uabi_class);
1438 unsigned int expected = engine->default_state ? bit : 0;
1439
1440 if ((found & bit) != expected) {
1441 DRM_ERROR("mismatching default context state for class %d on engine %s\n",
1442 engine->uabi_class, engine->name);
1443 }
1444 }
1445 }
1446
1447out_ctx:
Chris Wilson5e2a0412019-04-26 17:33:34 +01001448 i915_gem_context_unlock_engines(ctx);
Chris Wilsond2b4b972017-11-10 14:26:33 +00001449 i915_gem_context_set_closed(ctx);
1450 i915_gem_context_put(ctx);
1451 return err;
1452
1453err_active:
1454 /*
1455 * If we have to abandon now, we expect the engines to be idle
Chris Wilson604c37d2019-03-08 09:36:55 +00001456 * and ready to be torn-down. The quickest way we can accomplish
1457 * this is by declaring ourselves wedged.
Chris Wilsond2b4b972017-11-10 14:26:33 +00001458 */
Chris Wilson604c37d2019-03-08 09:36:55 +00001459 i915_gem_set_wedged(i915);
Chris Wilsond2b4b972017-11-10 14:26:33 +00001460 goto out_ctx;
1461}
1462
Chris Wilson51797492018-12-04 14:15:16 +00001463static int
1464i915_gem_init_scratch(struct drm_i915_private *i915, unsigned int size)
1465{
1466 struct drm_i915_gem_object *obj;
1467 struct i915_vma *vma;
1468 int ret;
1469
1470 obj = i915_gem_object_create_stolen(i915, size);
1471 if (!obj)
1472 obj = i915_gem_object_create_internal(i915, size);
1473 if (IS_ERR(obj)) {
1474 DRM_ERROR("Failed to allocate scratch page\n");
1475 return PTR_ERR(obj);
1476 }
1477
1478 vma = i915_vma_instance(obj, &i915->ggtt.vm, NULL);
1479 if (IS_ERR(vma)) {
1480 ret = PTR_ERR(vma);
1481 goto err_unref;
1482 }
1483
1484 ret = i915_vma_pin(vma, 0, 0, PIN_GLOBAL | PIN_HIGH);
1485 if (ret)
1486 goto err_unref;
1487
1488 i915->gt.scratch = vma;
1489 return 0;
1490
1491err_unref:
1492 i915_gem_object_put(obj);
1493 return ret;
1494}
1495
1496static void i915_gem_fini_scratch(struct drm_i915_private *i915)
1497{
1498 i915_vma_unpin_and_release(&i915->gt.scratch, 0);
1499}
1500
Chris Wilson254e1182019-04-17 08:56:28 +01001501static int intel_engines_verify_workarounds(struct drm_i915_private *i915)
1502{
1503 struct intel_engine_cs *engine;
1504 enum intel_engine_id id;
1505 int err = 0;
1506
1507 if (!IS_ENABLED(CONFIG_DRM_I915_DEBUG_GEM))
1508 return 0;
1509
1510 for_each_engine(engine, i915, id) {
1511 if (intel_engine_verify_workarounds(engine, "load"))
1512 err = -EIO;
1513 }
1514
1515 return err;
1516}
1517
Tvrtko Ursulinbf9e8422016-12-01 14:16:38 +00001518int i915_gem_init(struct drm_i915_private *dev_priv)
Chris Wilson1070a422012-04-24 15:47:41 +01001519{
Chris Wilson1070a422012-04-24 15:47:41 +01001520 int ret;
1521
Changbin Du52b24162018-05-08 17:07:05 +08001522 /* We need to fallback to 4K pages if host doesn't support huge gtt. */
1523 if (intel_vgpu_active(dev_priv) && !intel_vgpu_has_huge_gtt(dev_priv))
Matthew Auldda9fe3f32017-10-06 23:18:31 +01001524 mkwrite_device_info(dev_priv)->page_sizes =
1525 I915_GTT_PAGE_SIZE_4K;
1526
Chris Wilson94312822017-05-03 10:39:18 +01001527 dev_priv->mm.unordered_timeline = dma_fence_context_alloc(1);
Chris Wilson57822dc2017-02-22 11:40:48 +00001528
Chris Wilson1e345562019-01-28 10:23:56 +00001529 i915_timelines_init(dev_priv);
1530
Chris Wilsonee487002017-11-22 17:26:21 +00001531 ret = i915_gem_init_userptr(dev_priv);
1532 if (ret)
1533 return ret;
1534
Sagar Arun Kamble70deead2018-01-24 21:16:58 +05301535 ret = intel_uc_init_misc(dev_priv);
Michał Winiarski3176ff42017-12-13 23:13:47 +01001536 if (ret)
1537 return ret;
1538
Michal Wajdeczkof7dc0152018-06-28 14:15:21 +00001539 ret = intel_wopcm_init(&dev_priv->wopcm);
1540 if (ret)
1541 goto err_uc_misc;
1542
Chris Wilson5e4f5182015-02-13 14:35:59 +00001543 /* This is just a security blanket to placate dragons.
1544 * On some systems, we very sporadically observe that the first TLBs
1545 * used by the CS may be stale, despite us poking the TLB reset. If
1546 * we hold the forcewake during initialisation these problems
1547 * just magically go away.
1548 */
Chris Wilsonee487002017-11-22 17:26:21 +00001549 mutex_lock(&dev_priv->drm.struct_mutex);
Daniele Ceraolo Spurio3ceea6a2019-03-19 11:35:36 -07001550 intel_uncore_forcewake_get(&dev_priv->uncore, FORCEWAKE_ALL);
Chris Wilson5e4f5182015-02-13 14:35:59 +00001551
Chris Wilsonf6b9d5c2016-08-04 07:52:23 +01001552 ret = i915_gem_init_ggtt(dev_priv);
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001553 if (ret) {
1554 GEM_BUG_ON(ret == -EIO);
1555 goto err_unlock;
1556 }
Jesse Barnesd62b4892013-03-08 10:45:53 -08001557
Chris Wilson51797492018-12-04 14:15:16 +00001558 ret = i915_gem_init_scratch(dev_priv,
Lucas De Marchicf819ef2018-12-12 10:10:43 -08001559 IS_GEN(dev_priv, 2) ? SZ_256K : PAGE_SIZE);
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001560 if (ret) {
1561 GEM_BUG_ON(ret == -EIO);
1562 goto err_ggtt;
1563 }
Ben Widawsky2fa48d82013-12-06 14:11:04 -08001564
Chris Wilson11334c62019-04-26 17:33:33 +01001565 ret = intel_engines_setup(dev_priv);
1566 if (ret) {
1567 GEM_BUG_ON(ret == -EIO);
1568 goto err_unlock;
1569 }
1570
Chris Wilson51797492018-12-04 14:15:16 +00001571 ret = i915_gem_contexts_init(dev_priv);
1572 if (ret) {
1573 GEM_BUG_ON(ret == -EIO);
1574 goto err_scratch;
1575 }
1576
Tvrtko Ursulinbf9e8422016-12-01 14:16:38 +00001577 ret = intel_engines_init(dev_priv);
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001578 if (ret) {
1579 GEM_BUG_ON(ret == -EIO);
1580 goto err_context;
1581 }
Daniel Vetter53ca26c2012-04-26 23:28:03 +02001582
Chris Wilsonf58d13d2017-11-10 14:26:29 +00001583 intel_init_gt_powersave(dev_priv);
1584
Michał Winiarski61b5c152017-12-13 23:13:48 +01001585 ret = intel_uc_init(dev_priv);
Chris Wilsoncc6a8182017-11-10 14:26:30 +00001586 if (ret)
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001587 goto err_pm;
Chris Wilsoncc6a8182017-11-10 14:26:30 +00001588
Michał Winiarski61b5c152017-12-13 23:13:48 +01001589 ret = i915_gem_init_hw(dev_priv);
1590 if (ret)
1591 goto err_uc_init;
1592
Chris Wilsoncc6a8182017-11-10 14:26:30 +00001593 /*
1594 * Despite its name intel_init_clock_gating applies both display
1595 * clock gating workarounds; GT mmio workarounds and the occasional
1596 * GT power context workaround. Worse, sometimes it includes a context
1597 * register workaround which we need to apply before we record the
1598 * default HW state for all contexts.
1599 *
1600 * FIXME: break up the workarounds and apply them at the right time!
1601 */
1602 intel_init_clock_gating(dev_priv);
1603
Chris Wilson254e1182019-04-17 08:56:28 +01001604 ret = intel_engines_verify_workarounds(dev_priv);
1605 if (ret)
1606 goto err_init_hw;
1607
Chris Wilsond2b4b972017-11-10 14:26:33 +00001608 ret = __intel_engines_record_defaults(dev_priv);
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001609 if (ret)
1610 goto err_init_hw;
1611
1612 if (i915_inject_load_failure()) {
1613 ret = -ENODEV;
1614 goto err_init_hw;
1615 }
1616
1617 if (i915_inject_load_failure()) {
1618 ret = -EIO;
1619 goto err_init_hw;
1620 }
1621
Daniele Ceraolo Spurio3ceea6a2019-03-19 11:35:36 -07001622 intel_uncore_forcewake_put(&dev_priv->uncore, FORCEWAKE_ALL);
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001623 mutex_unlock(&dev_priv->drm.struct_mutex);
1624
1625 return 0;
1626
1627 /*
1628 * Unwinding is complicated by that we want to handle -EIO to mean
1629 * disable GPU submission but keep KMS alive. We want to mark the
1630 * HW as irrevisibly wedged, but keep enough state around that the
1631 * driver doesn't explode during runtime.
1632 */
1633err_init_hw:
Chris Wilson8571a052018-06-06 15:54:41 +01001634 mutex_unlock(&dev_priv->drm.struct_mutex);
1635
Chris Wilson79ffac852019-04-24 21:07:17 +01001636 i915_gem_set_wedged(dev_priv);
Chris Wilson5861b012019-03-08 09:36:54 +00001637 i915_gem_suspend(dev_priv);
Chris Wilson8571a052018-06-06 15:54:41 +01001638 i915_gem_suspend_late(dev_priv);
1639
Chris Wilson8bcf9f72018-07-10 10:44:20 +01001640 i915_gem_drain_workqueue(dev_priv);
1641
Chris Wilson8571a052018-06-06 15:54:41 +01001642 mutex_lock(&dev_priv->drm.struct_mutex);
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001643 intel_uc_fini_hw(dev_priv);
Michał Winiarski61b5c152017-12-13 23:13:48 +01001644err_uc_init:
1645 intel_uc_fini(dev_priv);
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001646err_pm:
1647 if (ret != -EIO) {
1648 intel_cleanup_gt_powersave(dev_priv);
Chris Wilson45b9c962019-05-01 11:32:04 +01001649 intel_engines_cleanup(dev_priv);
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001650 }
1651err_context:
1652 if (ret != -EIO)
1653 i915_gem_contexts_fini(dev_priv);
Chris Wilson51797492018-12-04 14:15:16 +00001654err_scratch:
1655 i915_gem_fini_scratch(dev_priv);
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001656err_ggtt:
1657err_unlock:
Daniele Ceraolo Spurio3ceea6a2019-03-19 11:35:36 -07001658 intel_uncore_forcewake_put(&dev_priv->uncore, FORCEWAKE_ALL);
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001659 mutex_unlock(&dev_priv->drm.struct_mutex);
1660
Michal Wajdeczkof7dc0152018-06-28 14:15:21 +00001661err_uc_misc:
Sagar Arun Kamble70deead2018-01-24 21:16:58 +05301662 intel_uc_fini_misc(dev_priv);
Sagar Arun Kambleda943b52018-01-10 18:24:16 +05301663
Chris Wilson1e345562019-01-28 10:23:56 +00001664 if (ret != -EIO) {
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001665 i915_gem_cleanup_userptr(dev_priv);
Chris Wilson1e345562019-01-28 10:23:56 +00001666 i915_timelines_fini(dev_priv);
1667 }
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001668
Chris Wilson60990322014-04-09 09:19:42 +01001669 if (ret == -EIO) {
Chris Wilson7ed43df2018-07-26 09:50:32 +01001670 mutex_lock(&dev_priv->drm.struct_mutex);
1671
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001672 /*
1673 * Allow engine initialisation to fail by marking the GPU as
Chris Wilson60990322014-04-09 09:19:42 +01001674 * wedged. But we only want to do this where the GPU is angry,
1675 * for all other failure, such as an allocation failure, bail.
1676 */
Chris Wilsonc41166f2019-02-20 14:56:37 +00001677 if (!i915_reset_failed(dev_priv)) {
Chris Wilson51c18bf2018-06-09 12:10:58 +01001678 i915_load_error(dev_priv,
1679 "Failed to initialize GPU, declaring it wedged!\n");
Chris Wilson6f74b362017-10-15 15:37:25 +01001680 i915_gem_set_wedged(dev_priv);
1681 }
Chris Wilson7ed43df2018-07-26 09:50:32 +01001682
1683 /* Minimal basic recovery for KMS */
1684 ret = i915_ggtt_enable_hw(dev_priv);
1685 i915_gem_restore_gtt_mappings(dev_priv);
1686 i915_gem_restore_fences(dev_priv);
1687 intel_init_clock_gating(dev_priv);
1688
1689 mutex_unlock(&dev_priv->drm.struct_mutex);
Chris Wilson1070a422012-04-24 15:47:41 +01001690 }
1691
Chris Wilson6ca9a2b2017-12-13 13:43:47 +00001692 i915_gem_drain_freed_objects(dev_priv);
Chris Wilson60990322014-04-09 09:19:42 +01001693 return ret;
Chris Wilson1070a422012-04-24 15:47:41 +01001694}
1695
Janusz Krzysztofik47bc28d2019-05-30 15:31:05 +02001696void i915_gem_fini_hw(struct drm_i915_private *dev_priv)
Michal Wajdeczko8979187a2018-06-04 09:00:32 +00001697{
Chris Wilson79ffac852019-04-24 21:07:17 +01001698 GEM_BUG_ON(dev_priv->gt.awake);
1699
Chris Wilson0cf289b2019-06-13 08:32:54 +01001700 intel_wakeref_auto_fini(&dev_priv->ggtt.userfault_wakeref);
Chris Wilsonb27e35a2019-05-27 12:51:14 +01001701
Michal Wajdeczko8979187a2018-06-04 09:00:32 +00001702 i915_gem_suspend_late(dev_priv);
Chris Wilson30b710842018-08-12 23:36:29 +01001703 intel_disable_gt_powersave(dev_priv);
Michal Wajdeczko8979187a2018-06-04 09:00:32 +00001704
1705 /* Flush any outstanding unpin_work. */
1706 i915_gem_drain_workqueue(dev_priv);
1707
1708 mutex_lock(&dev_priv->drm.struct_mutex);
1709 intel_uc_fini_hw(dev_priv);
1710 intel_uc_fini(dev_priv);
Janusz Krzysztofik47bc28d2019-05-30 15:31:05 +02001711 mutex_unlock(&dev_priv->drm.struct_mutex);
1712
1713 i915_gem_drain_freed_objects(dev_priv);
1714}
1715
1716void i915_gem_fini(struct drm_i915_private *dev_priv)
1717{
1718 mutex_lock(&dev_priv->drm.struct_mutex);
Chris Wilson45b9c962019-05-01 11:32:04 +01001719 intel_engines_cleanup(dev_priv);
Michal Wajdeczko8979187a2018-06-04 09:00:32 +00001720 i915_gem_contexts_fini(dev_priv);
Chris Wilson51797492018-12-04 14:15:16 +00001721 i915_gem_fini_scratch(dev_priv);
Michal Wajdeczko8979187a2018-06-04 09:00:32 +00001722 mutex_unlock(&dev_priv->drm.struct_mutex);
1723
Tvrtko Ursulin25d140f2018-12-03 13:33:19 +00001724 intel_wa_list_free(&dev_priv->gt_wa_list);
1725
Chris Wilson30b710842018-08-12 23:36:29 +01001726 intel_cleanup_gt_powersave(dev_priv);
1727
Michal Wajdeczko8979187a2018-06-04 09:00:32 +00001728 intel_uc_fini_misc(dev_priv);
1729 i915_gem_cleanup_userptr(dev_priv);
Chris Wilson1e345562019-01-28 10:23:56 +00001730 i915_timelines_fini(dev_priv);
Michal Wajdeczko8979187a2018-06-04 09:00:32 +00001731
1732 i915_gem_drain_freed_objects(dev_priv);
1733
1734 WARN_ON(!list_empty(&dev_priv->contexts.list));
1735}
1736
Chris Wilson24145512017-01-24 11:01:35 +00001737void i915_gem_init_mmio(struct drm_i915_private *i915)
1738{
1739 i915_gem_sanitize(i915);
1740}
1741
Chris Wilson9c52d1c2017-11-10 23:24:47 +00001742static void i915_gem_init__mm(struct drm_i915_private *i915)
1743{
Chris Wilson9c52d1c2017-11-10 23:24:47 +00001744 spin_lock_init(&i915->mm.obj_lock);
1745 spin_lock_init(&i915->mm.free_lock);
1746
1747 init_llist_head(&i915->mm.free_list);
1748
Chris Wilson3b4fa962019-05-30 21:34:59 +01001749 INIT_LIST_HEAD(&i915->mm.purge_list);
Chris Wilsonecab9be2019-06-12 11:57:20 +01001750 INIT_LIST_HEAD(&i915->mm.shrink_list);
Chris Wilson9c52d1c2017-11-10 23:24:47 +00001751
Chris Wilson84753552019-05-28 10:29:45 +01001752 i915_gem_init__objects(i915);
Chris Wilson9c52d1c2017-11-10 23:24:47 +00001753}
1754
Michal Wajdeczkoa0de9082018-03-23 12:34:49 +00001755int i915_gem_init_early(struct drm_i915_private *dev_priv)
Eric Anholt673a3942008-07-30 12:06:12 -07001756{
Chris Wilson13f1bfd2019-02-28 10:20:34 +00001757 int err;
Chris Wilsond1b48c12017-08-16 09:52:08 +01001758
Chris Wilson9c52d1c2017-11-10 23:24:47 +00001759 i915_gem_init__mm(dev_priv);
Chris Wilson23c3c3d2019-04-24 21:07:14 +01001760 i915_gem_init__pm(dev_priv);
Chris Wilsonf2123812017-10-16 12:40:37 +01001761
Chris Wilson1f15b762016-07-01 17:23:14 +01001762 init_waitqueue_head(&dev_priv->gpu_error.wait_queue);
Daniel Vetter1f83fee2012-11-15 17:17:22 +01001763 init_waitqueue_head(&dev_priv->gpu_error.reset_queue);
Chris Wilson18bb2bc2019-01-14 21:04:01 +00001764 mutex_init(&dev_priv->gpu_error.wedge_mutex);
Chris Wilson2caffbf2019-02-08 15:37:03 +00001765 init_srcu_struct(&dev_priv->gpu_error.reset_backoff_srcu);
Chris Wilson31169712009-09-14 16:50:28 +01001766
Joonas Lahtinen6f633402016-09-01 14:58:21 +03001767 atomic_set(&dev_priv->mm.bsd_engine_dispatch_index, 0);
1768
Chris Wilsonb5add952016-08-04 16:32:36 +01001769 spin_lock_init(&dev_priv->fb_tracking.lock);
Chris Wilson73cb9702016-10-28 13:58:46 +01001770
Matthew Auld465c4032017-10-06 23:18:14 +01001771 err = i915_gemfs_init(dev_priv);
1772 if (err)
1773 DRM_NOTE("Unable to create a private tmpfs mount, hugepage support will be disabled(%d).\n", err);
1774
Chris Wilson73cb9702016-10-28 13:58:46 +01001775 return 0;
Eric Anholt673a3942008-07-30 12:06:12 -07001776}
Dave Airlie71acb5e2008-12-30 20:31:46 +10001777
Michal Wajdeczkoa0de9082018-03-23 12:34:49 +00001778void i915_gem_cleanup_early(struct drm_i915_private *dev_priv)
Imre Deakd64aa092016-01-19 15:26:29 +02001779{
Chris Wilsonc4d4c1c2017-02-10 16:35:23 +00001780 i915_gem_drain_freed_objects(dev_priv);
Chris Wilsonc9c704712018-02-19 22:06:31 +00001781 GEM_BUG_ON(!llist_empty(&dev_priv->mm.free_list));
1782 GEM_BUG_ON(atomic_read(&dev_priv->mm.free_count));
Chris Wilsond82b4b22019-05-30 21:35:00 +01001783 WARN_ON(dev_priv->mm.shrink_count);
Matthew Auldea84aa72016-11-17 21:04:11 +00001784
Chris Wilson2caffbf2019-02-08 15:37:03 +00001785 cleanup_srcu_struct(&dev_priv->gpu_error.reset_backoff_srcu);
1786
Matthew Auld465c4032017-10-06 23:18:14 +01001787 i915_gemfs_fini(dev_priv);
Imre Deakd64aa092016-01-19 15:26:29 +02001788}
1789
Chris Wilson6a800ea2016-09-21 14:51:07 +01001790int i915_gem_freeze(struct drm_i915_private *dev_priv)
1791{
Chris Wilsond0aa3012017-04-07 11:25:49 +01001792 /* Discard all purgeable objects, let userspace recover those as
1793 * required after resuming.
1794 */
Chris Wilson6a800ea2016-09-21 14:51:07 +01001795 i915_gem_shrink_all(dev_priv);
Chris Wilson6a800ea2016-09-21 14:51:07 +01001796
Chris Wilson6a800ea2016-09-21 14:51:07 +01001797 return 0;
1798}
1799
Chris Wilson95c778d2018-06-01 15:41:25 +01001800int i915_gem_freeze_late(struct drm_i915_private *i915)
Chris Wilson461fb992016-05-14 07:26:33 +01001801{
1802 struct drm_i915_gem_object *obj;
Chris Wilsonecab9be2019-06-12 11:57:20 +01001803 intel_wakeref_t wakeref;
Chris Wilson461fb992016-05-14 07:26:33 +01001804
Chris Wilson95c778d2018-06-01 15:41:25 +01001805 /*
1806 * Called just before we write the hibernation image.
Chris Wilson461fb992016-05-14 07:26:33 +01001807 *
1808 * We need to update the domain tracking to reflect that the CPU
1809 * will be accessing all the pages to create and restore from the
1810 * hibernation, and so upon restoration those pages will be in the
1811 * CPU domain.
1812 *
1813 * To make sure the hibernation image contains the latest state,
1814 * we update that state just before writing out the image.
Chris Wilson7aab2d52016-09-09 20:02:18 +01001815 *
1816 * To try and reduce the hibernation image, we manually shrink
Chris Wilsond0aa3012017-04-07 11:25:49 +01001817 * the objects as well, see i915_gem_freeze()
Chris Wilson461fb992016-05-14 07:26:33 +01001818 */
1819
Daniele Ceraolo Spuriod858d562019-06-13 16:21:54 -07001820 wakeref = intel_runtime_pm_get(&i915->runtime_pm);
Chris Wilsonecab9be2019-06-12 11:57:20 +01001821
1822 i915_gem_shrink(i915, -1UL, NULL, ~0);
Chris Wilson95c778d2018-06-01 15:41:25 +01001823 i915_gem_drain_freed_objects(i915);
Chris Wilson461fb992016-05-14 07:26:33 +01001824
Chris Wilsonecab9be2019-06-12 11:57:20 +01001825 list_for_each_entry(obj, &i915->mm.shrink_list, mm.link) {
1826 i915_gem_object_lock(obj);
1827 WARN_ON(i915_gem_object_set_to_cpu_domain(obj, true));
1828 i915_gem_object_unlock(obj);
Chris Wilson461fb992016-05-14 07:26:33 +01001829 }
Chris Wilsonecab9be2019-06-12 11:57:20 +01001830
Daniele Ceraolo Spuriod858d562019-06-13 16:21:54 -07001831 intel_runtime_pm_put(&i915->runtime_pm, wakeref);
Chris Wilson461fb992016-05-14 07:26:33 +01001832
1833 return 0;
1834}
1835
Chris Wilsonf787a5f2010-09-24 16:02:42 +01001836void i915_gem_release(struct drm_device *dev, struct drm_file *file)
Eric Anholtb9624422009-06-03 07:27:35 +00001837{
Chris Wilsonf787a5f2010-09-24 16:02:42 +01001838 struct drm_i915_file_private *file_priv = file->driver_priv;
Chris Wilsone61e0f52018-02-21 09:56:36 +00001839 struct i915_request *request;
Eric Anholtb9624422009-06-03 07:27:35 +00001840
1841 /* Clean up our request list when the client is going away, so that
1842 * later retire_requests won't dereference our soon-to-be-gone
1843 * file_priv.
1844 */
Chris Wilson1c255952010-09-26 11:03:27 +01001845 spin_lock(&file_priv->mm.lock);
Chris Wilsonc8659ef2017-03-02 12:25:25 +00001846 list_for_each_entry(request, &file_priv->mm.request_list, client_link)
Chris Wilsonf787a5f2010-09-24 16:02:42 +01001847 request->file_priv = NULL;
Chris Wilson1c255952010-09-26 11:03:27 +01001848 spin_unlock(&file_priv->mm.lock);
Chris Wilsonb29c19b2013-09-25 17:34:56 +01001849}
1850
Chris Wilson829a0af2017-06-20 12:05:45 +01001851int i915_gem_open(struct drm_i915_private *i915, struct drm_file *file)
Chris Wilsonb29c19b2013-09-25 17:34:56 +01001852{
1853 struct drm_i915_file_private *file_priv;
Ben Widawskye422b882013-12-06 14:10:58 -08001854 int ret;
Chris Wilsonb29c19b2013-09-25 17:34:56 +01001855
Chris Wilsonc4c29d72016-11-09 10:45:07 +00001856 DRM_DEBUG("\n");
Chris Wilsonb29c19b2013-09-25 17:34:56 +01001857
1858 file_priv = kzalloc(sizeof(*file_priv), GFP_KERNEL);
1859 if (!file_priv)
1860 return -ENOMEM;
1861
1862 file->driver_priv = file_priv;
Chris Wilson829a0af2017-06-20 12:05:45 +01001863 file_priv->dev_priv = i915;
Chris Wilsonab0e7ff2014-02-25 17:11:24 +02001864 file_priv->file = file;
Chris Wilsonb29c19b2013-09-25 17:34:56 +01001865
1866 spin_lock_init(&file_priv->mm.lock);
1867 INIT_LIST_HEAD(&file_priv->mm.request_list);
Chris Wilsonb29c19b2013-09-25 17:34:56 +01001868
Chris Wilsonc80ff162016-07-27 09:07:27 +01001869 file_priv->bsd_engine = -1;
Mika Kuoppala14921f32018-06-15 13:44:29 +03001870 file_priv->hang_timestamp = jiffies;
Tvrtko Ursulinde1add32016-01-15 15:12:50 +00001871
Chris Wilson829a0af2017-06-20 12:05:45 +01001872 ret = i915_gem_context_open(i915, file);
Ben Widawskye422b882013-12-06 14:10:58 -08001873 if (ret)
1874 kfree(file_priv);
Chris Wilsonb29c19b2013-09-25 17:34:56 +01001875
Ben Widawskye422b882013-12-06 14:10:58 -08001876 return ret;
Chris Wilsonb29c19b2013-09-25 17:34:56 +01001877}
1878
Daniel Vetterb680c372014-09-19 18:27:27 +02001879/**
1880 * i915_gem_track_fb - update frontbuffer tracking
Geliang Tangd9072a32015-09-15 05:58:44 -07001881 * @old: current GEM buffer for the frontbuffer slots
1882 * @new: new GEM buffer for the frontbuffer slots
1883 * @frontbuffer_bits: bitmask of frontbuffer slots
Daniel Vetterb680c372014-09-19 18:27:27 +02001884 *
1885 * This updates the frontbuffer tracking bits @frontbuffer_bits by clearing them
1886 * from @old and setting them in @new. Both @old and @new can be NULL.
1887 */
Daniel Vettera071fa02014-06-18 23:28:09 +02001888void i915_gem_track_fb(struct drm_i915_gem_object *old,
1889 struct drm_i915_gem_object *new,
1890 unsigned frontbuffer_bits)
1891{
Chris Wilsonfaf5bf02016-08-04 16:32:37 +01001892 /* Control of individual bits within the mask are guarded by
1893 * the owning plane->mutex, i.e. we can never see concurrent
1894 * manipulation of individual bits. But since the bitfield as a whole
1895 * is updated using RMW, we need to use atomics in order to update
1896 * the bits.
1897 */
1898 BUILD_BUG_ON(INTEL_FRONTBUFFER_BITS_PER_PIPE * I915_MAX_PIPES >
Chris Wilson74f6e182018-09-26 11:47:07 +01001899 BITS_PER_TYPE(atomic_t));
Chris Wilsonfaf5bf02016-08-04 16:32:37 +01001900
Daniel Vettera071fa02014-06-18 23:28:09 +02001901 if (old) {
Chris Wilsonfaf5bf02016-08-04 16:32:37 +01001902 WARN_ON(!(atomic_read(&old->frontbuffer_bits) & frontbuffer_bits));
1903 atomic_andnot(frontbuffer_bits, &old->frontbuffer_bits);
Daniel Vettera071fa02014-06-18 23:28:09 +02001904 }
1905
1906 if (new) {
Chris Wilsonfaf5bf02016-08-04 16:32:37 +01001907 WARN_ON(atomic_read(&new->frontbuffer_bits) & frontbuffer_bits);
1908 atomic_or(frontbuffer_bits, &new->frontbuffer_bits);
Daniel Vettera071fa02014-06-18 23:28:09 +02001909 }
1910}
1911
Chris Wilson935a2f72017-02-13 17:15:13 +00001912#if IS_ENABLED(CONFIG_DRM_I915_SELFTEST)
Chris Wilson66d9cb52017-02-13 17:15:17 +00001913#include "selftests/mock_gem_device.c"
Chris Wilson3f51b7e12018-08-30 14:48:06 +01001914#include "selftests/i915_gem.c"
Chris Wilson935a2f72017-02-13 17:15:13 +00001915#endif