forked from mirrors/linux
		
	drm: convert drm context code to use Linux idr
This converts the drm context allocator to an idr, using the new idr interface features from Kristian. Fixes from Kristian Hoegsberg <krh@redhat.com> Signed-off-by: Dave Airlie <airlied@linux.ie>
This commit is contained in:
		
							parent
							
								
									7608a864e5
								
							
						
					
					
						commit
						62968144e6
					
				
					 2 changed files with 33 additions and 103 deletions
				
			
		| 
						 | 
					@ -75,6 +75,8 @@
 | 
				
			||||||
#include <asm/pgalloc.h>
 | 
					#include <asm/pgalloc.h>
 | 
				
			||||||
#include "drm.h"
 | 
					#include "drm.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <linux/idr.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define __OS_HAS_AGP (defined(CONFIG_AGP) || (defined(CONFIG_AGP_MODULE) && defined(MODULE)))
 | 
					#define __OS_HAS_AGP (defined(CONFIG_AGP) || (defined(CONFIG_AGP_MODULE) && defined(MODULE)))
 | 
				
			||||||
#define __OS_HAS_MTRR (defined(CONFIG_MTRR))
 | 
					#define __OS_HAS_MTRR (defined(CONFIG_MTRR))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -676,8 +678,7 @@ struct drm_device {
 | 
				
			||||||
	int ctx_count;			/**< Number of context handles */
 | 
						int ctx_count;			/**< Number of context handles */
 | 
				
			||||||
	struct mutex ctxlist_mutex;	/**< For ctxlist */
 | 
						struct mutex ctxlist_mutex;	/**< For ctxlist */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	struct drm_map **context_sareas;	    /**< per-context SAREA's */
 | 
						struct idr ctx_idr;
 | 
				
			||||||
	int max_context;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	struct list_head vmalist;	/**< List of vmas (for debugging) */
 | 
						struct list_head vmalist;	/**< List of vmas (for debugging) */
 | 
				
			||||||
	struct drm_lock_data lock;	/**< Information on hardware lock */
 | 
						struct drm_lock_data lock;	/**< Information on hardware lock */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -53,26 +53,14 @@
 | 
				
			||||||
 * \param ctx_handle context handle.
 | 
					 * \param ctx_handle context handle.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * Clears the bit specified by \p ctx_handle in drm_device::ctx_bitmap and the entry
 | 
					 * Clears the bit specified by \p ctx_handle in drm_device::ctx_bitmap and the entry
 | 
				
			||||||
 * in drm_device::context_sareas, while holding the drm_device::struct_mutex
 | 
					 * in drm_device::ctx_idr, while holding the drm_device::struct_mutex
 | 
				
			||||||
 * lock.
 | 
					 * lock.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
