blob: ae695dabafd15d7e18a9e993f425af8048cca8d4 [file] [log] [blame]
kz15163460405de2006-09-28 01:41:18 -07001/*
miao chen - Sun Microsystems - Beijing China0035d212009-12-05 13:25:40 +08002 * Copyright 2009 Sun Microsystems, Inc. All rights reserved.
kz15163460405de2006-09-28 01:41:18 -07003 * Use is subject to license terms.
4 */
5
6/*
7 * drm_agpsupport.h -- DRM support for AGP/GART backend -*- linux-c -*-
8 * Created: Mon Dec 13 09:56:45 1999 by faith@precisioninsight.com
9 */
10/*
11 * Copyright 1999 Precision Insight, Inc., Cedar Park, Texas.
12 * Copyright 2000 VA Linux Systems, Inc., Sunnyvale, California.
miao chen - Sun Microsystems - Beijing China0035d212009-12-05 13:25:40 +080013 * Copyright (c) 2009, Intel Corporation.
kz15163460405de2006-09-28 01:41:18 -070014 * All Rights Reserved.
15 *
16 * Permission is hereby granted, free of charge, to any person obtaining a
17 * copy of this software and associated documentation files (the "Software"),
18 * to deal in the Software without restriction, including without limitation
19 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
20 * and/or sell copies of the Software, and to permit persons to whom the
21 * Software is furnished to do so, subject to the following conditions:
22 *
23 * The above copyright notice and this permission notice (including the next
24 * paragraph) shall be included in all copies or substantial portions of the
25 * Software.
26 *
27 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
28 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
29 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
30 * VA LINUX SYSTEMS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
31 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
32 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
33 * OTHER DEALINGS IN THE SOFTWARE.
34 *
35 * Author:
36 * Rickard E. (Rik) Faith <faith@valinux.com>
37 * Gareth Hughes <gareth@valinux.com>
38 *
39 */
40
kz15163460405de2006-09-28 01:41:18 -070041#include "drm.h"
42#include "drmP.h"
43
44#ifndef AGP_PAGE_SIZE
45#define AGP_PAGE_SIZE 4096
46#define AGP_PAGE_SHIFT 12
47#endif
48
cg149915d0538f62008-01-09 19:45:15 -080049/*
50 * The agpa_key field of struct agp_allocate_t actually is
51 * an index to an array. It can be zero. But we will use
52 * this agpa_key as a handle returned to userland. Generally,
53 * 0 is not a valid value for a handle, so we add an offset
54 * to the key to get a handle.
55 */
56#define DRM_AGP_KEY_OFFSET 8
57
58extern int drm_supp_device_capability(void *handle, int capid);
59
kz15163460405de2006-09-28 01:41:18 -070060/*ARGSUSED*/
61int
cg149915d0538f62008-01-09 19:45:15 -080062drm_device_is_agp(drm_device_t *dev)
kz15163460405de2006-09-28 01:41:18 -070063{
cg149915d0538f62008-01-09 19:45:15 -080064 int ret;
65
66 if (dev->driver->device_is_agp != NULL) {
67 /*
68 * device_is_agp returns a tristate:
69 * 0 = not AGP;
70 * 1 = definitely AGP;
71 * 2 = fall back to PCI capability
72 */
73 ret = (*dev->driver->device_is_agp)(dev);
74 if (ret != DRM_MIGHT_BE_AGP)
75 return (ret);
76 }
77
78 return (drm_supp_device_capability(dev->drm_handle, PCIY_AGP));
79
kz15163460405de2006-09-28 01:41:18 -070080}
81
82/*ARGSUSED*/
83int
cg149915d0538f62008-01-09 19:45:15 -080084drm_device_is_pcie(drm_device_t *dev)
kz15163460405de2006-09-28 01:41:18 -070085{
cg149915d0538f62008-01-09 19:45:15 -080086 return (drm_supp_device_capability(dev->drm_handle, PCIY_EXPRESS));
kz15163460405de2006-09-28 01:41:18 -070087}
88
89
90/*ARGSUSED*/
91int
92drm_agp_info(DRM_IOCTL_ARGS)
93{
94 DRM_DEVICE;
cg149915d0538f62008-01-09 19:45:15 -080095 agp_info_t *agpinfo;
96 drm_agp_info_t info;
kz15163460405de2006-09-28 01:41:18 -070097
98 if (!dev->agp || !dev->agp->acquired)
cg149915d0538f62008-01-09 19:45:15 -080099 return (EINVAL);
kz15163460405de2006-09-28 01:41:18 -0700100
cg149915d0538f62008-01-09 19:45:15 -0800101 agpinfo = &dev->agp->agp_info;
102 info.agp_version_major = agpinfo->agpi_version.agpv_major;
103 info.agp_version_minor = agpinfo->agpi_version.agpv_minor;
104 info.mode = agpinfo->agpi_mode;
105 info.aperture_base = agpinfo->agpi_aperbase;
106 info.aperture_size = agpinfo->agpi_apersize* 1024 * 1024;
107 info.memory_allowed = agpinfo->agpi_pgtotal << PAGE_SHIFT;
108 info.memory_used = agpinfo->agpi_pgused << PAGE_SHIFT;
109 info.id_vendor = agpinfo->agpi_devid & 0xffff;
110 info.id_device = agpinfo->agpi_devid >> 16;
111
112 DRM_COPYTO_WITH_RETURN((void *)data, &info, sizeof (info));
kz15163460405de2006-09-28 01:41:18 -0700113 return (0);
114}
115
116/*ARGSUSED*/
117int
118drm_agp_acquire(DRM_IOCTL_ARGS)
119{
120 DRM_DEVICE;
121 int ret, rval;
122
kz15163460405de2006-09-28 01:41:18 -0700123 if (!dev->agp) {
cg149915d0538f62008-01-09 19:45:15 -0800124 DRM_ERROR("drm_agp_acquire : agp isn't initialized yet");
125 return (ENODEV);
kz15163460405de2006-09-28 01:41:18 -0700126 }
cg149915d0538f62008-01-09 19:45:15 -0800127 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_ACQUIRE,
kz15163460405de2006-09-28 01:41:18 -0700128 (uintptr_t)0, FKIOCTL, kcred, &rval);
129 if (ret) {
130 DRM_ERROR("drm_agp_acquired: AGPIOC_ACQUIRE failed\n");
cg149915d0538f62008-01-09 19:45:15 -0800131 return (EIO);
kz15163460405de2006-09-28 01:41:18 -0700132 }
kz15163460405de2006-09-28 01:41:18 -0700133 dev->agp->acquired = 1;
134
135 return (0);
136}
137
138/*ARGSUSED*/
139int
140drm_agp_release(DRM_IOCTL_ARGS)
141{
142 DRM_DEVICE;
kz15163460405de2006-09-28 01:41:18 -0700143 int ret, rval;
144
145 if (!dev->agp)
cg149915d0538f62008-01-09 19:45:15 -0800146 return (ENODEV);
kz15163460405de2006-09-28 01:41:18 -0700147 if (!dev->agp->acquired)
cg149915d0538f62008-01-09 19:45:15 -0800148 return (EBUSY);
kz15163460405de2006-09-28 01:41:18 -0700149
cg149915d0538f62008-01-09 19:45:15 -0800150 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_RELEASE,
151 (intptr_t)0, FKIOCTL, kcred, &rval);
152 if (ret) {
153 DRM_ERROR("drm_agp_release: AGPIOC_RELEASE failed\n");
154 return (ENXIO);
kz15163460405de2006-09-28 01:41:18 -0700155 }
kz15163460405de2006-09-28 01:41:18 -0700156 dev->agp->acquired = 0;
157
cg149915d0538f62008-01-09 19:45:15 -0800158 return (ret);
159}
160
161
162int
163drm_agp_do_release(drm_device_t *dev)
164{
165 int ret, rval;
166
167 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_RELEASE,
168 (intptr_t)0, FKIOCTL, kcred, &rval);
169
170 if (ret == 0)
171 dev->agp->acquired = 0;
172
173 return (ret);
kz15163460405de2006-09-28 01:41:18 -0700174}
175
176/*ARGSUSED*/
177int
178drm_agp_enable(DRM_IOCTL_ARGS)
179{
180 DRM_DEVICE;
181 drm_agp_mode_t modes;
182 agp_setup_t setup;
183 int ret, rval;
184
185 if (!dev->agp)
cg149915d0538f62008-01-09 19:45:15 -0800186 return (ENODEV);
kz15163460405de2006-09-28 01:41:18 -0700187 if (!dev->agp->acquired)
cg149915d0538f62008-01-09 19:45:15 -0800188 return (EBUSY);
kz15163460405de2006-09-28 01:41:18 -0700189
cg149915d0538f62008-01-09 19:45:15 -0800190 DRM_COPYFROM_WITH_RETURN(&modes, (void *)data, sizeof (modes));
kz15163460405de2006-09-28 01:41:18 -0700191
192 dev->agp->mode = modes.mode;
cg149915d0538f62008-01-09 19:45:15 -0800193 setup.agps_mode = (uint32_t)modes.mode;
kz15163460405de2006-09-28 01:41:18 -0700194
miao chen - Sun Microsystems - Beijing China0035d212009-12-05 13:25:40 +0800195
kz15163460405de2006-09-28 01:41:18 -0700196 DRM_DEBUG("drm_agp_enable: dev->agp->mode=%lx", modes.mode);
197
cg149915d0538f62008-01-09 19:45:15 -0800198 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_SETUP,
199 (intptr_t)&setup, FKIOCTL, kcred, &rval);
kz15163460405de2006-09-28 01:41:18 -0700200 if (ret) {
cg149915d0538f62008-01-09 19:45:15 -0800201 DRM_ERROR("drm_agp_enable: failed");
202 return (EIO);
kz15163460405de2006-09-28 01:41:18 -0700203 }
204
205 dev->agp->base = dev->agp->agp_info.agpi_aperbase;
206 dev->agp->enabled = 1;
207
miao chen - Sun Microsystems - Beijing China0035d212009-12-05 13:25:40 +0800208 DRM_DEBUG("drm_agp_enable: dev->agp->base=0x%lx", dev->agp->base);
kz15163460405de2006-09-28 01:41:18 -0700209 return (0);
210}
211
212/*ARGSUSED*/
213int
214drm_agp_alloc(DRM_IOCTL_ARGS)
215{
216 DRM_DEVICE;
217 drm_agp_mem_t *entry;
218 agp_allocate_t alloc;
219 drm_agp_buffer_t request;
220 int pages;
221 int ret, rval;
222
223 if (!dev->agp || !dev->agp->acquired)
cg149915d0538f62008-01-09 19:45:15 -0800224 return (EINVAL);
kz15163460405de2006-09-28 01:41:18 -0700225
cg149915d0538f62008-01-09 19:45:15 -0800226 DRM_COPYFROM_WITH_RETURN(&request, (void *)data, sizeof (request));
kz15163460405de2006-09-28 01:41:18 -0700227
cg149915d0538f62008-01-09 19:45:15 -0800228 entry = kmem_zalloc(sizeof (*entry), KM_SLEEP);
kz15163460405de2006-09-28 01:41:18 -0700229
kz15163460405de2006-09-28 01:41:18 -0700230 pages = btopr(request.size);
231 alloc.agpa_pgcount = pages;
cg149915d0538f62008-01-09 19:45:15 -0800232 alloc.agpa_type = AGP_NORMAL;
233 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_ALLOCATE,
234 (intptr_t)&alloc, FKIOCTL, kcred, &rval);
kz15163460405de2006-09-28 01:41:18 -0700235 if (ret) {
cg149915d0538f62008-01-09 19:45:15 -0800236 DRM_ERROR("drm_agp_alloc: AGPIOC_ALLOCATE failed, ret=%d", ret);
237 kmem_free(entry, sizeof (*entry));
238 return (ret);
kz15163460405de2006-09-28 01:41:18 -0700239 }
240
241 entry->bound = 0;
242 entry->pages = pages;
cg149915d0538f62008-01-09 19:45:15 -0800243 entry->handle = (void*)(uintptr_t)(alloc.agpa_key + DRM_AGP_KEY_OFFSET);
kz15163460405de2006-09-28 01:41:18 -0700244 entry->prev = NULL;
cg149915d0538f62008-01-09 19:45:15 -0800245 entry->phys_addr = (void*)(uintptr_t)alloc.agpa_physical;
kz15163460405de2006-09-28 01:41:18 -0700246 entry->next = dev->agp->memory;
247 if (dev->agp->memory)
cg149915d0538f62008-01-09 19:45:15 -0800248 dev->agp->memory->prev = entry;
kz15163460405de2006-09-28 01:41:18 -0700249 dev->agp->memory = entry;
250
miao chen - Sun Microsystems - Beijing China0035d212009-12-05 13:25:40 +0800251 DRM_DEBUG("entry->phys_addr %lx", entry->phys_addr);
252
cg149915d0538f62008-01-09 19:45:15 -0800253 /* physical is used only by i810 driver */
kz15163460405de2006-09-28 01:41:18 -0700254 request.physical = alloc.agpa_physical;
cg149915d0538f62008-01-09 19:45:15 -0800255 request.handle = (unsigned long)entry->handle;
kz15163460405de2006-09-28 01:41:18 -0700256
cg149915d0538f62008-01-09 19:45:15 -0800257 /*
258 * If failed to ddi_copyout(), we will free allocated AGP memory
259 * when closing drm
260 */
261 DRM_COPYTO_WITH_RETURN((void *)data, &request, sizeof (request));
kz15163460405de2006-09-28 01:41:18 -0700262
263 return (0);
264}
265
266/*ARGSUSED*/
267static drm_agp_mem_t *
cg149915d0538f62008-01-09 19:45:15 -0800268drm_agp_lookup_entry(drm_device_t *dev, void *handle)
kz15163460405de2006-09-28 01:41:18 -0700269{
270 drm_agp_mem_t *entry;
271
kz15163460405de2006-09-28 01:41:18 -0700272 for (entry = dev->agp->memory; entry; entry = entry->next) {
273 if (entry->handle == handle)
274 return (entry);
275 }
276
277 return (NULL);
278}
279
280/*ARGSUSED*/
281int
282drm_agp_unbind(DRM_IOCTL_ARGS)
283{
284 DRM_DEVICE;
285 agp_unbind_t unbind;
286 drm_agp_binding_t request;
287 drm_agp_mem_t *entry;
288 int ret, rval;
289
290 if (!dev->agp || !dev->agp->acquired)
cg149915d0538f62008-01-09 19:45:15 -0800291 return (EINVAL);
292
293 DRM_COPYFROM_WITH_RETURN(&request, (void *)data, sizeof (request));
kz15163460405de2006-09-28 01:41:18 -0700294
295 if (!(entry = drm_agp_lookup_entry(dev, (void *)request.handle)))
cg149915d0538f62008-01-09 19:45:15 -0800296 return (EINVAL);
kz15163460405de2006-09-28 01:41:18 -0700297 if (!entry->bound)
cg149915d0538f62008-01-09 19:45:15 -0800298 return (EINVAL);
kz15163460405de2006-09-28 01:41:18 -0700299
300 unbind.agpu_pri = 0;
cg149915d0538f62008-01-09 19:45:15 -0800301 unbind.agpu_key = (uintptr_t)entry->handle - DRM_AGP_KEY_OFFSET;
kz15163460405de2006-09-28 01:41:18 -0700302
cg149915d0538f62008-01-09 19:45:15 -0800303 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_UNBIND,
kz15163460405de2006-09-28 01:41:18 -0700304 (intptr_t)&unbind, FKIOCTL, kcred, &rval);
305 if (ret) {
306 DRM_ERROR("drm_agp_unbind: AGPIOC_UNBIND failed");
cg149915d0538f62008-01-09 19:45:15 -0800307 return (EIO);
kz15163460405de2006-09-28 01:41:18 -0700308 }
309 entry->bound = 0;
310 return (0);
311}
312
313/*ARGSUSED*/
314int
315drm_agp_bind(DRM_IOCTL_ARGS)
316{
317 DRM_DEVICE;
318 drm_agp_binding_t request;
319 drm_agp_mem_t *entry;
cg149915d0538f62008-01-09 19:45:15 -0800320 int start;
321 uint_t key;
kz15163460405de2006-09-28 01:41:18 -0700322
323 if (!dev->agp || !dev->agp->acquired)
cg149915d0538f62008-01-09 19:45:15 -0800324 return (EINVAL);
kz15163460405de2006-09-28 01:41:18 -0700325
cg149915d0538f62008-01-09 19:45:15 -0800326 DRM_COPYFROM_WITH_RETURN(&request, (void *)data, sizeof (request));
kz15163460405de2006-09-28 01:41:18 -0700327
cg149915d0538f62008-01-09 19:45:15 -0800328 entry = drm_agp_lookup_entry(dev, (void *)request.handle);
329 if (!entry || entry->bound)
330 return (EINVAL);
kz15163460405de2006-09-28 01:41:18 -0700331
cg149915d0538f62008-01-09 19:45:15 -0800332 key = (uintptr_t)entry->handle - DRM_AGP_KEY_OFFSET;
333 start = btopr(request.offset);
334 if (drm_agp_bind_memory(key, start, dev)) {
335 DRM_ERROR("drm_agp_bind: failed key=%x, start=0x%x, "
336 "agp_base=0x%lx", key, start, dev->agp->base);
337 return (EIO);
338 }
339
340 entry->bound = dev->agp->base + (start << AGP_PAGE_SHIFT);
kz15163460405de2006-09-28 01:41:18 -0700341
342 return (0);
343}
344
345/*ARGSUSED*/
346int
347drm_agp_free(DRM_IOCTL_ARGS)
348{
349 DRM_DEVICE;
350 drm_agp_buffer_t request;
351 drm_agp_mem_t *entry;
352 int ret, rval;
cg149915d0538f62008-01-09 19:45:15 -0800353 int agpu_key;
kz15163460405de2006-09-28 01:41:18 -0700354
cg149915d0538f62008-01-09 19:45:15 -0800355 DRM_COPYFROM_WITH_RETURN(&request, (void *)data, sizeof (request));
kz15163460405de2006-09-28 01:41:18 -0700356 if (!dev->agp || !dev->agp->acquired)
cg149915d0538f62008-01-09 19:45:15 -0800357 return (EINVAL);
kz15163460405de2006-09-28 01:41:18 -0700358 if (!(entry = drm_agp_lookup_entry(dev, (void *)request.handle)))
cg149915d0538f62008-01-09 19:45:15 -0800359 return (EINVAL);
kz15163460405de2006-09-28 01:41:18 -0700360 if (entry->bound)
cg149915d0538f62008-01-09 19:45:15 -0800361 (void) drm_agp_unbind_memory(request.handle, dev);
kz15163460405de2006-09-28 01:41:18 -0700362
cg149915d0538f62008-01-09 19:45:15 -0800363 if (entry == dev->agp->memory)
364 dev->agp->memory = entry->next;
kz15163460405de2006-09-28 01:41:18 -0700365 if (entry->prev)
366 entry->prev->next = entry->next;
367 if (entry->next)
368 entry->next->prev = entry->prev;
369
cg149915d0538f62008-01-09 19:45:15 -0800370 agpu_key = (uintptr_t)entry->handle - DRM_AGP_KEY_OFFSET;
371 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_DEALLOCATE,
372 (intptr_t)agpu_key, FKIOCTL, kcred, &rval);
kz15163460405de2006-09-28 01:41:18 -0700373 if (ret) {
cg149915d0538f62008-01-09 19:45:15 -0800374 DRM_ERROR("drm_agp_free: AGPIOC_DEALLOCATE failed,"
375 "akey=%d, ret=%d", agpu_key, ret);
376 return (EIO);
kz15163460405de2006-09-28 01:41:18 -0700377 }
cg149915d0538f62008-01-09 19:45:15 -0800378 drm_free(entry, sizeof (*entry), DRM_MEM_AGPLISTS);
kz15163460405de2006-09-28 01:41:18 -0700379 return (0);
380}
381
382/*ARGSUSED*/
383drm_agp_head_t *
cg149915d0538f62008-01-09 19:45:15 -0800384drm_agp_init(drm_device_t *dev)
kz15163460405de2006-09-28 01:41:18 -0700385{
cg149915d0538f62008-01-09 19:45:15 -0800386 drm_agp_head_t *agp = NULL;
387 int retval, rval;
kz15163460405de2006-09-28 01:41:18 -0700388
cg149915d0538f62008-01-09 19:45:15 -0800389 agp = kmem_zalloc(sizeof (drm_agp_head_t), KM_SLEEP);
390
391 retval = ldi_ident_from_dip(dev->dip, &agp->agpgart_li);
392 if (retval != 0) {
393 DRM_ERROR("drm_agp_init: failed to get layerd ident, retval=%d",
394 retval);
395 goto err_1;
396 }
397
398 retval = ldi_open_by_name(AGP_DEVICE, FEXCL, kcred,
399 &agp->agpgart_lh, agp->agpgart_li);
400 if (retval != 0) {
401 DRM_ERROR("drm_agp_init: failed to open %s, retval=%d",
402 AGP_DEVICE, retval);
403 goto err_2;
404 }
405
406 retval = ldi_ioctl(agp->agpgart_lh, AGPIOC_INFO,
407 (intptr_t)&agp->agp_info, FKIOCTL, kcred, &rval);
408
409 if (retval != 0) {
410 DRM_ERROR("drm_agp_init: failed to get agpinfo, retval=%d",
411 retval);
412 goto err_3;
413 }
414
415 return (agp);
416
417err_3:
418 (void) ldi_close(agp->agpgart_lh, FEXCL, kcred);
419
420err_2:
421 ldi_ident_release(agp->agpgart_li);
422
423err_1:
424 kmem_free(agp, sizeof (drm_agp_head_t));
kz15163460405de2006-09-28 01:41:18 -0700425 return (NULL);
426}
427
428/*ARGSUSED*/
429void
cg149915d0538f62008-01-09 19:45:15 -0800430drm_agp_fini(drm_device_t *dev)
kz15163460405de2006-09-28 01:41:18 -0700431{
cg149915d0538f62008-01-09 19:45:15 -0800432 drm_agp_head_t *agp = dev->agp;
433 (void) ldi_close(agp->agpgart_lh, FEXCL, kcred);
434 ldi_ident_release(agp->agpgart_li);
435 kmem_free(agp, sizeof (drm_agp_head_t));
436 dev->agp = NULL;
kz15163460405de2006-09-28 01:41:18 -0700437}
438
439
440/*ARGSUSED*/
441void *
miao chen - Sun Microsystems - Beijing China0035d212009-12-05 13:25:40 +0800442drm_agp_allocate_memory(size_t pages, uint32_t type, drm_device_t *dev)
kz15163460405de2006-09-28 01:41:18 -0700443{
444 return (NULL);
445}
446
447/*ARGSUSED*/
448int
miao chen - Sun Microsystems - Beijing China0035d212009-12-05 13:25:40 +0800449drm_agp_free_memory(agp_allocate_t *handle, drm_device_t *dev)
kz15163460405de2006-09-28 01:41:18 -0700450{
451 return (1);
452}
453
454/*ARGSUSED*/
455int
456drm_agp_bind_memory(unsigned int key, uint32_t start, drm_device_t *dev)
457{
458 agp_bind_t bind;
459 int ret, rval;
460
kz15163460405de2006-09-28 01:41:18 -0700461 bind.agpb_pgstart = start;
462 bind.agpb_key = key;
cg149915d0538f62008-01-09 19:45:15 -0800463 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_BIND,
kz15163460405de2006-09-28 01:41:18 -0700464 (intptr_t)&bind, FKIOCTL, kcred, &rval);
465 if (ret) {
466 DRM_DEBUG("drm_agp_bind_meory: AGPIOC_BIND failed");
cg149915d0538f62008-01-09 19:45:15 -0800467 return (EIO);
kz15163460405de2006-09-28 01:41:18 -0700468 }
469 return (0);
470}
471
472/*ARGSUSED*/
473int
cg149915d0538f62008-01-09 19:45:15 -0800474drm_agp_unbind_memory(unsigned long handle, drm_device_t *dev)
kz15163460405de2006-09-28 01:41:18 -0700475{
476 agp_unbind_t unbind;
477 drm_agp_mem_t *entry;
478 int ret, rval;
479
480 if (!dev->agp || !dev->agp->acquired)
cg149915d0538f62008-01-09 19:45:15 -0800481 return (EINVAL);
kz15163460405de2006-09-28 01:41:18 -0700482
cg149915d0538f62008-01-09 19:45:15 -0800483 entry = drm_agp_lookup_entry(dev, (void *)handle);
484 if (!entry || !entry->bound)
485 return (EINVAL);
kz15163460405de2006-09-28 01:41:18 -0700486
487 unbind.agpu_pri = 0;
cg149915d0538f62008-01-09 19:45:15 -0800488 unbind.agpu_key = (uintptr_t)entry->handle - DRM_AGP_KEY_OFFSET;
kz15163460405de2006-09-28 01:41:18 -0700489
cg149915d0538f62008-01-09 19:45:15 -0800490 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_UNBIND,
491 (intptr_t)&unbind, FKIOCTL, kcred, &rval);
kz15163460405de2006-09-28 01:41:18 -0700492 if (ret) {
493 DRM_ERROR("drm_agp_unbind: AGPIO_UNBIND failed");
cg149915d0538f62008-01-09 19:45:15 -0800494 return (EIO);
kz15163460405de2006-09-28 01:41:18 -0700495 }
496 entry->bound = 0;
497 return (0);
kz15163460405de2006-09-28 01:41:18 -0700498}
miao chen - Sun Microsystems - Beijing China0035d212009-12-05 13:25:40 +0800499
500/*
501 * Binds a collection of pages into AGP memory at the given offset, returning
502 * the AGP memory structure containing them.
503 *
504 * No reference is held on the pages during this time -- it is up to the
505 * caller to handle that.
506 */
507int
508drm_agp_bind_pages(drm_device_t *dev,
509 pfn_t *pages,
510 unsigned long num_pages,
511 uint32_t gtt_offset)
512{
513
514 agp_bind_pages_t bind;
515 int ret, rval;
516
517 bind.agpb_pgstart = gtt_offset / AGP_PAGE_SIZE;
518 bind.agpb_pgcount = num_pages;
519 bind.agpb_pages = pages;
520 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_PAGES_BIND,
521 (intptr_t)&bind, FKIOCTL, kcred, &rval);
522 if (ret) {
523 DRM_ERROR("AGPIOC_PAGES_BIND failed ret %d", ret);
524 return (ret);
525 }
526 return (0);
527}
528
529int
530drm_agp_unbind_pages(drm_device_t *dev,
531 unsigned long num_pages,
532 uint32_t gtt_offset,
533 uint32_t type)
534{
535
536 agp_unbind_pages_t unbind;
537 int ret, rval;
538
539 unbind.agpb_pgstart = gtt_offset / AGP_PAGE_SIZE;
540 unbind.agpb_pgcount = num_pages;
541 unbind.agpb_type = type;
542 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_PAGES_UNBIND,
543 (intptr_t)&unbind, FKIOCTL, kcred, &rval);
544 if (ret) {
545 DRM_DEBUG("drm_agp_unbind_pages AGPIOC_PAGES_UNBIND failed");
546 return (ret);
547 }
548 return (0);
549}
550
551/*
552 * Certain Intel chipsets contains a global write buffer, and this can require
553 * flushing from the drm or X.org to make sure all data has hit RAM before
554 * initiating a GPU transfer, due to a lack of coherency with the integrated
555 * graphics device and this buffer.
556 */
557void
558drm_agp_chipset_flush(struct drm_device *dev)
559{
560 int ret, rval;
561
562 DRM_DEBUG("agp_chipset_flush");
563 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_FLUSHCHIPSET,
564 (intptr_t)0, FKIOCTL, kcred, &rval);
565 if (ret != 0) {
566 DRM_ERROR("Failed to drm_agp_chipset_flush ret %d", ret);
567 }
568}
569
570/*
571 * The pages are evict on suspend, so re-bind it at resume time
572 */
573void
574drm_agp_rebind(struct drm_device *dev)
575{
576 int ret, rval;
577
578 if (!dev->agp) {
579 return;
580 }
581
582 ret = ldi_ioctl(dev->agp->agpgart_lh, AGPIOC_PAGES_REBIND,
583 (intptr_t)0, FKIOCTL, kcred, &rval);
584 if (ret != 0) {
585 DRM_ERROR("rebind failed %d", ret);
586 }
587}