Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux
1
fork

Configure Feed

Select the types of activity you want to include in your feed.

at 989a7241df87526bfef0396567e71ebe53a84ae4 293 lines 7.7 kB view raw
1/** 2 * \file drm_stub.h 3 * Stub support 4 * 5 * \author Rickard E. (Rik) Faith <faith@valinux.com> 6 */ 7 8/* 9 * Created: Fri Jan 19 10:48:35 2001 by faith@acm.org 10 * 11 * Copyright 2001 VA Linux Systems, Inc., Sunnyvale, California. 12 * All Rights Reserved. 13 * 14 * Permission is hereby granted, free of charge, to any person obtaining a 15 * copy of this software and associated documentation files (the "Software"), 16 * to deal in the Software without restriction, including without limitation 17 * the rights to use, copy, modify, merge, publish, distribute, sublicense, 18 * and/or sell copies of the Software, and to permit persons to whom the 19 * Software is furnished to do so, subject to the following conditions: 20 * 21 * The above copyright notice and this permission notice (including the next 22 * paragraph) shall be included in all copies or substantial portions of the 23 * Software. 24 * 25 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 26 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 27 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 28 * PRECISION INSIGHT AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR 29 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, 30 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER 31 * DEALINGS IN THE SOFTWARE. 32 */ 33 34#include <linux/module.h> 35#include <linux/moduleparam.h> 36#include "drmP.h" 37#include "drm_core.h" 38 39unsigned int drm_cards_limit = 16; /* Enough for one machine */ 40unsigned int drm_debug = 0; /* 1 to enable debug output */ 41EXPORT_SYMBOL(drm_debug); 42 43MODULE_AUTHOR(CORE_AUTHOR); 44MODULE_DESCRIPTION(CORE_DESC); 45MODULE_LICENSE("GPL and additional rights"); 46MODULE_PARM_DESC(cards_limit, "Maximum number of graphics cards"); 47MODULE_PARM_DESC(debug, "Enable debug output"); 48 49module_param_named(cards_limit, drm_cards_limit, int, 0444); 50module_param_named(debug, drm_debug, int, 0600); 51 52struct drm_head **drm_heads; 53struct class *drm_class; 54struct proc_dir_entry *drm_proc_root; 55 56static int drm_fill_in_dev(struct drm_device * dev, struct pci_dev *pdev, 57 const struct pci_device_id *ent, 58 struct drm_driver *driver) 59{ 60 int retcode; 61 62 INIT_LIST_HEAD(&dev->filelist); 63 INIT_LIST_HEAD(&dev->ctxlist); 64 INIT_LIST_HEAD(&dev->vmalist); 65 INIT_LIST_HEAD(&dev->maplist); 66 67 spin_lock_init(&dev->count_lock); 68 spin_lock_init(&dev->drw_lock); 69 spin_lock_init(&dev->tasklet_lock); 70 spin_lock_init(&dev->lock.spinlock); 71 init_timer(&dev->timer); 72 mutex_init(&dev->struct_mutex); 73 mutex_init(&dev->ctxlist_mutex); 74 75 idr_init(&dev->drw_idr); 76 77 dev->pdev = pdev; 78 dev->pci_device = pdev->device; 79 dev->pci_vendor = pdev->vendor; 80 81#ifdef __alpha__ 82 dev->hose = pdev->sysdata; 83#endif 84 dev->irq = pdev->irq; 85 86 if (drm_ht_create(&dev->map_hash, 12)) { 87 return -ENOMEM; 88 } 89 90 /* the DRM has 6 basic counters */ 91 dev->counters = 6; 92 dev->types[0] = _DRM_STAT_LOCK; 93 dev->types[1] = _DRM_STAT_OPENS; 94 dev->types[2] = _DRM_STAT_CLOSES; 95 dev->types[3] = _DRM_STAT_IOCTLS; 96 dev->types[4] = _DRM_STAT_LOCKS; 97 dev->types[5] = _DRM_STAT_UNLOCKS; 98 99 dev->driver = driver; 100 101 if (drm_core_has_AGP(dev)) { 102 if (drm_device_is_agp(dev)) 103 dev->agp = drm_agp_init(dev); 104 if (drm_core_check_feature(dev, DRIVER_REQUIRE_AGP) 105 && (dev->agp == NULL)) { 106 DRM_ERROR("Cannot initialize the agpgart module.\n"); 107 retcode = -EINVAL; 108 goto error_out_unreg; 109 } 110 if (drm_core_has_MTRR(dev)) { 111 if (dev->agp) 112 dev->agp->agp_mtrr = 113 mtrr_add(dev->agp->agp_info.aper_base, 114 dev->agp->agp_info.aper_size * 115 1024 * 1024, MTRR_TYPE_WRCOMB, 1); 116 } 117 } 118 119 if (dev->driver->load) 120 if ((retcode = dev->driver->load(dev, ent->driver_data))) 121 goto error_out_unreg; 122 123 retcode = drm_ctxbitmap_init(dev); 124 if (retcode) { 125 DRM_ERROR("Cannot allocate memory for context bitmap.\n"); 126 goto error_out_unreg; 127 } 128 129 return 0; 130 131 error_out_unreg: 132 drm_lastclose(dev); 133 return retcode; 134} 135 136 137/** 138 * Get a secondary minor number. 139 * 140 * \param dev device data structure 141 * \param sec-minor structure to hold the assigned minor 142 * \return negative number on failure. 143 * 144 * Search an empty entry and initialize it to the given parameters, and 145 * create the proc init entry via proc_init(). This routines assigns 146 * minor numbers to secondary heads of multi-headed cards 147 */ 148static int drm_get_head(struct drm_device * dev, struct drm_head * head) 149{ 150 struct drm_head **heads = drm_heads; 151 int ret; 152 int minor; 153 154 DRM_DEBUG("\n"); 155 156 for (minor = 0; minor < drm_cards_limit; minor++, heads++) { 157 if (!*heads) { 158 159 *head = (struct drm_head) { 160 .dev = dev,.device = 161 MKDEV(DRM_MAJOR, minor),.minor = minor,}; 162 163 if ((ret = 164 drm_proc_init(dev, minor, drm_proc_root, 165 &head->dev_root))) { 166 printk(KERN_ERR 167 "DRM: Failed to initialize /proc/dri.\n"); 168 goto err_g1; 169 } 170 171 ret = drm_sysfs_device_add(dev, head); 172 if (ret) { 173 printk(KERN_ERR 174 "DRM: Error sysfs_device_add.\n"); 175 goto err_g2; 176 } 177 *heads = head; 178 179 DRM_DEBUG("new minor assigned %d\n", minor); 180 return 0; 181 } 182 } 183 DRM_ERROR("out of minors\n"); 184 return -ENOMEM; 185 err_g2: 186 drm_proc_cleanup(minor, drm_proc_root, head->dev_root); 187 err_g1: 188 *head = (struct drm_head) { 189 .dev = NULL}; 190 return ret; 191} 192 193/** 194 * Register. 195 * 196 * \param pdev - PCI device structure 197 * \param ent entry from the PCI ID table with device type flags 198 * \return zero on success or a negative number on failure. 199 * 200 * Attempt to gets inter module "drm" information. If we are first 201 * then register the character device and inter module information. 202 * Try and register, if we fail to register, backout previous work. 203 */ 204int drm_get_dev(struct pci_dev *pdev, const struct pci_device_id *ent, 205 struct drm_driver *driver) 206{ 207 struct drm_device *dev; 208 int ret; 209 210 DRM_DEBUG("\n"); 211 212 dev = drm_calloc(1, sizeof(*dev), DRM_MEM_STUB); 213 if (!dev) 214 return -ENOMEM; 215 216 ret = pci_enable_device(pdev); 217 if (ret) 218 goto err_g1; 219 220 pci_set_master(pdev); 221 if ((ret = drm_fill_in_dev(dev, pdev, ent, driver))) { 222 printk(KERN_ERR "DRM: Fill_in_dev failed.\n"); 223 goto err_g2; 224 } 225 if ((ret = drm_get_head(dev, &dev->primary))) 226 goto err_g2; 227 228 DRM_INFO("Initialized %s %d.%d.%d %s on minor %d\n", 229 driver->name, driver->major, driver->minor, driver->patchlevel, 230 driver->date, dev->primary.minor); 231 232 return 0; 233 234err_g2: 235 pci_disable_device(pdev); 236err_g1: 237 drm_free(dev, sizeof(*dev), DRM_MEM_STUB); 238 return ret; 239} 240 241/** 242 * Put a device minor number. 243 * 244 * \param dev device data structure 245 * \return always zero 246 * 247 * Cleans up the proc resources. If it is the last minor then release the foreign 248 * "drm" data, otherwise unregisters the "drm" data, frees the dev list and 249 * unregisters the character device. 250 */ 251int drm_put_dev(struct drm_device * dev) 252{ 253 DRM_DEBUG("release primary %s\n", dev->driver->pci_driver.name); 254 255 if (dev->unique) { 256 drm_free(dev->unique, strlen(dev->unique) + 1, DRM_MEM_DRIVER); 257 dev->unique = NULL; 258 dev->unique_len = 0; 259 } 260 if (dev->devname) { 261 drm_free(dev->devname, strlen(dev->devname) + 1, 262 DRM_MEM_DRIVER); 263 dev->devname = NULL; 264 } 265 drm_free(dev, sizeof(*dev), DRM_MEM_STUB); 266 return 0; 267} 268 269/** 270 * Put a secondary minor number. 271 * 272 * \param sec_minor - structure to be released 273 * \return always zero 274 * 275 * Cleans up the proc resources. Not legal for this to be the 276 * last minor released. 277 * 278 */ 279int drm_put_head(struct drm_head * head) 280{ 281 int minor = head->minor; 282 283 DRM_DEBUG("release secondary minor %d\n", minor); 284 285 drm_proc_cleanup(minor, drm_proc_root, head->dev_root); 286 drm_sysfs_device_remove(head->dev); 287 288 *head = (struct drm_head) {.dev = NULL}; 289 290 drm_heads[minor] = NULL; 291 292 return 0; 293}