return &q->gt->uc.guc;
 }
 
-static void xe_devcoredump_deferred_snap_work(struct work_struct *work)
-{
-       struct xe_devcoredump_snapshot *ss = container_of(work, typeof(*ss), work);
-
-       /* keep going if fw fails as we still want to save the memory and SW data */
-       if (xe_force_wake_get(gt_to_fw(ss->gt), XE_FORCEWAKE_ALL))
-               xe_gt_info(ss->gt, "failed to get forcewake for coredump capture\n");
-       xe_vm_snapshot_capture_delayed(ss->vm);
-       xe_guc_exec_queue_snapshot_capture_delayed(ss->ge);
-       xe_force_wake_put(gt_to_fw(ss->gt), XE_FORCEWAKE_ALL);
-}
-
-static ssize_t xe_devcoredump_read(char *buffer, loff_t offset,
-                                  size_t count, void *data, size_t datalen)
+static ssize_t __xe_devcoredump_read(char *buffer, size_t count,
+                                    struct xe_devcoredump *coredump)
 {
-       struct xe_devcoredump *coredump = data;
        struct xe_device *xe;
        struct xe_devcoredump_snapshot *ss;
        struct drm_printer p;
        struct timespec64 ts;
        int i;
 
-       if (!coredump)
-               return -ENODEV;
-
        xe = coredump_to_xe(coredump);
        ss = &coredump->snapshot;
 
-       /* Ensure delayed work is captured before continuing */
-       flush_work(&ss->work);
-
        iter.data = buffer;
-       iter.offset = 0;
-       iter.start = offset;
+       iter.start = 0;
        iter.remain = count;
 
        p = drm_coredump_printer(&iter);
        return count - iter.remain;
 }
 
+static void xe_devcoredump_snapshot_free(struct xe_devcoredump_snapshot *ss)
+{
+       int i;
+
+       xe_guc_ct_snapshot_free(ss->ct);
+       ss->ct = NULL;
+
+       xe_guc_exec_queue_snapshot_free(ss->ge);
+       ss->ge = NULL;
+
+       xe_sched_job_snapshot_free(ss->job);
+       ss->job = NULL;
+
+       for (i = 0; i < XE_NUM_HW_ENGINES; i++)
+               if (ss->hwe[i]) {
+                       xe_hw_engine_snapshot_free(ss->hwe[i]);
+                       ss->hwe[i] = NULL;
+               }
+
+       xe_vm_snapshot_free(ss->vm);
+       ss->vm = NULL;
+}
+
+static void xe_devcoredump_deferred_snap_work(struct work_struct *work)
+{
+       struct xe_devcoredump_snapshot *ss = container_of(work, typeof(*ss), work);
+       struct xe_devcoredump *coredump = container_of(ss, typeof(*coredump), snapshot);
+
+       /* keep going if fw fails as we still want to save the memory and SW data */
+       if (xe_force_wake_get(gt_to_fw(ss->gt), XE_FORCEWAKE_ALL))
+               xe_gt_info(ss->gt, "failed to get forcewake for coredump capture\n");
+       xe_vm_snapshot_capture_delayed(ss->vm);
+       xe_guc_exec_queue_snapshot_capture_delayed(ss->ge);
+       xe_force_wake_put(gt_to_fw(ss->gt), XE_FORCEWAKE_ALL);
+
+       /* Calculate devcoredump size */
+       ss->read.size = __xe_devcoredump_read(NULL, INT_MAX, coredump);
+
+       ss->read.buffer = kvmalloc(ss->read.size, GFP_USER);
+       if (!ss->read.buffer)
+               return;
+
+       __xe_devcoredump_read(ss->read.buffer, ss->read.size, coredump);
+       xe_devcoredump_snapshot_free(ss);
+}
+
+static ssize_t xe_devcoredump_read(char *buffer, loff_t offset,
+                                  size_t count, void *data, size_t datalen)
+{
+       struct xe_devcoredump *coredump = data;
+       struct xe_devcoredump_snapshot *ss;
+       ssize_t byte_copied;
+
+       if (!coredump)
+               return -ENODEV;
+
+       ss = &coredump->snapshot;
+
+       /* Ensure delayed work is captured before continuing */
+       flush_work(&ss->work);
+
+       if (!ss->read.buffer)
+               return -ENODEV;
+
+       if (offset >= ss->read.size)
+               return 0;
+
+       byte_copied = count < ss->read.size - offset ? count :
+               ss->read.size - offset;
+       memcpy(buffer, ss->read.buffer + offset, byte_copied);
+
+       return byte_copied;
+}
+
 static void xe_devcoredump_free(void *data)
 {
        struct xe_devcoredump *coredump = data;
-       int i;
 
        /* Our device is gone. Nothing to do... */
        if (!data || !coredump_to_xe(coredump))
 
        cancel_work_sync(&coredump->snapshot.work);
 
-       xe_guc_ct_snapshot_free(coredump->snapshot.ct);
-       xe_guc_exec_queue_snapshot_free(coredump->snapshot.ge);
-       xe_sched_job_snapshot_free(coredump->snapshot.job);
-       for (i = 0; i < XE_NUM_HW_ENGINES; i++)
-               if (coredump->snapshot.hwe[i])
-                       xe_hw_engine_snapshot_free(coredump->snapshot.hwe[i]);
-       xe_vm_snapshot_free(coredump->snapshot.vm);
+       xe_devcoredump_snapshot_free(&coredump->snapshot);
+       kvfree(coredump->snapshot.read.buffer);
 
        /* To prevent stale data on next snapshot, clear everything */
        memset(&coredump->snapshot, 0, sizeof(coredump->snapshot));
 {
        return devm_add_action_or_reset(xe->drm.dev, xe_driver_devcoredump_fini, &xe->drm);
 }
+
 #endif