root/drivers/gpu/drm/i915/gt/uc/intel_gsc_uc_heci_cmd_submit.c
// SPDX-License-Identifier: MIT
/*
 * Copyright © 2023 Intel Corporation
 */

#include <drm/drm_print.h>

#include "gt/intel_context.h"
#include "gt/intel_engine_pm.h"
#include "gt/intel_gpu_commands.h"
#include "gt/intel_gt.h"
#include "gt/intel_ring.h"

#include "i915_wait_util.h"
#include "intel_gsc_uc_heci_cmd_submit.h"

struct gsc_heci_pkt {
        u64 addr_in;
        u32 size_in;
        u64 addr_out;
        u32 size_out;
};

static int emit_gsc_heci_pkt(struct i915_request *rq, struct gsc_heci_pkt *pkt)
{
        u32 *cs;

        cs = intel_ring_begin(rq, 8);
        if (IS_ERR(cs))
                return PTR_ERR(cs);

        *cs++ = GSC_HECI_CMD_PKT;
        *cs++ = lower_32_bits(pkt->addr_in);
        *cs++ = upper_32_bits(pkt->addr_in);
        *cs++ = pkt->size_in;
        *cs++ = lower_32_bits(pkt->addr_out);
        *cs++ = upper_32_bits(pkt->addr_out);
        *cs++ = pkt->size_out;
        *cs++ = 0;

        intel_ring_advance(rq, cs);

        return 0;
}

int intel_gsc_uc_heci_cmd_submit_packet(struct intel_gsc_uc *gsc, u64 addr_in,
                                        u32 size_in, u64 addr_out,
                                        u32 size_out)
{
        struct intel_context *ce = gsc->ce;
        struct i915_request *rq;
        struct gsc_heci_pkt pkt = {
        .addr_in = addr_in,
        .size_in = size_in,
        .addr_out = addr_out,
        .size_out = size_out
        };
        int err;

        if (!ce)
                return -ENODEV;

        rq = i915_request_create(ce);
        if (IS_ERR(rq))
                return PTR_ERR(rq);

        if (ce->engine->emit_init_breadcrumb) {
                err = ce->engine->emit_init_breadcrumb(rq);
                if (err)
                        goto out_rq;
        }

        err = emit_gsc_heci_pkt(rq, &pkt);

        if (err)
                goto out_rq;

        err = ce->engine->emit_flush(rq, 0);

out_rq:
        i915_request_get(rq);

        if (unlikely(err))
                i915_request_set_error_once(rq, err);

        i915_request_add(rq);

        if (!err) {
                /*
                 * Start timeout for i915_request_wait only after considering one possible
                 * pending GSC-HECI submission cycle on the other (non-privileged) path.
                 */
                if (wait_for(i915_request_started(rq), GSC_HECI_REPLY_LATENCY_MS))
                        drm_dbg(&gsc_uc_to_gt(gsc)->i915->drm,
                                "Delay in gsc-heci-priv submission to gsccs-hw");
                if (i915_request_wait(rq, 0, msecs_to_jiffies(GSC_HECI_REPLY_LATENCY_MS)) < 0)
                        err = -ETIME;
        }

        i915_request_put(rq);

        if (err)
                drm_err(&gsc_uc_to_gt(gsc)->i915->drm,
                        "Request submission for GSC heci cmd failed (%d)\n",
                        err);

        return err;
}

void intel_gsc_uc_heci_cmd_emit_mtl_header(struct intel_gsc_mtl_header *header,
                                           u8 heci_client_id, u32 message_size,
                                           u64 host_session_id)
{
        host_session_id &= ~HOST_SESSION_MASK;
        if (host_session_id && heci_client_id == HECI_MEADDRESS_PXP)
                host_session_id |= HOST_SESSION_PXP_SINGLE;

        header->validity_marker = GSC_HECI_VALIDITY_MARKER;
        header->heci_client_id = heci_client_id;
        header->host_session_handle = host_session_id;
        header->header_version = MTL_GSC_HEADER_VERSION;
        header->message_size = message_size;
}