void drm_ctxbitmap_free(struct drm_device * dev, int ctx_handle)
 | 
					void drm_ctxbitmap_free(struct drm_device * dev, int ctx_handle)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	if (ctx_handle < 0)
 | 
					 | 
				
			||||||
		goto failed;
 | 
					 | 
				
			||||||
	if (!dev->ctx_bitmap)
 | 
					 | 
				
			||||||
		goto failed;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (ctx_handle < DRM_MAX_CTXBITMAP) {
 | 
					 | 
				
			||||||
	mutex_lock(&dev->struct_mutex);
 | 
						mutex_lock(&dev->struct_mutex);
 | 
				
			||||||
		clear_bit(ctx_handle, dev->ctx_bitmap);
 | 
						idr_remove(&dev->ctx_idr, ctx_handle);
 | 
				
			||||||
		dev->context_sareas[ctx_handle] = NULL;
 | 
					 | 
				
			||||||
	mutex_unlock(&dev->struct_mutex);
 | 
						mutex_unlock(&dev->struct_mutex);
 | 
				
			||||||
		return;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
      failed:
 | 
					 | 
				
			||||||
	DRM_ERROR("Attempt to free invalid context handle: %d\n", ctx_handle);
 | 
					 | 
				
			||||||
	return;
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
| 
						 | 
					@ -81,62 +69,28 @@ void drm_ctxbitmap_free(struct drm_device * dev, int ctx_handle)
 | 
				
			||||||
 * \param dev DRM device.
 | 
					 * \param dev DRM device.
 | 
				
			||||||
 * \return (non-negative) context handle on success or a negative number on failure.
 | 
					 * \return (non-negative) context handle on success or a negative number on failure.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * Find the first zero bit in drm_device::ctx_bitmap and (re)allocates
 | 
					 * Allocate a new idr from drm_device::ctx_idr while holding the
 | 
				
			||||||
 * drm_device::context_sareas to accommodate the new entry while holding the
 | 
					 | 
				
			||||||
 * drm_device::struct_mutex lock.
 | 
					 * drm_device::struct_mutex lock.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
static int drm_ctxbitmap_next(struct drm_device * dev)
 | 
					static int drm_ctxbitmap_next(struct drm_device * dev)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int bit;
 | 
						int new_id;
 | 
				
			||||||
 | 
						int ret;
 | 
				
			||||||
	if (!dev->ctx_bitmap)
 | 
					 | 
				
			||||||
		return -1;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					again:
 | 
				
			||||||
 | 
						if (idr_pre_get(&dev->ctx_idr, GFP_KERNEL) == 0) {
 | 
				
			||||||
 | 
							DRM_ERROR("Out of memory expanding drawable idr\n");
 | 
				
			||||||
 | 
							return -ENOMEM;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	mutex_lock(&dev->struct_mutex);
 | 
						mutex_lock(&dev->struct_mutex);
 | 
				
			||||||
	bit = find_first_zero_bit(dev->ctx_bitmap, DRM_MAX_CTXBITMAP);
 | 
						ret = idr_get_new_above(&dev->ctx_idr, NULL,
 | 
				
			||||||
	if (bit < DRM_MAX_CTXBITMAP) {
 | 
									DRM_RESERVED_CONTEXTS, &new_id);
 | 
				
			||||||
		set_bit(bit, dev->ctx_bitmap);
 | 
						if (ret == -EAGAIN) {
 | 
				
			||||||
		DRM_DEBUG("drm_ctxbitmap_next bit : %d\n", bit);
 | 
					 | 
				
			||||||
		if ((bit + 1) > dev->max_context) {
 | 
					 | 
				
			||||||
			dev->max_context = (bit + 1);
 | 
					 | 
				
			||||||
			if (dev->context_sareas) {
 | 
					 | 
				
			||||||
				struct drm_map **ctx_sareas;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
				ctx_sareas = drm_realloc(dev->context_sareas,
 | 
					 | 
				
			||||||
							 (dev->max_context -
 | 
					 | 
				
			||||||
							  1) *
 | 
					 | 
				
			||||||
							 sizeof(*dev->
 | 
					 | 
				
			||||||
								context_sareas),
 | 
					 | 
				
			||||||
							 dev->max_context *
 | 
					 | 
				
			||||||
							 sizeof(*dev->
 | 
					 | 
				
			||||||
								context_sareas),
 | 
					 | 
				
			||||||
							 DRM_MEM_MAPS);
 | 
					 | 
				
			||||||
				if (!ctx_sareas) {
 | 
					 | 
				
			||||||
					clear_bit(bit, dev->ctx_bitmap);
 | 
					 | 
				
			||||||
		mutex_unlock(&dev->struct_mutex);
 | 
							mutex_unlock(&dev->struct_mutex);
 | 
				
			||||||
					return -1;
 | 
							goto again;
 | 
				
			||||||
				}
 | 
					 | 
				
			||||||
				dev->context_sareas = ctx_sareas;
 | 
					 | 
				
			||||||
				dev->context_sareas[bit] = NULL;
 | 
					 | 
				
			||||||
			} else {
 | 
					 | 
				
			||||||
				/* max_context == 1 at this point */
 | 
					 | 
				
			||||||
				dev->context_sareas =
 | 
					 | 
				
			||||||
				    drm_alloc(dev->max_context *
 | 
					 | 
				
			||||||
					      sizeof(*dev->context_sareas),
 | 
					 | 
				
			||||||
					      DRM_MEM_MAPS);
 | 
					 | 
				
			||||||
				if (!dev->context_sareas) {
 | 
					 | 
				
			||||||
					clear_bit(bit, dev->ctx_bitmap);
 | 
					 | 
				
			||||||
					mutex_unlock(&dev->struct_mutex);
 | 
					 | 
				
			||||||
					return -1;
 | 
					 | 
				
			||||||
				}
 | 
					 | 
				
			||||||
				dev->context_sareas[bit] = NULL;
 | 
					 | 
				
			||||||
			}
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	mutex_unlock(&dev->struct_mutex);
 | 
						mutex_unlock(&dev->struct_mutex);
 | 
				
			||||||
		return bit;
 | 
						return new_id;
 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	mutex_unlock(&dev->struct_mutex);
 | 
					 | 
				
			||||||
	return -1;
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
| 
						 | 
					@ -144,31 +98,11 @@ static int drm_ctxbitmap_next(struct drm_device * dev)
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * \param dev DRM device.
 | 
					 * \param dev DRM device.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * Allocates and initialize drm_device::ctx_bitmap and drm_device::context_sareas, while holding
 | 
					 * Initialise the drm_device::ctx_idr
 | 
				
			||||||
 * the drm_device::struct_mutex lock.
 | 
					 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
int drm_ctxbitmap_init(struct drm_device * dev)
 | 
					int drm_ctxbitmap_init(struct drm_device * dev)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int i;
 | 
						idr_init(&dev->ctx_idr);
 | 
				
			||||||
	int temp;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	mutex_lock(&dev->struct_mutex);
 | 
					 | 
				
			||||||
	dev->ctx_bitmap = (unsigned long *)drm_alloc(PAGE_SIZE,
 | 
					 | 
				
			||||||
						     DRM_MEM_CTXBITMAP);
 | 
					 | 
				
			||||||
	if (dev->ctx_bitmap == NULL) {
 | 
					 | 
				
			||||||
		mutex_unlock(&dev->struct_mutex);
 | 
					 | 
				
			||||||
		return -ENOMEM;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	memset((void *)dev->ctx_bitmap, 0, PAGE_SIZE);
 | 
					 | 
				
			||||||
	dev->context_sareas = NULL;
 | 
					 | 
				
			||||||
	dev->max_context = -1;
 | 
					 | 
				
			||||||
	mutex_unlock(&dev->struct_mutex);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	for (i = 0; i < DRM_RESERVED_CONTEXTS; i++) {
 | 
					 | 
				
			||||||
		temp = drm_ctxbitmap_next(dev);
 | 
					 | 
				
			||||||
		DRM_DEBUG("drm_ctxbitmap_init : %d\n", temp);
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -177,17 +111,13 @@ int drm_ctxbitmap_init(struct drm_device * dev)
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * \param dev DRM device.
 | 
					 * \param dev DRM device.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * Frees drm_device::ctx_bitmap and drm_device::context_sareas, while holding
 | 
					 * Free all idr members using drm_ctx_sarea_free helper function
 | 
				
			||||||
 * the drm_device::struct_mutex lock.
 | 
					 * while holding the drm_device::struct_mutex lock.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
void drm_ctxbitmap_cleanup(struct drm_device * dev)
 | 
					void drm_ctxbitmap_cleanup(struct drm_device * dev)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	mutex_lock(&dev->struct_mutex);
 | 
						mutex_lock(&dev->struct_mutex);
 | 
				
			||||||
	if (dev->context_sareas)
 | 
						idr_remove_all(&dev->ctx_idr);
 | 
				
			||||||
		drm_free(dev->context_sareas,
 | 
					 | 
				
			||||||
			 sizeof(*dev->context_sareas) *
 | 
					 | 
				
			||||||
			 dev->max_context, DRM_MEM_MAPS);
 | 
					 | 
				
			||||||
	drm_free((void *)dev->ctx_bitmap, PAGE_SIZE, DRM_MEM_CTXBITMAP);
 | 
					 | 
				
			||||||
	mutex_unlock(&dev->struct_mutex);
 | 
						mutex_unlock(&dev->struct_mutex);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -206,7 +136,7 @@ void drm_ctxbitmap_cleanup(struct drm_device * dev)
 | 
				
			||||||
 * \param arg user argument pointing to a drm_ctx_priv_map structure.
 | 
					 * \param arg user argument pointing to a drm_ctx_priv_map structure.
 | 
				
			||||||
 * \return zero on success or a negative number on failure.
 | 
					 * \return zero on success or a negative number on failure.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * Gets the map from drm_device::context_sareas with the handle specified and
 | 
					 * Gets the map from drm_device::ctx_idr with the handle specified and
 | 
				
			||||||
 * returns its handle.
 | 
					 * returns its handle.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
int drm_getsareactx(struct inode *inode, struct file *filp,
 | 
					int drm_getsareactx(struct inode *inode, struct file *filp,
 | 
				
			||||||
| 
						 | 
					@ -223,13 +153,13 @@ int drm_getsareactx(struct inode *inode, struct file *filp,
 | 
				
			||||||
		return -EFAULT;
 | 
							return -EFAULT;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	mutex_lock(&dev->struct_mutex);
 | 
						mutex_lock(&dev->struct_mutex);
 | 
				
			||||||
	if (dev->max_context < 0
 | 
					
 | 
				
			||||||
	    || request.ctx_id >= (unsigned)dev->max_context) {
 | 
						map = idr_find(&dev->ctx_idr, request.ctx_id);
 | 
				
			||||||
 | 
						if (!map) {
 | 
				
			||||||
		mutex_unlock(&dev->struct_mutex);
 | 
							mutex_unlock(&dev->struct_mutex);
 | 
				
			||||||
		return -EINVAL;
 | 
							return -EINVAL;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	map = dev->context_sareas[request.ctx_id];
 | 
					 | 
				
			||||||
	mutex_unlock(&dev->struct_mutex);
 | 
						mutex_unlock(&dev->struct_mutex);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	request.handle = NULL;
 | 
						request.handle = NULL;
 | 
				
			||||||
| 
						 | 
					@ -258,7 +188,7 @@ int drm_getsareactx(struct inode *inode, struct file *filp,
 | 
				
			||||||
 * \return zero on success or a negative number on failure.
 | 
					 * \return zero on success or a negative number on failure.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * Searches the mapping specified in \p arg and update the entry in
 | 
					 * Searches the mapping specified in \p arg and update the entry in
 | 
				
			||||||
 * drm_device::context_sareas with it.
 | 
					 * drm_device::ctx_idr with it.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
int drm_setsareactx(struct inode *inode, struct file *filp,
 | 
					int drm_setsareactx(struct inode *inode, struct file *filp,
 | 
				
			||||||
		    unsigned int cmd, unsigned long arg)
 | 
							    unsigned int cmd, unsigned long arg)
 | 
				
			||||||
| 
						 | 
					@ -288,11 +218,10 @@ int drm_setsareactx(struct inode *inode, struct file *filp,
 | 
				
			||||||
	map = r_list->map;
 | 
						map = r_list->map;
 | 
				
			||||||
	if (!map)
 | 
						if (!map)
 | 
				
			||||||
		goto bad;
 | 
							goto bad;
 | 
				
			||||||
	if (dev->max_context < 0)
 | 
					
 | 
				
			||||||
 | 
						if (IS_ERR(idr_replace(&dev->ctx_idr, map, request.ctx_id)))
 | 
				
			||||||
		goto bad;
 | 
							goto bad;
 | 
				
			||||||
	if (request.ctx_id >= (unsigned)dev->max_context)
 | 
					
 | 
				
			||||||
		goto bad;
 | 
					 | 
				
			||||||
	dev->context_sareas[request.ctx_id] = map;
 | 
					 | 
				
			||||||
	mutex_unlock(&dev->struct_mutex);
 | 
						mutex_unlock(&dev->struct_mutex);
 | 
				
			||||||
	return 0;
 | 
						return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue