18c2ecf20Sopenharmony_ci/*
28c2ecf20Sopenharmony_ci * Copyright © 2016 Intel Corporation
38c2ecf20Sopenharmony_ci *
48c2ecf20Sopenharmony_ci * Permission is hereby granted, free of charge, to any person obtaining a
58c2ecf20Sopenharmony_ci * copy of this software and associated documentation files (the "Software"),
68c2ecf20Sopenharmony_ci * to deal in the Software without restriction, including without limitation
78c2ecf20Sopenharmony_ci * the rights to use, copy, modify, merge, publish, distribute, sublicense,
88c2ecf20Sopenharmony_ci * and/or sell copies of the Software, and to permit persons to whom the
98c2ecf20Sopenharmony_ci * Software is furnished to do so, subject to the following conditions:
108c2ecf20Sopenharmony_ci *
118c2ecf20Sopenharmony_ci * The above copyright notice and this permission notice (including the next
128c2ecf20Sopenharmony_ci * paragraph) shall be included in all copies or substantial portions of the
138c2ecf20Sopenharmony_ci * Software.
148c2ecf20Sopenharmony_ci *
158c2ecf20Sopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
168c2ecf20Sopenharmony_ci * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
178c2ecf20Sopenharmony_ci * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
188c2ecf20Sopenharmony_ci * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
198c2ecf20Sopenharmony_ci * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
208c2ecf20Sopenharmony_ci * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
218c2ecf20Sopenharmony_ci * IN THE SOFTWARE.
228c2ecf20Sopenharmony_ci *
238c2ecf20Sopenharmony_ci */
248c2ecf20Sopenharmony_ci
258c2ecf20Sopenharmony_ci#include "gem/i915_gem_context.h"
268c2ecf20Sopenharmony_ci#include "gt/intel_ring.h"
278c2ecf20Sopenharmony_ci
288c2ecf20Sopenharmony_ci#include "i915_drv.h"
298c2ecf20Sopenharmony_ci#include "intel_context.h"
308c2ecf20Sopenharmony_ci#include "intel_engine_pm.h"
318c2ecf20Sopenharmony_ci
328c2ecf20Sopenharmony_ci#include "mock_engine.h"
338c2ecf20Sopenharmony_ci#include "selftests/mock_request.h"
348c2ecf20Sopenharmony_ci
358c2ecf20Sopenharmony_cistatic void mock_timeline_pin(struct intel_timeline *tl)
368c2ecf20Sopenharmony_ci{
378c2ecf20Sopenharmony_ci	atomic_inc(&tl->pin_count);
388c2ecf20Sopenharmony_ci}
398c2ecf20Sopenharmony_ci
408c2ecf20Sopenharmony_cistatic void mock_timeline_unpin(struct intel_timeline *tl)
418c2ecf20Sopenharmony_ci{
428c2ecf20Sopenharmony_ci	GEM_BUG_ON(!atomic_read(&tl->pin_count));
438c2ecf20Sopenharmony_ci	atomic_dec(&tl->pin_count);
448c2ecf20Sopenharmony_ci}
458c2ecf20Sopenharmony_ci
468c2ecf20Sopenharmony_cistatic struct intel_ring *mock_ring(struct intel_engine_cs *engine)
478c2ecf20Sopenharmony_ci{
488c2ecf20Sopenharmony_ci	const unsigned long sz = PAGE_SIZE / 2;
498c2ecf20Sopenharmony_ci	struct intel_ring *ring;
508c2ecf20Sopenharmony_ci
518c2ecf20Sopenharmony_ci	ring = kzalloc(sizeof(*ring) + sz, GFP_KERNEL);
528c2ecf20Sopenharmony_ci	if (!ring)
538c2ecf20Sopenharmony_ci		return NULL;
548c2ecf20Sopenharmony_ci
558c2ecf20Sopenharmony_ci	kref_init(&ring->ref);
568c2ecf20Sopenharmony_ci	ring->size = sz;
578c2ecf20Sopenharmony_ci	ring->effective_size = sz;
588c2ecf20Sopenharmony_ci	ring->vaddr = (void *)(ring + 1);
598c2ecf20Sopenharmony_ci	atomic_set(&ring->pin_count, 1);
608c2ecf20Sopenharmony_ci
618c2ecf20Sopenharmony_ci	ring->vma = i915_vma_alloc();
628c2ecf20Sopenharmony_ci	if (!ring->vma) {
638c2ecf20Sopenharmony_ci		kfree(ring);
648c2ecf20Sopenharmony_ci		return NULL;
658c2ecf20Sopenharmony_ci	}
668c2ecf20Sopenharmony_ci	i915_active_init(&ring->vma->active, NULL, NULL);
678c2ecf20Sopenharmony_ci	__set_bit(I915_VMA_GGTT_BIT, __i915_vma_flags(ring->vma));
688c2ecf20Sopenharmony_ci	__set_bit(DRM_MM_NODE_ALLOCATED_BIT, &ring->vma->node.flags);
698c2ecf20Sopenharmony_ci	ring->vma->node.size = sz;
708c2ecf20Sopenharmony_ci
718c2ecf20Sopenharmony_ci	intel_ring_update_space(ring);
728c2ecf20Sopenharmony_ci
738c2ecf20Sopenharmony_ci	return ring;
748c2ecf20Sopenharmony_ci}
758c2ecf20Sopenharmony_ci
768c2ecf20Sopenharmony_cistatic void mock_ring_free(struct intel_ring *ring)
778c2ecf20Sopenharmony_ci{
788c2ecf20Sopenharmony_ci	i915_active_fini(&ring->vma->active);
798c2ecf20Sopenharmony_ci	i915_vma_free(ring->vma);
808c2ecf20Sopenharmony_ci
818c2ecf20Sopenharmony_ci	kfree(ring);
828c2ecf20Sopenharmony_ci}
838c2ecf20Sopenharmony_ci
848c2ecf20Sopenharmony_cistatic struct i915_request *first_request(struct mock_engine *engine)
858c2ecf20Sopenharmony_ci{
868c2ecf20Sopenharmony_ci	return list_first_entry_or_null(&engine->hw_queue,
878c2ecf20Sopenharmony_ci					struct i915_request,
888c2ecf20Sopenharmony_ci					mock.link);
898c2ecf20Sopenharmony_ci}
908c2ecf20Sopenharmony_ci
918c2ecf20Sopenharmony_cistatic void advance(struct i915_request *request)
928c2ecf20Sopenharmony_ci{
938c2ecf20Sopenharmony_ci	list_del_init(&request->mock.link);
948c2ecf20Sopenharmony_ci	i915_request_mark_complete(request);
958c2ecf20Sopenharmony_ci	GEM_BUG_ON(!i915_request_completed(request));
968c2ecf20Sopenharmony_ci
978c2ecf20Sopenharmony_ci	intel_engine_signal_breadcrumbs(request->engine);
988c2ecf20Sopenharmony_ci}
998c2ecf20Sopenharmony_ci
1008c2ecf20Sopenharmony_cistatic void hw_delay_complete(struct timer_list *t)
1018c2ecf20Sopenharmony_ci{
1028c2ecf20Sopenharmony_ci	struct mock_engine *engine = from_timer(engine, t, hw_delay);
1038c2ecf20Sopenharmony_ci	struct i915_request *request;
1048c2ecf20Sopenharmony_ci	unsigned long flags;
1058c2ecf20Sopenharmony_ci
1068c2ecf20Sopenharmony_ci	spin_lock_irqsave(&engine->hw_lock, flags);
1078c2ecf20Sopenharmony_ci
1088c2ecf20Sopenharmony_ci	/* Timer fired, first request is complete */
1098c2ecf20Sopenharmony_ci	request = first_request(engine);
1108c2ecf20Sopenharmony_ci	if (request)
1118c2ecf20Sopenharmony_ci		advance(request);
1128c2ecf20Sopenharmony_ci
1138c2ecf20Sopenharmony_ci	/*
1148c2ecf20Sopenharmony_ci	 * Also immediately signal any subsequent 0-delay requests, but
1158c2ecf20Sopenharmony_ci	 * requeue the timer for the next delayed request.
1168c2ecf20Sopenharmony_ci	 */
1178c2ecf20Sopenharmony_ci	while ((request = first_request(engine))) {
1188c2ecf20Sopenharmony_ci		if (request->mock.delay) {
1198c2ecf20Sopenharmony_ci			mod_timer(&engine->hw_delay,
1208c2ecf20Sopenharmony_ci				  jiffies + request->mock.delay);
1218c2ecf20Sopenharmony_ci			break;
1228c2ecf20Sopenharmony_ci		}
1238c2ecf20Sopenharmony_ci
1248c2ecf20Sopenharmony_ci		advance(request);
1258c2ecf20Sopenharmony_ci	}
1268c2ecf20Sopenharmony_ci
1278c2ecf20Sopenharmony_ci	spin_unlock_irqrestore(&engine->hw_lock, flags);
1288c2ecf20Sopenharmony_ci}
1298c2ecf20Sopenharmony_ci
1308c2ecf20Sopenharmony_cistatic void mock_context_unpin(struct intel_context *ce)
1318c2ecf20Sopenharmony_ci{
1328c2ecf20Sopenharmony_ci}
1338c2ecf20Sopenharmony_ci
1348c2ecf20Sopenharmony_cistatic void mock_context_post_unpin(struct intel_context *ce)
1358c2ecf20Sopenharmony_ci{
1368c2ecf20Sopenharmony_ci}
1378c2ecf20Sopenharmony_ci
1388c2ecf20Sopenharmony_cistatic void mock_context_destroy(struct kref *ref)
1398c2ecf20Sopenharmony_ci{
1408c2ecf20Sopenharmony_ci	struct intel_context *ce = container_of(ref, typeof(*ce), ref);
1418c2ecf20Sopenharmony_ci
1428c2ecf20Sopenharmony_ci	GEM_BUG_ON(intel_context_is_pinned(ce));
1438c2ecf20Sopenharmony_ci
1448c2ecf20Sopenharmony_ci	if (test_bit(CONTEXT_ALLOC_BIT, &ce->flags)) {
1458c2ecf20Sopenharmony_ci		mock_ring_free(ce->ring);
1468c2ecf20Sopenharmony_ci		mock_timeline_unpin(ce->timeline);
1478c2ecf20Sopenharmony_ci	}
1488c2ecf20Sopenharmony_ci
1498c2ecf20Sopenharmony_ci	intel_context_fini(ce);
1508c2ecf20Sopenharmony_ci	intel_context_free(ce);
1518c2ecf20Sopenharmony_ci}
1528c2ecf20Sopenharmony_ci
1538c2ecf20Sopenharmony_cistatic int mock_context_alloc(struct intel_context *ce)
1548c2ecf20Sopenharmony_ci{
1558c2ecf20Sopenharmony_ci	ce->ring = mock_ring(ce->engine);
1568c2ecf20Sopenharmony_ci	if (!ce->ring)
1578c2ecf20Sopenharmony_ci		return -ENOMEM;
1588c2ecf20Sopenharmony_ci
1598c2ecf20Sopenharmony_ci	ce->timeline = intel_timeline_create(ce->engine->gt);
1608c2ecf20Sopenharmony_ci	if (IS_ERR(ce->timeline)) {
1618c2ecf20Sopenharmony_ci		kfree(ce->engine);
1628c2ecf20Sopenharmony_ci		return PTR_ERR(ce->timeline);
1638c2ecf20Sopenharmony_ci	}
1648c2ecf20Sopenharmony_ci
1658c2ecf20Sopenharmony_ci	mock_timeline_pin(ce->timeline);
1668c2ecf20Sopenharmony_ci
1678c2ecf20Sopenharmony_ci	return 0;
1688c2ecf20Sopenharmony_ci}
1698c2ecf20Sopenharmony_ci
1708c2ecf20Sopenharmony_cistatic int mock_context_pre_pin(struct intel_context *ce,
1718c2ecf20Sopenharmony_ci				struct i915_gem_ww_ctx *ww, void **unused)
1728c2ecf20Sopenharmony_ci{
1738c2ecf20Sopenharmony_ci	return 0;
1748c2ecf20Sopenharmony_ci}
1758c2ecf20Sopenharmony_ci
1768c2ecf20Sopenharmony_cistatic int mock_context_pin(struct intel_context *ce, void *unused)
1778c2ecf20Sopenharmony_ci{
1788c2ecf20Sopenharmony_ci	return 0;
1798c2ecf20Sopenharmony_ci}
1808c2ecf20Sopenharmony_ci
1818c2ecf20Sopenharmony_cistatic void mock_context_reset(struct intel_context *ce)
1828c2ecf20Sopenharmony_ci{
1838c2ecf20Sopenharmony_ci}
1848c2ecf20Sopenharmony_ci
1858c2ecf20Sopenharmony_cistatic const struct intel_context_ops mock_context_ops = {
1868c2ecf20Sopenharmony_ci	.alloc = mock_context_alloc,
1878c2ecf20Sopenharmony_ci
1888c2ecf20Sopenharmony_ci	.pre_pin = mock_context_pre_pin,
1898c2ecf20Sopenharmony_ci	.pin = mock_context_pin,
1908c2ecf20Sopenharmony_ci	.unpin = mock_context_unpin,
1918c2ecf20Sopenharmony_ci	.post_unpin = mock_context_post_unpin,
1928c2ecf20Sopenharmony_ci
1938c2ecf20Sopenharmony_ci	.enter = intel_context_enter_engine,
1948c2ecf20Sopenharmony_ci	.exit = intel_context_exit_engine,
1958c2ecf20Sopenharmony_ci
1968c2ecf20Sopenharmony_ci	.reset = mock_context_reset,
1978c2ecf20Sopenharmony_ci	.destroy = mock_context_destroy,
1988c2ecf20Sopenharmony_ci};
1998c2ecf20Sopenharmony_ci
2008c2ecf20Sopenharmony_cistatic int mock_request_alloc(struct i915_request *request)
2018c2ecf20Sopenharmony_ci{
2028c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&request->mock.link);
2038c2ecf20Sopenharmony_ci	request->mock.delay = 0;
2048c2ecf20Sopenharmony_ci
2058c2ecf20Sopenharmony_ci	return 0;
2068c2ecf20Sopenharmony_ci}
2078c2ecf20Sopenharmony_ci
2088c2ecf20Sopenharmony_cistatic int mock_emit_flush(struct i915_request *request,
2098c2ecf20Sopenharmony_ci			   unsigned int flags)
2108c2ecf20Sopenharmony_ci{
2118c2ecf20Sopenharmony_ci	return 0;
2128c2ecf20Sopenharmony_ci}
2138c2ecf20Sopenharmony_ci
2148c2ecf20Sopenharmony_cistatic u32 *mock_emit_breadcrumb(struct i915_request *request, u32 *cs)
2158c2ecf20Sopenharmony_ci{
2168c2ecf20Sopenharmony_ci	return cs;
2178c2ecf20Sopenharmony_ci}
2188c2ecf20Sopenharmony_ci
2198c2ecf20Sopenharmony_cistatic void mock_submit_request(struct i915_request *request)
2208c2ecf20Sopenharmony_ci{
2218c2ecf20Sopenharmony_ci	struct mock_engine *engine =
2228c2ecf20Sopenharmony_ci		container_of(request->engine, typeof(*engine), base);
2238c2ecf20Sopenharmony_ci	unsigned long flags;
2248c2ecf20Sopenharmony_ci
2258c2ecf20Sopenharmony_ci	i915_request_submit(request);
2268c2ecf20Sopenharmony_ci
2278c2ecf20Sopenharmony_ci	spin_lock_irqsave(&engine->hw_lock, flags);
2288c2ecf20Sopenharmony_ci	list_add_tail(&request->mock.link, &engine->hw_queue);
2298c2ecf20Sopenharmony_ci	if (list_is_first(&request->mock.link, &engine->hw_queue)) {
2308c2ecf20Sopenharmony_ci		if (request->mock.delay)
2318c2ecf20Sopenharmony_ci			mod_timer(&engine->hw_delay,
2328c2ecf20Sopenharmony_ci				  jiffies + request->mock.delay);
2338c2ecf20Sopenharmony_ci		else
2348c2ecf20Sopenharmony_ci			advance(request);
2358c2ecf20Sopenharmony_ci	}
2368c2ecf20Sopenharmony_ci	spin_unlock_irqrestore(&engine->hw_lock, flags);
2378c2ecf20Sopenharmony_ci}
2388c2ecf20Sopenharmony_ci
2398c2ecf20Sopenharmony_cistatic void mock_reset_prepare(struct intel_engine_cs *engine)
2408c2ecf20Sopenharmony_ci{
2418c2ecf20Sopenharmony_ci}
2428c2ecf20Sopenharmony_ci
2438c2ecf20Sopenharmony_cistatic void mock_reset_rewind(struct intel_engine_cs *engine, bool stalled)
2448c2ecf20Sopenharmony_ci{
2458c2ecf20Sopenharmony_ci	GEM_BUG_ON(stalled);
2468c2ecf20Sopenharmony_ci}
2478c2ecf20Sopenharmony_ci
2488c2ecf20Sopenharmony_cistatic void mock_reset_cancel(struct intel_engine_cs *engine)
2498c2ecf20Sopenharmony_ci{
2508c2ecf20Sopenharmony_ci	struct i915_request *request;
2518c2ecf20Sopenharmony_ci	unsigned long flags;
2528c2ecf20Sopenharmony_ci
2538c2ecf20Sopenharmony_ci	spin_lock_irqsave(&engine->active.lock, flags);
2548c2ecf20Sopenharmony_ci
2558c2ecf20Sopenharmony_ci	/* Mark all submitted requests as skipped. */
2568c2ecf20Sopenharmony_ci	list_for_each_entry(request, &engine->active.requests, sched.link) {
2578c2ecf20Sopenharmony_ci		i915_request_set_error_once(request, -EIO);
2588c2ecf20Sopenharmony_ci		i915_request_mark_complete(request);
2598c2ecf20Sopenharmony_ci	}
2608c2ecf20Sopenharmony_ci
2618c2ecf20Sopenharmony_ci	spin_unlock_irqrestore(&engine->active.lock, flags);
2628c2ecf20Sopenharmony_ci}
2638c2ecf20Sopenharmony_ci
2648c2ecf20Sopenharmony_cistatic void mock_reset_finish(struct intel_engine_cs *engine)
2658c2ecf20Sopenharmony_ci{
2668c2ecf20Sopenharmony_ci}
2678c2ecf20Sopenharmony_ci
2688c2ecf20Sopenharmony_cistatic void mock_engine_release(struct intel_engine_cs *engine)
2698c2ecf20Sopenharmony_ci{
2708c2ecf20Sopenharmony_ci	struct mock_engine *mock =
2718c2ecf20Sopenharmony_ci		container_of(engine, typeof(*mock), base);
2728c2ecf20Sopenharmony_ci
2738c2ecf20Sopenharmony_ci	GEM_BUG_ON(timer_pending(&mock->hw_delay));
2748c2ecf20Sopenharmony_ci
2758c2ecf20Sopenharmony_ci	intel_breadcrumbs_free(engine->breadcrumbs);
2768c2ecf20Sopenharmony_ci
2778c2ecf20Sopenharmony_ci	intel_context_unpin(engine->kernel_context);
2788c2ecf20Sopenharmony_ci	intel_context_put(engine->kernel_context);
2798c2ecf20Sopenharmony_ci
2808c2ecf20Sopenharmony_ci	intel_engine_fini_retire(engine);
2818c2ecf20Sopenharmony_ci}
2828c2ecf20Sopenharmony_ci
2838c2ecf20Sopenharmony_cistruct intel_engine_cs *mock_engine(struct drm_i915_private *i915,
2848c2ecf20Sopenharmony_ci				    const char *name,
2858c2ecf20Sopenharmony_ci				    int id)
2868c2ecf20Sopenharmony_ci{
2878c2ecf20Sopenharmony_ci	struct mock_engine *engine;
2888c2ecf20Sopenharmony_ci
2898c2ecf20Sopenharmony_ci	GEM_BUG_ON(id >= I915_NUM_ENGINES);
2908c2ecf20Sopenharmony_ci	GEM_BUG_ON(!i915->gt.uncore);
2918c2ecf20Sopenharmony_ci
2928c2ecf20Sopenharmony_ci	engine = kzalloc(sizeof(*engine) + PAGE_SIZE, GFP_KERNEL);
2938c2ecf20Sopenharmony_ci	if (!engine)
2948c2ecf20Sopenharmony_ci		return NULL;
2958c2ecf20Sopenharmony_ci
2968c2ecf20Sopenharmony_ci	/* minimal engine setup for requests */
2978c2ecf20Sopenharmony_ci	engine->base.i915 = i915;
2988c2ecf20Sopenharmony_ci	engine->base.gt = &i915->gt;
2998c2ecf20Sopenharmony_ci	engine->base.uncore = i915->gt.uncore;
3008c2ecf20Sopenharmony_ci	snprintf(engine->base.name, sizeof(engine->base.name), "%s", name);
3018c2ecf20Sopenharmony_ci	engine->base.id = id;
3028c2ecf20Sopenharmony_ci	engine->base.mask = BIT(id);
3038c2ecf20Sopenharmony_ci	engine->base.legacy_idx = INVALID_ENGINE;
3048c2ecf20Sopenharmony_ci	engine->base.instance = id;
3058c2ecf20Sopenharmony_ci	engine->base.status_page.addr = (void *)(engine + 1);
3068c2ecf20Sopenharmony_ci
3078c2ecf20Sopenharmony_ci	engine->base.cops = &mock_context_ops;
3088c2ecf20Sopenharmony_ci	engine->base.request_alloc = mock_request_alloc;
3098c2ecf20Sopenharmony_ci	engine->base.emit_flush = mock_emit_flush;
3108c2ecf20Sopenharmony_ci	engine->base.emit_fini_breadcrumb = mock_emit_breadcrumb;
3118c2ecf20Sopenharmony_ci	engine->base.submit_request = mock_submit_request;
3128c2ecf20Sopenharmony_ci
3138c2ecf20Sopenharmony_ci	engine->base.reset.prepare = mock_reset_prepare;
3148c2ecf20Sopenharmony_ci	engine->base.reset.rewind = mock_reset_rewind;
3158c2ecf20Sopenharmony_ci	engine->base.reset.cancel = mock_reset_cancel;
3168c2ecf20Sopenharmony_ci	engine->base.reset.finish = mock_reset_finish;
3178c2ecf20Sopenharmony_ci
3188c2ecf20Sopenharmony_ci	engine->base.release = mock_engine_release;
3198c2ecf20Sopenharmony_ci
3208c2ecf20Sopenharmony_ci	i915->gt.engine[id] = &engine->base;
3218c2ecf20Sopenharmony_ci	i915->gt.engine_class[0][id] = &engine->base;
3228c2ecf20Sopenharmony_ci
3238c2ecf20Sopenharmony_ci	/* fake hw queue */
3248c2ecf20Sopenharmony_ci	spin_lock_init(&engine->hw_lock);
3258c2ecf20Sopenharmony_ci	timer_setup(&engine->hw_delay, hw_delay_complete, 0);
3268c2ecf20Sopenharmony_ci	INIT_LIST_HEAD(&engine->hw_queue);
3278c2ecf20Sopenharmony_ci
3288c2ecf20Sopenharmony_ci	intel_engine_add_user(&engine->base);
3298c2ecf20Sopenharmony_ci
3308c2ecf20Sopenharmony_ci	return &engine->base;
3318c2ecf20Sopenharmony_ci}
3328c2ecf20Sopenharmony_ci
3338c2ecf20Sopenharmony_ciint mock_engine_init(struct intel_engine_cs *engine)
3348c2ecf20Sopenharmony_ci{
3358c2ecf20Sopenharmony_ci	struct intel_context *ce;
3368c2ecf20Sopenharmony_ci
3378c2ecf20Sopenharmony_ci	intel_engine_init_active(engine, ENGINE_MOCK);
3388c2ecf20Sopenharmony_ci	intel_engine_init_execlists(engine);
3398c2ecf20Sopenharmony_ci	intel_engine_init__pm(engine);
3408c2ecf20Sopenharmony_ci	intel_engine_init_retire(engine);
3418c2ecf20Sopenharmony_ci
3428c2ecf20Sopenharmony_ci	engine->breadcrumbs = intel_breadcrumbs_create(NULL);
3438c2ecf20Sopenharmony_ci	if (!engine->breadcrumbs)
3448c2ecf20Sopenharmony_ci		return -ENOMEM;
3458c2ecf20Sopenharmony_ci
3468c2ecf20Sopenharmony_ci	ce = create_kernel_context(engine);
3478c2ecf20Sopenharmony_ci	if (IS_ERR(ce))
3488c2ecf20Sopenharmony_ci		goto err_breadcrumbs;
3498c2ecf20Sopenharmony_ci
3508c2ecf20Sopenharmony_ci	/* We insist the kernel context is using the status_page */
3518c2ecf20Sopenharmony_ci	engine->status_page.vma = ce->timeline->hwsp_ggtt;
3528c2ecf20Sopenharmony_ci
3538c2ecf20Sopenharmony_ci	engine->kernel_context = ce;
3548c2ecf20Sopenharmony_ci	return 0;
3558c2ecf20Sopenharmony_ci
3568c2ecf20Sopenharmony_cierr_breadcrumbs:
3578c2ecf20Sopenharmony_ci	intel_breadcrumbs_free(engine->breadcrumbs);
3588c2ecf20Sopenharmony_ci	return -ENOMEM;
3598c2ecf20Sopenharmony_ci}
3608c2ecf20Sopenharmony_ci
3618c2ecf20Sopenharmony_civoid mock_engine_flush(struct intel_engine_cs *engine)
3628c2ecf20Sopenharmony_ci{
3638c2ecf20Sopenharmony_ci	struct mock_engine *mock =
3648c2ecf20Sopenharmony_ci		container_of(engine, typeof(*mock), base);
3658c2ecf20Sopenharmony_ci	struct i915_request *request, *rn;
3668c2ecf20Sopenharmony_ci
3678c2ecf20Sopenharmony_ci	del_timer_sync(&mock->hw_delay);
3688c2ecf20Sopenharmony_ci
3698c2ecf20Sopenharmony_ci	spin_lock_irq(&mock->hw_lock);
3708c2ecf20Sopenharmony_ci	list_for_each_entry_safe(request, rn, &mock->hw_queue, mock.link)
3718c2ecf20Sopenharmony_ci		advance(request);
3728c2ecf20Sopenharmony_ci	spin_unlock_irq(&mock->hw_lock);
3738c2ecf20Sopenharmony_ci}
3748c2ecf20Sopenharmony_ci
3758c2ecf20Sopenharmony_civoid mock_engine_reset(struct intel_engine_cs *engine)
3768c2ecf20Sopenharmony_ci{
3778c2ecf20Sopenharmony_ci}
378