xref: /onnv-gate/usr/src/uts/common/io/drm/drm_drv.c (revision 11260:eb8c6f2097e8)
12820Skz151634 /*
22820Skz151634  * drm_drv.h -- Generic driver template -*- linux-c -*-
32820Skz151634  * Created: Thu Nov 23 03:10:50 2000 by gareth@valinux.com
42820Skz151634  */
52820Skz151634 /*
62820Skz151634  * Copyright 1999, 2000 Precision Insight, Inc., Cedar Park, Texas.
72820Skz151634  * Copyright 2000 VA Linux Systems, Inc., Sunnyvale, California.
8*11260SMiao.Chen@Sun.COM  * Copyright (c) 2009, Intel Corporation.
92820Skz151634  * All Rights Reserved.
102820Skz151634  *
112820Skz151634  * Permission is hereby granted, free of charge, to any person obtaining a
122820Skz151634  * copy of this software and associated documentation files (the "Software"),
132820Skz151634  * to deal in the Software without restriction, including without limitation
142820Skz151634  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
152820Skz151634  * and/or sell copies of the Software, and to permit persons to whom the
162820Skz151634  * Software is furnished to do so, subject to the following conditions:
172820Skz151634  *
182820Skz151634  * The above copyright notice and this permission notice (including the next
192820Skz151634  * paragraph) shall be included in all copies or substantial portions of the
202820Skz151634  * Software.
212820Skz151634  *
222820Skz151634  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
232820Skz151634  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
242820Skz151634  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
252820Skz151634  * VA LINUX SYSTEMS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
262820Skz151634  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
272820Skz151634  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
282820Skz151634  * OTHER DEALINGS IN THE SOFTWARE.
292820Skz151634  *
302820Skz151634  * Authors:
312820Skz151634  *    Rickard E. (Rik) Faith <faith@valinux.com>
322820Skz151634  *    Gareth Hughes <gareth@valinux.com>
332820Skz151634  *
342820Skz151634  */
352820Skz151634 
364194Szw161486 /*
378832SMiao.Chen@Sun.COM  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
384194Szw161486  * Use is subject to license terms.
394194Szw161486  */
404194Szw161486 
412820Skz151634 #include "drmP.h"
422820Skz151634 #include "drm.h"
432820Skz151634 #include "drm_sarea.h"
442820Skz151634 
452820Skz151634 int drm_debug_flag = 1;
462820Skz151634 
472820Skz151634 #define	DRIVER_IOCTL_COUNT	256
488019SMiao.Chen@Sun.COM drm_ioctl_desc_t drm_ioctls[DRIVER_IOCTL_COUNT] = {
498019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_VERSION)] =
508019SMiao.Chen@Sun.COM 	    {drm_version, 0},
518019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_GET_UNIQUE)] =
528019SMiao.Chen@Sun.COM 	    {drm_getunique, 0},
538019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_GET_MAGIC)] =
548019SMiao.Chen@Sun.COM 	    {drm_getmagic, 0},
558019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_IRQ_BUSID)] =
568019SMiao.Chen@Sun.COM 	    {drm_irq_by_busid, DRM_MASTER|DRM_ROOT_ONLY},
578019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_GET_MAP)] =
588019SMiao.Chen@Sun.COM 	    {drm_getmap, 0},
598019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_GET_CLIENT)] =
608019SMiao.Chen@Sun.COM 	    {drm_getclient, 0},
618019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_GET_STATS)] =
628019SMiao.Chen@Sun.COM 	    {drm_getstats, 0},
638019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_SET_VERSION)] =
648019SMiao.Chen@Sun.COM 	    {drm_setversion, DRM_MASTER|DRM_ROOT_ONLY},
65*11260SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_MODESET_CTL)] =
66*11260SMiao.Chen@Sun.COM 	    {drm_modeset_ctl, 0},
67*11260SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_GEM_CLOSE)] =
68*11260SMiao.Chen@Sun.COM 	    {drm_gem_close_ioctl, 0},
69*11260SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_GEM_FLINK)] =
70*11260SMiao.Chen@Sun.COM 	    {drm_gem_flink_ioctl, DRM_AUTH},
71*11260SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_GEM_OPEN)] =
72*11260SMiao.Chen@Sun.COM 	    {drm_gem_open_ioctl, DRM_AUTH},
738019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_SET_UNIQUE)] =
748019SMiao.Chen@Sun.COM 	    {drm_setunique, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
758019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_BLOCK)] =
768019SMiao.Chen@Sun.COM 	    {drm_noop, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
778019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_UNBLOCK)] =
788019SMiao.Chen@Sun.COM 	    {drm_noop, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
798019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_AUTH_MAGIC)] =
808019SMiao.Chen@Sun.COM 	    {drm_authmagic, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
818019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_ADD_MAP)] =
828019SMiao.Chen@Sun.COM 	    {drm_addmap_ioctl, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
838019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_RM_MAP)] =
848019SMiao.Chen@Sun.COM 	    {drm_rmmap_ioctl, DRM_AUTH},
858019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_SET_SAREA_CTX)] =
868019SMiao.Chen@Sun.COM 	    {drm_setsareactx, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
878019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_GET_SAREA_CTX)] =
888019SMiao.Chen@Sun.COM 	    {drm_getsareactx, DRM_AUTH},
898019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_ADD_CTX)] =
908019SMiao.Chen@Sun.COM 	    {drm_addctx, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
918019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_RM_CTX)] =
928019SMiao.Chen@Sun.COM 	    {drm_rmctx, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
938019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_MOD_CTX)] =
948019SMiao.Chen@Sun.COM 	    {drm_modctx, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
958019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_GET_CTX)] =
968019SMiao.Chen@Sun.COM 	    {drm_getctx, DRM_AUTH},
978019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_SWITCH_CTX)] =
988019SMiao.Chen@Sun.COM 	    {drm_switchctx, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
998019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_NEW_CTX)] =
1008019SMiao.Chen@Sun.COM 	    {drm_newctx, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1018019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_RES_CTX)] =
1028019SMiao.Chen@Sun.COM 	    {drm_resctx, DRM_AUTH},
1038019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_ADD_DRAW)] =
1048019SMiao.Chen@Sun.COM 	    {drm_adddraw, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1058019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_RM_DRAW)] =
1068019SMiao.Chen@Sun.COM 	    {drm_rmdraw, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1078019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_LOCK)] =
1088019SMiao.Chen@Sun.COM 	    {drm_lock, DRM_AUTH},
1098019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_UNLOCK)] =
1108019SMiao.Chen@Sun.COM 	    {drm_unlock, DRM_AUTH},
1118019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_FINISH)] =
1128019SMiao.Chen@Sun.COM 	    {drm_noop, DRM_AUTH},
1138019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_ADD_BUFS)] =
1148019SMiao.Chen@Sun.COM 	    {drm_addbufs_ioctl, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1158019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_MARK_BUFS)] =
1168019SMiao.Chen@Sun.COM 	    {drm_markbufs, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1178019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_INFO_BUFS)] =
1188019SMiao.Chen@Sun.COM 	    {drm_infobufs, DRM_AUTH},
1198019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_MAP_BUFS)] =
1208019SMiao.Chen@Sun.COM 	    {drm_mapbufs, DRM_AUTH},
1218019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_FREE_BUFS)] =
1228019SMiao.Chen@Sun.COM 	    {drm_freebufs, DRM_AUTH},
1238019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_DMA)] =
1248019SMiao.Chen@Sun.COM 	    {drm_dma, DRM_AUTH},
1258019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_CONTROL)] =
1268019SMiao.Chen@Sun.COM 	    {drm_control, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1278019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_AGP_ACQUIRE)] =
1288019SMiao.Chen@Sun.COM 	    {drm_agp_acquire, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1298019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_AGP_RELEASE)] =
1308019SMiao.Chen@Sun.COM 	    {drm_agp_release, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1318019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_AGP_ENABLE)] =
1328019SMiao.Chen@Sun.COM 	    {drm_agp_enable, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1338019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_AGP_INFO)] =
1348019SMiao.Chen@Sun.COM 	    {drm_agp_info, DRM_AUTH},
1358019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_AGP_ALLOC)] =
1368019SMiao.Chen@Sun.COM 	    {drm_agp_alloc, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1378019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_AGP_FREE)] =
1388019SMiao.Chen@Sun.COM 	    {drm_agp_free, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1398019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_AGP_BIND)] =
1408019SMiao.Chen@Sun.COM 	    {drm_agp_bind, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1418019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_AGP_UNBIND)] =
1428019SMiao.Chen@Sun.COM 	    {drm_agp_unbind, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1438019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_SG_ALLOC)] =
1448019SMiao.Chen@Sun.COM 	    {drm_sg_alloc, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1458019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_SG_FREE)] =
1468019SMiao.Chen@Sun.COM 	    {drm_sg_free, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1478019SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_WAIT_VBLANK)] =
1488019SMiao.Chen@Sun.COM 	    {drm_wait_vblank, 0},
1498832SMiao.Chen@Sun.COM 	[DRM_IOCTL_NR(DRM_IOCTL_UPDATE_DRAW)] =
1508832SMiao.Chen@Sun.COM 	    {drm_update_draw, DRM_AUTH|DRM_MASTER|DRM_ROOT_ONLY},
1518019SMiao.Chen@Sun.COM };
1522820Skz151634 
153*11260SMiao.Chen@Sun.COM extern void idr_list_free(struct idr_list *head);
154*11260SMiao.Chen@Sun.COM 
1552820Skz151634 const char *
drm_find_description(int vendor,int device,drm_pci_id_list_t * idlist)1562820Skz151634 drm_find_description(int vendor, int device, drm_pci_id_list_t *idlist)
1572820Skz151634 {
1582820Skz151634 	int i = 0;
1592820Skz151634 	for (i = 0; idlist[i].vendor != 0; i++) {
1602820Skz151634 	if ((idlist[i].vendor == vendor) &&
1615804Scg149915 	    (idlist[i].device == device)) {
1622820Skz151634 			return (idlist[i].name);
1632820Skz151634 		}
1642820Skz151634 	}
1652820Skz151634 	return ((char *)NULL);
1662820Skz151634 }
1672820Skz151634 
1682820Skz151634 static int
drm_firstopen(drm_device_t * dev)1695804Scg149915 drm_firstopen(drm_device_t *dev)
1702820Skz151634 {
1712820Skz151634 	int i;
1725804Scg149915 	int retval;
1735804Scg149915 	drm_local_map_t *map;
1742820Skz151634 
1755804Scg149915 	/* prebuild the SAREA */
1765804Scg149915 	retval = drm_addmap(dev, 0, SAREA_MAX, _DRM_SHM,
1775804Scg149915 	    _DRM_CONTAINS_LOCK, &map);
1785804Scg149915 	if (retval != 0) {
1795804Scg149915 		DRM_ERROR("firstopen: failed to prebuild SAREA");
1805804Scg149915 		return (retval);
1815804Scg149915 	}
1825804Scg149915 
1835804Scg149915 	if (dev->driver->use_agp) {
1845804Scg149915 		DRM_DEBUG("drm_firstopen: use_agp=%d", dev->driver->use_agp);
1855804Scg149915 		if (drm_device_is_agp(dev))
1865804Scg149915 			dev->agp = drm_agp_init(dev);
1875804Scg149915 		if (dev->driver->require_agp && dev->agp == NULL) {
1885804Scg149915 			DRM_ERROR("couldn't initialize AGP");
1895804Scg149915 			return (EIO);
1905804Scg149915 		}
1915804Scg149915 	}
1925804Scg149915 
1935804Scg149915 	if (dev->driver->firstopen)
1945804Scg149915 		retval = dev->driver->firstopen(dev);
1955804Scg149915 
1965804Scg149915 	if (retval != 0) {
1975804Scg149915 		DRM_ERROR("drm_firstopen: driver-specific firstopen failed");
1985804Scg149915 		return (retval);
1995804Scg149915 	}
2002820Skz151634 
2012820Skz151634 	dev->buf_use = 0;
2022820Skz151634 
2035804Scg149915 	if (dev->driver->use_dma) {
2042820Skz151634 		i = drm_dma_setup(dev);
2052820Skz151634 		if (i != 0)
2062820Skz151634 			return (i);
2072820Skz151634 	}
2082820Skz151634 	dev->counters  = 6;
2092820Skz151634 	dev->types[0]  = _DRM_STAT_LOCK;
2102820Skz151634 	dev->types[1]  = _DRM_STAT_OPENS;
2112820Skz151634 	dev->types[2]  = _DRM_STAT_CLOSES;
2122820Skz151634 	dev->types[3]  = _DRM_STAT_IOCTLS;
2132820Skz151634 	dev->types[4]  = _DRM_STAT_LOCKS;
2142820Skz151634 	dev->types[5]  = _DRM_STAT_UNLOCKS;
2152820Skz151634 
2162820Skz151634 	for (i = 0; i < DRM_ARRAY_SIZE(dev->counts); i++)
2172820Skz151634 		*(&dev->counts[i]) = 0;
2182820Skz151634 
2192820Skz151634 	for (i = 0; i < DRM_HASH_SIZE; i++) {
2202820Skz151634 		dev->magiclist[i].head = NULL;
2212820Skz151634 		dev->magiclist[i].tail = NULL;
2222820Skz151634 	}
2232820Skz151634 
2242820Skz151634 	dev->irq_enabled	= 0;
2252820Skz151634 	dev->context_flag	= 0;
2262820Skz151634 	dev->last_context	= 0;
2272820Skz151634 	dev->if_version		= 0;
2282820Skz151634 
2295804Scg149915 	return (0);
2302820Skz151634 }
2312820Skz151634 
2322820Skz151634 /* Free resources associated with the DRM on the last close. */
2332820Skz151634 static int
drm_lastclose(drm_device_t * dev)2345804Scg149915 drm_lastclose(drm_device_t *dev)
2352820Skz151634 {
2362820Skz151634 	drm_magic_entry_t *pt, *next;
2372820Skz151634 	drm_local_map_t *map, *mapsave;
2382820Skz151634 	int i;
2392820Skz151634 
2402820Skz151634 	DRM_SPINLOCK_ASSERT(&dev->dev_lock);
2412820Skz151634 
2425804Scg149915 	if (dev->driver->lastclose != NULL)
2435804Scg149915 		dev->driver->lastclose(dev);
2442820Skz151634 
2452820Skz151634 	if (dev->irq_enabled)
2462820Skz151634 		(void) drm_irq_uninstall(dev);
2472820Skz151634 
2482820Skz151634 	if (dev->unique) {
2492820Skz151634 		drm_free(dev->unique, dev->unique_len + 1, DRM_MEM_DRIVER);
2502820Skz151634 		dev->unique = NULL;
2512820Skz151634 		dev->unique_len = 0;
2522820Skz151634 	}
2532820Skz151634 
2542820Skz151634 	/* Clear pid list */
2552820Skz151634 	for (i = 0; i < DRM_HASH_SIZE; i++) {
2562820Skz151634 		for (pt = dev->magiclist[i].head; pt; pt = next) {
2572820Skz151634 			next = pt->next;
2582820Skz151634 			drm_free(pt, sizeof (*pt), DRM_MEM_MAGIC);
2592820Skz151634 		}
2602820Skz151634 		dev->magiclist[i].head = dev->magiclist[i].tail = NULL;
2612820Skz151634 	}
2622820Skz151634 
2632820Skz151634 	/* Clear AGP information */
2642820Skz151634 	if (dev->agp) {
2652820Skz151634 		drm_agp_mem_t *entry;
2662820Skz151634 		drm_agp_mem_t *nexte;
2672820Skz151634 
2682820Skz151634 		/*
2692820Skz151634 		 * Remove AGP resources, but leave dev->agp
2702820Skz151634 		 * intact until drm_cleanup is called.
2712820Skz151634 		 */
2722820Skz151634 		for (entry = dev->agp->memory; entry; entry = nexte) {
2732820Skz151634 			nexte = entry->next;
2742820Skz151634 			if (entry->bound)
2752820Skz151634 				(void) drm_agp_unbind_memory(
2765804Scg149915 				    (unsigned long)entry->handle, dev);
277*11260SMiao.Chen@Sun.COM 			(void) drm_agp_free_memory(entry->handle, dev);
2782820Skz151634 			drm_free(entry, sizeof (*entry), DRM_MEM_AGPLISTS);
2792820Skz151634 		}
2802820Skz151634 		dev->agp->memory = NULL;
2812820Skz151634 
2822820Skz151634 		if (dev->agp->acquired)
2832820Skz151634 			(void) drm_agp_do_release(dev);
2842820Skz151634 
2852820Skz151634 		dev->agp->acquired = 0;
2862820Skz151634 		dev->agp->enabled  = 0;
2875804Scg149915 		drm_agp_fini(dev);
2882820Skz151634 	}
2895804Scg149915 
2902820Skz151634 	if (dev->sg != NULL) {
2915804Scg149915 		drm_sg_mem_t *entry;
2925804Scg149915 		entry = dev->sg;
2932820Skz151634 		dev->sg = NULL;
2945804Scg149915 		drm_sg_cleanup(dev, entry);
2952820Skz151634 	}
2962820Skz151634 
2975804Scg149915 
2982820Skz151634 	/* Clean up maps that weren't set up by the driver. */
2992820Skz151634 	TAILQ_FOREACH_SAFE(map, &dev->maplist, link, mapsave) {
3002820Skz151634 		if (!map->kernel_owned)
3012820Skz151634 			drm_rmmap(dev, map);
3022820Skz151634 	}
3032820Skz151634 
3042820Skz151634 	drm_dma_takedown(dev);
3052820Skz151634 	if (dev->lock.hw_lock) {
3062820Skz151634 		dev->lock.hw_lock = NULL; /* SHM removed */
3072820Skz151634 		dev->lock.filp = NULL;
3082820Skz151634 
3092820Skz151634 		mutex_enter(&(dev->lock.lock_mutex));
3102820Skz151634 		cv_broadcast(&(dev->lock.lock_cv));
3112820Skz151634 		mutex_exit(&(dev->lock.lock_mutex));
3122820Skz151634 	}
3132820Skz151634 
3142820Skz151634 	return (0);
3152820Skz151634 }
3162820Skz151634 
3172820Skz151634 static int
drm_load(drm_device_t * dev)3185804Scg149915 drm_load(drm_device_t *dev)
3192820Skz151634 {
3202820Skz151634 	int retcode;
3212820Skz151634 
3222820Skz151634 	cv_init(&(dev->lock.lock_cv), NULL, CV_DRIVER, NULL);
3232820Skz151634 	mutex_init(&(dev->lock.lock_mutex), NULL, MUTEX_DRIVER, NULL);
3248832SMiao.Chen@Sun.COM 	mutex_init(&(dev->dev_lock), "drmdev", MUTEX_DRIVER, NULL);
3258832SMiao.Chen@Sun.COM 	mutex_init(&dev->irq_lock, "drmirq", MUTEX_DRIVER,
3268832SMiao.Chen@Sun.COM 	    (void *)dev->intr_block);
3278832SMiao.Chen@Sun.COM 	mutex_init(&dev->drw_lock, "drmdrw", MUTEX_DRIVER, NULL);
3288832SMiao.Chen@Sun.COM 	mutex_init(&dev->tasklet_lock, "drmtsk", MUTEX_DRIVER, NULL);
3294194Szw161486 
3308832SMiao.Chen@Sun.COM 	dev->irq = pci_get_irq(dev);
3314194Szw161486 	dev->pci_vendor = pci_get_vendor(dev);
3324194Szw161486 	dev->pci_device = pci_get_device(dev);
3332820Skz151634 
3342820Skz151634 	TAILQ_INIT(&dev->maplist);
3356114Scg149915 	TAILQ_INIT(&dev->minordevs);
3362820Skz151634 	TAILQ_INIT(&dev->files);
3375804Scg149915 	if (dev->driver->load != NULL) {
3385804Scg149915 		retcode = dev->driver->load(dev, 0);
3392820Skz151634 		if (retcode != 0) {
3405804Scg149915 			DRM_ERROR("drm_load: failed\n");
3412820Skz151634 			goto error;
3422820Skz151634 		}
3432820Skz151634 	}
3442820Skz151634 
3452820Skz151634 	retcode = drm_ctxbitmap_init(dev);
3462820Skz151634 	if (retcode != 0) {
3472820Skz151634 		DRM_ERROR("drm_load: Cannot allocate memory for ctx bitmap");
3482820Skz151634 		goto error;
3492820Skz151634 	}
3502820Skz151634 
351*11260SMiao.Chen@Sun.COM 	if (dev->driver->use_gem == 1) {
352*11260SMiao.Chen@Sun.COM 		retcode = drm_gem_init(dev);
353*11260SMiao.Chen@Sun.COM 		if (retcode) {
354*11260SMiao.Chen@Sun.COM 			DRM_ERROR("Cannot initialize graphics execution "
355*11260SMiao.Chen@Sun.COM 			    "manager (GEM)\n");
356*11260SMiao.Chen@Sun.COM 			goto error;
357*11260SMiao.Chen@Sun.COM 		}
358*11260SMiao.Chen@Sun.COM 	}
359*11260SMiao.Chen@Sun.COM 
3602820Skz151634 	if (drm_init_kstats(dev)) {
3612820Skz151634 		DRM_ERROR("drm_attach => drm_load: init kstats error");
3625804Scg149915 		retcode = EFAULT;
3632820Skz151634 		goto error;
3642820Skz151634 	}
3652820Skz151634 
3662820Skz151634 	DRM_INFO("!drm: Initialized %s %d.%d.%d %s ",
3675804Scg149915 	    dev->driver->driver_name,
3685804Scg149915 	    dev->driver->driver_major,
3695804Scg149915 	    dev->driver->driver_minor,
3705804Scg149915 	    dev->driver->driver_patchlevel,
3715804Scg149915 	    dev->driver->driver_date);
3725804Scg149915 	return (0);
3732820Skz151634 
3742820Skz151634 error:
3752820Skz151634 	DRM_LOCK();
3762820Skz151634 	(void) drm_lastclose(dev);
3772820Skz151634 	DRM_UNLOCK();
3782820Skz151634 	cv_destroy(&(dev->lock.lock_cv));
3792820Skz151634 	mutex_destroy(&(dev->lock.lock_mutex));
3808832SMiao.Chen@Sun.COM 	mutex_destroy(&dev->irq_lock);
3812820Skz151634 	mutex_destroy(&(dev->dev_lock));
3824194Szw161486 	mutex_destroy(&dev->drw_lock);
3838832SMiao.Chen@Sun.COM 	mutex_destroy(&dev->tasklet_lock);
3842820Skz151634 
3852820Skz151634 	return (retcode);
3862820Skz151634 }
3872820Skz151634 
3882820Skz151634 /* called when cleanup this module */
3892820Skz151634 static void
drm_unload(drm_device_t * dev)3905804Scg149915 drm_unload(drm_device_t *dev)
3912820Skz151634 {
3922820Skz151634 	drm_local_map_t *map;
3932820Skz151634 
3949207SEdward.Shu@Sun.COM 	drm_vblank_cleanup(dev);
3959207SEdward.Shu@Sun.COM 
3962820Skz151634 	drm_ctxbitmap_cleanup(dev);
3972820Skz151634 
398*11260SMiao.Chen@Sun.COM 	if (dev->driver->use_gem == 1) {
399*11260SMiao.Chen@Sun.COM 		idr_list_free(&dev->object_name_idr);
400*11260SMiao.Chen@Sun.COM 		mutex_destroy(&dev->object_name_lock);
401*11260SMiao.Chen@Sun.COM 	}
402*11260SMiao.Chen@Sun.COM 
4032820Skz151634 	DRM_LOCK();
4042820Skz151634 	(void) drm_lastclose(dev);
4052820Skz151634 	DRM_UNLOCK();
4062820Skz151634 
4072820Skz151634 	while ((map = TAILQ_FIRST(&dev->maplist)) != NULL) {
4082820Skz151634 		drm_rmmap(dev, map);
4092820Skz151634 	}
4102820Skz151634 
4115804Scg149915 	if (dev->driver->unload != NULL)
4125804Scg149915 		dev->driver->unload(dev);
4132820Skz151634 
4142820Skz151634 	drm_mem_uninit();
4152820Skz151634 	cv_destroy(&dev->lock.lock_cv);
4162820Skz151634 	mutex_destroy(&dev->lock.lock_mutex);
4178832SMiao.Chen@Sun.COM 	mutex_destroy(&dev->irq_lock);
4182820Skz151634 	mutex_destroy(&dev->dev_lock);
4194194Szw161486 	mutex_destroy(&dev->drw_lock);
4208832SMiao.Chen@Sun.COM 	mutex_destroy(&dev->tasklet_lock);
421*11260SMiao.Chen@Sun.COM 
422*11260SMiao.Chen@Sun.COM 	dev->gtt_total = 0;
423*11260SMiao.Chen@Sun.COM 	atomic_set(&dev->pin_memory, 0);
424*11260SMiao.Chen@Sun.COM 	DRM_ERROR("drm_unload");
4252820Skz151634 }
4262820Skz151634 
4272820Skz151634 
4282820Skz151634 /*ARGSUSED*/
4292820Skz151634 int
drm_open(drm_device_t * dev,drm_cminor_t * mp,int openflags,int otyp,cred_t * credp)4306114Scg149915 drm_open(drm_device_t *dev, drm_cminor_t *mp, int openflags,
4312820Skz151634     int otyp, cred_t *credp)
4322820Skz151634 {
4332820Skz151634 	int retcode;
4342820Skz151634 
4356114Scg149915 	retcode = drm_open_helper(dev, mp, openflags, otyp, credp);
4362820Skz151634 
4372820Skz151634 	if (!retcode) {
4382820Skz151634 		atomic_inc_32(&dev->counts[_DRM_STAT_OPENS]);
4392820Skz151634 		DRM_LOCK();
4402820Skz151634 		if (!dev->open_count ++)
4412820Skz151634 			retcode = drm_firstopen(dev);
4422820Skz151634 		DRM_UNLOCK();
4432820Skz151634 	}
4442820Skz151634 
4452820Skz151634 	return (retcode);
4462820Skz151634 }
4472820Skz151634 
4482820Skz151634 /*ARGSUSED*/
4492820Skz151634 int
drm_close(drm_device_t * dev,int minor,int flag,int otyp,cred_t * credp)4506114Scg149915 drm_close(drm_device_t *dev, int minor, int flag, int otyp,
4512820Skz151634     cred_t *credp)
4522820Skz151634 {
4536114Scg149915 	drm_cminor_t	*mp;
4546114Scg149915 	drm_file_t		*fpriv;
4556114Scg149915 	int		retcode = 0;
4562820Skz151634 
4572820Skz151634 	DRM_LOCK();
4586114Scg149915 	mp = drm_find_file_by_minor(dev, minor);
4596114Scg149915 	if (!mp) {
4602820Skz151634 		DRM_UNLOCK();
4612820Skz151634 		DRM_ERROR("drm_close: can't find authenticator");
4625804Scg149915 		return (EACCES);
4632820Skz151634 	}
4642820Skz151634 
4656114Scg149915 	fpriv = mp->fpriv;
4666114Scg149915 	ASSERT(fpriv);
4676114Scg149915 
4685804Scg149915 	if (--fpriv->refs != 0)
4695804Scg149915 		goto done;
4705804Scg149915 
4715804Scg149915 	if (dev->driver->preclose != NULL)
4725804Scg149915 		dev->driver->preclose(dev, fpriv);
4732820Skz151634 
4742820Skz151634 	/*
4752820Skz151634 	 * Begin inline drm_release
4762820Skz151634 	 */
4772820Skz151634 	DRM_DEBUG("drm_close :pid = %d , open_count = %d",
4785804Scg149915 	    DRM_CURRENTPID, dev->open_count);
4792820Skz151634 
4802820Skz151634 	if (dev->lock.hw_lock &&
4815804Scg149915 	    _DRM_LOCK_IS_HELD(dev->lock.hw_lock->lock) &&
4825804Scg149915 	    dev->lock.filp == fpriv) {
4832820Skz151634 		DRM_DEBUG("Process %d dead, freeing lock for context %d",
4842820Skz151634 		    DRM_CURRENTPID,
4852820Skz151634 		    _DRM_LOCKING_CONTEXT(dev->lock.hw_lock->lock));
4865804Scg149915 		if (dev->driver->reclaim_buffers_locked != NULL)
4875804Scg149915 			dev->driver->reclaim_buffers_locked(dev, fpriv);
4882820Skz151634 		(void) drm_lock_free(dev, &dev->lock.hw_lock->lock,
4892820Skz151634 		    _DRM_LOCKING_CONTEXT(dev->lock.hw_lock->lock));
4905804Scg149915 	} else if (dev->driver->reclaim_buffers_locked != NULL &&
4912820Skz151634 	    dev->lock.hw_lock != NULL) {
4922820Skz151634 		DRM_ERROR("drm_close: "
4932820Skz151634 		    "retake lock not implemented yet");
4942820Skz151634 	}
4952820Skz151634 
496*11260SMiao.Chen@Sun.COM 	if (dev->driver->use_dma) {
4975804Scg149915 		drm_reclaim_buffers(dev, fpriv);
498*11260SMiao.Chen@Sun.COM 	}
4992820Skz151634 
500*11260SMiao.Chen@Sun.COM 	if (dev->driver->use_gem == 1) {
501*11260SMiao.Chen@Sun.COM 		drm_gem_release(dev, fpriv);
502*11260SMiao.Chen@Sun.COM 	}
5032820Skz151634 
504*11260SMiao.Chen@Sun.COM 	if (dev->driver->postclose != NULL) {
5055804Scg149915 		dev->driver->postclose(dev, fpriv);
506*11260SMiao.Chen@Sun.COM 	}
5075804Scg149915 	TAILQ_REMOVE(&dev->files, fpriv, link);
5085804Scg149915 	drm_free(fpriv, sizeof (*fpriv), DRM_MEM_FILES);
5092820Skz151634 
5105804Scg149915 done:
5112820Skz151634 	atomic_inc_32(&dev->counts[_DRM_STAT_CLOSES]);
5122820Skz151634 
5136114Scg149915 	TAILQ_REMOVE(&dev->minordevs, mp, link);
5146114Scg149915 	drm_free(mp, sizeof (*mp), DRM_MEM_FILES);
5156114Scg149915 
5162820Skz151634 	if (--dev->open_count == 0) {
5172820Skz151634 		retcode = drm_lastclose(dev);
5182820Skz151634 	}
5192820Skz151634 	DRM_UNLOCK();
5202820Skz151634 
5212820Skz151634 	return (retcode);
5222820Skz151634 }
5232820Skz151634 
5242820Skz151634 int
drm_attach(drm_device_t * dev)5255804Scg149915 drm_attach(drm_device_t *dev)
5262820Skz151634 {
5272820Skz151634 	return (drm_load(dev));
5282820Skz151634 }
5292820Skz151634 
5302820Skz151634 int
drm_detach(drm_device_t * dev)5315804Scg149915 drm_detach(drm_device_t *dev)
5322820Skz151634 {
5332820Skz151634 	drm_unload(dev);
5342820Skz151634 	drm_fini_kstats(dev);
5352820Skz151634 	return (DDI_SUCCESS);
5362820Skz151634 }
5372820Skz151634 
5382820Skz151634 static int
drm_get_businfo(drm_device_t * dev)5395804Scg149915 drm_get_businfo(drm_device_t *dev)
5402820Skz151634 {
5412820Skz151634 	dev->irq = pci_get_irq(dev);
5422820Skz151634 	if (dev->irq == -1) {
5432820Skz151634 		DRM_ERROR("drm_get_businfo: get irq error");
5442820Skz151634 		return (DDI_FAILURE);
5452820Skz151634 	}
5462820Skz151634 	/* XXX Fix domain number (alpha hoses) */
5472820Skz151634 	dev->pci_domain = 0;
5482820Skz151634 	if (pci_get_info(dev, &dev->pci_bus,
5492820Skz151634 	    &dev->pci_slot, &dev->pci_func) != DDI_SUCCESS) {
5502820Skz151634 		DRM_ERROR("drm_get_businfo: get bus slot func error ");
5512820Skz151634 		return (DDI_FAILURE);
5522820Skz151634 	}
5532820Skz151634 	DRM_DEBUG("drm_get_businfo: pci bus: %d, pci slot :%d pci func %d",
5542820Skz151634 	    dev->pci_bus, dev->pci_slot, dev->pci_func);
5552820Skz151634 	return (DDI_SUCCESS);
5562820Skz151634 }
5572820Skz151634 
5582820Skz151634 int
drm_probe(drm_device_t * dev,drm_pci_id_list_t * idlist)5595804Scg149915 drm_probe(drm_device_t *dev, drm_pci_id_list_t *idlist)
5602820Skz151634 {
5612820Skz151634 	const char *s = NULL;
5622820Skz151634 	int vendor, device;
5632820Skz151634 
5642820Skz151634 	vendor = pci_get_vendor(dev);
5652820Skz151634 	device = pci_get_device(dev);
5662820Skz151634 
5672820Skz151634 	s = drm_find_description(vendor, device, idlist);
5682820Skz151634 	if (s != NULL) {
5692820Skz151634 		dev->desc = s;
5702820Skz151634 		if (drm_get_businfo(dev) != DDI_SUCCESS) {
5712820Skz151634 			DRM_ERROR("drm_probe: drm get bus info error");
5722820Skz151634 			return (DDI_FAILURE);
5732820Skz151634 		}
5742820Skz151634 		return (DDI_SUCCESS);
5752820Skz151634 	}
5762820Skz151634 	return (DDI_FAILURE);
5772820Skz151634 }
578