drm_agpsupport.c 12.7 KB
Newer Older
L
Linus Torvalds 已提交
1
/**
D
Dave Airlie 已提交
2
 * \file drm_agpsupport.h
L
Linus Torvalds 已提交
3
 * DRM support for AGP/GART backend
D
Dave Airlie 已提交
4
 *
L
Linus Torvalds 已提交
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39
 * \author Rickard E. (Rik) Faith <faith@valinux.com>
 * \author Gareth Hughes <gareth@valinux.com>
 */

/*
 * Copyright 1999 Precision Insight, Inc., Cedar Park, Texas.
 * Copyright 2000 VA Linux Systems, Inc., Sunnyvale, California.
 * All Rights Reserved.
 *
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 * and/or sell copies of the Software, and to permit persons to whom the
 * Software is furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice (including the next
 * paragraph) shall be included in all copies or substantial portions of the
 * Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 * VA LINUX SYSTEMS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
 * OTHER DEALINGS IN THE SOFTWARE.
 */

#include "drmP.h"
#include <linux/module.h>

#if __OS_HAS_AGP

/**
40
 * Get AGP information.
L
Linus Torvalds 已提交
41 42 43 44 45 46 47 48 49 50
 *
 * \param inode device inode.
 * \param filp file pointer.
 * \param cmd command.
 * \param arg pointer to a (output) drm_agp_info structure.
 * \return zero on success or a negative number on failure.
 *
 * Verifies the AGP device has been initialized and acquired and fills in the
 * drm_agp_info structure with the information in drm_agp_head::agp_info.
 */
D
Dave Airlie 已提交
51
int drm_agp_info(drm_device_t * dev, drm_agp_info_t * info)
L
Linus Torvalds 已提交
52
{
D
Dave Airlie 已提交
53
	DRM_AGP_KERN *kern;
L
Linus Torvalds 已提交
54 55 56 57

	if (!dev->agp || !dev->agp->acquired)
		return -EINVAL;

D
Dave Airlie 已提交
58
	kern = &dev->agp->agp_info;
59 60
	info->agp_version_major = kern->version.major;
	info->agp_version_minor = kern->version.minor;
D
Dave Airlie 已提交
61 62 63 64 65 66 67
	info->mode = kern->mode;
	info->aperture_base = kern->aper_base;
	info->aperture_size = kern->aper_size * 1024 * 1024;
	info->memory_allowed = kern->max_memory << PAGE_SHIFT;
	info->memory_used = kern->current_memory << PAGE_SHIFT;
	info->id_vendor = kern->device->vendor;
	info->id_device = kern->device->device;
68 69 70

	return 0;
}
D
Dave Airlie 已提交
71

72 73 74
EXPORT_SYMBOL(drm_agp_info);

int drm_agp_info_ioctl(struct inode *inode, struct file *filp,
D
Dave Airlie 已提交
75
		       unsigned int cmd, unsigned long arg)
76 77 78 79 80 81 82 83 84
{
	drm_file_t *priv = filp->private_data;
	drm_device_t *dev = priv->head->dev;
	drm_agp_info_t info;
	int err;

	err = drm_agp_info(dev, &info);
	if (err)
		return err;
D
Dave Airlie 已提交
85

86
	if (copy_to_user((drm_agp_info_t __user *) arg, &info, sizeof(info)))
L
Linus Torvalds 已提交
87 88 89 90 91
		return -EFAULT;
	return 0;
}

/**
92
 * Acquire the AGP device.
L
Linus Torvalds 已提交
93
 *
94
 * \param dev DRM device that is to acquire AGP
D
Dave Airlie 已提交
95
 * \return zero on success or a negative number on failure.
L
Linus Torvalds 已提交
96 97
 *
 * Verifies the AGP device hasn't been acquired before and calls
98
 * \c agp_backend_acquire.
L
Linus Torvalds 已提交
99
 */
D
Dave Airlie 已提交
100
int drm_agp_acquire(drm_device_t * dev)
L
Linus Torvalds 已提交
101 102 103 104 105 106 107 108 109 110
{
	if (!dev->agp)
		return -ENODEV;
	if (dev->agp->acquired)
		return -EBUSY;
	if (!(dev->agp->bridge = agp_backend_acquire(dev->pdev)))
		return -ENODEV;
	dev->agp->acquired = 1;
	return 0;
}
D
Dave Airlie 已提交
111

