From: "Huang, Sean Z" <[email protected]>

The HW will generate a teardown interrupt when session termination is
required, which requires i915 to submit a terminating batch. Once the HW
is done with the termination it will generate another interrupt, at
which point it is safe to re-create the session.

Signed-off-by: Huang, Sean Z <[email protected]>
Signed-off-by: Daniele Ceraolo Spurio <[email protected]>
---
 drivers/gpu/drm/i915/Makefile                |   1 +
 drivers/gpu/drm/i915/gt/intel_gt_irq.c       |   7 +
 drivers/gpu/drm/i915/i915_reg.h              |   1 +
 drivers/gpu/drm/i915/pxp/intel_pxp.c         |   7 +
 drivers/gpu/drm/i915/pxp/intel_pxp.h         |   9 ++
 drivers/gpu/drm/i915/pxp/intel_pxp_irq.c     | 145 +++++++++++++++++++
 drivers/gpu/drm/i915/pxp/intel_pxp_irq.h     |  33 +++++
 drivers/gpu/drm/i915/pxp/intel_pxp_session.c |   1 +
 drivers/gpu/drm/i915/pxp/intel_pxp_tee.c     |   9 +-
 drivers/gpu/drm/i915/pxp/intel_pxp_types.h   |   7 +
 10 files changed, 219 insertions(+), 1 deletion(-)
 create mode 100644 drivers/gpu/drm/i915/pxp/intel_pxp_irq.c
 create mode 100644 drivers/gpu/drm/i915/pxp/intel_pxp_irq.h

diff --git a/drivers/gpu/drm/i915/Makefile b/drivers/gpu/drm/i915/Makefile
index 9698fec810ae..f45275c0034e 100644
--- a/drivers/gpu/drm/i915/Makefile
+++ b/drivers/gpu/drm/i915/Makefile
@@ -272,6 +272,7 @@ i915-y += i915_perf.o
 i915-$(CONFIG_DRM_I915_PXP) += \
        pxp/intel_pxp.o \
        pxp/intel_pxp_cmd.o \
+       pxp/intel_pxp_irq.o \
        pxp/intel_pxp_session.o \
        pxp/intel_pxp_tee.o
 
diff --git a/drivers/gpu/drm/i915/gt/intel_gt_irq.c 
b/drivers/gpu/drm/i915/gt/intel_gt_irq.c
index d29126c458ba..0d3585efe2b8 100644
--- a/drivers/gpu/drm/i915/gt/intel_gt_irq.c
+++ b/drivers/gpu/drm/i915/gt/intel_gt_irq.c
@@ -13,6 +13,7 @@
 #include "intel_lrc_reg.h"
 #include "intel_uncore.h"
 #include "intel_rps.h"
+#include "pxp/intel_pxp_irq.h"
 
 static void guc_irq_handler(struct intel_guc *guc, u16 iir)
 {
@@ -64,6 +65,9 @@ gen11_other_irq_handler(struct intel_gt *gt, const u8 
instance,
        if (instance == OTHER_GTPM_INSTANCE)
                return gen11_rps_irq_handler(&gt->rps, iir);
 
+       if (instance == OTHER_KCR_INSTANCE)
+               return intel_pxp_irq_handler(&gt->pxp, iir);
+
        WARN_ONCE(1, "unhandled other interrupt instance=0x%x, iir=0x%x\n",
                  instance, iir);
 }
@@ -190,6 +194,9 @@ void gen11_gt_irq_reset(struct intel_gt *gt)
        intel_uncore_write(uncore, GEN11_GPM_WGBOXPERF_INTR_MASK,  ~0);
        intel_uncore_write(uncore, GEN11_GUC_SG_INTR_ENABLE, 0);
        intel_uncore_write(uncore, GEN11_GUC_SG_INTR_MASK,  ~0);
+
+       intel_uncore_write(uncore, GEN11_CRYPTO_RSVD_INTR_ENABLE, 0);
+       intel_uncore_write(uncore, GEN11_CRYPTO_RSVD_INTR_MASK,  ~0);
 }
 
 void gen11_gt_irq_postinstall(struct intel_gt *gt)
