2005-04-17 02:20:36 +04:00
/*
2005-05-20 22:33:25 +04:00
* w1 . c
2005-04-17 02:20:36 +04:00
*
* Copyright ( c ) 2004 Evgeniy Polyakov < johnpol @ 2 ka . mipt . ru >
2005-05-20 22:33:25 +04:00
*
2005-04-17 02:20:36 +04:00
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include <linux/delay.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/list.h>
# include <linux/interrupt.h>
# include <linux/spinlock.h>
# include <linux/timer.h>
# include <linux/device.h>
# include <linux/slab.h>
# include <linux/sched.h>
# include <asm/atomic.h>
# include "w1.h"
# include "w1_io.h"
# include "w1_log.h"
# include "w1_int.h"
# include "w1_family.h"
# include "w1_netlink.h"
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " Evgeniy Polyakov <johnpol@2ka.mipt.ru> " ) ;
MODULE_DESCRIPTION ( " Driver for 1-wire Dallas network protocol. " ) ;
static int w1_timeout = 10 ;
2005-08-11 17:27:50 +04:00
static int w1_control_timeout = 1 ;
2005-04-17 02:20:36 +04:00
int w1_max_slave_count = 10 ;
int w1_max_slave_ttl = 10 ;
module_param_named ( timeout , w1_timeout , int , 0 ) ;
2005-08-11 17:27:50 +04:00
module_param_named ( control_timeout , w1_control_timeout , int , 0 ) ;
2005-04-17 02:20:36 +04:00
module_param_named ( max_slave_count , w1_max_slave_count , int , 0 ) ;
module_param_named ( slave_ttl , w1_max_slave_ttl , int , 0 ) ;
DEFINE_SPINLOCK ( w1_mlock ) ;
LIST_HEAD ( w1_masters ) ;
static pid_t control_thread ;
static int control_needs_exit ;
static DECLARE_COMPLETION ( w1_control_complete ) ;
static int w1_master_match ( struct device * dev , struct device_driver * drv )
{
return 1 ;
}
static int w1_master_probe ( struct device * dev )
{
return - ENODEV ;
}
static void w1_master_release ( struct device * dev )
{
2005-08-11 17:27:49 +04:00
struct w1_master * md = dev_to_w1_master ( dev ) ;
2005-08-11 17:27:50 +04:00
dev_dbg ( dev , " %s: Releasing %s. \n " , __func__ , md - > name ) ;
if ( md - > nls & & md - > nls - > sk_socket )
sock_release ( md - > nls - > sk_socket ) ;
memset ( md , 0 , sizeof ( struct w1_master ) + sizeof ( struct w1_bus_master ) ) ;
kfree ( md ) ;
2005-04-17 02:20:36 +04:00
}
static void w1_slave_release ( struct device * dev )
{
2005-08-11 17:27:49 +04:00
struct w1_slave * sl = dev_to_w1_slave ( dev ) ;
2005-08-11 17:27:50 +04:00
dev_dbg ( dev , " %s: Releasing %s. \n " , __func__ , sl - > name ) ;
while ( atomic_read ( & sl - > refcnt ) ) {
dev_dbg ( dev , " Waiting for %s to become free: refcnt=%d. \n " ,
sl - > name , atomic_read ( & sl - > refcnt ) ) ;
if ( msleep_interruptible ( 1000 ) )
flush_signals ( current ) ;
}
w1_family_put ( sl - > family ) ;
sl - > master - > slave_count - - ;
complete ( & sl - > released ) ;
2005-04-17 02:20:36 +04:00
}
2005-08-11 17:27:50 +04:00
static ssize_t w1_slave_read_name ( struct device * dev , struct device_attribute * attr , char * buf )
2005-04-17 02:20:36 +04:00
{
2005-08-11 17:27:50 +04:00
struct w1_slave * sl = dev_to_w1_slave ( dev ) ;
2005-08-11 17:27:50 +04:00
2005-08-11 17:27:50 +04:00
return sprintf ( buf , " %s \n " , sl - > name ) ;
2005-04-17 02:20:36 +04:00
}
2005-08-11 17:27:50 +04:00
static ssize_t w1_slave_read_id ( struct kobject * kobj , char * buf , loff_t off , size_t count )
2005-04-17 02:20:36 +04:00
{
2005-08-11 17:27:50 +04:00
struct w1_slave * sl = kobj_to_w1_slave ( kobj ) ;
2005-04-17 02:20:36 +04:00
2005-08-11 17:27:50 +04:00
atomic_inc ( & sl - > refcnt ) ;
if ( off > 8 ) {
count = 0 ;
2005-08-11 17:27:50 +04:00
} else {
if ( off + count > 8 )
count = 8 - off ;
memcpy ( buf , ( u8 * ) & sl - > reg_num , count ) ;
}
atomic_dec ( & sl - > refcnt ) ;
return count ;
2005-08-11 17:27:50 +04:00
}
2005-08-11 17:27:50 +04:00
static struct device_attribute w1_slave_attr_name =
__ATTR ( name , S_IRUGO , w1_slave_read_name , NULL ) ;
static struct bin_attribute w1_slave_attr_bin_id = {
. attr = {
. name = " id " ,
. mode = S_IRUGO ,
. owner = THIS_MODULE ,
} ,
. size = 8 ,
. read = w1_slave_read_id ,
2005-05-20 22:33:25 +04:00
} ;
2005-08-11 17:27:50 +04:00
/* Default family */
static struct w1_family w1_default_family ;
2005-08-11 13:20:07 +04:00
static int w1_hotplug ( struct device * dev , char * * envp , int num_envp , char * buffer , int buffer_size ) ;
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
static struct bus_type w1_bus_type = {
. name = " w1 " ,
. match = w1_master_match ,
2005-08-11 13:20:07 +04:00
. hotplug = w1_hotplug ,
2005-04-17 02:20:36 +04:00
} ;
2005-08-11 13:20:07 +04:00
struct device_driver w1_master_driver = {
. name = " w1_master_driver " ,
2005-04-17 02:20:36 +04:00
. bus = & w1_bus_type ,
. probe = w1_master_probe ,
} ;
2005-08-11 13:20:07 +04:00
struct device w1_master_device = {
2005-04-17 02:20:36 +04:00
. parent = NULL ,
. bus = & w1_bus_type ,
. bus_id = " w1 bus master " ,
2005-08-11 13:20:07 +04:00
. driver = & w1_master_driver ,
2005-04-17 02:20:36 +04:00
. release = & w1_master_release
} ;
2005-08-11 13:20:07 +04:00
struct device_driver w1_slave_driver = {
. name = " w1_slave_driver " ,
. bus = & w1_bus_type ,
} ;
struct device w1_slave_device = {
. parent = NULL ,
. bus = & w1_bus_type ,
. bus_id = " w1 bus slave " ,
. driver = & w1_slave_driver ,
2005-08-11 17:27:50 +04:00
. release = & w1_slave_release
2005-08-11 13:20:07 +04:00
} ;
2005-05-17 14:44:04 +04:00
static ssize_t w1_master_attribute_show_name ( struct device * dev , struct device_attribute * attr , char * buf )
2005-04-17 02:20:36 +04:00
{
2005-08-11 17:27:49 +04:00
struct w1_master * md = dev_to_w1_master ( dev ) ;
2005-04-17 02:20:36 +04:00
ssize_t count ;
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
if ( down_interruptible ( & md - > mutex ) )
return - EBUSY ;
count = sprintf ( buf , " %s \n " , md - > name ) ;
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
up ( & md - > mutex ) ;
return count ;
}
2005-06-04 01:31:02 +04:00
static ssize_t w1_master_attribute_store_search ( struct device * dev ,
struct device_attribute * attr ,
const char * buf , size_t count )
{
2005-08-11 17:27:49 +04:00
struct w1_master * md = dev_to_w1_master ( dev ) ;
2005-06-04 01:31:02 +04:00
if ( down_interruptible ( & md - > mutex ) )
return - EBUSY ;
md - > search_count = simple_strtol ( buf , NULL , 0 ) ;
up ( & md - > mutex ) ;
return count ;
}
static ssize_t w1_master_attribute_show_search ( struct device * dev ,
struct device_attribute * attr ,
char * buf )
{
2005-08-11 17:27:49 +04:00
struct w1_master * md = dev_to_w1_master ( dev ) ;
2005-06-04 01:31:02 +04:00
ssize_t count ;
if ( down_interruptible ( & md - > mutex ) )
return - EBUSY ;
count = sprintf ( buf , " %d \n " , md - > search_count ) ;
up ( & md - > mutex ) ;
return count ;
}
2005-05-17 14:44:04 +04:00
static ssize_t w1_master_attribute_show_pointer ( struct device * dev , struct device_attribute * attr , char * buf )
2005-04-17 02:20:36 +04:00
{
2005-08-11 17:27:49 +04:00
struct w1_master * md = dev_to_w1_master ( dev ) ;
2005-04-17 02:20:36 +04:00
ssize_t count ;
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
if ( down_interruptible ( & md - > mutex ) )
return - EBUSY ;
count = sprintf ( buf , " 0x%p \n " , md - > bus_master ) ;
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
up ( & md - > mutex ) ;
return count ;
}
2005-05-17 14:44:04 +04:00
static ssize_t w1_master_attribute_show_timeout ( struct device * dev , struct device_attribute * attr , char * buf )
2005-04-17 02:20:36 +04:00
{
ssize_t count ;
count = sprintf ( buf , " %d \n " , w1_timeout ) ;
return count ;
}
2005-05-17 14:44:04 +04:00
static ssize_t w1_master_attribute_show_max_slave_count ( struct device * dev , struct device_attribute * attr , char * buf )
2005-04-17 02:20:36 +04:00
{
2005-08-11 17:27:49 +04:00
struct w1_master * md = dev_to_w1_master ( dev ) ;
2005-04-17 02:20:36 +04:00
ssize_t count ;
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
if ( down_interruptible ( & md - > mutex ) )
return - EBUSY ;
count = sprintf ( buf , " %d \n " , md - > max_slave_count ) ;
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
up ( & md - > mutex ) ;
return count ;
}
2005-05-17 14:44:04 +04:00
static ssize_t w1_master_attribute_show_attempts ( struct device * dev , struct device_attribute * attr , char * buf )
2005-04-17 02:20:36 +04:00
{
2005-08-11 17:27:49 +04:00
struct w1_master * md = dev_to_w1_master ( dev ) ;
2005-04-17 02:20:36 +04:00
ssize_t count ;
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
if ( down_interruptible ( & md - > mutex ) )
return - EBUSY ;
count = sprintf ( buf , " %lu \n " , md - > attempts ) ;
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
up ( & md - > mutex ) ;
return count ;
}
2005-05-17 14:44:04 +04:00
static ssize_t w1_master_attribute_show_slave_count ( struct device * dev , struct device_attribute * attr , char * buf )
2005-04-17 02:20:36 +04:00
{
2005-08-11 17:27:49 +04:00
struct w1_master * md = dev_to_w1_master ( dev ) ;
2005-04-17 02:20:36 +04:00
ssize_t count ;
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
if ( down_interruptible ( & md - > mutex ) )
return - EBUSY ;
count = sprintf ( buf , " %d \n " , md - > slave_count ) ;
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
up ( & md - > mutex ) ;
return count ;
}
2005-05-17 14:44:04 +04:00
static ssize_t w1_master_attribute_show_slaves ( struct device * dev , struct device_attribute * attr , char * buf )
2005-04-17 02:20:36 +04:00
{
2005-08-11 17:27:49 +04:00
struct w1_master * md = dev_to_w1_master ( dev ) ;
2005-04-17 02:20:36 +04:00
int c = PAGE_SIZE ;
if ( down_interruptible ( & md - > mutex ) )
return - EBUSY ;
if ( md - > slave_count = = 0 )
c - = snprintf ( buf + PAGE_SIZE - c , c , " not found. \n " ) ;
else {
struct list_head * ent , * n ;
struct w1_slave * sl ;
list_for_each_safe ( ent , n , & md - > slist ) {
sl = list_entry ( ent , struct w1_slave , w1_slave_entry ) ;
2005-05-20 22:33:25 +04:00
c - = snprintf ( buf + PAGE_SIZE - c , c , " %s \n " , sl - > name ) ;
2005-04-17 02:20:36 +04:00
}
}
up ( & md - > mutex ) ;
return PAGE_SIZE - c ;
}
2005-05-20 22:33:25 +04:00
# define W1_MASTER_ATTR_RO(_name, _mode) \
struct device_attribute w1_master_attribute_ # # _name = \
__ATTR ( w1_master_ # # _name , _mode , \
w1_master_attribute_show_ # # _name , NULL )
2005-06-04 01:31:02 +04:00
# define W1_MASTER_ATTR_RW(_name, _mode) \
struct device_attribute w1_master_attribute_ # # _name = \
__ATTR ( w1_master_ # # _name , _mode , \
w1_master_attribute_show_ # # _name , \
w1_master_attribute_store_ # # _name )
2005-05-20 22:33:25 +04:00
static W1_MASTER_ATTR_RO ( name , S_IRUGO ) ;
static W1_MASTER_ATTR_RO ( slaves , S_IRUGO ) ;
static W1_MASTER_ATTR_RO ( slave_count , S_IRUGO ) ;
static W1_MASTER_ATTR_RO ( max_slave_count , S_IRUGO ) ;
static W1_MASTER_ATTR_RO ( attempts , S_IRUGO ) ;
static W1_MASTER_ATTR_RO ( timeout , S_IRUGO ) ;
static W1_MASTER_ATTR_RO ( pointer , S_IRUGO ) ;
2005-06-04 01:31:02 +04:00
static W1_MASTER_ATTR_RW ( search , S_IRUGO | S_IWUGO ) ;
2005-05-20 22:33:25 +04:00
static struct attribute * w1_master_default_attrs [ ] = {
& w1_master_attribute_name . attr ,
& w1_master_attribute_slaves . attr ,
& w1_master_attribute_slave_count . attr ,
& w1_master_attribute_max_slave_count . attr ,
& w1_master_attribute_attempts . attr ,
& w1_master_attribute_timeout . attr ,
& w1_master_attribute_pointer . attr ,
2005-06-04 01:31:02 +04:00
& w1_master_attribute_search . attr ,
2005-05-20 22:33:25 +04:00
NULL
2005-04-17 02:20:36 +04:00
} ;
2005-05-20 22:33:25 +04:00
static struct attribute_group w1_master_defattr_group = {
. attrs = w1_master_default_attrs ,
2005-04-17 02:20:36 +04:00
} ;
2005-05-20 22:33:25 +04:00
int w1_create_master_attributes ( struct w1_master * master )
{
return sysfs_create_group ( & master - > dev . kobj , & w1_master_defattr_group ) ;
}
void w1_destroy_master_attributes ( struct w1_master * master )
{
sysfs_remove_group ( & master - > dev . kobj , & w1_master_defattr_group ) ;
}
2005-08-11 13:20:07 +04:00
# ifdef CONFIG_HOTPLUG
static int w1_hotplug ( struct device * dev , char * * envp , int num_envp , char * buffer , int buffer_size )
{
struct w1_master * md = NULL ;
struct w1_slave * sl = NULL ;
char * event_owner , * name ;
int err , cur_index = 0 , cur_len = 0 ;
if ( dev - > driver = = & w1_master_driver ) {
md = container_of ( dev , struct w1_master , dev ) ;
event_owner = " master " ;
name = md - > name ;
} else if ( dev - > driver = = & w1_slave_driver ) {
sl = container_of ( dev , struct w1_slave , dev ) ;
event_owner = " slave " ;
name = sl - > name ;
} else {
dev_dbg ( dev , " Unknown hotplug event. \n " ) ;
return - EINVAL ;
}
dev_dbg ( dev , " Hotplug event for %s %s, bus_id=%s. \n " , event_owner , name , dev - > bus_id ) ;
if ( dev - > driver ! = & w1_slave_driver | | ! sl )
return 0 ;
err = add_hotplug_env_var ( envp , num_envp , & cur_index , buffer , buffer_size , & cur_len , " W1_FID=%02X " , sl - > reg_num . family ) ;
if ( err )
return err ;
2005-08-11 13:45:54 +04:00
err = add_hotplug_env_var ( envp , num_envp , & cur_index , buffer , buffer_size , & cur_len , " W1_SLAVE_ID=%024LX " , ( u64 ) sl - > reg_num . id ) ;
2005-08-11 13:20:07 +04:00
if ( err )
return err ;
return 0 ;
} ;
# else
static int w1_hotplug ( struct device * dev , char * * envp , int num_envp , char * buffer , int buffer_size )
{
return 0 ;
}
# endif
2005-04-17 02:20:36 +04:00
static int __w1_attach_slave_device ( struct w1_slave * sl )
{
int err ;
sl - > dev . parent = & sl - > master - > dev ;
2005-08-11 13:20:07 +04:00
sl - > dev . driver = & w1_slave_driver ;
2005-04-17 02:20:36 +04:00
sl - > dev . bus = & w1_bus_type ;
sl - > dev . release = & w1_slave_release ;
snprintf ( & sl - > dev . bus_id [ 0 ] , sizeof ( sl - > dev . bus_id ) ,
2005-06-04 01:30:43 +04:00
" %02x-%012llx " ,
( unsigned int ) sl - > reg_num . family ,
( unsigned long long ) sl - > reg_num . id ) ;
snprintf ( & sl - > name [ 0 ] , sizeof ( sl - > name ) ,
" %02x-%012llx " ,
( unsigned int ) sl - > reg_num . family ,
( unsigned long long ) sl - > reg_num . id ) ;
2005-04-17 02:20:36 +04:00
2005-08-11 17:27:50 +04:00
dev_dbg ( & sl - > dev , " %s: registering %s as %p. \n " , __func__ , & sl - > dev . bus_id [ 0 ] ) ;
2005-04-17 02:20:36 +04:00
err = device_register ( & sl - > dev ) ;
if ( err < 0 ) {
dev_err ( & sl - > dev ,
2005-06-04 01:30:43 +04:00
" Device registration [%s] failed. err=%d \n " ,
sl - > dev . bus_id , err ) ;
2005-04-17 02:20:36 +04:00
return err ;
}
2005-08-11 17:27:50 +04:00
/* Create "name" entry */
err = device_create_file ( & sl - > dev , & w1_slave_attr_name ) ;
if ( err < 0 ) {
dev_err ( & sl - > dev ,
" sysfs file creation for [%s] failed. err=%d \n " ,
sl - > dev . bus_id , err ) ;
goto out_unreg ;
}
2005-04-17 02:20:36 +04:00
2005-08-11 17:27:50 +04:00
/* Create "id" entry */
err = sysfs_create_bin_file ( & sl - > dev . kobj , & w1_slave_attr_bin_id ) ;
2005-04-17 02:20:36 +04:00
if ( err < 0 ) {
dev_err ( & sl - > dev ,
2005-06-04 01:30:43 +04:00
" sysfs file creation for [%s] failed. err=%d \n " ,
sl - > dev . bus_id , err ) ;
2005-08-11 17:27:50 +04:00
goto out_rem1 ;
2005-04-17 02:20:36 +04:00
}
2005-08-11 17:27:50 +04:00
/* if the family driver needs to initialize something... */
if ( sl - > family - > fops & & sl - > family - > fops - > add_slave & &
( ( err = sl - > family - > fops - > add_slave ( sl ) ) < 0 ) ) {
dev_err ( & sl - > dev ,
" sysfs file creation for [%s] failed. err=%d \n " ,
sl - > dev . bus_id , err ) ;
goto out_rem2 ;
2005-04-17 02:20:36 +04:00
}
list_add_tail ( & sl - > w1_slave_entry , & sl - > master - > slist ) ;
return 0 ;
2005-08-11 17:27:50 +04:00
out_rem2 :
sysfs_remove_bin_file ( & sl - > dev . kobj , & w1_slave_attr_bin_id ) ;
out_rem1 :
device_remove_file ( & sl - > dev , & w1_slave_attr_name ) ;
out_unreg :
device_unregister ( & sl - > dev ) ;
return err ;
2005-04-17 02:20:36 +04:00
}
static int w1_attach_slave_device ( struct w1_master * dev , struct w1_reg_num * rn )
{
struct w1_slave * sl ;
struct w1_family * f ;
int err ;
struct w1_netlink_msg msg ;
sl = kmalloc ( sizeof ( struct w1_slave ) , GFP_KERNEL ) ;
if ( ! sl ) {
dev_err ( & dev - > dev ,
" %s: failed to allocate new slave device. \n " ,
__func__ ) ;
return - ENOMEM ;
}
memset ( sl , 0 , sizeof ( * sl ) ) ;
sl - > owner = THIS_MODULE ;
sl - > master = dev ;
set_bit ( W1_SLAVE_ACTIVE , ( long * ) & sl - > flags ) ;
memcpy ( & sl - > reg_num , rn , sizeof ( sl - > reg_num ) ) ;
atomic_set ( & sl - > refcnt , 0 ) ;
2005-08-11 17:27:50 +04:00
init_completion ( & sl - > released ) ;
2005-04-17 02:20:36 +04:00
spin_lock ( & w1_flock ) ;
f = w1_family_registered ( rn - > family ) ;
if ( ! f ) {
2005-06-04 01:31:26 +04:00
f = & w1_default_family ;
2005-04-17 02:20:36 +04:00
dev_info ( & dev - > dev , " Family %x for %02x.%012llx.%02x is not registered. \n " ,
rn - > family , rn - > family ,
( unsigned long long ) rn - > id , rn - > crc ) ;
}
__w1_family_get ( f ) ;
spin_unlock ( & w1_flock ) ;
sl - > family = f ;
err = __w1_attach_slave_device ( sl ) ;
if ( err < 0 ) {
dev_err ( & dev - > dev , " %s: Attaching %s failed. \n " , __func__ ,
sl - > name ) ;
w1_family_put ( sl - > family ) ;
kfree ( sl ) ;
return err ;
}
sl - > ttl = dev - > slave_ttl ;
dev - > slave_count + + ;
memcpy ( & msg . id . id , rn , sizeof ( msg . id . id ) ) ;
msg . type = W1_SLAVE_ADD ;
w1_netlink_send ( dev , & msg ) ;
return 0 ;
}
static void w1_slave_detach ( struct w1_slave * sl )
{
struct w1_netlink_msg msg ;
2005-05-20 22:33:25 +04:00
2005-08-11 17:27:50 +04:00
dev_dbg ( & sl - > dev , " %s: detaching %s [%p]. \n " , __func__ , sl - > name , sl ) ;
2005-04-17 02:20:36 +04:00
2005-08-11 17:27:50 +04:00
list_del ( & sl - > w1_slave_entry ) ;
2005-04-17 02:20:36 +04:00
2005-08-11 17:27:50 +04:00
if ( sl - > family - > fops & & sl - > family - > fops - > remove_slave )
sl - > family - > fops - > remove_slave ( sl ) ;
2005-08-11 17:27:50 +04:00
memcpy ( & msg . id . id , & sl - > reg_num , sizeof ( msg . id . id ) ) ;
msg . type = W1_SLAVE_REMOVE ;
w1_netlink_send ( sl - > master , & msg ) ;
2005-08-11 17:27:50 +04:00
sysfs_remove_bin_file ( & sl - > dev . kobj , & w1_slave_attr_bin_id ) ;
device_remove_file ( & sl - > dev , & w1_slave_attr_name ) ;
2005-04-17 02:20:36 +04:00
device_unregister ( & sl - > dev ) ;
2005-08-11 17:27:50 +04:00
wait_for_completion ( & sl - > released ) ;
kfree ( sl ) ;
2005-04-17 02:20:36 +04:00
}
static struct w1_master * w1_search_master ( unsigned long data )
{
struct w1_master * dev ;
int found = 0 ;
2005-05-20 22:33:25 +04:00
spin_lock_bh ( & w1_mlock ) ;
2005-04-17 02:20:36 +04:00
list_for_each_entry ( dev , & w1_masters , w1_master_entry ) {
if ( dev - > bus_master - > data = = data ) {
found = 1 ;
atomic_inc ( & dev - > refcnt ) ;
break ;
}
}
2005-05-20 22:33:25 +04:00
spin_unlock_bh ( & w1_mlock ) ;
2005-04-17 02:20:36 +04:00
return ( found ) ? dev : NULL ;
}
2005-06-04 01:29:25 +04:00
void w1_reconnect_slaves ( struct w1_family * f )
{
struct w1_master * dev ;
spin_lock_bh ( & w1_mlock ) ;
list_for_each_entry ( dev , & w1_masters , w1_master_entry ) {
2005-08-11 17:27:50 +04:00
dev_dbg ( & dev - > dev , " Reconnecting slaves in %s into new family %02x. \n " ,
2005-06-04 01:29:25 +04:00
dev - > name , f - > fid ) ;
set_bit ( W1_MASTER_NEED_RECONNECT , & dev - > flags ) ;
}
spin_unlock_bh ( & w1_mlock ) ;
}
2005-05-20 22:33:25 +04:00
static void w1_slave_found ( unsigned long data , u64 rn )
2005-04-17 02:20:36 +04:00
{
int slave_count ;
struct w1_slave * sl ;
struct list_head * ent ;
struct w1_reg_num * tmp ;
int family_found = 0 ;
struct w1_master * dev ;
2005-06-30 22:52:38 +04:00
u64 rn_le = cpu_to_le64 ( rn ) ;
2005-04-17 02:20:36 +04:00
dev = w1_search_master ( data ) ;
if ( ! dev ) {
printk ( KERN_ERR " Failed to find w1 master device for data %08lx, it is impossible. \n " ,
data ) ;
return ;
}
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
tmp = ( struct w1_reg_num * ) & rn ;
slave_count = 0 ;
list_for_each ( ent , & dev - > slist ) {
sl = list_entry ( ent , struct w1_slave , w1_slave_entry ) ;
if ( sl - > reg_num . family = = tmp - > family & &
sl - > reg_num . id = = tmp - > id & &
sl - > reg_num . crc = = tmp - > crc ) {
set_bit ( W1_SLAVE_ACTIVE , ( long * ) & sl - > flags ) ;
break ;
2005-05-20 22:33:25 +04:00
} else if ( sl - > reg_num . family = = tmp - > family ) {
2005-04-17 02:20:36 +04:00
family_found = 1 ;
break ;
}
slave_count + + ;
}
2005-04-19 08:16:57 +04:00
if ( slave_count = = dev - > slave_count & &
2005-06-30 22:52:38 +04:00
rn & & ( ( rn > > 56 ) & 0xff ) = = w1_calc_crc8 ( ( u8 * ) & rn_le , 7 ) ) {
2005-04-19 08:16:57 +04:00
w1_attach_slave_device ( dev , tmp ) ;
2005-04-17 02:20:36 +04:00
}
2005-05-20 22:33:25 +04:00
2005-04-17 02:20:36 +04:00
atomic_dec ( & dev - > refcnt ) ;
}
2005-06-04 01:30:43 +04:00
/**
* Performs a ROM Search & registers any devices found .
* The 1 - wire search is a simple binary tree search .
* For each bit of the address , we read two bits and write one bit .
* The bit written will put to sleep all devies that don ' t match that bit .
* When the two reads differ , the direction choice is obvious .
* When both bits are 0 , we must choose a path to take .
* When we can scan all 64 bits without having to choose a path , we are done .
*
* See " Application note 187 1-wire search algorithm " at www . maxim - ic . com
*
* @ dev The master device to search
* @ cb Function to call when a device is found
*/
void w1_search ( struct w1_master * dev , w1_slave_found_callback cb )
2005-04-17 02:20:36 +04:00
{
2005-06-04 01:30:43 +04:00
u64 last_rn , rn , tmp64 ;
int i , slave_count = 0 ;
int last_zero , last_device ;
int search_bit , desc_bit ;
u8 triplet_ret = 0 ;
2005-04-17 02:20:36 +04:00
2005-06-04 01:30:43 +04:00
search_bit = 0 ;
rn = last_rn = 0 ;
last_device = 0 ;
last_zero = - 1 ;
2005-04-17 02:20:36 +04:00
desc_bit = 64 ;
2005-06-04 01:30:43 +04:00
while ( ! last_device & & ( slave_count + + < dev - > max_slave_count ) ) {
last_rn = rn ;
2005-04-17 02:20:36 +04:00
rn = 0 ;
/*
* Reset bus and all 1 - wire device state machines
* so they can respond to our requests .
*
* Return 0 - device ( s ) present , 1 - no devices present .
*/
if ( w1_reset_bus ( dev ) ) {
2005-08-12 22:46:22 +04:00
dev_dbg ( & dev - > dev , " No devices present on the wire. \n " ) ;
2005-04-17 02:20:36 +04:00
break ;
}
2005-06-04 01:30:43 +04:00
/* Start the search */
2005-04-17 02:20:36 +04:00
w1_write_8 ( dev , W1_SEARCH ) ;
for ( i = 0 ; i < 64 ; + + i ) {
2005-06-04 01:30:43 +04:00
/* Determine the direction/search bit */
if ( i = = desc_bit )
search_bit = 1 ; /* took the 0 path last time, so take the 1 path */
else if ( i > desc_bit )
search_bit = 0 ; /* take the 0 path on the next branch */
2005-04-17 02:20:36 +04:00
else
2005-06-04 01:30:43 +04:00
search_bit = ( ( last_rn > > i ) & 0x1 ) ;
2005-04-17 02:20:36 +04:00
2005-06-04 01:30:43 +04:00
/** Read two bits and write one bit */
triplet_ret = w1_triplet ( dev , search_bit ) ;
/* quit if no device responded */
if ( ( triplet_ret & 0x03 ) = = 0x03 )
break ;
2005-04-17 02:20:36 +04:00
2005-06-04 01:30:43 +04:00
/* If both directions were valid, and we took the 0 path... */
if ( triplet_ret = = 0 )
last_zero = i ;
2005-04-17 02:20:36 +04:00
2005-06-04 01:30:43 +04:00
/* extract the direction taken & update the device number */
tmp64 = ( triplet_ret > > 2 ) ;
rn | = ( tmp64 < < i ) ;
}
2005-05-20 22:33:25 +04:00
2005-06-04 01:30:43 +04:00
if ( ( triplet_ret & 0x03 ) ! = 0x03 ) {
if ( ( desc_bit = = last_zero ) | | ( last_zero < 0 ) )
last_device = 1 ;
desc_bit = last_zero ;
cb ( dev - > bus_master - > data , rn ) ;
}
2005-04-17 02:20:36 +04:00
}
}
2005-05-20 22:33:25 +04:00
static int w1_control ( void * data )
2005-04-17 02:20:36 +04:00
{
2005-05-20 22:33:25 +04:00
struct w1_slave * sl , * sln ;
struct w1_master * dev , * n ;
2005-04-17 02:20:36 +04:00
int err , have_to_wait = 0 ;
daemonize ( " w1_control " ) ;
allow_signal ( SIGTERM ) ;
while ( ! control_needs_exit | | have_to_wait ) {
have_to_wait = 0 ;
2005-06-25 10:13:50 +04:00
try_to_freeze ( ) ;
2005-08-11 17:27:50 +04:00
msleep_interruptible ( w1_control_timeout * 1000 ) ;
2005-04-17 02:20:36 +04:00
if ( signal_pending ( current ) )
flush_signals ( current ) ;
2005-05-20 22:33:25 +04:00
list_for_each_entry_safe ( dev , n , & w1_masters , w1_master_entry ) {
2005-06-04 01:29:25 +04:00
if ( ! control_needs_exit & & ! dev - > flags )
2005-04-17 02:20:36 +04:00
continue ;
/*
* Little race : we can create thread but not set the flag .
* Get a chance for external process to set flag up .
*/
if ( ! dev - > initialized ) {
have_to_wait = 1 ;
continue ;
}
if ( control_needs_exit ) {
2005-06-04 01:29:25 +04:00
set_bit ( W1_MASTER_NEED_EXIT , & dev - > flags ) ;
2005-04-17 02:20:36 +04:00
err = kill_proc ( dev - > kpid , SIGTERM , 1 ) ;
if ( err )
dev_err ( & dev - > dev ,
" Failed to send signal to w1 kernel thread %d. \n " ,
dev - > kpid ) ;
}
2005-06-04 01:29:25 +04:00
if ( test_bit ( W1_MASTER_NEED_EXIT , & dev - > flags ) ) {
wait_for_completion ( & dev - > dev_exited ) ;
spin_lock_bh ( & w1_mlock ) ;
list_del ( & dev - > w1_master_entry ) ;
spin_unlock_bh ( & w1_mlock ) ;
2005-08-11 17:27:50 +04:00
down ( & dev - > mutex ) ;
2005-06-04 01:29:25 +04:00
list_for_each_entry_safe ( sl , sln , & dev - > slist , w1_slave_entry ) {
w1_slave_detach ( sl ) ;
}
w1_destroy_master_attributes ( dev ) ;
2005-08-11 17:27:50 +04:00
up ( & dev - > mutex ) ;
2005-06-04 01:29:25 +04:00
atomic_dec ( & dev - > refcnt ) ;
continue ;
}
if ( test_bit ( W1_MASTER_NEED_RECONNECT , & dev - > flags ) ) {
2005-08-11 17:27:50 +04:00
dev_dbg ( & dev - > dev , " Reconnecting slaves in device %s. \n " , dev - > name ) ;
2005-06-04 01:29:25 +04:00
down ( & dev - > mutex ) ;
2005-08-11 17:27:50 +04:00
list_for_each_entry_safe ( sl , sln , & dev - > slist , w1_slave_entry ) {
2005-06-04 01:29:25 +04:00
if ( sl - > family - > fid = = W1_FAMILY_DEFAULT ) {
struct w1_reg_num rn ;
memcpy ( & rn , & sl - > reg_num , sizeof ( rn ) ) ;
2005-08-11 17:27:50 +04:00
w1_slave_detach ( sl ) ;
2005-04-17 02:20:36 +04:00
2005-06-04 01:29:25 +04:00
w1_attach_slave_device ( dev , & rn ) ;
}
}
2005-08-11 17:27:50 +04:00
dev_dbg ( & dev - > dev , " Reconnecting slaves in device %s has been finished. \n " , dev - > name ) ;
2005-06-04 01:29:25 +04:00
clear_bit ( W1_MASTER_NEED_RECONNECT , & dev - > flags ) ;
up ( & dev - > mutex ) ;
2005-04-17 02:20:36 +04:00
}
}
}
complete_and_exit ( & w1_control_complete , 0 ) ;
}
int w1_process ( void * data )
{
struct w1_master * dev = ( struct w1_master * ) data ;
2005-05-20 22:33:25 +04:00
struct w1_slave * sl , * sln ;
2005-04-17 02:20:36 +04:00
daemonize ( " %s " , dev - > name ) ;
allow_signal ( SIGTERM ) ;
2005-06-04 01:29:25 +04:00
while ( ! test_bit ( W1_MASTER_NEED_EXIT , & dev - > flags ) ) {
2005-06-25 10:13:50 +04:00
try_to_freeze ( ) ;
2005-04-17 02:20:36 +04:00
msleep_interruptible ( w1_timeout * 1000 ) ;
if ( signal_pending ( current ) )
flush_signals ( current ) ;
2005-06-04 01:29:25 +04:00
if ( test_bit ( W1_MASTER_NEED_EXIT , & dev - > flags ) )
2005-04-17 02:20:36 +04:00
break ;
if ( ! dev - > initialized )
continue ;
2005-06-04 01:31:02 +04:00
if ( dev - > search_count = = 0 )
continue ;
2005-04-17 02:20:36 +04:00
if ( down_interruptible ( & dev - > mutex ) )
continue ;
2005-05-20 22:33:25 +04:00
list_for_each_entry ( sl , & dev - > slist , w1_slave_entry )
2005-06-04 01:31:02 +04:00
clear_bit ( W1_SLAVE_ACTIVE , ( long * ) & sl - > flags ) ;
2005-04-17 02:20:36 +04:00
2005-05-20 22:33:25 +04:00
w1_search_devices ( dev , w1_slave_found ) ;
2005-04-17 02:20:36 +04:00
2005-05-20 22:33:25 +04:00
list_for_each_entry_safe ( sl , sln , & dev - > slist , w1_slave_entry ) {
if ( ! test_bit ( W1_SLAVE_ACTIVE , ( unsigned long * ) & sl - > flags ) & & ! - - sl - > ttl ) {
2005-08-11 17:27:50 +04:00
w1_slave_detach ( sl ) ;
2005-04-17 02:20:36 +04:00
dev - > slave_count - - ;
2005-05-20 22:33:25 +04:00
} else if ( test_bit ( W1_SLAVE_ACTIVE , ( unsigned long * ) & sl - > flags ) )
2005-04-17 02:20:36 +04:00
sl - > ttl = dev - > slave_ttl ;
}
2005-06-04 01:31:02 +04:00
if ( dev - > search_count > 0 )
dev - > search_count - - ;
2005-04-17 02:20:36 +04:00
up ( & dev - > mutex ) ;
}
atomic_dec ( & dev - > refcnt ) ;
complete_and_exit ( & dev - > dev_exited , 0 ) ;
return 0 ;
}
2005-05-20 22:33:25 +04:00
static int w1_init ( void )
2005-04-17 02:20:36 +04:00
{
int retval ;
printk ( KERN_INFO " Driver for 1-wire Dallas network protocol. \n " ) ;
retval = bus_register ( & w1_bus_type ) ;
if ( retval ) {
printk ( KERN_ERR " Failed to register bus. err=%d. \n " , retval ) ;
goto err_out_exit_init ;
}
2005-08-11 13:20:07 +04:00
retval = driver_register ( & w1_master_driver ) ;
2005-04-17 02:20:36 +04:00
if ( retval ) {
printk ( KERN_ERR
" Failed to register master driver. err=%d. \n " ,
retval ) ;
goto err_out_bus_unregister ;
}
2005-08-11 13:20:07 +04:00
retval = driver_register ( & w1_slave_driver ) ;
if ( retval ) {
printk ( KERN_ERR
" Failed to register master driver. err=%d. \n " ,
retval ) ;
goto err_out_master_unregister ;
}
2005-04-17 02:20:36 +04:00
control_thread = kernel_thread ( & w1_control , NULL , 0 ) ;
if ( control_thread < 0 ) {
printk ( KERN_ERR " Failed to create control thread. err=%d \n " ,
control_thread ) ;
retval = control_thread ;
2005-08-11 13:20:07 +04:00
goto err_out_slave_unregister ;
2005-04-17 02:20:36 +04:00
}
return 0 ;
2005-08-11 13:20:07 +04:00
err_out_slave_unregister :
driver_unregister ( & w1_slave_driver ) ;
err_out_master_unregister :
driver_unregister ( & w1_master_driver ) ;
2005-04-17 02:20:36 +04:00
err_out_bus_unregister :
bus_unregister ( & w1_bus_type ) ;
err_out_exit_init :
return retval ;
}
2005-05-20 22:33:25 +04:00
static void w1_fini ( void )
2005-04-17 02:20:36 +04:00
{
struct w1_master * dev ;
2005-05-20 22:33:25 +04:00
list_for_each_entry ( dev , & w1_masters , w1_master_entry )
2005-04-17 02:20:36 +04:00
__w1_remove_master_device ( dev ) ;
control_needs_exit = 1 ;
wait_for_completion ( & w1_control_complete ) ;
2005-08-11 13:20:07 +04:00
driver_unregister ( & w1_slave_driver ) ;
driver_unregister ( & w1_master_driver ) ;
2005-04-17 02:20:36 +04:00
bus_unregister ( & w1_bus_type ) ;
}
module_init ( w1_init ) ;
module_exit ( w1_fini ) ;