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 v2.6.19-rc2 284 lines 7.6 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 52drm_head_t **drm_heads; 53struct class *drm_class; 54struct proc_dir_entry *drm_proc_root; 55 56static int drm_fill_in_dev(drm_device_t * dev, struct pci_dev *pdev, 57 const struct pci_device_id *ent, 58 struct drm_driver *driver) 59{ 60 int retcode; 61 62 spin_lock_init(&dev->count_lock); 63 init_timer(&dev->timer); 64 mutex_init(&dev->struct_mutex); 65 mutex_init(&dev->ctxlist_mutex); 66 67 dev->pdev = pdev; 68 dev->pci_device = pdev->device; 69 dev->pci_vendor = pdev->vendor; 70 71#ifdef __alpha__ 72 dev->hose = pdev->sysdata; 73#endif 74 dev->irq = pdev->irq; 75 76 dev->maplist = drm_calloc(1, sizeof(*dev->maplist), DRM_MEM_MAPS); 77 if (dev->maplist == NULL) 78 return -ENOMEM; 79 INIT_LIST_HEAD(&dev->maplist->head); 80 if (drm_ht_create(&dev->map_hash, 12)) { 81 drm_free(dev->maplist, sizeof(*dev->maplist), DRM_MEM_MAPS); 82 return -ENOMEM; 83 } 84 85 /* the DRM has 6 basic counters */ 86 dev->counters = 6; 87 dev->types[0] = _DRM_STAT_LOCK; 88 dev->types[1] = _DRM_STAT_OPENS; 89 dev->types[2] = _DRM_STAT_CLOSES; 90 dev->types[3] = _DRM_STAT_IOCTLS; 91 dev->types[4] = _DRM_STAT_LOCKS; 92 dev->types[5] = _DRM_STAT_UNLOCKS; 93 94 dev->driver = driver; 95 96 if (dev->driver->load) 97 if ((retcode = dev->driver->load(dev, ent->driver_data))) 98 goto error_out_unreg; 99 100 if (drm_core_has_AGP(dev)) { 101 if (drm_device_is_agp(dev)) 102 dev->agp = drm_agp_init(dev); 103 if (drm_core_check_feature(dev, DRIVER_REQUIRE_AGP) 104 && (dev->agp == NULL)) { 105 DRM_ERROR("Cannot initialize the agpgart module.\n"); 106 retcode = -EINVAL; 107 goto error_out_unreg; 108 } 109 if (drm_core_has_MTRR(dev)) { 110 if (dev->agp) 111 dev->agp->agp_mtrr = 112 mtrr_add(dev->agp->agp_info.aper_base, 113 dev->agp->agp_info.aper_size * 114 1024 * 1024, MTRR_TYPE_WRCOMB, 1); 115 } 116 } 117 118 retcode = drm_ctxbitmap_init(dev); 119 if (retcode) { 120 DRM_ERROR("Cannot allocate memory for context bitmap.\n"); 121 goto error_out_unreg; 122 } 123 124 return 0; 125 126 error_out_unreg: 127 drm_lastclose(dev); 128 return retcode; 129} 130 131 132/** 133 * Get a secondary minor number. 134 * 135 * \param dev device data structure 136 * \param sec-minor structure to hold the assigned minor 137 * \return negative number on failure. 138 * 139 * Search an empty entry and initialize it to the given parameters, and 140 * create the proc init entry via proc_init(). This routines assigns 141 * minor numbers to secondary heads of multi-headed cards 142 */ 143static int drm_get_head(drm_device_t * dev, drm_head_t * head) 144{ 145 drm_head_t **heads = drm_heads; 146 int ret; 147 int minor; 148 149 DRM_DEBUG("\n"); 150 151 for (minor = 0; minor < drm_cards_limit; minor++, heads++) { 152 if (!*heads) { 153 154 *head = (drm_head_t) { 155 .dev = dev,.device = 156 MKDEV(DRM_MAJOR, minor),.minor = minor,}; 157 158 if ((ret = 159 drm_proc_init(dev, minor, drm_proc_root, 160 &head->dev_root))) { 161 printk(KERN_ERR 162 "DRM: Failed to initialize /proc/dri.\n"); 163 goto err_g1; 164 } 165 166 head->dev_class = drm_sysfs_device_add(drm_class, head); 167 if (IS_ERR(head->dev_class)) { 168 printk(KERN_ERR 169 "DRM: Error sysfs_device_add.\n"); 170 ret = PTR_ERR(head->dev_class); 171 goto err_g2; 172 } 173 *heads = head; 174 175 DRM_DEBUG("new minor assigned %d\n", minor); 176 return 0; 177 } 178 } 179 DRM_ERROR("out of minors\n"); 180 return -ENOMEM; 181 err_g2: 182 drm_proc_cleanup(minor, drm_proc_root, head->dev_root); 183 err_g1: 184 *head = (drm_head_t) { 185 .dev = NULL}; 186 return ret; 187} 188 189/** 190 * Register. 191 * 192 * \param pdev - PCI device structure 193 * \param ent entry from the PCI ID table with device type flags 194 * \return zero on success or a negative number on failure. 195 * 196 * Attempt to gets inter module "drm" information. If we are first 197 * then register the character device and inter module information. 198 * Try and register, if we fail to register, backout previous work. 199 */ 200int drm_get_dev(struct pci_dev *pdev, const struct pci_device_id *ent, 201 struct drm_driver *driver) 202{ 203 drm_device_t *dev; 204 int ret; 205 206 DRM_DEBUG("\n"); 207 208 dev = drm_calloc(1, sizeof(*dev), DRM_MEM_STUB); 209 if (!dev) 210 return -ENOMEM; 211 212 pci_enable_device(pdev); 213 214 if ((ret = drm_fill_in_dev(dev, pdev, ent, driver))) { 215 printk(KERN_ERR "DRM: Fill_in_dev failed.\n"); 216 goto err_g1; 217 } 218 if ((ret = drm_get_head(dev, &dev->primary))) 219 goto err_g1; 220 221 DRM_INFO("Initialized %s %d.%d.%d %s on minor %d\n", 222 driver->name, driver->major, driver->minor, driver->patchlevel, 223 driver->date, dev->primary.minor); 224 225 return 0; 226 227 err_g1: 228 drm_free(dev, sizeof(*dev), DRM_MEM_STUB); 229 return ret; 230} 231 232/** 233 * Put a device minor number. 234 * 235 * \param dev device data structure 236 * \return always zero 237 * 238 * Cleans up the proc resources. If it is the last minor then release the foreign 239 * "drm" data, otherwise unregisters the "drm" data, frees the dev list and 240 * unregisters the character device. 241 */ 242int drm_put_dev(drm_device_t * dev) 243{ 244 DRM_DEBUG("release primary %s\n", dev->driver->pci_driver.name); 245 246 if (dev->unique) { 247 drm_free(dev->unique, strlen(dev->unique) + 1, DRM_MEM_DRIVER); 248 dev->unique = NULL; 249 dev->unique_len = 0; 250 } 251 if (dev->devname) { 252 drm_free(dev->devname, strlen(dev->devname) + 1, 253 DRM_MEM_DRIVER); 254 dev->devname = NULL; 255 } 256 drm_free(dev, sizeof(*dev), DRM_MEM_STUB); 257 return 0; 258} 259 260/** 261 * Put a secondary minor number. 262 * 263 * \param sec_minor - structure to be released 264 * \return always zero 265 * 266 * Cleans up the proc resources. Not legal for this to be the 267 * last minor released. 268 * 269 */ 270int drm_put_head(drm_head_t * head) 271{ 272 int minor = head->minor; 273 274 DRM_DEBUG("release secondary minor %d\n", minor); 275 276 drm_proc_cleanup(minor, drm_proc_root, head->dev_root); 277 drm_sysfs_device_remove(head->dev_class); 278 279 *head = (drm_head_t) {.dev = NULL}; 280 281 drm_heads[minor] = NULL; 282 283 return 0; 284}