#include "i915_drv.h"
 #include "intel_gt.h"
 #include "intel_gt_clock_utils.h"
+#include "intel_gt_pm.h"
 #include "intel_llc.h"
 #include "intel_rc6.h"
 #include "intel_rps.h"
 
        seq_printf(m, "RPS enabled? %s\n", yesno(intel_rps_is_enabled(rps)));
        seq_printf(m, "RPS active? %s\n", yesno(intel_rps_is_active(rps)));
-       seq_printf(m, "GPU busy? %s\n", yesno(gt->awake));
+       seq_printf(m, "GPU busy? %s, %llums\n",
+                  yesno(gt->awake),
+                  ktime_to_ms(intel_gt_get_awake_time(gt)));
        seq_printf(m, "Boosts outstanding? %d\n",
                   atomic_read(&rps->num_waiters));
        seq_printf(m, "Interactive? %d\n", READ_ONCE(rps->power.interactive));
 
        intel_gt_pm_put(gt);
 }
 
+static void runtime_begin(struct intel_gt *gt)
+{
+       local_irq_disable();
+       write_seqcount_begin(>->stats.lock);
+       gt->stats.start = ktime_get();
+       gt->stats.active = true;
+       write_seqcount_end(>->stats.lock);
+       local_irq_enable();
+}
+
+static void runtime_end(struct intel_gt *gt)
+{
+       local_irq_disable();
+       write_seqcount_begin(>->stats.lock);
+       gt->stats.active = false;
+       gt->stats.total =
+               ktime_add(gt->stats.total,
+                         ktime_sub(ktime_get(), gt->stats.start));
+       write_seqcount_end(>->stats.lock);
+       local_irq_enable();
+}
+
 static int __gt_unpark(struct intel_wakeref *wf)
 {
        struct intel_gt *gt = container_of(wf, typeof(*gt), wakeref);
        i915_pmu_gt_unparked(i915);
 
        intel_gt_unpark_requests(gt);
+       runtime_begin(gt);
 
        return 0;
 }
 
        GT_TRACE(gt, "\n");
 
+       runtime_end(gt);
        intel_gt_park_requests(gt);
 
        i915_vma_parked(gt);
 void intel_gt_pm_init_early(struct intel_gt *gt)
 {
        intel_wakeref_init(>->wakeref, gt->uncore->rpm, &wf_ops);
+       seqcount_mutex_init(>->stats.lock, >->wakeref.mutex);
 }
 
 void intel_gt_pm_init(struct intel_gt *gt)
        return intel_uc_runtime_resume(>->uc);
 }
 
+static ktime_t __intel_gt_get_awake_time(const struct intel_gt *gt)
+{
+       ktime_t total = gt->stats.total;
+
+       if (gt->stats.active)
+               total = ktime_add(total,
+                                 ktime_sub(ktime_get(), gt->stats.start));
+
+       return total;
+}
+
+ktime_t intel_gt_get_awake_time(const struct intel_gt *gt)
+{
+       unsigned int seq;
+       ktime_t total;
+
+       do {
+               seq = read_seqcount_begin(>->stats.lock);
+               total = __intel_gt_get_awake_time(gt);
+       } while (read_seqcount_retry(>->stats.lock, seq));
+
+       return total;
+}
+
 #if IS_ENABLED(CONFIG_DRM_I915_SELFTEST)
 #include "selftest_gt_pm.c"
 #endif
 
 void intel_gt_runtime_suspend(struct intel_gt *gt);
 int intel_gt_runtime_resume(struct intel_gt *gt);
 
+ktime_t intel_gt_get_awake_time(const struct intel_gt *gt);
+
 static inline bool is_mock_gt(const struct intel_gt *gt)
 {
        return I915_SELFTEST_ONLY(gt->awake == -ENODEV);
 
 
        u32 pm_guc_events;
 
+       struct {
+               bool active;
+
+               /**
+                * @lock: Lock protecting the below fields.
+                */
+               seqcount_mutex_t lock;
+
+               /**
+                * @total: Total time this engine was busy.
+                *
+                * Accumulated time not counting the most recent block in cases
+                * where engine is currently busy (active > 0).
+                */
+               ktime_t total;
+
+               /**
+                * @start: Timestamp of the last idle to active transition.
+                *
+                * Idle is defined as active == 0, active is active > 0.
+                */
+               ktime_t start;
+       } stats;
+
        struct intel_engine_cs *engine[I915_NUM_ENGINES];
        struct intel_engine_cs *engine_class[MAX_ENGINE_CLASS + 1]
                                            [MAX_ENGINE_INSTANCE + 1];
 
 
        wakeref = intel_runtime_pm_get(&i915->runtime_pm);
 
-       seq_printf(m, "GT awake? %s [%d]\n",
+       seq_printf(m, "GT awake? %s [%d], %llums\n",
                   yesno(i915->gt.awake),
-                  atomic_read(&i915->gt.wakeref.count));
+                  atomic_read(&i915->gt.wakeref.count),
+                  ktime_to_ms(intel_gt_get_awake_time(&i915->gt)));
        seq_printf(m, "CS timestamp frequency: %u Hz\n",
                   RUNTIME_INFO(i915)->cs_timestamp_frequency_hz);
 
 
                if (!HAS_RC6(i915))
                        return -ENODEV;
                break;
+       case I915_PMU_SOFTWARE_GT_AWAKE_TIME:
+               break;
        default:
                return -ENOENT;
        }
                case I915_PMU_RC6_RESIDENCY:
                        val = get_rc6(&i915->gt);
                        break;
+               case I915_PMU_SOFTWARE_GT_AWAKE_TIME:
+                       val = ktime_to_ns(intel_gt_get_awake_time(&i915->gt));
+                       break;
                }
        }
 
                __event(I915_PMU_REQUESTED_FREQUENCY, "requested-frequency", "M"),
                __event(I915_PMU_INTERRUPTS, "interrupts", NULL),
                __event(I915_PMU_RC6_RESIDENCY, "rc6-residency", "ns"),
+               __event(I915_PMU_SOFTWARE_GT_AWAKE_TIME, "software-gt-awake-time", "ns"),
        };
        static const struct {
                enum drm_i915_pmu_engine_sample sample;
 
 #define I915_PMU_REQUESTED_FREQUENCY   __I915_PMU_OTHER(1)
 #define I915_PMU_INTERRUPTS            __I915_PMU_OTHER(2)
 #define I915_PMU_RC6_RESIDENCY         __I915_PMU_OTHER(3)
+#define I915_PMU_SOFTWARE_GT_AWAKE_TIME        __I915_PMU_OTHER(4)
 
 #define I915_PMU_LAST /* Deprecated - do not use */ I915_PMU_RC6_RESIDENCY