112
EXPORT_SYMBOL(drm_agp_acquire);
L
Linus Torvalds 已提交
113 114

/**
115
 * Acquire the AGP device (ioctl).
L
Linus Torvalds 已提交
116 117 118 119 120 121 122
 *
 * \param inode device inode.
 * \param filp file pointer.
 * \param cmd command.
 * \param arg user argument.
 * \return zero on success or a negative number on failure.
 *
123 124
 * Verifies the AGP device hasn't been acquired before and calls
 * \c agp_backend_acquire.
L
Linus Torvalds 已提交
125
 */
126 127
int drm_agp_acquire_ioctl(struct inode *inode, struct file *filp,
			  unsigned int cmd, unsigned long arg)
L
Linus Torvalds 已提交
128
{
129
	drm_file_t *priv = filp->private_data;
D
Dave Airlie 已提交
130 131

	return drm_agp_acquire((drm_device_t *) priv->head->dev);
132
}
L
Linus Torvalds 已提交
133

134 135 136 137 138 139 140 141
/**
 * Release the AGP device.
 *
 * \param dev DRM device that is to release AGP
 * \return zero on success or a negative number on failure.
 *
 * Verifies the AGP device has been acquired and calls \c agp_backend_release.
 */
D
Dave Airlie 已提交
142
int drm_agp_release(drm_device_t * dev)
143
{
L
Linus Torvalds 已提交
144 145 146 147 148 149
	if (!dev->agp || !dev->agp->acquired)
		return -EINVAL;
	agp_backend_release(dev->agp->bridge);
	dev->agp->acquired = 0;
	return 0;
}
D
Dave Airlie 已提交
150

151
EXPORT_SYMBOL(drm_agp_release);
L
Linus Torvalds 已提交
152

153 154
int drm_agp_release_ioctl(struct inode *inode, struct file *filp,
			  unsigned int cmd, unsigned long arg)
L
Linus Torvalds 已提交
155
{
156 157
	drm_file_t *priv = filp->private_data;
	drm_device_t *dev = priv->head->dev;
D
Dave Airlie 已提交
158

159
	return drm_agp_release(dev);
L
Linus Torvalds 已提交
160 161 162 163
}

/**
 * Enable the AGP bus.
D
Dave Airlie 已提交
164
 *
165 166
 * \param dev DRM device that has previously acquired AGP.
 * \param mode Requested AGP mode.
L
Linus Torvalds 已提交
167 168 169
 * \return zero on success or a negative number on failure.
 *
 * Verifies the AGP device has been acquired but not enabled, and calls
170
 * \c agp_enable.
L
Linus Torvalds 已提交
171
 */
D
Dave Airlie 已提交
172
int drm_agp_enable(drm_device_t * dev, drm_agp_mode_t mode)
L
Linus Torvalds 已提交
173 174 175 176
{
	if (!dev->agp || !dev->agp->acquired)
		return -EINVAL;

D
Dave Airlie 已提交
177
	dev->agp->mode = mode.mode;
L
Linus Torvalds 已提交
178
	agp_enable(dev->agp->bridge, mode.mode);
D
Dave Airlie 已提交
179
	dev->agp->base = dev->agp->agp_info.aper_base;
L
Linus Torvalds 已提交
180 181 182
	dev->agp->enabled = 1;
	return 0;
}
D
Dave Airlie 已提交
183

184 185 186
EXPORT_SYMBOL(drm_agp_enable);

int drm_agp_enable_ioctl(struct inode *inode, struct file *filp,
D
Dave Airlie 已提交
187
			 unsigned int cmd, unsigned long arg)
188 189 190 191 192 193 194 195 196 197
{
	drm_file_t *priv = filp->private_data;
	drm_device_t *dev = priv->head->dev;
	drm_agp_mode_t mode;

	if (copy_from_user(&mode, (drm_agp_mode_t __user *) arg, sizeof(mode)))
		return -EFAULT;

	return drm_agp_enable(dev, mode);
}
L
Linus Torvalds 已提交
198 199 200 201 202 203 204 205 206

