static void cancel_userptr(struct work_struct *work) { struct i915_mmu_object *mo = container_of(work, typeof(*mo), work); struct drm_i915_gem_object *obj = mo->obj; struct work_struct *active; /* Cancel any active worker and force us to re-evaluate gup */ mutex_lock(&obj->mm.lock); active = fetch_and_zero(&obj->userptr.work); mutex_unlock(&obj->mm.lock); if (active) goto out; i915_gem_object_wait(obj, I915_WAIT_ALL, MAX_SCHEDULE_TIMEOUT, NULL); mutex_lock(&obj->base.dev->struct_mutex); /* We are inside a kthread context and can't be interrupted */ if (i915_gem_object_unbind(obj) == 0) __i915_gem_object_put_pages(obj, I915_MM_NORMAL); WARN_ONCE(i915_gem_object_has_pages(obj), "Failed to release pages: bind_count=%d, pages_pin_count=%d, pin_global=%d\n", obj->bind_count, atomic_read(&obj->mm.pages_pin_count), obj->pin_global); mutex_unlock(&obj->base.dev->struct_mutex); out: i915_gem_object_put(obj); }
static void intel_csr_runtime_pm_put(struct drm_i915_private *dev_priv) { intel_wakeref_t wakeref __maybe_unused = fetch_and_zero(&dev_priv->csr.wakeref); intel_display_power_put(dev_priv, POWER_DOMAIN_INIT, wakeref); }
/* Cleans up uC firmware by releasing the firmware GEM obj. */ static void __intel_uc_fw_fini(struct intel_uc_fw *uc_fw) { struct drm_i915_gem_object *obj; obj = fetch_and_zero(&uc_fw->obj); if (obj) i915_gem_object_put(obj); uc_fw->fetch_status = INTEL_UC_FIRMWARE_NONE; }
static void i915_gem_object_release_stolen(struct drm_i915_gem_object *obj) { struct drm_i915_private *dev_priv = to_i915(obj->base.dev); struct drm_mm_node *stolen = fetch_and_zero(&obj->stolen); GEM_BUG_ON(!stolen); __i915_gem_object_unpin_pages(obj); i915_gem_stolen_remove_node(dev_priv, stolen); kfree(stolen); }
static void i915_gem_userptr_release__mmu_notifier(struct drm_i915_gem_object *obj) { struct i915_mmu_object *mo; mo = fetch_and_zero(&obj->userptr.mmu_object); if (!mo) return; spin_lock(&mo->mn->lock); del_object(mo); spin_unlock(&mo->mn->lock); kfree(mo); }
void i915_vma_unpin_and_release(struct i915_vma **p_vma) { struct i915_vma *vma; struct drm_i915_gem_object *obj; vma = fetch_and_zero(p_vma); if (!vma) return; obj = vma->obj; i915_vma_unpin(vma); i915_vma_close(vma); __i915_gem_object_release_unless_active(obj); }
void i915_gem_render_state_fini(struct intel_engine_cs *engine) { struct intel_render_state *so; struct drm_i915_gem_object *obj; so = fetch_and_zero(&engine->render_state); if (!so) return; obj = so->vma->obj; i915_vma_close(so->vma); __i915_gem_object_release_unless_active(obj); kfree(so); }
static struct i915_mmu_notifier * i915_mmu_notifier_find(struct i915_mm_struct *mm) { struct i915_mmu_notifier *mn; int err = 0; mn = mm->mn; if (mn) return mn; mn = i915_mmu_notifier_create(mm->mm); if (IS_ERR(mn)) err = PTR_ERR(mn); down_write(&mm->mm->mmap_sem); mutex_lock(&mm->i915->mm_lock); if (mm->mn == NULL && !err) { /* Protected by mmap_sem (write-lock) */ err = __mmu_notifier_register(&mn->mn, mm->mm); if (!err) { /* Protected by mm_lock */ mm->mn = fetch_and_zero(&mn); } } else if (mm->mn) { /* * Someone else raced and successfully installed the mmu * notifier, we can cancel our own errors. */ err = 0; } mutex_unlock(&mm->i915->mm_lock); up_write(&mm->mm->mmap_sem); if (mn && !IS_ERR(mn)) { destroy_workqueue(mn->wq); kfree(mn); } return err ? ERR_PTR(err) : mm->mn; }