cachepc-linux

Fork of AMDESE/linux with modifications for CachePC side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-linux
Log | Files | Refs | README | LICENSE | sfeed.txt

intel_pxp_irq.c (2649B)


      1// SPDX-License-Identifier: MIT
      2/*
      3 * Copyright(c) 2020 Intel Corporation.
      4 */
      5#include <linux/workqueue.h>
      6#include "intel_pxp.h"
      7#include "intel_pxp_irq.h"
      8#include "intel_pxp_session.h"
      9#include "gt/intel_gt_irq.h"
     10#include "gt/intel_gt_regs.h"
     11#include "gt/intel_gt_types.h"
     12#include "i915_irq.h"
     13#include "i915_reg.h"
     14#include "intel_runtime_pm.h"
     15
     16/**
     17 * intel_pxp_irq_handler - Handles PXP interrupts.
     18 * @pxp: pointer to pxp struct
     19 * @iir: interrupt vector
     20 */
     21void intel_pxp_irq_handler(struct intel_pxp *pxp, u16 iir)
     22{
     23	struct intel_gt *gt = pxp_to_gt(pxp);
     24
     25	if (GEM_WARN_ON(!intel_pxp_is_enabled(pxp)))
     26		return;
     27
     28	lockdep_assert_held(&gt->irq_lock);
     29
     30	if (unlikely(!iir))
     31		return;
     32
     33	if (iir & (GEN12_DISPLAY_PXP_STATE_TERMINATED_INTERRUPT |
     34		   GEN12_DISPLAY_APP_TERMINATED_PER_FW_REQ_INTERRUPT)) {
     35		/* immediately mark PXP as inactive on termination */
     36		intel_pxp_mark_termination_in_progress(pxp);
     37		pxp->session_events |= PXP_TERMINATION_REQUEST | PXP_INVAL_REQUIRED;
     38	}
     39
     40	if (iir & GEN12_DISPLAY_STATE_RESET_COMPLETE_INTERRUPT)
     41		pxp->session_events |= PXP_TERMINATION_COMPLETE;
     42
     43	if (pxp->session_events)
     44		queue_work(system_unbound_wq, &pxp->session_work);
     45}
     46
     47static inline void __pxp_set_interrupts(struct intel_gt *gt, u32 interrupts)
     48{
     49	struct intel_uncore *uncore = gt->uncore;
     50	const u32 mask = interrupts << 16;
     51
     52	intel_uncore_write(uncore, GEN11_CRYPTO_RSVD_INTR_ENABLE, mask);
     53	intel_uncore_write(uncore, GEN11_CRYPTO_RSVD_INTR_MASK,  ~mask);
     54}
     55
     56static inline void pxp_irq_reset(struct intel_gt *gt)
     57{
     58	spin_lock_irq(&gt->irq_lock);
     59	gen11_gt_reset_one_iir(gt, 0, GEN11_KCR);
     60	spin_unlock_irq(&gt->irq_lock);
     61}
     62
     63void intel_pxp_irq_enable(struct intel_pxp *pxp)
     64{
     65	struct intel_gt *gt = pxp_to_gt(pxp);
     66
     67	spin_lock_irq(&gt->irq_lock);
     68
     69	if (!pxp->irq_enabled)
     70		WARN_ON_ONCE(gen11_gt_reset_one_iir(gt, 0, GEN11_KCR));
     71
     72	__pxp_set_interrupts(gt, GEN12_PXP_INTERRUPTS);
     73	pxp->irq_enabled = true;
     74
     75	spin_unlock_irq(&gt->irq_lock);
     76}
     77
     78void intel_pxp_irq_disable(struct intel_pxp *pxp)
     79{
     80	struct intel_gt *gt = pxp_to_gt(pxp);
     81
     82	/*
     83	 * We always need to submit a global termination when we re-enable the
     84	 * interrupts, so there is no need to make sure that the session state
     85	 * makes sense at the end of this function. Just make sure this is not
     86	 * called in a path were the driver consider the session as valid and
     87	 * doesn't call a termination on restart.
     88	 */
     89	GEM_WARN_ON(intel_pxp_is_active(pxp));
     90
     91	spin_lock_irq(&gt->irq_lock);
     92
     93	pxp->irq_enabled = false;
     94	__pxp_set_interrupts(gt, 0);
     95
     96	spin_unlock_irq(&gt->irq_lock);
     97	intel_synchronize_irq(gt->i915);
     98
     99	pxp_irq_reset(gt);
    100
    101	flush_work(&pxp->session_work);
    102}