drm/i915: Correct duplicated/misplaced GT register definitions
[linux-block.git] / drivers / gpu / drm / drm_pci.c
CommitLineData
1da177e4 1/*
96de0e25 2 * Copyright 2003 José Fonseca.
1da177e4
LT
3 * Copyright 2003 Leif Delgass.
4 * All Rights Reserved.
5 *
6 * Permission is hereby granted, free of charge, to any person obtaining a
7 * copy of this software and associated documentation files (the "Software"),
8 * to deal in the Software without restriction, including without limitation
9 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
10 * and/or sell copies of the Software, and to permit persons to whom the
11 * Software is furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice (including the next
14 * paragraph) shall be included in all copies or substantial portions of the
15 * Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
20 * AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
21 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22 * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23 */
24
195b3a2d 25#include <linux/dma-mapping.h>
2d1a8a48 26#include <linux/export.h>
8dbe1b4a
LP
27#include <linux/list.h>
28#include <linux/mutex.h>
0500c04e
SR
29#include <linux/pci.h>
30#include <linux/slab.h>
31
32#include <drm/drm.h>
0500c04e 33#include <drm/drm_drv.h>
0500c04e
SR
34#include <drm/drm_print.h>
35
43fc884e 36#include "drm_internal.h"
ba8286fa 37#include "drm_legacy.h"
1da177e4 38
33775336 39#ifdef CONFIG_DRM_LEGACY
8dbe1b4a
LP
40/* List of devices hanging off drivers with stealth attach. */
41static LIST_HEAD(legacy_dev_list);
42static DEFINE_MUTEX(legacy_dev_list_lock);
33775336 43#endif
1da177e4 44
8410ea3b
DA
45static int drm_get_pci_domain(struct drm_device *dev)
46{
47#ifndef __alpha__
48 /* For historical reasons, drm_get_pci_domain() is busticated
49 * on most archs and has to remain so for userspace interface
50 * < 1.4, except on alpha which was right from the beginning
51 */
52 if (dev->if_version < 0x10004)
53 return 0;
54#endif /* __alpha__ */
55
36b73b05 56 return pci_domain_nr(to_pci_dev(dev->dev)->bus);
8410ea3b
DA
57}
58
915b4d11 59int drm_pci_set_busid(struct drm_device *dev, struct drm_master *master)
8410ea3b 60{
36b73b05
TZ
61 struct pci_dev *pdev = to_pci_dev(dev->dev);
62
d0a39164
DH
63 master->unique = kasprintf(GFP_KERNEL, "pci:%04x:%02x:%02x.%d",
64 drm_get_pci_domain(dev),
36b73b05
TZ
65 pdev->bus->number,
66 PCI_SLOT(pdev->devfn),
67 PCI_FUNC(pdev->devfn));
d0a39164 68 if (!master->unique)
8410ea3b
DA
69 return -ENOMEM;
70
d0a39164 71 master->unique_len = strlen(master->unique);
8410ea3b 72 return 0;
8410ea3b
DA
73}
74
644adc3d
TZ
75#ifdef CONFIG_DRM_LEGACY
76
77static int drm_legacy_pci_irq_by_busid(struct drm_device *dev, struct drm_irq_busid *p)
8410ea3b 78{
36b73b05
TZ
79 struct pci_dev *pdev = to_pci_dev(dev->dev);
80
8410ea3b 81 if ((p->busnum >> 8) != drm_get_pci_domain(dev) ||
36b73b05
TZ
82 (p->busnum & 0xff) != pdev->bus->number ||
83 p->devnum != PCI_SLOT(pdev->devfn) || p->funcnum != PCI_FUNC(pdev->devfn))
8410ea3b
DA
84 return -EINVAL;
85
36b73b05 86 p->irq = pdev->irq;
8410ea3b
DA
87
88 DRM_DEBUG("%d:%d:%d => IRQ %d\n", p->busnum, p->devnum, p->funcnum,
89 p->irq);
90 return 0;
91}
92
eaaf8f0f 93/**
8df4ec51 94 * drm_legacy_irq_by_busid - Get interrupt from bus ID
c6a1af8a
TR
95 * @dev: DRM device
96 * @data: IOCTL parameter pointing to a drm_irq_busid structure
97 * @file_priv: DRM file private.
eaaf8f0f
DV
98 *
99 * Finds the PCI device with the specified bus id and gets its IRQ number.
100 * This IOCTL is deprecated, and will now return EINVAL for any busid not equal
101 * to that of the device that this DRM instance attached to.
c6a1af8a
TR
102 *
103 * Return: 0 on success or a negative error code on failure.
eaaf8f0f 104 */
8df4ec51
DV
105int drm_legacy_irq_by_busid(struct drm_device *dev, void *data,
106 struct drm_file *file_priv)
eaaf8f0f
DV
107{
108 struct drm_irq_busid *p = data;
109
fa538645 110 if (!drm_core_check_feature(dev, DRIVER_LEGACY))
69fdf420 111 return -EOPNOTSUPP;
eaaf8f0f
DV
112
113 /* UMS was only ever support on PCI devices. */
36b73b05 114 if (WARN_ON(!dev_is_pci(dev->dev)))
eaaf8f0f
DV
115 return -EINVAL;
116
117 if (!drm_core_check_feature(dev, DRIVER_HAVE_IRQ))
69fdf420 118 return -EOPNOTSUPP;
eaaf8f0f 119
644adc3d 120 return drm_legacy_pci_irq_by_busid(dev, p);
eaaf8f0f
DV
121}
122
6bff2279 123void drm_legacy_pci_agp_destroy(struct drm_device *dev)
ee21ec77
CW
124{
125 if (dev->agp) {
126 arch_phys_wc_del(dev->agp->agp_mtrr);
127 drm_legacy_agp_clear(dev);
128 kfree(dev->agp);
129 dev->agp = NULL;
130 }
131}
132
6bff2279 133static void drm_legacy_pci_agp_init(struct drm_device *dev)
8410ea3b 134{
d9906753 135 if (drm_core_check_feature(dev, DRIVER_USE_AGP)) {
36b73b05 136 if (pci_find_capability(to_pci_dev(dev->dev), PCI_CAP_ID_AGP))
04dfe19a 137 dev->agp = drm_legacy_agp_init(dev);
28185647
DV
138 if (dev->agp) {
139 dev->agp->agp_mtrr = arch_phys_wc_add(
140 dev->agp->agp_info.aper_base,
141 dev->agp->agp_info.aper_size *
142 1024 * 1024);
8410ea3b
DA
143 }
144 }
8410ea3b
DA
145}
146
6bff2279
TZ
147static int drm_legacy_get_pci_dev(struct pci_dev *pdev,
148 const struct pci_device_id *ent,
149 const struct drm_driver *driver)
dcdb1674
JC
150{
151 struct drm_device *dev;
152 int ret;
153
154 DRM_DEBUG("\n");
155
1bb72532 156 dev = drm_dev_alloc(driver, &pdev->dev);
0f288605
TG
157 if (IS_ERR(dev))
158 return PTR_ERR(dev);
dcdb1674
JC
159
160 ret = pci_enable_device(pdev);
161 if (ret)
c22f0ace 162 goto err_free;
dcdb1674 163
dcdb1674
JC
164#ifdef __alpha__
165 dev->hose = pdev->sysdata;
166#endif
167
6bff2279 168 drm_legacy_pci_agp_init(dev);
2c695fa0 169
c22f0ace
DH
170 ret = drm_dev_register(dev, ent->driver_data);
171 if (ret)
2c695fa0 172 goto err_agp;
dcdb1674 173
8dbe1b4a
LP
174 if (drm_core_check_feature(dev, DRIVER_LEGACY)) {
175 mutex_lock(&legacy_dev_list_lock);
176 list_add_tail(&dev->legacy_dev_list, &legacy_dev_list);
177 mutex_unlock(&legacy_dev_list_lock);
178 }
b3f2333d 179
dcdb1674
JC
180 return 0;
181
2c695fa0 182err_agp:
6bff2279 183 drm_legacy_pci_agp_destroy(dev);
dcdb1674 184 pci_disable_device(pdev);
c22f0ace 185err_free:
ffeeeed0 186 drm_dev_put(dev);
dcdb1674
JC
187 return ret;
188}
1be9d5f0 189
dcdb1674 190/**
10631d72 191 * drm_legacy_pci_init - shadow-attach a legacy DRM PCI driver
c6a1af8a
TR
192 * @driver: DRM device driver
193 * @pdriver: PCI device driver
dcdb1674 194 *
10631d72 195 * This is only used by legacy dri1 drivers and deprecated.
6e3f797c 196 *
c6a1af8a 197 * Return: 0 on success or a negative error code on failure.
dcdb1674 198 */
b1dda997
LP
199int drm_legacy_pci_init(const struct drm_driver *driver,
200 struct pci_driver *pdriver)
dcdb1674
JC
201{
202 struct pci_dev *pdev = NULL;
203 const struct pci_device_id *pid;
204 int i;
205
8410ea3b
DA
206 DRM_DEBUG("\n");
207
10631d72
DV
208 if (WARN_ON(!(driver->driver_features & DRIVER_LEGACY)))
209 return -EINVAL;
dcdb1674
JC
210
211 /* If not using KMS, fall back to stealth mode manual scanning. */
8410ea3b
DA
212 for (i = 0; pdriver->id_table[i].vendor != 0; i++) {
213 pid = &pdriver->id_table[i];
dcdb1674
JC
214
215 /* Loop around setting up a DRM device for each PCI device
216 * matching our ID and device class. If we had the internal
217 * function that pci_get_subsys and pci_get_class used, we'd
218 * be able to just pass pid in instead of doing a two-stage
219 * thing.
220 */
221 pdev = NULL;
222 while ((pdev =
223 pci_get_subsys(pid->vendor, pid->device, pid->subvendor,
224 pid->subdevice, pdev)) != NULL) {
225 if ((pdev->class & pid->class_mask) != pid->class)
226 continue;
227
228 /* stealth mode requires a manual probe */
229 pci_dev_get(pdev);
6bff2279 230 drm_legacy_get_pci_dev(pdev, pid, driver);
dcdb1674
JC
231 }
232 }
233 return 0;
234}
10631d72 235EXPORT_SYMBOL(drm_legacy_pci_init);
dcdb1674 236
c6a1af8a 237/**
10631d72 238 * drm_legacy_pci_exit - unregister shadow-attach legacy DRM driver
c6a1af8a
TR
239 * @driver: DRM device driver
240 * @pdriver: PCI device driver
241 *
10631d72
DV
242 * Unregister a DRM driver shadow-attached through drm_legacy_pci_init(). This
243 * is deprecated and only used by dri1 drivers.
c6a1af8a 244 */
b1dda997
LP
245void drm_legacy_pci_exit(const struct drm_driver *driver,
246 struct pci_driver *pdriver)
93711d8b
BH
247{
248 struct drm_device *dev, *tmp;
948de842 249
93711d8b
BH
250 DRM_DEBUG("\n");
251
fa538645 252 if (!(driver->driver_features & DRIVER_LEGACY)) {
10631d72 253 WARN_ON(1);
93711d8b 254 } else {
8dbe1b4a
LP
255 mutex_lock(&legacy_dev_list_lock);
256 list_for_each_entry_safe(dev, tmp, &legacy_dev_list,
b3f2333d 257 legacy_dev_list) {
8dbe1b4a
LP
258 if (dev->driver == driver) {
259 list_del(&dev->legacy_dev_list);
260 drm_put_dev(dev);
261 }
b3f2333d 262 }
8dbe1b4a 263 mutex_unlock(&legacy_dev_list_lock);
93711d8b
BH
264 }
265 DRM_INFO("Module unloaded\n");
266}
10631d72 267EXPORT_SYMBOL(drm_legacy_pci_exit);
1be9d5f0
TZ
268
269#endif