/**
 * Allocate AGP memory.
 *
 * \param inode device inode.
 * \param filp file pointer.
 * \param cmd command.
 * \param arg pointer to a drm_agp_buffer structure.
 * \return zero on success or a negative number on failure.
D
Dave Airlie 已提交
207
 *
L
Linus Torvalds 已提交
208 209 210 211
 * Verifies the AGP device is present and has been acquired, allocates the
 * memory via alloc_agp() and creates a drm_agp_mem entry for it.
 */
int drm_agp_alloc(struct inode *inode, struct file *filp,
D
Dave Airlie 已提交
212
		  unsigned int cmd, unsigned long arg)
L
Linus Torvalds 已提交
213
{
D
Dave Airlie 已提交
214 215
	drm_file_t *priv = filp->private_data;
	drm_device_t *dev = priv->head->dev;
L
Linus Torvalds 已提交
216
	drm_agp_buffer_t request;
D
Dave Airlie 已提交
217 218 219 220
	drm_agp_mem_t *entry;
	DRM_AGP_MEM *memory;
	unsigned long pages;
	u32 type;
L
Linus Torvalds 已提交
221 222 223 224 225 226 227 228 229
	drm_agp_buffer_t __user *argp = (void __user *)arg;

	if (!dev->agp || !dev->agp->acquired)
		return -EINVAL;
	if (copy_from_user(&request, argp, sizeof(request)))
		return -EFAULT;
	if (!(entry = drm_alloc(sizeof(*entry), DRM_MEM_AGPLISTS)))
		return -ENOMEM;

D
Dave Airlie 已提交
230
	memset(entry, 0, sizeof(*entry));
L
Linus Torvalds 已提交
231 232 233 234

	pages = (request.size + PAGE_SIZE - 1) / PAGE_SIZE;
	type = (u32) request.type;

235
	if (!(memory = drm_alloc_agp(dev, pages, type))) {
L
Linus Torvalds 已提交
236 237 238 239
		drm_free(entry, sizeof(*entry), DRM_MEM_AGPLISTS);
		return -ENOMEM;
	}

D
Dave Airlie 已提交
240 241 242 243 244 245
	entry->handle = (unsigned long)memory->key + 1;
	entry->memory = memory;
	entry->bound = 0;
	entry->pages = pages;
	entry->prev = NULL;
	entry->next = dev->agp->memory;
L
Linus Torvalds 已提交
246 247 248 249
	if (dev->agp->memory)
		dev->agp->memory->prev = entry;
	dev->agp->memory = entry;

D
Dave Airlie 已提交
250
	request.handle = entry->handle;
L
Linus Torvalds 已提交
251 252 253
	request.physical = memory->physical;

	if (copy_to_user(argp, &request, sizeof(request))) {
D
Dave Airlie 已提交
254
		dev->agp->memory = entry->next;
L
Linus Torvalds 已提交
255 256 257 258 259 260 261 262 263 264 265 266 267 268
		dev->agp->memory->prev = NULL;
		drm_free_agp(memory, pages);
		drm_free(entry, sizeof(*entry), DRM_MEM_AGPLISTS);
		return -EFAULT;
	}
	return 0;
}

/**
 * Search for the AGP memory entry associated with a handle.
 *
 * \param dev DRM device structure.
 * \param handle AGP memory handle.
 * \return pointer to the drm_agp_mem structure associated with \p handle.
D
Dave Airlie 已提交
269
 *
L
Linus Torvalds 已提交
270 271
 * Walks through drm_agp_head::memory until finding a matching handle.
 */
D
Dave Airlie 已提交
272 273
static drm_agp_mem_t *drm_agp_lookup_entry(drm_device_t * dev,
					   unsigned long handle)
L
Linus Torvalds 已提交
274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296
{
	drm_agp_mem_t *entry;

	for (entry = dev->agp->memory; entry; entry = entry->next) {
		if (entry->handle == handle)
			return entry;
	}
	return NULL;
}

