forked from mirrors/linux
		
	 53c165e0a6
			
		
	
	
		53c165e0a6
		
	
	
	
	
		
			
			One of the changes in the attribute_container code in the scsi-misc tree was to add a lock to protect the list of devices per container. This, unfortunately, leads to potential scheduling while atomic problems if there's a sleep in the function called by a trigger. The correct solution is to use the kernel klist infrastructure instead which allows lockless traversal of a list. Signed-off-by: James Bottomley <James.Bottomley@SteelEye.com>
		
			
				
	
	
		
			71 lines
		
	
	
	
		
			2.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			71 lines
		
	
	
	
		
			2.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * class_container.h - a generic container for all classes
 | |
|  *
 | |
|  * Copyright (c) 2005 - James Bottomley <James.Bottomley@steeleye.com>
 | |
|  *
 | |
|  * This file is licensed under GPLv2
 | |
|  */
 | |
| 
 | |
| #ifndef _ATTRIBUTE_CONTAINER_H_
 | |
| #define _ATTRIBUTE_CONTAINER_H_
 | |
| 
 | |
| #include <linux/device.h>
 | |
| #include <linux/list.h>
 | |
| #include <linux/klist.h>
 | |
| #include <linux/spinlock.h>
 | |
| 
 | |
| struct attribute_container {
 | |
| 	struct list_head	node;
 | |
| 	struct klist		containers;
 | |
| 	struct class		*class;
 | |
| 	struct class_device_attribute **attrs;
 | |
| 	int (*match)(struct attribute_container *, struct device *);
 | |
| #define	ATTRIBUTE_CONTAINER_NO_CLASSDEVS	0x01
 | |
| 	unsigned long		flags;
 | |
| };
 | |
| 
 | |
| static inline int
 | |
| attribute_container_no_classdevs(struct attribute_container *atc)
 | |
| {
 | |
| 	return atc->flags & ATTRIBUTE_CONTAINER_NO_CLASSDEVS;
 | |
| }
 | |
| 
 | |
| static inline void
 | |
| attribute_container_set_no_classdevs(struct attribute_container *atc)
 | |
| {
 | |
| 	atc->flags |= ATTRIBUTE_CONTAINER_NO_CLASSDEVS;
 | |
| }
 | |
| 
 | |
| int attribute_container_register(struct attribute_container *cont);
 | |
| int attribute_container_unregister(struct attribute_container *cont);
 | |
| void attribute_container_create_device(struct device *dev,
 | |
| 				       int (*fn)(struct attribute_container *,
 | |
| 						 struct device *,
 | |
| 						 struct class_device *));
 | |
| void attribute_container_add_device(struct device *dev,
 | |
| 				    int (*fn)(struct attribute_container *,
 | |
| 					      struct device *,
 | |
| 					      struct class_device *));
 | |
| void attribute_container_remove_device(struct device *dev,
 | |
| 				       void (*fn)(struct attribute_container *,
 | |
| 						  struct device *,
 | |
| 						  struct class_device *));
 | |
| void attribute_container_device_trigger(struct device *dev, 
 | |
| 					int (*fn)(struct attribute_container *,
 | |
| 						  struct device *,
 | |
| 						  struct class_device *));
 | |
| void attribute_container_trigger(struct device *dev, 
 | |
| 				 int (*fn)(struct attribute_container *,
 | |
| 					   struct device *));
 | |
| int attribute_container_add_attrs(struct class_device *classdev);
 | |
| int attribute_container_add_class_device(struct class_device *classdev);
 | |
| int attribute_container_add_class_device_adapter(struct attribute_container *cont,
 | |
| 						 struct device *dev,
 | |
| 						 struct class_device *classdev);
 | |
| void attribute_container_remove_attrs(struct class_device *classdev);
 | |
| void attribute_container_class_device_del(struct class_device *classdev);
 | |
| struct attribute_container *attribute_container_classdev_to_container(struct class_device *);
 | |
| struct class_device *attribute_container_find_class_device(struct attribute_container *, struct device *);
 | |
| struct class_device_attribute **attribute_container_classdev_to_attrs(const struct class_device *classdev);
 | |
| 
 | |
| #endif
 |