2005-04-16 15:20:36 -07:00
/*
* core . c - contains all core device and protocol registration functions
*
* Copyright 2002 Adam Belay < ambx1 @ neo . rr . com >
*
*/
# include <linux/pnp.h>
# include <linux/types.h>
# include <linux/list.h>
# include <linux/device.h>
# include <linux/module.h>
# include <linux/init.h>
# include <linux/string.h>
# include <linux/slab.h>
# include <linux/errno.h>
2007-05-08 00:25:29 -07:00
# include <linux/dma-mapping.h>
2005-04-16 15:20:36 -07:00
# include "base.h"
static LIST_HEAD ( pnp_protocols ) ;
LIST_HEAD ( pnp_global ) ;
DEFINE_SPINLOCK ( pnp_lock ) ;
2007-05-08 00:35:54 -07:00
/*
* ACPI or PNPBIOS should tell us about all platform devices , so we can
* skip some blind probes . ISAPNP typically enumerates only plug - in ISA
* devices , not built - in things like COM ports .
*/
int pnp_platform_devices ;
EXPORT_SYMBOL ( pnp_platform_devices ) ;
2005-04-16 15:20:36 -07:00
void * pnp_alloc ( long size )
{
void * result ;
some kmalloc/memset ->kzalloc (tree wide)
Transform some calls to kmalloc/memset to a single kzalloc (or kcalloc).
Here is a short excerpt of the semantic patch performing
this transformation:
@@
type T2;
expression x;
identifier f,fld;
expression E;
expression E1,E2;
expression e1,e2,e3,y;
statement S;
@@
x =
- kmalloc
+ kzalloc
(E1,E2)
... when != \(x->fld=E;\|y=f(...,x,...);\|f(...,x,...);\|x=E;\|while(...) S\|for(e1;e2;e3) S\)
- memset((T2)x,0,E1);
@@
expression E1,E2,E3;
@@
- kzalloc(E1 * E2,E3)
+ kcalloc(E1,E2,E3)
[akpm@linux-foundation.org: get kcalloc args the right way around]
Signed-off-by: Yoann Padioleau <padator@wanadoo.fr>
Cc: Richard Henderson <rth@twiddle.net>
Cc: Ivan Kokshaysky <ink@jurassic.park.msu.ru>
Acked-by: Russell King <rmk@arm.linux.org.uk>
Cc: Bryan Wu <bryan.wu@analog.com>
Acked-by: Jiri Slaby <jirislaby@gmail.com>
Cc: Dave Airlie <airlied@linux.ie>
Acked-by: Roland Dreier <rolandd@cisco.com>
Cc: Jiri Kosina <jkosina@suse.cz>
Acked-by: Dmitry Torokhov <dtor@mail.ru>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Acked-by: Mauro Carvalho Chehab <mchehab@infradead.org>
Acked-by: Pierre Ossman <drzeus-list@drzeus.cx>
Cc: Jeff Garzik <jeff@garzik.org>
Cc: "David S. Miller" <davem@davemloft.net>
Acked-by: Greg KH <greg@kroah.com>
Cc: James Bottomley <James.Bottomley@steeleye.com>
Cc: "Antonino A. Daplas" <adaplas@pol.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-07-19 01:49:03 -07:00
result = kzalloc ( size , GFP_KERNEL ) ;
2005-04-16 15:20:36 -07:00
if ( ! result ) {
printk ( KERN_ERR " pnp: Out of Memory \n " ) ;
return NULL ;
}
return result ;
}
/**
* pnp_protocol_register - adds a pnp protocol to the pnp layer
* @ protocol : pointer to the corresponding pnp_protocol structure
*
* Ex protocols : ISAPNP , PNPBIOS , etc
*/
int pnp_register_protocol ( struct pnp_protocol * protocol )
{
int nodenum ;
struct list_head * pos ;
if ( ! protocol )
return - EINVAL ;
INIT_LIST_HEAD ( & protocol - > devices ) ;
INIT_LIST_HEAD ( & protocol - > cards ) ;
nodenum = 0 ;
spin_lock ( & pnp_lock ) ;
/* assign the lowest unused number */
list_for_each ( pos , & pnp_protocols ) {
struct pnp_protocol * cur = to_pnp_protocol ( pos ) ;
if ( cur - > number = = nodenum ) {
pos = & pnp_protocols ;
nodenum + + ;
}
}
list_add_tail ( & protocol - > protocol_list , & pnp_protocols ) ;
spin_unlock ( & pnp_lock ) ;
protocol - > number = nodenum ;
sprintf ( protocol - > dev . bus_id , " pnp%d " , nodenum ) ;
return device_register ( & protocol - > dev ) ;
}
/**
* pnp_protocol_unregister - removes a pnp protocol from the pnp layer
* @ protocol : pointer to the corresponding pnp_protocol structure
*
*/
void pnp_unregister_protocol ( struct pnp_protocol * protocol )
{
spin_lock ( & pnp_lock ) ;
list_del ( & protocol - > protocol_list ) ;
spin_unlock ( & pnp_lock ) ;
device_unregister ( & protocol - > dev ) ;
}
static void pnp_free_ids ( struct pnp_dev * dev )
{
struct pnp_id * id ;
struct pnp_id * next ;
if ( ! dev )
return ;
id = dev - > id ;
while ( id ) {
next = id - > next ;
kfree ( id ) ;
id = next ;
}
}
static void pnp_release_device ( struct device * dmdev )
{
struct pnp_dev * dev = to_pnp_dev ( dmdev ) ;
pnp_free_option ( dev - > independent ) ;
pnp_free_option ( dev - > dependent ) ;
pnp_free_ids ( dev ) ;
kfree ( dev ) ;
}
int __pnp_add_device ( struct pnp_dev * dev )
{
int ret ;
pnp_fixup_device ( dev ) ;
dev - > dev . bus = & pnp_bus_type ;
2007-05-08 00:25:29 -07:00
dev - > dev . dma_mask = & dev - > dma_mask ;
dev - > dma_mask = dev - > dev . coherent_dma_mask = DMA_24BIT_MASK ;
2005-04-16 15:20:36 -07:00
dev - > dev . release = & pnp_release_device ;
dev - > status = PNP_READY ;
spin_lock ( & pnp_lock ) ;
list_add_tail ( & dev - > global_list , & pnp_global ) ;
list_add_tail ( & dev - > protocol_list , & dev - > protocol - > devices ) ;
spin_unlock ( & pnp_lock ) ;
ret = device_register ( & dev - > dev ) ;
if ( ret = = 0 )
pnp_interface_attach_device ( dev ) ;
return ret ;
}
/*
* pnp_add_device - adds a pnp device to the pnp layer
* @ dev : pointer to dev to add
*
* adds to driver model , name database , fixups , interface , etc .
*/
int pnp_add_device ( struct pnp_dev * dev )
{
if ( ! dev | | ! dev - > protocol | | dev - > card )
return - EINVAL ;
dev - > dev . parent = & dev - > protocol - > dev ;
sprintf ( dev - > dev . bus_id , " %02x:%02x " , dev - > protocol - > number , dev - > number ) ;
return __pnp_add_device ( dev ) ;
}
void __pnp_remove_device ( struct pnp_dev * dev )
{
spin_lock ( & pnp_lock ) ;
list_del ( & dev - > global_list ) ;
list_del ( & dev - > protocol_list ) ;
spin_unlock ( & pnp_lock ) ;
device_unregister ( & dev - > dev ) ;
}
/**
* pnp_remove_device - removes a pnp device from the pnp layer
* @ dev : pointer to dev to add
*
* this function will free all mem used by dev
*/
2005-11-07 01:01:48 -08:00
#if 0
2005-04-16 15:20:36 -07:00
void pnp_remove_device ( struct pnp_dev * dev )
{
if ( ! dev | | dev - > card )
return ;
__pnp_remove_device ( dev ) ;
}
2005-11-07 01:01:48 -08:00
# endif /* 0 */
2005-04-16 15:20:36 -07:00
static int __init pnp_init ( void )
{
printk ( KERN_INFO " Linux Plug and Play Support v0.97 (c) Adam Belay \n " ) ;
return bus_register ( & pnp_bus_type ) ;
}
subsys_initcall ( pnp_init ) ;
2005-11-07 01:01:48 -08:00
#if 0
2005-04-16 15:20:36 -07:00
EXPORT_SYMBOL ( pnp_register_protocol ) ;
EXPORT_SYMBOL ( pnp_unregister_protocol ) ;
EXPORT_SYMBOL ( pnp_add_device ) ;
EXPORT_SYMBOL ( pnp_remove_device ) ;
2005-11-07 01:01:48 -08:00
# endif /* 0 */