/**
 * Unbind AGP memory from the GATT (ioctl).
 *
 * \param inode device inode.
 * \param filp file pointer.
 * \param cmd command.
 * \param arg pointer to a drm_agp_binding structure.
 * \return zero on success or a negative number on failure.
 *
 * Verifies the AGP device is present and acquired, looks-up the AGP memory
 * entry and passes it to the unbind_agp() function.
 */
int drm_agp_unbind(struct inode *inode, struct file *filp,
D
Dave Airlie 已提交
297
		   unsigned int cmd, unsigned long arg)
L
Linus Torvalds 已提交
298
{
D
Dave Airlie 已提交
299 300
	drm_file_t *priv = filp->private_data;
	drm_device_t *dev = priv->head->dev;
L
Linus Torvalds 已提交
301
	drm_agp_binding_t request;
D
Dave Airlie 已提交
302
	drm_agp_mem_t *entry;
L
Linus Torvalds 已提交
303 304 305 306
	int ret;

	if (!dev->agp || !dev->agp->acquired)
		return -EINVAL;
D
Dave Airlie 已提交
307 308
	if (copy_from_user
	    (&request, (drm_agp_binding_t __user *) arg, sizeof(request)))
L
Linus Torvalds 已提交
309 310 311 312 313 314 315
		return -EFAULT;
	if (!(entry = drm_agp_lookup_entry(dev, request.handle)))
		return -EINVAL;
	if (!entry->bound)
		return -EINVAL;
	ret = drm_unbind_agp(entry->memory);
	if (ret == 0)
D
Dave Airlie 已提交
316
		entry->bound = 0;
L
Linus Torvalds 已提交
317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333
	return ret;
}

/**
 * Bind AGP memory into the GATT (ioctl)
 *
 * \param inode device inode.
 * \param filp file pointer.
 * \param cmd command.
 * \param arg pointer to a drm_agp_binding structure.
 * \return zero on success or a negative number on failure.
 *
 * Verifies the AGP device is present and has been acquired and that no memory
 * is currently bound into the GATT. Looks-up the AGP memory entry and passes
 * it to bind_agp() function.
 */
int drm_agp_bind(struct inode *inode, struct file *filp,
D
Dave Airlie 已提交
334
		 unsigned int cmd, unsigned long arg)
L
Linus Torvalds 已提交
335
{
D
Dave Airlie 已提交
336 337
	drm_file_t *priv = filp->private_data;
	drm_device_t *dev = priv->head->dev;
L
Linus Torvalds 已提交
338
	drm_agp_binding_t request;
D
Dave Airlie 已提交
339 340 341
	drm_agp_mem_t *entry;
	int retcode;
	int page;
L
Linus Torvalds 已提交
342 343 344

	if (!dev->agp || !dev->agp->acquired)
		return -EINVAL;
D
Dave Airlie 已提交
345 346
	if (copy_from_user
	    (&request, (drm_agp_binding_t __user *) arg, sizeof(request)))
L
Linus Torvalds 已提交
347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375
		return -EFAULT;
	if (!(entry = drm_agp_lookup_entry(dev, request.handle)))
		return -EINVAL;
	if (entry->bound)
		return -EINVAL;
	page = (request.offset + PAGE_SIZE - 1) / PAGE_SIZE;
	if ((retcode = drm_bind_agp(entry->memory, page)))
		return retcode;
	entry->bound = dev->agp->base + (page << PAGE_SHIFT);
	DRM_DEBUG("base = 0x%lx entry->bound = 0x%lx\n",
		  dev->agp->base, entry->bound);
	return 0;
}

/**
 * Free AGP memory (ioctl).
 *
 * \param inode device inode.
 * \param filp file pointer.
 * \param cmd command.
 * \param arg pointer to a drm_agp_buffer structure.
 * \return zero on success or a negative number on failure.
 *
 * Verifies the AGP device is present and has been acquired and looks up the
 * AGP memory entry. If the memory it's currently bound, unbind it via
 * unbind_agp(). Frees it via free_agp() as well as the entry itself
 * and unlinks from the doubly linked list it's inserted in.
 */
int drm_agp_free(struct inode *inode, struct file *filp,
D
Dave Airlie 已提交
376
		 unsigned int cmd, unsigned long arg)