static void
emit_gsc_heci_pkt_nonpriv(u32 *cmd, struct intel_gsc_heci_non_priv_pkt *pkt)
{
        *cmd++ = GSC_HECI_CMD_PKT;
        *cmd++ = lower_32_bits(pkt->addr_in);
        *cmd++ = upper_32_bits(pkt->addr_in);
        *cmd++ = pkt->size_in;
        *cmd++ = lower_32_bits(pkt->addr_out);
        *cmd++ = upper_32_bits(pkt->addr_out);
        *cmd++ = pkt->size_out;
        *cmd++ = 0;
        *cmd++ = MI_BATCH_BUFFER_END;
}

int
intel_gsc_uc_heci_cmd_submit_nonpriv(struct intel_gsc_uc *gsc,
                                     struct intel_context *ce,
                                     struct intel_gsc_heci_non_priv_pkt *pkt,
                                     u32 *cmd, int timeout_ms)
{
        struct intel_engine_cs *engine;
        struct i915_gem_ww_ctx ww;
        struct i915_request *rq;
        int err, trials = 0;

        i915_gem_ww_ctx_init(&ww, false);
retry:
        err = i915_gem_object_lock(pkt->bb_vma->obj, &ww);
        if (err)
                goto out_ww;
        err = i915_gem_object_lock(pkt->heci_pkt_vma->obj, &ww);
        if (err)
                goto out_ww;
        err = intel_context_pin_ww(ce, &ww);
        if (err)
                goto out_ww;

        rq = i915_request_create(ce);
        if (IS_ERR(rq)) {
                err = PTR_ERR(rq);
                goto out_unpin_ce;
        }

        emit_gsc_heci_pkt_nonpriv(cmd, pkt);

        err = i915_vma_move_to_active(pkt->bb_vma, rq, 0);
        if (err)
                goto out_rq;
        err = i915_vma_move_to_active(pkt->heci_pkt_vma, rq, EXEC_OBJECT_WRITE);
        if (err)
                goto out_rq;

        engine = rq->context->engine;
        if (engine->emit_init_breadcrumb) {
                err = engine->emit_init_breadcrumb(rq);
                if (err)
                        goto out_rq;
        }

        err = engine->emit_bb_start(rq, i915_vma_offset(pkt->bb_vma), PAGE_SIZE, 0);
        if (err)
                goto out_rq;

        err = ce->engine->emit_flush(rq, 0);
        if (err)
                drm_err(&gsc_uc_to_gt(gsc)->i915->drm,
                        "Failed emit-flush for gsc-heci-non-priv-pkterr=%d\n", err);

out_rq:
        i915_request_get(rq);

        if (unlikely(err))
                i915_request_set_error_once(rq, err);

        i915_request_add(rq);

        if (!err) {
                /*
                 * Start timeout for i915_request_wait only after considering one possible
                 * pending GSC-HECI submission cycle on the other (privileged) path.
                 */
                if (wait_for(i915_request_started(rq), GSC_HECI_REPLY_LATENCY_MS))
                        drm_dbg(&gsc_uc_to_gt(gsc)->i915->drm,
                                "Delay in gsc-heci-non-priv submission to gsccs-hw");
                if (i915_request_wait(rq, I915_WAIT_INTERRUPTIBLE,
                                      msecs_to_jiffies(timeout_ms)) < 0)
                        err = -ETIME;
        }

        i915_request_put(rq);

out_unpin_ce:
        intel_context_unpin(ce);
out_ww:
        if (err == -EDEADLK) {
                err = i915_gem_ww_ctx_backoff(&ww);
                if (!err) {
                        if (++trials < 10)
                                goto retry;
                        else
                                err = -EAGAIN;
                }
        }
        i915_gem_ww_ctx_fini(&ww);

        return err;
}