Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2012 Red Hat Inc. |
| 3 | * |
| 4 | * Permission is hereby granted, free of charge, to any person obtaining a |
| 5 | * copy of this software and associated documentation files (the |
| 6 | * "Software"), to deal in the Software without restriction, including |
| 7 | * without limitation the rights to use, copy, modify, merge, publish, |
| 8 | * distribute, sub license, and/or sell copies of the Software, and to |
| 9 | * permit persons to whom the Software is furnished to do so, subject to |
| 10 | * the following conditions: |
| 11 | * |
| 12 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
| 13 | * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| 14 | * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL |
| 15 | * THE COPYRIGHT HOLDERS, AUTHORS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, |
| 16 | * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR |
| 17 | * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE |
| 18 | * USE OR OTHER DEALINGS IN THE SOFTWARE. |
| 19 | * |
| 20 | * The above copyright notice and this permission notice (including the |
| 21 | * next paragraph) shall be included in all copies or substantial portions |
| 22 | * of the Software. |
| 23 | * |
| 24 | */ |
| 25 | /* |
| 26 | * Authors: Dave Airlie <airlied@redhat.com> |
| 27 | */ |
David Howells | 760285e | 2012-10-02 18:01:07 +0100 | [diff] [blame] | 28 | #include <drm/drmP.h> |
Masahiro Yamada | ea35134 | 2017-04-24 13:50:22 +0900 | [diff] [blame] | 29 | #include <drm/ttm/ttm_page_alloc.h> |
| 30 | |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 31 | #include "ast_drv.h" |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 32 | |
| 33 | static inline struct ast_private * |
| 34 | ast_bdev(struct ttm_bo_device *bd) |
| 35 | { |
| 36 | return container_of(bd, struct ast_private, ttm.bdev); |
| 37 | } |
| 38 | |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 39 | static void ast_bo_ttm_destroy(struct ttm_buffer_object *tbo) |
| 40 | { |
| 41 | struct ast_bo *bo; |
| 42 | |
| 43 | bo = container_of(tbo, struct ast_bo, bo); |
| 44 | |
| 45 | drm_gem_object_release(&bo->gem); |
| 46 | kfree(bo); |
| 47 | } |
| 48 | |
Rashika | a1537f3 | 2014-01-06 20:27:48 +0530 | [diff] [blame] | 49 | static bool ast_ttm_bo_is_ast_bo(struct ttm_buffer_object *bo) |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 50 | { |
| 51 | if (bo->destroy == &ast_bo_ttm_destroy) |
| 52 | return true; |
| 53 | return false; |
| 54 | } |
| 55 | |
| 56 | static int |
| 57 | ast_bo_init_mem_type(struct ttm_bo_device *bdev, uint32_t type, |
| 58 | struct ttm_mem_type_manager *man) |
| 59 | { |
| 60 | switch (type) { |
| 61 | case TTM_PL_SYSTEM: |
| 62 | man->flags = TTM_MEMTYPE_FLAG_MAPPABLE; |
| 63 | man->available_caching = TTM_PL_MASK_CACHING; |
| 64 | man->default_caching = TTM_PL_FLAG_CACHED; |
| 65 | break; |
| 66 | case TTM_PL_VRAM: |
| 67 | man->func = &ttm_bo_manager_func; |
| 68 | man->flags = TTM_MEMTYPE_FLAG_FIXED | |
| 69 | TTM_MEMTYPE_FLAG_MAPPABLE; |
| 70 | man->available_caching = TTM_PL_FLAG_UNCACHED | |
| 71 | TTM_PL_FLAG_WC; |
| 72 | man->default_caching = TTM_PL_FLAG_WC; |
| 73 | break; |
| 74 | default: |
| 75 | DRM_ERROR("Unsupported memory type %u\n", (unsigned)type); |
| 76 | return -EINVAL; |
| 77 | } |
| 78 | return 0; |
| 79 | } |
| 80 | |
| 81 | static void |
| 82 | ast_bo_evict_flags(struct ttm_buffer_object *bo, struct ttm_placement *pl) |
| 83 | { |
| 84 | struct ast_bo *astbo = ast_bo(bo); |
| 85 | |
| 86 | if (!ast_ttm_bo_is_ast_bo(bo)) |
| 87 | return; |
| 88 | |
| 89 | ast_ttm_placement(astbo, TTM_PL_FLAG_SYSTEM); |
| 90 | *pl = astbo->placement; |
| 91 | } |
| 92 | |
| 93 | static int ast_bo_verify_access(struct ttm_buffer_object *bo, struct file *filp) |
| 94 | { |
David Herrmann | acb4652 | 2013-08-25 18:28:59 +0200 | [diff] [blame] | 95 | struct ast_bo *astbo = ast_bo(bo); |
| 96 | |
David Herrmann | d9a1f0b | 2016-09-01 14:48:33 +0200 | [diff] [blame] | 97 | return drm_vma_node_verify_access(&astbo->gem.vma_node, |
| 98 | filp->private_data); |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 99 | } |
| 100 | |
| 101 | static int ast_ttm_io_mem_reserve(struct ttm_bo_device *bdev, |
| 102 | struct ttm_mem_reg *mem) |
| 103 | { |
| 104 | struct ttm_mem_type_manager *man = &bdev->man[mem->mem_type]; |
| 105 | struct ast_private *ast = ast_bdev(bdev); |
| 106 | |
| 107 | mem->bus.addr = NULL; |
| 108 | mem->bus.offset = 0; |
| 109 | mem->bus.size = mem->num_pages << PAGE_SHIFT; |
| 110 | mem->bus.base = 0; |
| 111 | mem->bus.is_iomem = false; |
| 112 | if (!(man->flags & TTM_MEMTYPE_FLAG_MAPPABLE)) |
| 113 | return -EINVAL; |
| 114 | switch (mem->mem_type) { |
| 115 | case TTM_PL_SYSTEM: |
| 116 | /* system memory */ |
| 117 | return 0; |
| 118 | case TTM_PL_VRAM: |
| 119 | mem->bus.offset = mem->start << PAGE_SHIFT; |
| 120 | mem->bus.base = pci_resource_start(ast->dev->pdev, 0); |
| 121 | mem->bus.is_iomem = true; |
| 122 | break; |
| 123 | default: |
| 124 | return -EINVAL; |
| 125 | break; |
| 126 | } |
| 127 | return 0; |
| 128 | } |
| 129 | |
| 130 | static void ast_ttm_io_mem_free(struct ttm_bo_device *bdev, struct ttm_mem_reg *mem) |
| 131 | { |
| 132 | } |
| 133 | |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 134 | static void ast_ttm_backend_destroy(struct ttm_tt *tt) |
| 135 | { |
| 136 | ttm_tt_fini(tt); |
| 137 | kfree(tt); |
| 138 | } |
| 139 | |
| 140 | static struct ttm_backend_func ast_tt_backend_func = { |
| 141 | .destroy = &ast_ttm_backend_destroy, |
| 142 | }; |
| 143 | |
| 144 | |
Christian König | dde5da2 | 2018-02-22 10:18:14 +0100 | [diff] [blame] | 145 | static struct ttm_tt *ast_ttm_tt_create(struct ttm_buffer_object *bo, |
| 146 | uint32_t page_flags) |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 147 | { |
| 148 | struct ttm_tt *tt; |
| 149 | |
| 150 | tt = kzalloc(sizeof(struct ttm_tt), GFP_KERNEL); |
| 151 | if (tt == NULL) |
| 152 | return NULL; |
| 153 | tt->func = &ast_tt_backend_func; |
Christian König | dde5da2 | 2018-02-22 10:18:14 +0100 | [diff] [blame] | 154 | if (ttm_tt_init(tt, bo, page_flags)) { |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 155 | kfree(tt); |
| 156 | return NULL; |
| 157 | } |
| 158 | return tt; |
| 159 | } |
| 160 | |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 161 | struct ttm_bo_driver ast_bo_driver = { |
| 162 | .ttm_tt_create = ast_ttm_tt_create, |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 163 | .init_mem_type = ast_bo_init_mem_type, |
Christian König | a2ab19fe | 2016-08-30 17:26:04 +0200 | [diff] [blame] | 164 | .eviction_valuable = ttm_bo_eviction_valuable, |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 165 | .evict_flags = ast_bo_evict_flags, |
Christian König | 1888577 | 2016-06-06 10:17:52 +0200 | [diff] [blame] | 166 | .move = NULL, |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 167 | .verify_access = ast_bo_verify_access, |
| 168 | .io_mem_reserve = &ast_ttm_io_mem_reserve, |
| 169 | .io_mem_free = &ast_ttm_io_mem_free, |
| 170 | }; |
| 171 | |
| 172 | int ast_mm_init(struct ast_private *ast) |
| 173 | { |
| 174 | int ret; |
| 175 | struct drm_device *dev = ast->dev; |
| 176 | struct ttm_bo_device *bdev = &ast->ttm.bdev; |
| 177 | |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 178 | ret = ttm_bo_device_init(&ast->ttm.bdev, |
David Herrmann | 44d847b | 2013-08-13 19:10:30 +0200 | [diff] [blame] | 179 | &ast_bo_driver, |
| 180 | dev->anon_inode->i_mapping, |
| 181 | DRM_FILE_PAGE_OFFSET, |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 182 | true); |
| 183 | if (ret) { |
| 184 | DRM_ERROR("Error initialising bo driver; %d\n", ret); |
| 185 | return ret; |
| 186 | } |
| 187 | |
| 188 | ret = ttm_bo_init_mm(bdev, TTM_PL_VRAM, |
| 189 | ast->vram_size >> PAGE_SHIFT); |
| 190 | if (ret) { |
| 191 | DRM_ERROR("Failed ttm VRAM init: %d\n", ret); |
| 192 | return ret; |
| 193 | } |
| 194 | |
Dave Airlie | 7cf321d | 2016-10-24 15:37:48 +1000 | [diff] [blame] | 195 | arch_io_reserve_memtype_wc(pci_resource_start(dev->pdev, 0), |
| 196 | pci_resource_len(dev->pdev, 0)); |
Andy Lutomirski | 247d36d | 2013-05-13 23:58:41 +0000 | [diff] [blame] | 197 | ast->fb_mtrr = arch_phys_wc_add(pci_resource_start(dev->pdev, 0), |
| 198 | pci_resource_len(dev->pdev, 0)); |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 199 | |
| 200 | return 0; |
| 201 | } |
| 202 | |
| 203 | void ast_mm_fini(struct ast_private *ast) |
| 204 | { |
Dave Airlie | 7cf321d | 2016-10-24 15:37:48 +1000 | [diff] [blame] | 205 | struct drm_device *dev = ast->dev; |
| 206 | |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 207 | ttm_bo_device_release(&ast->ttm.bdev); |
| 208 | |
Andy Lutomirski | 247d36d | 2013-05-13 23:58:41 +0000 | [diff] [blame] | 209 | arch_phys_wc_del(ast->fb_mtrr); |
Dave Airlie | 7cf321d | 2016-10-24 15:37:48 +1000 | [diff] [blame] | 210 | arch_io_free_memtype_wc(pci_resource_start(dev->pdev, 0), |
| 211 | pci_resource_len(dev->pdev, 0)); |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 212 | } |
| 213 | |
| 214 | void ast_ttm_placement(struct ast_bo *bo, int domain) |
| 215 | { |
| 216 | u32 c = 0; |
Christian König | f1217ed | 2014-08-27 13:16:04 +0200 | [diff] [blame] | 217 | unsigned i; |
| 218 | |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 219 | bo->placement.placement = bo->placements; |
| 220 | bo->placement.busy_placement = bo->placements; |
| 221 | if (domain & TTM_PL_FLAG_VRAM) |
Christian König | f1217ed | 2014-08-27 13:16:04 +0200 | [diff] [blame] | 222 | bo->placements[c++].flags = TTM_PL_FLAG_WC | TTM_PL_FLAG_UNCACHED | TTM_PL_FLAG_VRAM; |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 223 | if (domain & TTM_PL_FLAG_SYSTEM) |
Dave Airlie | a9d6dd2 | 2014-09-12 14:32:40 +1000 | [diff] [blame] | 224 | bo->placements[c++].flags = TTM_PL_FLAG_CACHED | TTM_PL_FLAG_SYSTEM; |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 225 | if (!c) |
Dave Airlie | a9d6dd2 | 2014-09-12 14:32:40 +1000 | [diff] [blame] | 226 | bo->placements[c++].flags = TTM_PL_FLAG_CACHED | TTM_PL_FLAG_SYSTEM; |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 227 | bo->placement.num_placement = c; |
| 228 | bo->placement.num_busy_placement = c; |
Christian König | f1217ed | 2014-08-27 13:16:04 +0200 | [diff] [blame] | 229 | for (i = 0; i < c; ++i) { |
| 230 | bo->placements[i].fpfn = 0; |
| 231 | bo->placements[i].lpfn = 0; |
| 232 | } |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 233 | } |
| 234 | |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 235 | int ast_bo_create(struct drm_device *dev, int size, int align, |
| 236 | uint32_t flags, struct ast_bo **pastbo) |
| 237 | { |
| 238 | struct ast_private *ast = dev->dev_private; |
| 239 | struct ast_bo *astbo; |
| 240 | size_t acc_size; |
| 241 | int ret; |
| 242 | |
| 243 | astbo = kzalloc(sizeof(struct ast_bo), GFP_KERNEL); |
| 244 | if (!astbo) |
| 245 | return -ENOMEM; |
| 246 | |
| 247 | ret = drm_gem_object_init(dev, &astbo->gem, size); |
Egbert Eich | b2d44e2 | 2017-07-18 16:47:38 +0200 | [diff] [blame] | 248 | if (ret) |
| 249 | goto error; |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 250 | |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 251 | astbo->bo.bdev = &ast->ttm.bdev; |
| 252 | |
| 253 | ast_ttm_placement(astbo, TTM_PL_FLAG_VRAM | TTM_PL_FLAG_SYSTEM); |
| 254 | |
| 255 | acc_size = ttm_bo_dma_acc_size(&ast->ttm.bdev, size, |
| 256 | sizeof(struct ast_bo)); |
| 257 | |
| 258 | ret = ttm_bo_init(&ast->ttm.bdev, &astbo->bo, size, |
| 259 | ttm_bo_type_device, &astbo->placement, |
Christian König | 724daa4 | 2018-02-22 15:52:31 +0100 | [diff] [blame] | 260 | align >> PAGE_SHIFT, false, acc_size, |
Maarten Lankhorst | f4f4e3e | 2014-01-09 11:03:15 +0100 | [diff] [blame] | 261 | NULL, NULL, ast_bo_ttm_destroy); |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 262 | if (ret) |
Egbert Eich | b2d44e2 | 2017-07-18 16:47:38 +0200 | [diff] [blame] | 263 | goto error; |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 264 | |
| 265 | *pastbo = astbo; |
| 266 | return 0; |
Egbert Eich | b2d44e2 | 2017-07-18 16:47:38 +0200 | [diff] [blame] | 267 | error: |
| 268 | kfree(astbo); |
| 269 | return ret; |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 270 | } |
| 271 | |
| 272 | static inline u64 ast_bo_gpu_offset(struct ast_bo *bo) |
| 273 | { |
| 274 | return bo->bo.offset; |
| 275 | } |
| 276 | |
| 277 | int ast_bo_pin(struct ast_bo *bo, u32 pl_flag, u64 *gpu_addr) |
| 278 | { |
Christian König | 19be557 | 2017-04-12 14:24:39 +0200 | [diff] [blame] | 279 | struct ttm_operation_ctx ctx = { false, false }; |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 280 | int i, ret; |
| 281 | |
| 282 | if (bo->pin_count) { |
| 283 | bo->pin_count++; |
| 284 | if (gpu_addr) |
| 285 | *gpu_addr = ast_bo_gpu_offset(bo); |
| 286 | } |
| 287 | |
| 288 | ast_ttm_placement(bo, pl_flag); |
| 289 | for (i = 0; i < bo->placement.num_placement; i++) |
Christian König | f1217ed | 2014-08-27 13:16:04 +0200 | [diff] [blame] | 290 | bo->placements[i].flags |= TTM_PL_FLAG_NO_EVICT; |
Christian König | 19be557 | 2017-04-12 14:24:39 +0200 | [diff] [blame] | 291 | ret = ttm_bo_validate(&bo->bo, &bo->placement, &ctx); |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 292 | if (ret) |
| 293 | return ret; |
| 294 | |
| 295 | bo->pin_count = 1; |
| 296 | if (gpu_addr) |
| 297 | *gpu_addr = ast_bo_gpu_offset(bo); |
| 298 | return 0; |
| 299 | } |
| 300 | |
| 301 | int ast_bo_unpin(struct ast_bo *bo) |
| 302 | { |
Christian König | 19be557 | 2017-04-12 14:24:39 +0200 | [diff] [blame] | 303 | struct ttm_operation_ctx ctx = { false, false }; |
Egbert Eich | 587b9b1 | 2017-07-18 16:47:36 +0200 | [diff] [blame] | 304 | int i; |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 305 | if (!bo->pin_count) { |
| 306 | DRM_ERROR("unpin bad %p\n", bo); |
| 307 | return 0; |
| 308 | } |
| 309 | bo->pin_count--; |
| 310 | if (bo->pin_count) |
| 311 | return 0; |
| 312 | |
| 313 | for (i = 0; i < bo->placement.num_placement ; i++) |
Christian König | f1217ed | 2014-08-27 13:16:04 +0200 | [diff] [blame] | 314 | bo->placements[i].flags &= ~TTM_PL_FLAG_NO_EVICT; |
Christian König | 19be557 | 2017-04-12 14:24:39 +0200 | [diff] [blame] | 315 | return ttm_bo_validate(&bo->bo, &bo->placement, &ctx); |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 316 | } |
| 317 | |
| 318 | int ast_bo_push_sysram(struct ast_bo *bo) |
| 319 | { |
Christian König | 19be557 | 2017-04-12 14:24:39 +0200 | [diff] [blame] | 320 | struct ttm_operation_ctx ctx = { false, false }; |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 321 | int i, ret; |
| 322 | if (!bo->pin_count) { |
| 323 | DRM_ERROR("unpin bad %p\n", bo); |
| 324 | return 0; |
| 325 | } |
| 326 | bo->pin_count--; |
| 327 | if (bo->pin_count) |
| 328 | return 0; |
| 329 | |
| 330 | if (bo->kmap.virtual) |
| 331 | ttm_bo_kunmap(&bo->kmap); |
| 332 | |
| 333 | ast_ttm_placement(bo, TTM_PL_FLAG_SYSTEM); |
| 334 | for (i = 0; i < bo->placement.num_placement ; i++) |
Christian König | f1217ed | 2014-08-27 13:16:04 +0200 | [diff] [blame] | 335 | bo->placements[i].flags |= TTM_PL_FLAG_NO_EVICT; |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 336 | |
Christian König | 19be557 | 2017-04-12 14:24:39 +0200 | [diff] [blame] | 337 | ret = ttm_bo_validate(&bo->bo, &bo->placement, &ctx); |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 338 | if (ret) { |
| 339 | DRM_ERROR("pushing to VRAM failed\n"); |
| 340 | return ret; |
| 341 | } |
| 342 | return 0; |
| 343 | } |
| 344 | |
| 345 | int ast_mmap(struct file *filp, struct vm_area_struct *vma) |
| 346 | { |
| 347 | struct drm_file *file_priv; |
| 348 | struct ast_private *ast; |
| 349 | |
| 350 | if (unlikely(vma->vm_pgoff < DRM_FILE_PAGE_OFFSET)) |
Daniel Vetter | 884c6da | 2014-09-23 15:46:47 +0200 | [diff] [blame] | 351 | return -EINVAL; |
Dave Airlie | 312fec1 | 2012-02-29 13:40:04 +0000 | [diff] [blame] | 352 | |
| 353 | file_priv = filp->private_data; |
| 354 | ast = file_priv->minor->dev->dev_private; |
| 355 | return ttm_bo_mmap(filp, vma, &ast->ttm.bdev); |
| 356 | } |