L
Linus Torvalds 已提交
377
{
D
Dave Airlie 已提交
378 379
	drm_file_t *priv = filp->private_data;
	drm_device_t *dev = priv->head->dev;
L
Linus Torvalds 已提交
380
	drm_agp_buffer_t request;
D
Dave Airlie 已提交
381
	drm_agp_mem_t *entry;
L
Linus Torvalds 已提交
382 383 384

	if (!dev->agp || !dev->agp->acquired)
		return -EINVAL;
D
Dave Airlie 已提交
385 386
	if (copy_from_user
	    (&request, (drm_agp_buffer_t __user *) arg, sizeof(request)))
L
Linus Torvalds 已提交
387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411
		return -EFAULT;
	if (!(entry = drm_agp_lookup_entry(dev, request.handle)))
		return -EINVAL;
	if (entry->bound)
		drm_unbind_agp(entry->memory);

	if (entry->prev)
		entry->prev->next = entry->next;
	else
		dev->agp->memory = entry->next;

	if (entry->next)
		entry->next->prev = entry->prev;

	drm_free_agp(entry->memory, entry->pages);
	drm_free(entry, sizeof(*entry), DRM_MEM_AGPLISTS);
	return 0;
}

/**
 * Initialize the AGP resources.
 *
 * \return pointer to a drm_agp_head structure.
 *
 */
D
Dave Airlie 已提交
412
drm_agp_head_t *drm_agp_init(drm_device_t * dev)
L
Linus Torvalds 已提交
413
{
D
Dave Airlie 已提交
414
	drm_agp_head_t *head = NULL;
L
Linus Torvalds 已提交
415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441

	if (!(head = drm_alloc(sizeof(*head), DRM_MEM_AGPLISTS)))
		return NULL;
	memset((void *)head, 0, sizeof(*head));
	head->bridge = agp_find_bridge(dev->pdev);
	if (!head->bridge) {
		if (!(head->bridge = agp_backend_acquire(dev->pdev))) {
			drm_free(head, sizeof(*head), DRM_MEM_AGPLISTS);
			return NULL;
		}
		agp_copy_info(head->bridge, &head->agp_info);
		agp_backend_release(head->bridge);
	} else {
		agp_copy_info(head->bridge, &head->agp_info);
	}
	if (head->agp_info.chipset == NOT_SUPPORTED) {
		drm_free(head, sizeof(*head), DRM_MEM_AGPLISTS);
		return NULL;
	}
	head->memory = NULL;
	head->cant_use_aperture = head->agp_info.cant_use_aperture;
	head->page_mask = head->agp_info.page_mask;

	return head;
}

/** Calls agp_allocate_memory() */
D
Dave Airlie 已提交
442 443
DRM_AGP_MEM *drm_agp_allocate_memory(struct agp_bridge_data * bridge,
				     size_t pages, u32 type)
L
Linus Torvalds 已提交
444 445 446 447 448
{
	return agp_allocate_memory(bridge, pages, type);
}

/** Calls agp_free_memory() */
D
Dave Airlie 已提交
449
int drm_agp_free_memory(DRM_AGP_MEM * handle)
L
Linus Torvalds 已提交
450 451 452 453 454 455 456 457
{
	if (!handle)
		return 0;
	agp_free_memory(handle);
	return 1;
}

/** Calls agp_bind_memory() */
D
Dave Airlie 已提交
458
int drm_agp_bind_memory(DRM_AGP_MEM * handle, off_t start)
L
Linus Torvalds 已提交
459 460 461 462 463
{
	if (!handle)
		return -EINVAL;
	return agp_bind_memory(handle, start);
}
D
Dave Airlie 已提交
464

465
EXPORT_SYMBOL(drm_agp_bind_memory);
L
Linus Torvalds 已提交
466 467

/** Calls agp_unbind_memory() */
D
Dave Airlie 已提交
468
int drm_agp_unbind_memory(DRM_AGP_MEM * handle)
L
Linus Torvalds 已提交
469 470 471 472 473 474
{
	if (!handle)
		return -EINVAL;
	return agp_unbind_memory(handle);
}

D
Dave Airlie 已提交
475
#endif				/* __OS_HAS_AGP */