198932149SChris Wilson /* 298932149SChris Wilson * SPDX-License-Identifier: MIT 398932149SChris Wilson * 498932149SChris Wilson * Copyright © 2016 Intel Corporation 598932149SChris Wilson */ 698932149SChris Wilson 798932149SChris Wilson #ifndef __I915_GEM_OBJECT_H__ 898932149SChris Wilson #define __I915_GEM_OBJECT_H__ 998932149SChris Wilson 1098932149SChris Wilson #include <drm/drm_gem.h> 1198932149SChris Wilson #include <drm/drm_file.h> 1298932149SChris Wilson #include <drm/drm_device.h> 1398932149SChris Wilson 14da42104fSChris Wilson #include "display/intel_frontbuffer.h" 1598932149SChris Wilson #include "i915_gem_object_types.h" 16f0e4a063SChris Wilson #include "i915_gem_gtt.h" 172c86e55dSMatthew Auld #include "i915_vma_types.h" 18f0e4a063SChris Wilson 198475355fSChris Wilson void i915_gem_init__objects(struct drm_i915_private *i915); 208475355fSChris Wilson 2198932149SChris Wilson struct drm_i915_gem_object *i915_gem_object_alloc(void); 2298932149SChris Wilson void i915_gem_object_free(struct drm_i915_gem_object *obj); 2398932149SChris Wilson 248475355fSChris Wilson void i915_gem_object_init(struct drm_i915_gem_object *obj, 257867d709SChris Wilson const struct drm_i915_gem_object_ops *ops, 26c471748dSMaarten Lankhorst struct lock_class_key *key, 27c471748dSMaarten Lankhorst unsigned alloc_flags); 288475355fSChris Wilson struct drm_i915_gem_object * 29da1184cdSMatthew Auld i915_gem_object_create_shmem(struct drm_i915_private *i915, 30da1184cdSMatthew Auld resource_size_t size); 318475355fSChris Wilson struct drm_i915_gem_object * 328475355fSChris Wilson i915_gem_object_create_shmem_from_data(struct drm_i915_private *i915, 33da1184cdSMatthew Auld const void *data, resource_size_t size); 348475355fSChris Wilson 358475355fSChris Wilson extern const struct drm_i915_gem_object_ops i915_gem_shmem_ops; 36ed29c269SMaarten Lankhorst 378475355fSChris Wilson void __i915_gem_object_release_shmem(struct drm_i915_gem_object *obj, 388475355fSChris Wilson struct sg_table *pages, 398475355fSChris Wilson bool needs_clflush); 408475355fSChris Wilson 41a6117097SMaarten Lankhorst int i915_gem_object_pwrite_phys(struct drm_i915_gem_object *obj, 42a6117097SMaarten Lankhorst const struct drm_i915_gem_pwrite *args); 43a6117097SMaarten Lankhorst int i915_gem_object_pread_phys(struct drm_i915_gem_object *obj, 44a6117097SMaarten Lankhorst const struct drm_i915_gem_pread *args); 45a6117097SMaarten Lankhorst 46f033428dSChris Wilson int i915_gem_object_attach_phys(struct drm_i915_gem_object *obj, int align); 47a85fffe3SMaarten Lankhorst void i915_gem_object_put_pages_shmem(struct drm_i915_gem_object *obj, 48a85fffe3SMaarten Lankhorst struct sg_table *pages); 49a6117097SMaarten Lankhorst void i915_gem_object_put_pages_phys(struct drm_i915_gem_object *obj, 50a6117097SMaarten Lankhorst struct sg_table *pages); 51a6117097SMaarten Lankhorst 528475355fSChris Wilson void i915_gem_flush_free_objects(struct drm_i915_private *i915); 538475355fSChris Wilson 54f033428dSChris Wilson struct sg_table * 55f033428dSChris Wilson __i915_gem_object_unset_pages(struct drm_i915_gem_object *obj); 56f033428dSChris Wilson void i915_gem_object_truncate(struct drm_i915_gem_object *obj); 57f033428dSChris Wilson 5898932149SChris Wilson /** 5998932149SChris Wilson * i915_gem_object_lookup_rcu - look up a temporary GEM object from its handle 6098932149SChris Wilson * @filp: DRM file private date 6198932149SChris Wilson * @handle: userspace handle 6298932149SChris Wilson * 6398932149SChris Wilson * Returns: 6498932149SChris Wilson * 6598932149SChris Wilson * A pointer to the object named by the handle if such exists on @filp, NULL 6698932149SChris Wilson * otherwise. This object is only valid whilst under the RCU read lock, and 6798932149SChris Wilson * note carefully the object may be in the process of being destroyed. 6898932149SChris Wilson */ 6998932149SChris Wilson static inline struct drm_i915_gem_object * 7098932149SChris Wilson i915_gem_object_lookup_rcu(struct drm_file *file, u32 handle) 7198932149SChris Wilson { 7298932149SChris Wilson #ifdef CONFIG_LOCKDEP 7398932149SChris Wilson WARN_ON(debug_locks && !lock_is_held(&rcu_lock_map)); 7498932149SChris Wilson #endif 7598932149SChris Wilson return idr_find(&file->object_idr, handle); 7698932149SChris Wilson } 7798932149SChris Wilson 7898932149SChris Wilson static inline struct drm_i915_gem_object * 79280d14a6SChris Wilson i915_gem_object_get_rcu(struct drm_i915_gem_object *obj) 80280d14a6SChris Wilson { 81280d14a6SChris Wilson if (obj && !kref_get_unless_zero(&obj->base.refcount)) 82280d14a6SChris Wilson obj = NULL; 83280d14a6SChris Wilson 84280d14a6SChris Wilson return obj; 85280d14a6SChris Wilson } 86280d14a6SChris Wilson 87280d14a6SChris Wilson static inline struct drm_i915_gem_object * 8898932149SChris Wilson i915_gem_object_lookup(struct drm_file *file, u32 handle) 8998932149SChris Wilson { 9098932149SChris Wilson struct drm_i915_gem_object *obj; 9198932149SChris Wilson 9298932149SChris Wilson rcu_read_lock(); 9398932149SChris Wilson obj = i915_gem_object_lookup_rcu(file, handle); 94280d14a6SChris Wilson obj = i915_gem_object_get_rcu(obj); 9598932149SChris Wilson rcu_read_unlock(); 9698932149SChris Wilson 9798932149SChris Wilson return obj; 9898932149SChris Wilson } 9998932149SChris Wilson 10098932149SChris Wilson __deprecated 101b5893ffcSJanusz Krzysztofik struct drm_gem_object * 10298932149SChris Wilson drm_gem_object_lookup(struct drm_file *file, u32 handle); 10398932149SChris Wilson 10498932149SChris Wilson __attribute__((nonnull)) 10598932149SChris Wilson static inline struct drm_i915_gem_object * 10698932149SChris Wilson i915_gem_object_get(struct drm_i915_gem_object *obj) 10798932149SChris Wilson { 10898932149SChris Wilson drm_gem_object_get(&obj->base); 10998932149SChris Wilson return obj; 11098932149SChris Wilson } 11198932149SChris Wilson 11298932149SChris Wilson __attribute__((nonnull)) 11398932149SChris Wilson static inline void 11498932149SChris Wilson i915_gem_object_put(struct drm_i915_gem_object *obj) 11598932149SChris Wilson { 11698932149SChris Wilson __drm_gem_object_put(&obj->base); 11798932149SChris Wilson } 11898932149SChris Wilson 11952791eeeSChristian König #define assert_object_held(obj) dma_resv_assert_held((obj)->base.resv) 1206951e589SChris Wilson 121*a3258dbdSThomas Hellström /* 122*a3258dbdSThomas Hellström * If more than one potential simultaneous locker, assert held. 123*a3258dbdSThomas Hellström */ 124*a3258dbdSThomas Hellström static inline void assert_object_held_shared(struct drm_i915_gem_object *obj) 125*a3258dbdSThomas Hellström { 126*a3258dbdSThomas Hellström /* 127*a3258dbdSThomas Hellström * Note mm list lookup is protected by 128*a3258dbdSThomas Hellström * kref_get_unless_zero(). 129*a3258dbdSThomas Hellström */ 130*a3258dbdSThomas Hellström if (IS_ENABLED(CONFIG_LOCKDEP) && 131*a3258dbdSThomas Hellström kref_read(&obj->base.refcount) > 0) 132*a3258dbdSThomas Hellström lockdep_assert_held(&obj->mm.lock); 133*a3258dbdSThomas Hellström } 134*a3258dbdSThomas Hellström 13580f0b679SMaarten Lankhorst static inline int __i915_gem_object_lock(struct drm_i915_gem_object *obj, 13680f0b679SMaarten Lankhorst struct i915_gem_ww_ctx *ww, 13780f0b679SMaarten Lankhorst bool intr) 13898932149SChris Wilson { 13980f0b679SMaarten Lankhorst int ret; 14080f0b679SMaarten Lankhorst 14180f0b679SMaarten Lankhorst if (intr) 14280f0b679SMaarten Lankhorst ret = dma_resv_lock_interruptible(obj->base.resv, ww ? &ww->ctx : NULL); 14380f0b679SMaarten Lankhorst else 14480f0b679SMaarten Lankhorst ret = dma_resv_lock(obj->base.resv, ww ? &ww->ctx : NULL); 14580f0b679SMaarten Lankhorst 14680f0b679SMaarten Lankhorst if (!ret && ww) 14780f0b679SMaarten Lankhorst list_add_tail(&obj->obj_link, &ww->obj_list); 14880f0b679SMaarten Lankhorst if (ret == -EALREADY) 14980f0b679SMaarten Lankhorst ret = 0; 15080f0b679SMaarten Lankhorst 15180f0b679SMaarten Lankhorst if (ret == -EDEADLK) 15280f0b679SMaarten Lankhorst ww->contended = obj; 15380f0b679SMaarten Lankhorst 15480f0b679SMaarten Lankhorst return ret; 15580f0b679SMaarten Lankhorst } 15680f0b679SMaarten Lankhorst 15780f0b679SMaarten Lankhorst static inline int i915_gem_object_lock(struct drm_i915_gem_object *obj, 15880f0b679SMaarten Lankhorst struct i915_gem_ww_ctx *ww) 15980f0b679SMaarten Lankhorst { 16080f0b679SMaarten Lankhorst return __i915_gem_object_lock(obj, ww, ww && ww->intr); 16180f0b679SMaarten Lankhorst } 16280f0b679SMaarten Lankhorst 16380f0b679SMaarten Lankhorst static inline int i915_gem_object_lock_interruptible(struct drm_i915_gem_object *obj, 16480f0b679SMaarten Lankhorst struct i915_gem_ww_ctx *ww) 16580f0b679SMaarten Lankhorst { 16680f0b679SMaarten Lankhorst WARN_ON(ww && !ww->intr); 16780f0b679SMaarten Lankhorst return __i915_gem_object_lock(obj, ww, true); 16898932149SChris Wilson } 16998932149SChris Wilson 1702850748eSChris Wilson static inline bool i915_gem_object_trylock(struct drm_i915_gem_object *obj) 1712850748eSChris Wilson { 1722850748eSChris Wilson return dma_resv_trylock(obj->base.resv); 1732850748eSChris Wilson } 1742850748eSChris Wilson 17598932149SChris Wilson static inline void i915_gem_object_unlock(struct drm_i915_gem_object *obj) 17698932149SChris Wilson { 17752791eeeSChristian König dma_resv_unlock(obj->base.resv); 17898932149SChris Wilson } 17998932149SChris Wilson 18098932149SChris Wilson static inline void 18198932149SChris Wilson i915_gem_object_set_readonly(struct drm_i915_gem_object *obj) 18298932149SChris Wilson { 183cc662126SAbdiel Janulgue obj->flags |= I915_BO_READONLY; 18498932149SChris Wilson } 18598932149SChris Wilson 18698932149SChris Wilson static inline bool 18798932149SChris Wilson i915_gem_object_is_readonly(const struct drm_i915_gem_object *obj) 18898932149SChris Wilson { 189cc662126SAbdiel Janulgue return obj->flags & I915_BO_READONLY; 19098932149SChris Wilson } 19198932149SChris Wilson 19298932149SChris Wilson static inline bool 1932f0b97caSMatthew Auld i915_gem_object_is_contiguous(const struct drm_i915_gem_object *obj) 1942f0b97caSMatthew Auld { 1952f0b97caSMatthew Auld return obj->flags & I915_BO_ALLOC_CONTIGUOUS; 1962f0b97caSMatthew Auld } 1972f0b97caSMatthew Auld 1982f0b97caSMatthew Auld static inline bool 1997c98501aSMatthew Auld i915_gem_object_is_volatile(const struct drm_i915_gem_object *obj) 2007c98501aSMatthew Auld { 2017c98501aSMatthew Auld return obj->flags & I915_BO_ALLOC_VOLATILE; 2027c98501aSMatthew Auld } 2037c98501aSMatthew Auld 2047c98501aSMatthew Auld static inline void 2057c98501aSMatthew Auld i915_gem_object_set_volatile(struct drm_i915_gem_object *obj) 2067c98501aSMatthew Auld { 2077c98501aSMatthew Auld obj->flags |= I915_BO_ALLOC_VOLATILE; 2087c98501aSMatthew Auld } 2097c98501aSMatthew Auld 2107c98501aSMatthew Auld static inline bool 2110175969eSChris Wilson i915_gem_object_has_tiling_quirk(struct drm_i915_gem_object *obj) 2120175969eSChris Wilson { 2130175969eSChris Wilson return test_bit(I915_TILING_QUIRK_BIT, &obj->flags); 2140175969eSChris Wilson } 2150175969eSChris Wilson 2160175969eSChris Wilson static inline void 2170175969eSChris Wilson i915_gem_object_set_tiling_quirk(struct drm_i915_gem_object *obj) 2180175969eSChris Wilson { 2190175969eSChris Wilson set_bit(I915_TILING_QUIRK_BIT, &obj->flags); 2200175969eSChris Wilson } 2210175969eSChris Wilson 2220175969eSChris Wilson static inline void 2230175969eSChris Wilson i915_gem_object_clear_tiling_quirk(struct drm_i915_gem_object *obj) 2240175969eSChris Wilson { 2250175969eSChris Wilson clear_bit(I915_TILING_QUIRK_BIT, &obj->flags); 2260175969eSChris Wilson } 2270175969eSChris Wilson 2280175969eSChris Wilson static inline bool 2293cbad5d7SChris Wilson i915_gem_object_type_has(const struct drm_i915_gem_object *obj, 2303cbad5d7SChris Wilson unsigned long flags) 2313cbad5d7SChris Wilson { 2323cbad5d7SChris Wilson return obj->ops->flags & flags; 2333cbad5d7SChris Wilson } 2343cbad5d7SChris Wilson 2353cbad5d7SChris Wilson static inline bool 23698932149SChris Wilson i915_gem_object_has_struct_page(const struct drm_i915_gem_object *obj) 23798932149SChris Wilson { 238c471748dSMaarten Lankhorst return obj->flags & I915_BO_ALLOC_STRUCT_PAGE; 23998932149SChris Wilson } 24098932149SChris Wilson 24198932149SChris Wilson static inline bool 2425fbc2c2bSImre Deak i915_gem_object_has_iomem(const struct drm_i915_gem_object *obj) 2435fbc2c2bSImre Deak { 2445fbc2c2bSImre Deak return i915_gem_object_type_has(obj, I915_GEM_OBJECT_HAS_IOMEM); 2455fbc2c2bSImre Deak } 2465fbc2c2bSImre Deak 2475fbc2c2bSImre Deak static inline bool 24898932149SChris Wilson i915_gem_object_is_shrinkable(const struct drm_i915_gem_object *obj) 24998932149SChris Wilson { 2503cbad5d7SChris Wilson return i915_gem_object_type_has(obj, I915_GEM_OBJECT_IS_SHRINKABLE); 25198932149SChris Wilson } 25298932149SChris Wilson 25398932149SChris Wilson static inline bool 25498932149SChris Wilson i915_gem_object_is_proxy(const struct drm_i915_gem_object *obj) 25598932149SChris Wilson { 2563cbad5d7SChris Wilson return i915_gem_object_type_has(obj, I915_GEM_OBJECT_IS_PROXY); 25798932149SChris Wilson } 25898932149SChris Wilson 25998932149SChris Wilson static inline bool 260f6c26b55SJanusz Krzysztofik i915_gem_object_never_mmap(const struct drm_i915_gem_object *obj) 261a4311745SChris Wilson { 262f6c26b55SJanusz Krzysztofik return i915_gem_object_type_has(obj, I915_GEM_OBJECT_NO_MMAP); 263a4311745SChris Wilson } 264a4311745SChris Wilson 265a4311745SChris Wilson static inline bool 26698932149SChris Wilson i915_gem_object_is_framebuffer(const struct drm_i915_gem_object *obj) 26798932149SChris Wilson { 2688e7cb179SChris Wilson return READ_ONCE(obj->frontbuffer); 26998932149SChris Wilson } 27098932149SChris Wilson 27198932149SChris Wilson static inline unsigned int 27298932149SChris Wilson i915_gem_object_get_tiling(const struct drm_i915_gem_object *obj) 27398932149SChris Wilson { 27498932149SChris Wilson return obj->tiling_and_stride & TILING_MASK; 27598932149SChris Wilson } 27698932149SChris Wilson 27798932149SChris Wilson static inline bool 27898932149SChris Wilson i915_gem_object_is_tiled(const struct drm_i915_gem_object *obj) 27998932149SChris Wilson { 28098932149SChris Wilson return i915_gem_object_get_tiling(obj) != I915_TILING_NONE; 28198932149SChris Wilson } 28298932149SChris Wilson 28398932149SChris Wilson static inline unsigned int 28498932149SChris Wilson i915_gem_object_get_stride(const struct drm_i915_gem_object *obj) 28598932149SChris Wilson { 28698932149SChris Wilson return obj->tiling_and_stride & STRIDE_MASK; 28798932149SChris Wilson } 28898932149SChris Wilson 28998932149SChris Wilson static inline unsigned int 29098932149SChris Wilson i915_gem_tile_height(unsigned int tiling) 29198932149SChris Wilson { 29298932149SChris Wilson GEM_BUG_ON(!tiling); 29398932149SChris Wilson return tiling == I915_TILING_Y ? 32 : 8; 29498932149SChris Wilson } 29598932149SChris Wilson 29698932149SChris Wilson static inline unsigned int 29798932149SChris Wilson i915_gem_object_get_tile_height(const struct drm_i915_gem_object *obj) 29898932149SChris Wilson { 29998932149SChris Wilson return i915_gem_tile_height(i915_gem_object_get_tiling(obj)); 30098932149SChris Wilson } 30198932149SChris Wilson 30298932149SChris Wilson static inline unsigned int 30398932149SChris Wilson i915_gem_object_get_tile_row_size(const struct drm_i915_gem_object *obj) 30498932149SChris Wilson { 30598932149SChris Wilson return (i915_gem_object_get_stride(obj) * 30698932149SChris Wilson i915_gem_object_get_tile_height(obj)); 30798932149SChris Wilson } 30898932149SChris Wilson 30998932149SChris Wilson int i915_gem_object_set_tiling(struct drm_i915_gem_object *obj, 31098932149SChris Wilson unsigned int tiling, unsigned int stride); 31198932149SChris Wilson 31298932149SChris Wilson struct scatterlist * 313934941edSTvrtko Ursulin __i915_gem_object_get_sg(struct drm_i915_gem_object *obj, 314934941edSTvrtko Ursulin struct i915_gem_object_page_iter *iter, 315934941edSTvrtko Ursulin unsigned int n, 3160edbb9baSMaarten Lankhorst unsigned int *offset, bool allow_alloc); 317934941edSTvrtko Ursulin 318934941edSTvrtko Ursulin static inline struct scatterlist * 31998932149SChris Wilson i915_gem_object_get_sg(struct drm_i915_gem_object *obj, 320934941edSTvrtko Ursulin unsigned int n, 3210edbb9baSMaarten Lankhorst unsigned int *offset, bool allow_alloc) 322934941edSTvrtko Ursulin { 3230edbb9baSMaarten Lankhorst return __i915_gem_object_get_sg(obj, &obj->mm.get_page, n, offset, allow_alloc); 324934941edSTvrtko Ursulin } 325934941edSTvrtko Ursulin 326934941edSTvrtko Ursulin static inline struct scatterlist * 327934941edSTvrtko Ursulin i915_gem_object_get_sg_dma(struct drm_i915_gem_object *obj, 328934941edSTvrtko Ursulin unsigned int n, 3290edbb9baSMaarten Lankhorst unsigned int *offset, bool allow_alloc) 330934941edSTvrtko Ursulin { 3310edbb9baSMaarten Lankhorst return __i915_gem_object_get_sg(obj, &obj->mm.get_dma_page, n, offset, allow_alloc); 332934941edSTvrtko Ursulin } 33398932149SChris Wilson 33498932149SChris Wilson struct page * 33598932149SChris Wilson i915_gem_object_get_page(struct drm_i915_gem_object *obj, 33698932149SChris Wilson unsigned int n); 33798932149SChris Wilson 3384993a8a3SDave Airlie struct page * 3394993a8a3SDave Airlie i915_gem_object_get_dirty_page(struct drm_i915_gem_object *obj, 3404993a8a3SDave Airlie unsigned int n); 3414993a8a3SDave Airlie 34298932149SChris Wilson dma_addr_t 34398932149SChris Wilson i915_gem_object_get_dma_address_len(struct drm_i915_gem_object *obj, 34498932149SChris Wilson unsigned long n, 34598932149SChris Wilson unsigned int *len); 34698932149SChris Wilson 34798932149SChris Wilson dma_addr_t 34898932149SChris Wilson i915_gem_object_get_dma_address(struct drm_i915_gem_object *obj, 34998932149SChris Wilson unsigned long n); 35098932149SChris Wilson 35198932149SChris Wilson void __i915_gem_object_set_pages(struct drm_i915_gem_object *obj, 35298932149SChris Wilson struct sg_table *pages, 35398932149SChris Wilson unsigned int sg_page_sizes); 354f033428dSChris Wilson 355f033428dSChris Wilson int ____i915_gem_object_get_pages(struct drm_i915_gem_object *obj); 35698932149SChris Wilson int __i915_gem_object_get_pages(struct drm_i915_gem_object *obj); 35798932149SChris Wilson 35898932149SChris Wilson static inline int __must_check 35998932149SChris Wilson i915_gem_object_pin_pages(struct drm_i915_gem_object *obj) 36098932149SChris Wilson { 361abd2f577SMaarten Lankhorst might_lock(&obj->mm.lock); 36298932149SChris Wilson 36398932149SChris Wilson if (atomic_inc_not_zero(&obj->mm.pages_pin_count)) 36498932149SChris Wilson return 0; 36598932149SChris Wilson 36698932149SChris Wilson return __i915_gem_object_get_pages(obj); 36798932149SChris Wilson } 36898932149SChris Wilson 36998932149SChris Wilson static inline bool 37098932149SChris Wilson i915_gem_object_has_pages(struct drm_i915_gem_object *obj) 37198932149SChris Wilson { 37298932149SChris Wilson return !IS_ERR_OR_NULL(READ_ONCE(obj->mm.pages)); 37398932149SChris Wilson } 37498932149SChris Wilson 37598932149SChris Wilson static inline void 37698932149SChris Wilson __i915_gem_object_pin_pages(struct drm_i915_gem_object *obj) 37798932149SChris Wilson { 37898932149SChris Wilson GEM_BUG_ON(!i915_gem_object_has_pages(obj)); 37998932149SChris Wilson 38098932149SChris Wilson atomic_inc(&obj->mm.pages_pin_count); 38198932149SChris Wilson } 38298932149SChris Wilson 38398932149SChris Wilson static inline bool 38498932149SChris Wilson i915_gem_object_has_pinned_pages(struct drm_i915_gem_object *obj) 38598932149SChris Wilson { 38698932149SChris Wilson return atomic_read(&obj->mm.pages_pin_count); 38798932149SChris Wilson } 38898932149SChris Wilson 38998932149SChris Wilson static inline void 39098932149SChris Wilson __i915_gem_object_unpin_pages(struct drm_i915_gem_object *obj) 39198932149SChris Wilson { 39298932149SChris Wilson GEM_BUG_ON(!i915_gem_object_has_pages(obj)); 39398932149SChris Wilson GEM_BUG_ON(!i915_gem_object_has_pinned_pages(obj)); 39498932149SChris Wilson 39598932149SChris Wilson atomic_dec(&obj->mm.pages_pin_count); 39698932149SChris Wilson } 39798932149SChris Wilson 39898932149SChris Wilson static inline void 39998932149SChris Wilson i915_gem_object_unpin_pages(struct drm_i915_gem_object *obj) 40098932149SChris Wilson { 40198932149SChris Wilson __i915_gem_object_unpin_pages(obj); 40298932149SChris Wilson } 40398932149SChris Wilson 404f86dbacbSDaniel Vetter int __i915_gem_object_put_pages(struct drm_i915_gem_object *obj); 405abd2f577SMaarten Lankhorst int __i915_gem_object_put_pages_locked(struct drm_i915_gem_object *obj); 406f033428dSChris Wilson void i915_gem_object_truncate(struct drm_i915_gem_object *obj); 407f033428dSChris Wilson void i915_gem_object_writeback(struct drm_i915_gem_object *obj); 40898932149SChris Wilson 40998932149SChris Wilson /** 41098932149SChris Wilson * i915_gem_object_pin_map - return a contiguous mapping of the entire object 41198932149SChris Wilson * @obj: the object to map into kernel address space 41298932149SChris Wilson * @type: the type of mapping, used to select pgprot_t 41398932149SChris Wilson * 41498932149SChris Wilson * Calls i915_gem_object_pin_pages() to prevent reaping of the object's 41598932149SChris Wilson * pages and then returns a contiguous mapping of the backing storage into 41698932149SChris Wilson * the kernel address space. Based on the @type of mapping, the PTE will be 41798932149SChris Wilson * set to either WriteBack or WriteCombine (via pgprot_t). 41898932149SChris Wilson * 41998932149SChris Wilson * The caller is responsible for calling i915_gem_object_unpin_map() when the 42098932149SChris Wilson * mapping is no longer required. 42198932149SChris Wilson * 42298932149SChris Wilson * Returns the pointer through which to access the mapped object, or an 42398932149SChris Wilson * ERR_PTR() on error. 42498932149SChris Wilson */ 42598932149SChris Wilson void *__must_check i915_gem_object_pin_map(struct drm_i915_gem_object *obj, 42698932149SChris Wilson enum i915_map_type type); 42798932149SChris Wilson 42874827b53SMaarten Lankhorst void *__must_check i915_gem_object_pin_map_unlocked(struct drm_i915_gem_object *obj, 42974827b53SMaarten Lankhorst enum i915_map_type type); 43074827b53SMaarten Lankhorst 43198932149SChris Wilson void __i915_gem_object_flush_map(struct drm_i915_gem_object *obj, 43298932149SChris Wilson unsigned long offset, 43398932149SChris Wilson unsigned long size); 43498932149SChris Wilson static inline void i915_gem_object_flush_map(struct drm_i915_gem_object *obj) 43598932149SChris Wilson { 43698932149SChris Wilson __i915_gem_object_flush_map(obj, 0, obj->base.size); 43798932149SChris Wilson } 43898932149SChris Wilson 43998932149SChris Wilson /** 44098932149SChris Wilson * i915_gem_object_unpin_map - releases an earlier mapping 44198932149SChris Wilson * @obj: the object to unmap 44298932149SChris Wilson * 44398932149SChris Wilson * After pinning the object and mapping its pages, once you are finished 44498932149SChris Wilson * with your access, call i915_gem_object_unpin_map() to release the pin 44598932149SChris Wilson * upon the mapping. Once the pin count reaches zero, that mapping may be 44698932149SChris Wilson * removed. 44798932149SChris Wilson */ 44898932149SChris Wilson static inline void i915_gem_object_unpin_map(struct drm_i915_gem_object *obj) 44998932149SChris Wilson { 45098932149SChris Wilson i915_gem_object_unpin_pages(obj); 45198932149SChris Wilson } 45298932149SChris Wilson 45389d19b2bSChris Wilson void __i915_gem_object_release_map(struct drm_i915_gem_object *obj); 45489d19b2bSChris Wilson 455f0e4a063SChris Wilson int i915_gem_object_prepare_read(struct drm_i915_gem_object *obj, 456f0e4a063SChris Wilson unsigned int *needs_clflush); 457f0e4a063SChris Wilson int i915_gem_object_prepare_write(struct drm_i915_gem_object *obj, 458f0e4a063SChris Wilson unsigned int *needs_clflush); 459f0e4a063SChris Wilson #define CLFLUSH_BEFORE BIT(0) 460f0e4a063SChris Wilson #define CLFLUSH_AFTER BIT(1) 461f0e4a063SChris Wilson #define CLFLUSH_FLAGS (CLFLUSH_BEFORE | CLFLUSH_AFTER) 462f0e4a063SChris Wilson 463f0e4a063SChris Wilson static inline void 464f0e4a063SChris Wilson i915_gem_object_finish_access(struct drm_i915_gem_object *obj) 465f0e4a063SChris Wilson { 466f0e4a063SChris Wilson i915_gem_object_unpin_pages(obj); 467f0e4a063SChris Wilson } 468f0e4a063SChris Wilson 46998932149SChris Wilson static inline struct intel_engine_cs * 47098932149SChris Wilson i915_gem_object_last_write_engine(struct drm_i915_gem_object *obj) 47198932149SChris Wilson { 47298932149SChris Wilson struct intel_engine_cs *engine = NULL; 47398932149SChris Wilson struct dma_fence *fence; 47498932149SChris Wilson 47598932149SChris Wilson rcu_read_lock(); 47652791eeeSChristian König fence = dma_resv_get_excl_rcu(obj->base.resv); 47798932149SChris Wilson rcu_read_unlock(); 47898932149SChris Wilson 47998932149SChris Wilson if (fence && dma_fence_is_i915(fence) && !dma_fence_is_signaled(fence)) 48098932149SChris Wilson engine = to_request(fence)->engine; 48198932149SChris Wilson dma_fence_put(fence); 48298932149SChris Wilson 48398932149SChris Wilson return engine; 48498932149SChris Wilson } 48598932149SChris Wilson 48698932149SChris Wilson void i915_gem_object_set_cache_coherency(struct drm_i915_gem_object *obj, 48798932149SChris Wilson unsigned int cache_level); 48898932149SChris Wilson void i915_gem_object_flush_if_display(struct drm_i915_gem_object *obj); 489c1793ba8SMaarten Lankhorst void i915_gem_object_flush_if_display_locked(struct drm_i915_gem_object *obj); 49098932149SChris Wilson 491f0e4a063SChris Wilson int __must_check 492f0e4a063SChris Wilson i915_gem_object_set_to_wc_domain(struct drm_i915_gem_object *obj, bool write); 493f0e4a063SChris Wilson int __must_check 494f0e4a063SChris Wilson i915_gem_object_set_to_gtt_domain(struct drm_i915_gem_object *obj, bool write); 495f0e4a063SChris Wilson int __must_check 496f0e4a063SChris Wilson i915_gem_object_set_to_cpu_domain(struct drm_i915_gem_object *obj, bool write); 497f0e4a063SChris Wilson struct i915_vma * __must_check 498f0e4a063SChris Wilson i915_gem_object_pin_to_display_plane(struct drm_i915_gem_object *obj, 4991b321026SMaarten Lankhorst struct i915_gem_ww_ctx *ww, 500f0e4a063SChris Wilson u32 alignment, 501f0e4a063SChris Wilson const struct i915_ggtt_view *view, 502f0e4a063SChris Wilson unsigned int flags); 503f0e4a063SChris Wilson 5041aff1903SChris Wilson void i915_gem_object_make_unshrinkable(struct drm_i915_gem_object *obj); 5051aff1903SChris Wilson void i915_gem_object_make_shrinkable(struct drm_i915_gem_object *obj); 5061aff1903SChris Wilson void i915_gem_object_make_purgeable(struct drm_i915_gem_object *obj); 5071aff1903SChris Wilson 5088475355fSChris Wilson static inline bool cpu_write_needs_clflush(struct drm_i915_gem_object *obj) 5098475355fSChris Wilson { 5108475355fSChris Wilson if (obj->cache_dirty) 5118475355fSChris Wilson return false; 5128475355fSChris Wilson 5138475355fSChris Wilson if (!(obj->cache_coherent & I915_BO_CACHE_COHERENT_FOR_WRITE)) 5148475355fSChris Wilson return true; 5158475355fSChris Wilson 5165a90606dSChris Wilson /* Currently in use by HW (display engine)? Keep flushed. */ 5175a90606dSChris Wilson return i915_gem_object_is_framebuffer(obj); 5188475355fSChris Wilson } 5198475355fSChris Wilson 5208475355fSChris Wilson static inline void __start_cpu_write(struct drm_i915_gem_object *obj) 5218475355fSChris Wilson { 5228475355fSChris Wilson obj->read_domains = I915_GEM_DOMAIN_CPU; 5238475355fSChris Wilson obj->write_domain = I915_GEM_DOMAIN_CPU; 5248475355fSChris Wilson if (cpu_write_needs_clflush(obj)) 5258475355fSChris Wilson obj->cache_dirty = true; 5268475355fSChris Wilson } 52798932149SChris Wilson 528f99e67f1SChris Wilson void i915_gem_fence_wait_priority(struct dma_fence *fence, 529f99e67f1SChris Wilson const struct i915_sched_attr *attr); 530f99e67f1SChris Wilson 531d45a1a53SChris Wilson int i915_gem_object_wait(struct drm_i915_gem_object *obj, 532d45a1a53SChris Wilson unsigned int flags, 533d45a1a53SChris Wilson long timeout); 534d45a1a53SChris Wilson int i915_gem_object_wait_priority(struct drm_i915_gem_object *obj, 535d45a1a53SChris Wilson unsigned int flags, 536d45a1a53SChris Wilson const struct i915_sched_attr *attr); 537d45a1a53SChris Wilson 538da42104fSChris Wilson void __i915_gem_object_flush_frontbuffer(struct drm_i915_gem_object *obj, 539da42104fSChris Wilson enum fb_op_origin origin); 540da42104fSChris Wilson void __i915_gem_object_invalidate_frontbuffer(struct drm_i915_gem_object *obj, 541da42104fSChris Wilson enum fb_op_origin origin); 542da42104fSChris Wilson 543da42104fSChris Wilson static inline void 544da42104fSChris Wilson i915_gem_object_flush_frontbuffer(struct drm_i915_gem_object *obj, 545da42104fSChris Wilson enum fb_op_origin origin) 546da42104fSChris Wilson { 547da42104fSChris Wilson if (unlikely(rcu_access_pointer(obj->frontbuffer))) 548da42104fSChris Wilson __i915_gem_object_flush_frontbuffer(obj, origin); 549da42104fSChris Wilson } 550da42104fSChris Wilson 551da42104fSChris Wilson static inline void 552da42104fSChris Wilson i915_gem_object_invalidate_frontbuffer(struct drm_i915_gem_object *obj, 553da42104fSChris Wilson enum fb_op_origin origin) 554da42104fSChris Wilson { 555da42104fSChris Wilson if (unlikely(rcu_access_pointer(obj->frontbuffer))) 556da42104fSChris Wilson __i915_gem_object_invalidate_frontbuffer(obj, origin); 557da42104fSChris Wilson } 558da42104fSChris Wilson 5595fbc2c2bSImre Deak int i915_gem_object_read_from_page(struct drm_i915_gem_object *obj, u64 offset, void *dst, int size); 5605fbc2c2bSImre Deak 56141a9c75dSChris Wilson bool i915_gem_object_is_shmem(const struct drm_i915_gem_object *obj); 56241a9c75dSChris Wilson 563ed29c269SMaarten Lankhorst #ifdef CONFIG_MMU_NOTIFIER 564ed29c269SMaarten Lankhorst static inline bool 565ed29c269SMaarten Lankhorst i915_gem_object_is_userptr(struct drm_i915_gem_object *obj) 566ed29c269SMaarten Lankhorst { 567ed29c269SMaarten Lankhorst return obj->userptr.notifier.mm; 568ed29c269SMaarten Lankhorst } 569ed29c269SMaarten Lankhorst 570ed29c269SMaarten Lankhorst int i915_gem_object_userptr_submit_init(struct drm_i915_gem_object *obj); 571ed29c269SMaarten Lankhorst int i915_gem_object_userptr_submit_done(struct drm_i915_gem_object *obj); 572ed29c269SMaarten Lankhorst void i915_gem_object_userptr_submit_fini(struct drm_i915_gem_object *obj); 573ed29c269SMaarten Lankhorst int i915_gem_object_userptr_validate(struct drm_i915_gem_object *obj); 574ed29c269SMaarten Lankhorst #else 575ed29c269SMaarten Lankhorst static inline bool i915_gem_object_is_userptr(struct drm_i915_gem_object *obj) { return false; } 576ed29c269SMaarten Lankhorst 577ed29c269SMaarten Lankhorst static inline int i915_gem_object_userptr_submit_init(struct drm_i915_gem_object *obj) { GEM_BUG_ON(1); return -ENODEV; } 578ed29c269SMaarten Lankhorst static inline int i915_gem_object_userptr_submit_done(struct drm_i915_gem_object *obj) { GEM_BUG_ON(1); return -ENODEV; } 579ed29c269SMaarten Lankhorst static inline void i915_gem_object_userptr_submit_fini(struct drm_i915_gem_object *obj) { GEM_BUG_ON(1); } 580ed29c269SMaarten Lankhorst static inline int i915_gem_object_userptr_validate(struct drm_i915_gem_object *obj) { GEM_BUG_ON(1); return -ENODEV; } 581ed29c269SMaarten Lankhorst 582ed29c269SMaarten Lankhorst #endif 583ed29c269SMaarten Lankhorst 58498932149SChris Wilson #endif 585