| 
									
										
										
										
											2014-06-09 12:39:22 +02:00
										 |  |  | /**************************************************************************
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright © 2014 VMware, Inc., Palo Alto, CA., USA | 
					
						
							|  |  |  |  * All Rights Reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Permission is hereby granted, free of charge, to any person obtaining a | 
					
						
							|  |  |  |  * copy of this software and associated documentation files (the | 
					
						
							|  |  |  |  * "Software"), to deal in the Software without restriction, including | 
					
						
							|  |  |  |  * without limitation the rights to use, copy, modify, merge, publish, | 
					
						
							|  |  |  |  * distribute, sub license, and/or sell copies of the Software, and to | 
					
						
							|  |  |  |  * permit persons to whom the Software is furnished to do so, subject to | 
					
						
							|  |  |  |  * the following conditions: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The above copyright notice and this permission notice (including the | 
					
						
							|  |  |  |  * next paragraph) shall be included in all copies or substantial portions | 
					
						
							|  |  |  |  * of the Software. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
					
						
							|  |  |  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  |  * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL | 
					
						
							|  |  |  |  * THE COPYRIGHT HOLDERS, AUTHORS AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, | 
					
						
							|  |  |  |  * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR | 
					
						
							|  |  |  |  * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE | 
					
						
							|  |  |  |  * USE OR OTHER DEALINGS IN THE SOFTWARE. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  **************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "vmwgfx_drv.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define VMW_CMDBUF_RES_MAN_HT_ORDER 12
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum vmw_cmdbuf_res_state { | 
					
						
							|  |  |  | 	VMW_CMDBUF_RES_COMMITED, | 
					
						
							|  |  |  | 	VMW_CMDBUF_RES_ADD, | 
					
						
							|  |  |  | 	VMW_CMDBUF_RES_DEL | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * struct vmw_cmdbuf_res - Command buffer managed resource entry. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @res: Refcounted pointer to a struct vmw_resource. | 
					
						
							|  |  |  |  * @hash: Hash entry for the manager hash table. | 
					
						
							|  |  |  |  * @head: List head used either by the staging list or the manager list | 
					
						
							|  |  |  |  * of commited resources. | 
					
						
							|  |  |  |  * @state: Staging state of this resource entry. | 
					
						
							|  |  |  |  * @man: Pointer to a resource manager for this entry. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct vmw_cmdbuf_res { | 
					
						
							|  |  |  | 	struct vmw_resource *res; | 
					
						
							|  |  |  | 	struct drm_hash_item hash; | 
					
						
							|  |  |  | 	struct list_head head; | 
					
						
							|  |  |  | 	enum vmw_cmdbuf_res_state state; | 
					
						
							|  |  |  | 	struct vmw_cmdbuf_res_manager *man; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * struct vmw_cmdbuf_res_manager - Command buffer resource manager. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @resources: Hash table containing staged and commited command buffer | 
					
						
							|  |  |  |  * resources | 
					
						
							|  |  |  |  * @list: List of commited command buffer resources. | 
					
						
							|  |  |  |  * @dev_priv: Pointer to a device private structure. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @resources and @list are protected by the cmdbuf mutex for now. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct vmw_cmdbuf_res_manager { | 
					
						
							|  |  |  | 	struct drm_open_hash resources; | 
					
						
							|  |  |  | 	struct list_head list; | 
					
						
							|  |  |  | 	struct vmw_private *dev_priv; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * vmw_cmdbuf_res_lookup - Look up a command buffer resource | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @man: Pointer to the command buffer resource manager | 
					
						
							|  |  |  |  * @resource_type: The resource type, that combined with the user key | 
					
						
							|  |  |  |  * identifies the resource. | 
					
						
							|  |  |  |  * @user_key: The user key. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Returns a valid refcounted struct vmw_resource pointer on success, | 
					
						
							|  |  |  |  * an error pointer on failure. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct vmw_resource * | 
					
						
							|  |  |  | vmw_cmdbuf_res_lookup(struct vmw_cmdbuf_res_manager *man, | 
					
						
							|  |  |  | 		      enum vmw_cmdbuf_res_type res_type, | 
					
						
							|  |  |  | 		      u32 user_key) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct drm_hash_item *hash; | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 	unsigned long key = user_key | (res_type << 24); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = drm_ht_find_item(&man->resources, key, &hash); | 
					
						
							|  |  |  | 	if (unlikely(ret != 0)) | 
					
						
							|  |  |  | 		return ERR_PTR(ret); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return vmw_resource_reference | 
					
						
							|  |  |  | 		(drm_hash_entry(hash, struct vmw_cmdbuf_res, hash)->res); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * vmw_cmdbuf_res_free - Free a command buffer resource. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @man: Pointer to the command buffer resource manager | 
					
						
							|  |  |  |  * @entry: Pointer to a struct vmw_cmdbuf_res. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Frees a struct vmw_cmdbuf_res entry and drops its reference to the | 
					
						
							|  |  |  |  * struct vmw_resource. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void vmw_cmdbuf_res_free(struct vmw_cmdbuf_res_manager *man, | 
					
						
							|  |  |  | 				struct vmw_cmdbuf_res *entry) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	list_del(&entry->head); | 
					
						
							|  |  |  | 	WARN_ON(drm_ht_remove_item(&man->resources, &entry->hash)); | 
					
						
							|  |  |  | 	vmw_resource_unreference(&entry->res); | 
					
						
							|  |  |  | 	kfree(entry); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * vmw_cmdbuf_res_commit - Commit a list of command buffer resource actions | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @list: Caller's list of command buffer resource actions. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function commits a list of command buffer resource | 
					
						
							|  |  |  |  * additions or removals. | 
					
						
							|  |  |  |  * It is typically called when the execbuf ioctl call triggering these | 
					
						
							|  |  |  |  * actions has commited the fifo contents to the device. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void vmw_cmdbuf_res_commit(struct list_head *list) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct vmw_cmdbuf_res *entry, *next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry_safe(entry, next, list, head) { | 
					
						
							|  |  |  | 		list_del(&entry->head); | 
					
						
							|  |  |  | 		switch (entry->state) { | 
					
						
							|  |  |  | 		case VMW_CMDBUF_RES_ADD: | 
					
						
							|  |  |  | 			entry->state = VMW_CMDBUF_RES_COMMITED; | 
					
						
							|  |  |  | 			list_add_tail(&entry->head, &entry->man->list); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case VMW_CMDBUF_RES_DEL: | 
					
						
							|  |  |  | 			vmw_resource_unreference(&entry->res); | 
					
						
							|  |  |  | 			kfree(entry); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			BUG(); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * vmw_cmdbuf_res_revert - Revert a list of command buffer resource actions | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @man: Pointer to the command buffer resource manager | 
					
						
							|  |  |  |  * @list: Caller's list of command buffer resource action | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function reverts a list of command buffer resource | 
					
						
							|  |  |  |  * additions or removals. | 
					
						
							|  |  |  |  * It is typically called when the execbuf ioctl call triggering these | 
					
						
							|  |  |  |  * actions failed for some reason, and the command stream was never | 
					
						
							|  |  |  |  * submitted. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void vmw_cmdbuf_res_revert(struct list_head *list) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct vmw_cmdbuf_res *entry, *next; | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry_safe(entry, next, list, head) { | 
					
						
							|  |  |  | 		switch (entry->state) { | 
					
						
							|  |  |  | 		case VMW_CMDBUF_RES_ADD: | 
					
						
							|  |  |  | 			vmw_cmdbuf_res_free(entry->man, entry); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case VMW_CMDBUF_RES_DEL: | 
					
						
							|  |  |  | 			ret = drm_ht_insert_item(&entry->man->resources, | 
					
						
							|  |  |  | 						 &entry->hash); | 
					
						
							|  |  |  | 			list_del(&entry->head); | 
					
						
							|  |  |  | 			list_add_tail(&entry->head, &entry->man->list); | 
					
						
							|  |  |  | 			entry->state = VMW_CMDBUF_RES_COMMITED; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			BUG(); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * vmw_cmdbuf_res_add - Stage a command buffer managed resource for addition. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @man: Pointer to the command buffer resource manager. | 
					
						
							|  |  |  |  * @res_type: The resource type. | 
					
						
							|  |  |  |  * @user_key: The user-space id of the resource. | 
					
						
							|  |  |  |  * @res: Valid (refcount != 0) pointer to a struct vmw_resource. | 
					
						
							|  |  |  |  * @list: The staging list. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function allocates a struct vmw_cmdbuf_res entry and adds the | 
					
						
							|  |  |  |  * resource to the hash table of the manager identified by @man. The | 
					
						
							|  |  |  |  * entry is then put on the staging list identified by @list. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int vmw_cmdbuf_res_add(struct vmw_cmdbuf_res_manager *man, | 
					
						
							|  |  |  | 		       enum vmw_cmdbuf_res_type res_type, | 
					
						
							|  |  |  | 		       u32 user_key, | 
					
						
							|  |  |  | 		       struct vmw_resource *res, | 
					
						
							|  |  |  | 		       struct list_head *list) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct vmw_cmdbuf_res *cres; | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cres = kzalloc(sizeof(*cres), GFP_KERNEL); | 
					
						
							|  |  |  | 	if (unlikely(cres == NULL)) | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cres->hash.key = user_key | (res_type << 24); | 
					
						
							|  |  |  | 	ret = drm_ht_insert_item(&man->resources, &cres->hash); | 
					
						
							|  |  |  | 	if (unlikely(ret != 0)) | 
					
						
							|  |  |  | 		goto out_invalid_key; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cres->state = VMW_CMDBUF_RES_ADD; | 
					
						
							|  |  |  | 	cres->res = vmw_resource_reference(res); | 
					
						
							|  |  |  | 	cres->man = man; | 
					
						
							|  |  |  | 	list_add_tail(&cres->head, list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | out_invalid_key: | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * vmw_cmdbuf_res_remove - Stage a command buffer managed resource for removal. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @man: Pointer to the command buffer resource manager. | 
					
						
							|  |  |  |  * @res_type: The resource type. | 
					
						
							|  |  |  |  * @user_key: The user-space id of the resource. | 
					
						
							|  |  |  |  * @list: The staging list. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function looks up the struct vmw_cmdbuf_res entry from the manager | 
					
						
							|  |  |  |  * hash table and, if it exists, removes it. Depending on its current staging | 
					
						
							|  |  |  |  * state it then either removes the entry from the staging list or adds it | 
					
						
							|  |  |  |  * to it with a staging state of removal. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int vmw_cmdbuf_res_remove(struct vmw_cmdbuf_res_manager *man, | 
					
						
							|  |  |  | 			  enum vmw_cmdbuf_res_type res_type, | 
					
						
							|  |  |  | 			  u32 user_key, | 
					
						
							|  |  |  | 			  struct list_head *list) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct vmw_cmdbuf_res *entry; | 
					
						
							|  |  |  | 	struct drm_hash_item *hash; | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-31 09:54:22 +01:00
										 |  |  | 	ret = drm_ht_find_item(&man->resources, user_key | (res_type << 24), | 
					
						
							|  |  |  | 			       &hash); | 
					
						
							| 
									
										
										
										
											2014-06-09 12:39:22 +02:00
										 |  |  | 	if (likely(ret != 0)) | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	entry = drm_hash_entry(hash, struct vmw_cmdbuf_res, hash); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (entry->state) { | 
					
						
							|  |  |  | 	case VMW_CMDBUF_RES_ADD: | 
					
						
							|  |  |  | 		vmw_cmdbuf_res_free(man, entry); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case VMW_CMDBUF_RES_COMMITED: | 
					
						
							|  |  |  | 		(void) drm_ht_remove_item(&man->resources, &entry->hash); | 
					
						
							|  |  |  | 		list_del(&entry->head); | 
					
						
							|  |  |  | 		entry->state = VMW_CMDBUF_RES_DEL; | 
					
						
							|  |  |  | 		list_add_tail(&entry->head, list); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		BUG(); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * vmw_cmdbuf_res_man_create - Allocate a command buffer managed resource | 
					
						
							|  |  |  |  * manager. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @dev_priv: Pointer to a struct vmw_private | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Allocates and initializes a command buffer managed resource manager. Returns | 
					
						
							|  |  |  |  * an error pointer on failure. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct vmw_cmdbuf_res_manager * | 
					
						
							|  |  |  | vmw_cmdbuf_res_man_create(struct vmw_private *dev_priv) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct vmw_cmdbuf_res_manager *man; | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	man = kzalloc(sizeof(*man), GFP_KERNEL); | 
					
						
							|  |  |  | 	if (man == NULL) | 
					
						
							|  |  |  | 		return ERR_PTR(-ENOMEM); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	man->dev_priv = dev_priv; | 
					
						
							|  |  |  | 	INIT_LIST_HEAD(&man->list); | 
					
						
							|  |  |  | 	ret = drm_ht_create(&man->resources, VMW_CMDBUF_RES_MAN_HT_ORDER); | 
					
						
							|  |  |  | 	if (ret == 0) | 
					
						
							|  |  |  | 		return man; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	kfree(man); | 
					
						
							|  |  |  | 	return ERR_PTR(ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * vmw_cmdbuf_res_man_destroy - Destroy a command buffer managed resource | 
					
						
							|  |  |  |  * manager. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @man: Pointer to the  manager to destroy. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function destroys a command buffer managed resource manager and | 
					
						
							|  |  |  |  * unreferences / frees all command buffer managed resources and -entries | 
					
						
							|  |  |  |  * associated with it. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void vmw_cmdbuf_res_man_destroy(struct vmw_cmdbuf_res_manager *man) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct vmw_cmdbuf_res *entry, *next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry_safe(entry, next, &man->list, head) | 
					
						
							|  |  |  | 		vmw_cmdbuf_res_free(man, entry); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	kfree(man); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * vmw_cmdbuf_res_man_size - Return the size of a command buffer managed | 
					
						
							|  |  |  |  * resource manager | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Returns the approximate allocation size of a command buffer managed | 
					
						
							|  |  |  |  * resource manager. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | size_t vmw_cmdbuf_res_man_size(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	static size_t res_man_size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (unlikely(res_man_size == 0)) | 
					
						
							|  |  |  | 		res_man_size = | 
					
						
							|  |  |  | 			ttm_round_pot(sizeof(struct vmw_cmdbuf_res_manager)) + | 
					
						
							|  |  |  | 			ttm_round_pot(sizeof(struct hlist_head) << | 
					
						
							|  |  |  | 				      VMW_CMDBUF_RES_MAN_HT_ORDER); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return res_man_size; | 
					
						
							|  |  |  | } |