diff --git a/drivers/gpu/drm/i915/i915_reg.h b/drivers/gpu/drm/i915/i915_reg.h
index 224ad897af34..0dcc04c7d2d8 100644
--- a/drivers/gpu/drm/i915/i915_reg.h
+++ b/drivers/gpu/drm/i915/i915_reg.h
@@ -7958,6 +7958,7 @@ enum {
 /* irq instances for OTHER_CLASS */
 #define OTHER_GUC_INSTANCE     0
 #define OTHER_GTPM_INSTANCE    1
+#define OTHER_KCR_INSTANCE     4
 
 #define GEN11_INTR_IDENTITY_REG(x)     _MMIO(0x190060 + ((x) * 4))
 
diff --git a/drivers/gpu/drm/i915/pxp/intel_pxp.c 
b/drivers/gpu/drm/i915/pxp/intel_pxp.c
index 916187a0e566..f6317ce645c7 100644
--- a/drivers/gpu/drm/i915/pxp/intel_pxp.c
+++ b/drivers/gpu/drm/i915/pxp/intel_pxp.c
@@ -2,7 +2,9 @@
 /*
  * Copyright(c) 2020 Intel Corporation.
  */
+#include <linux/workqueue.h>
 #include "intel_pxp.h"
+#include "intel_pxp_irq.h"
 #include "intel_pxp_tee.h"
 #include "gt/intel_context.h"
 #include "i915_drv.h"
@@ -70,6 +72,9 @@ void intel_pxp_init(struct intel_pxp *pxp)
        if (ret)
                goto out_kcr;
 
+       intel_pxp_irq_init(pxp);
+       intel_pxp_irq_enable(pxp);
+
        ret = intel_pxp_tee_component_init(pxp);
        if (ret)
                goto out_context;
@@ -91,6 +96,8 @@ void intel_pxp_fini(struct intel_pxp *pxp)
        if (!intel_pxp_is_enabled(pxp))
                return;
 
+       intel_pxp_irq_disable(pxp);
+
        intel_pxp_tee_component_fini(pxp);
 
        destroy_vcs_context(pxp);
diff --git a/drivers/gpu/drm/i915/pxp/intel_pxp.h 
b/drivers/gpu/drm/i915/pxp/intel_pxp.h
index e2acd06402cd..dce7b1a23850 100644
--- a/drivers/gpu/drm/i915/pxp/intel_pxp.h
+++ b/drivers/gpu/drm/i915/pxp/intel_pxp.h
@@ -9,6 +9,15 @@
 #include "gt/intel_gt_types.h"
 #include "intel_pxp_types.h"
 
+#define GEN12_DISPLAY_PXP_STATE_TERMINATED_INTERRUPT BIT(1)
+#define GEN12_DISPLAY_APP_TERMINATED_PER_FW_REQ_INTERRUPT BIT(2)
+#define GEN12_DISPLAY_STATE_RESET_COMPLETE_INTERRUPT BIT(3)
+
+#define GEN12_PXP_INTERRUPTS \
+       (GEN12_DISPLAY_PXP_STATE_TERMINATED_INTERRUPT | \
+        GEN12_DISPLAY_APP_TERMINATED_PER_FW_REQ_INTERRUPT | \
+        GEN12_DISPLAY_STATE_RESET_COMPLETE_INTERRUPT)
+
 static inline struct intel_gt *pxp_to_gt(struct intel_pxp *pxp)
 {
        return container_of(pxp, struct intel_gt, pxp);
diff --git a/drivers/gpu/drm/i915/pxp/intel_pxp_irq.c 
b/drivers/gpu/drm/i915/pxp/intel_pxp_irq.c
new file mode 100644
index 000000000000..5ca0218a27d8
--- /dev/null
+++ b/drivers/gpu/drm/i915/pxp/intel_pxp_irq.c
@@ -0,0 +1,145 @@
+// SPDX-License-Identifier: MIT
+/*
+ * Copyright(c) 2020 Intel Corporation.
+ */
+#include <linux/workqueue.h>
+#include "intel_pxp.h"
+#include "intel_pxp_irq.h"
+#include "intel_pxp_session.h"
+#include "gt/intel_gt_irq.h"
+#include "i915_irq.h"
+#include "i915_reg.h"
+
+static int pxp_terminate(struct intel_pxp *pxp)
+{
+       int ret = 0;
+
+       mutex_lock(&pxp->mutex);
+
+       pxp->global_state_attacked = true;
+
+       ret = intel_pxp_arb_terminate_session_with_global_terminate(pxp);
+
+       mutex_unlock(&pxp->mutex);
+
+       return ret;
+}
+
+static int pxp_terminate_complete(struct intel_pxp *pxp)
+{
+       int ret = 0;
+
+       mutex_lock(&pxp->mutex);
+
+       pxp->termination_in_progress = false;
+
+       if (pxp->global_state_attacked) {
+               pxp->global_state_attacked = false;
+
+               /* Re-create the arb session after teardown handle complete */
+               ret = intel_pxp_create_arb_session(pxp);
+       }
+
+       mutex_unlock(&pxp->mutex);
+       return ret;
+}
+
+static void intel_pxp_irq_work(struct work_struct *work)
+{
+       struct intel_pxp *pxp = container_of(work, typeof(*pxp), irq_work);
+       struct intel_gt *gt = pxp_to_gt(pxp);
+       u32 events = 0;
+
+       spin_lock_irq(&gt->irq_lock);
+       events = fetch_and_zero(&pxp->current_events);
+       spin_unlock_irq(&gt->irq_lock);
+
+       if (!events)
+               return;
+
+       if (events & (GEN12_DISPLAY_PXP_STATE_TERMINATED_INTERRUPT |
+                     GEN12_DISPLAY_APP_TERMINATED_PER_FW_REQ_INTERRUPT))
+               pxp_terminate(pxp);
+
+       if (events & GEN12_DISPLAY_STATE_RESET_COMPLETE_INTERRUPT)
+               pxp_terminate_complete(pxp);
+
+       /*
+        * we expect the terminate complete to arrive quickly after emitting
+        * the terminate, so check back on it
+        */
+       if (pxp->irq_enabled)
+               queue_work(system_unbound_wq, &pxp->irq_work);
+}
+
+/**
+ * intel_pxp_irq_handler - Handles PXP interrupts.
+ * @pxp: pointer to pxp struct
+ * @iir: interrupt vector
+ */
+void intel_pxp_irq_handler(struct intel_pxp *pxp, u16 iir)
+{
+       struct intel_gt *gt = pxp_to_gt(pxp);
+
+       if (GEM_WARN_ON(!intel_pxp_is_enabled(pxp)))
+               return;
+
+       lockdep_assert_held(&gt->irq_lock);
+
+       if (unlikely(!iir))
+               return;
+
+       pxp->current_events |= iir;
+       queue_work(system_unbound_wq, &pxp->irq_work);
+}
+
+static inline void __pxp_set_interrupts(struct intel_gt *gt, u32 interrupts)
+{
+       struct intel_uncore *uncore = gt->uncore;
+       const u32 mask = interrupts << 16;
+
+       intel_uncore_write(uncore, GEN11_CRYPTO_RSVD_INTR_ENABLE, mask);
+       intel_uncore_write(uncore, GEN11_CRYPTO_RSVD_INTR_MASK,  ~mask);
+}
+
+static inline void pxp_irq_reset(struct intel_gt *gt)
+{
+       spin_lock_irq(&gt->irq_lock);
+       gen11_gt_reset_one_iir(gt, 0, GEN11_KCR);
+       spin_unlock_irq(&gt->irq_lock);
+}
+
+void intel_pxp_irq_init(struct intel_pxp *pxp)
+{
+       INIT_WORK(&pxp->irq_work, intel_pxp_irq_work);
+}
+
+void intel_pxp_irq_enable(struct intel_pxp *pxp)
+{
+       struct intel_gt *gt = pxp_to_gt(pxp);
+
+       spin_lock_irq(&gt->irq_lock);
+       if (!pxp->irq_enabled) {
+               WARN_ON_ONCE(gen11_gt_reset_one_iir(gt, 0, GEN11_KCR));
+               __pxp_set_interrupts(gt, GEN12_PXP_INTERRUPTS);
+               pxp->irq_enabled = true;
+       }
+       spin_unlock_irq(&gt->irq_lock);
+}
+
+void intel_pxp_irq_disable(struct intel_pxp *pxp)
+{
+       struct intel_gt *gt = pxp_to_gt(pxp);
+
+       spin_lock_irq(&gt->irq_lock);
+
+       pxp->irq_enabled = false;
+       __pxp_set_interrupts(gt, 0);
+
+       spin_unlock_irq(&gt->irq_lock);
+       intel_synchronize_irq(gt->i915);
+
+       pxp_irq_reset(gt);
+
+       flush_work(&pxp->irq_work);
+}
diff --git a/drivers/gpu/drm/i915/pxp/intel_pxp_irq.h 
b/drivers/gpu/drm/i915/pxp/intel_pxp_irq.h
new file mode 100644
index 000000000000..7a875831636d
--- /dev/null
+++ b/drivers/gpu/drm/i915/pxp/intel_pxp_irq.h
@@ -0,0 +1,33 @@
+/* SPDX-License-Identifier: MIT */
+/*
+ * Copyright(c) 2020, Intel Corporation. All rights reserved.
+ */
+
+#ifndef __INTEL_PXP_IRQ_H__
+#define __INTEL_PXP_IRQ_H__
+
+#include <linux/types.h>
+
+struct intel_pxp;
+
+#ifdef CONFIG_DRM_I915_PXP
+void intel_pxp_irq_init(struct intel_pxp *pxp);
+void intel_pxp_irq_enable(struct intel_pxp *pxp);
+void intel_pxp_irq_disable(struct intel_pxp *pxp);
+void intel_pxp_irq_handler(struct intel_pxp *pxp, u16 iir);
+#else
+void intel_pxp_irq_init(struct intel_pxp *pxp)
+{
+}
+void intel_pxp_irq_enable(struct intel_pxp *pxp)
+{
+}
+void intel_pxp_irq_disable(struct intel_pxp *pxp)
+{
+}
+static inline void intel_pxp_irq_handler(struct intel_pxp *pxp, u16 iir)
+{
+}
+#endif
+
+#endif /* __INTEL_PXP_IRQ_H__ */
diff --git a/drivers/gpu/drm/i915/pxp/intel_pxp_session.c 
b/drivers/gpu/drm/i915/pxp/intel_pxp_session.c
index fe733cc69f27..d0cb35a8b991 100644
--- a/drivers/gpu/drm/i915/pxp/intel_pxp_session.c
+++ b/drivers/gpu/drm/i915/pxp/intel_pxp_session.c
@@ -117,6 +117,7 @@ int 
intel_pxp_arb_terminate_session_with_global_terminate(struct intel_pxp *pxp)
        }
 
        intel_uncore_write(gt->uncore, PXP_GLOBAL_TERMINATE, 1);
+       pxp->termination_in_progress = true;
 
        return ret;
 }
diff --git a/drivers/gpu/drm/i915/pxp/intel_pxp_tee.c 
b/drivers/gpu/drm/i915/pxp/intel_pxp_tee.c
index dd7161f176f1..f52150aca02b 100644
--- a/drivers/gpu/drm/i915/pxp/intel_pxp_tee.c
+++ b/drivers/gpu/drm/i915/pxp/intel_pxp_tee.c
@@ -95,9 +95,16 @@ static int i915_pxp_tee_component_bind(struct device 
*i915_kdev,
        mutex_lock(&pxp->mutex);
 
        /* Create arb session only if tee is ready, during system boot or 
sleep/resume */
-       if (intel_pxp_arb_session_is_in_play(pxp))
+       if (intel_pxp_arb_session_is_in_play(pxp)) {
                ret = 
intel_pxp_arb_terminate_session_with_global_terminate(pxp);
 
+               if (!ret) {
+                       mutex_unlock(&pxp->mutex);
+                       ret = wait_for(!pxp->termination_in_progress, 10);
+                       mutex_lock(&pxp->mutex);
+               }
+       }
+
        if (!ret)
                ret = intel_pxp_create_arb_session(pxp);
 
diff --git a/drivers/gpu/drm/i915/pxp/intel_pxp_types.h 
b/drivers/gpu/drm/i915/pxp/intel_pxp_types.h
index 30a6013b5d95..51915f7b1521 100644
--- a/drivers/gpu/drm/i915/pxp/intel_pxp_types.h
+++ b/drivers/gpu/drm/i915/pxp/intel_pxp_types.h
@@ -8,6 +8,7 @@
 
 #include <linux/types.h>
 #include <linux/mutex.h>
+#include <linux/workqueue.h>
 
 struct intel_context;
 
@@ -16,6 +17,12 @@ struct intel_pxp {
 
        struct mutex mutex;
        bool arb_is_in_play;
+       bool termination_in_progress;
+       bool global_state_attacked;
+
+       struct work_struct irq_work;
+       bool irq_enabled;
+       u32 current_events; /* protected with gt->irq_lock */
 };
 
 #endif /* __INTEL_PXP_TYPES_H__ */
-- 
2.29.2

_______________________________________________
Intel-gfx mailing list
[email protected]
https://lists.freedesktop.org/mailman/listinfo/intel-gfx

Reply via email to