2007-02-09 17:24:33 +03:00
/*
2005-04-17 02:20:36 +04:00
RFCOMM implementation for Linux Bluetooth stack ( BlueZ ) .
Copyright ( C ) 2002 Maxim Krasnyansky < maxk @ qualcomm . com >
Copyright ( C ) 2002 Marcel Holtmann < marcel @ holtmann . org >
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License version 2 as
published by the Free Software Foundation ;
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 NONINFRINGEMENT OF THIRD PARTY RIGHTS .
IN NO EVENT SHALL THE COPYRIGHT HOLDER ( S ) AND AUTHOR ( S ) BE LIABLE FOR ANY
2007-02-09 17:24:33 +03:00
CLAIM , OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL DAMAGES , OR ANY DAMAGES
WHATSOEVER RESULTING FROM LOSS OF USE , DATA OR PROFITS , WHETHER IN AN
ACTION OF CONTRACT , NEGLIGENCE OR OTHER TORTIOUS ACTION , ARISING OUT OF
2005-04-17 02:20:36 +04:00
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE .
2007-02-09 17:24:33 +03:00
ALL LIABILITY , INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS ,
COPYRIGHTS , TRADEMARKS OR OTHER RIGHTS , RELATING TO USE OF THIS
2005-04-17 02:20:36 +04:00
SOFTWARE IS DISCLAIMED .
*/
/*
* RFCOMM TTY .
*/
# include <linux/module.h>
# include <linux/tty.h>
# include <linux/tty_driver.h>
# include <linux/tty_flip.h>
2006-01-11 23:17:47 +03:00
# include <linux/capability.h>
2005-04-17 02:20:36 +04:00
# include <linux/slab.h>
# include <linux/skbuff.h>
# include <net/bluetooth/bluetooth.h>
2006-07-06 15:09:02 +04:00
# include <net/bluetooth/hci_core.h>
2005-04-17 02:20:36 +04:00
# include <net/bluetooth/rfcomm.h>
# define RFCOMM_TTY_MAGIC 0x6d02 /* magic number for rfcomm struct */
# define RFCOMM_TTY_PORTS RFCOMM_MAX_DEV /* whole lotta rfcomm devices */
# define RFCOMM_TTY_MAJOR 216 /* device node major id of the usb/bluetooth.c driver */
# define RFCOMM_TTY_MINOR 0
static struct tty_driver * rfcomm_tty_driver ;
struct rfcomm_dev {
struct list_head list ;
atomic_t refcnt ;
char name [ 12 ] ;
int id ;
unsigned long flags ;
2008-11-30 14:17:29 +03:00
atomic_t opened ;
2005-04-17 02:20:36 +04:00
int err ;
bdaddr_t src ;
bdaddr_t dst ;
u8 channel ;
uint modem_status ;
struct rfcomm_dlc * dlc ;
struct tty_struct * tty ;
wait_queue_head_t wait ;
struct tasklet_struct wakeup_task ;
2007-02-18 01:58:57 +03:00
struct device * tty_dev ;
2005-04-17 02:20:36 +04:00
atomic_t wmem_alloc ;
2008-07-14 22:13:52 +04:00
struct sk_buff_head pending ;
2005-04-17 02:20:36 +04:00
} ;
static LIST_HEAD ( rfcomm_dev_list ) ;
static DEFINE_RWLOCK ( rfcomm_dev_lock ) ;
static void rfcomm_dev_data_ready ( struct rfcomm_dlc * dlc , struct sk_buff * skb ) ;
static void rfcomm_dev_state_change ( struct rfcomm_dlc * dlc , int err ) ;
static void rfcomm_dev_modem_status ( struct rfcomm_dlc * dlc , u8 v24_sig ) ;
static void rfcomm_tty_wakeup ( unsigned long arg ) ;
/* ---- Device functions ---- */
static void rfcomm_dev_destruct ( struct rfcomm_dev * dev )
{
struct rfcomm_dlc * dlc = dev - > dlc ;
BT_DBG ( " dev %p dlc %p " , dev , dlc ) ;
2008-01-11 09:22:52 +03:00
/* Refcount should only hit zero when called from rfcomm_dev_del()
which will have taken us off the list . Everything else are
refcounting bugs . */
BUG_ON ( ! list_empty ( & dev - > list ) ) ;
2007-07-11 11:23:41 +04:00
2005-04-17 02:20:36 +04:00
rfcomm_dlc_lock ( dlc ) ;
/* Detach DLC if it's owned by this dev */
if ( dlc - > owner = = dev )
dlc - > owner = NULL ;
rfcomm_dlc_unlock ( dlc ) ;
rfcomm_dlc_put ( dlc ) ;
tty_unregister_device ( rfcomm_tty_driver , dev - > id ) ;
kfree ( dev ) ;
2007-02-09 17:24:33 +03:00
/* It's safe to call module_put() here because socket still
2005-04-17 02:20:36 +04:00
holds reference to this module . */
module_put ( THIS_MODULE ) ;
}
static inline void rfcomm_dev_hold ( struct rfcomm_dev * dev )
{
atomic_inc ( & dev - > refcnt ) ;
}
static inline void rfcomm_dev_put ( struct rfcomm_dev * dev )
{
/* The reason this isn't actually a race, as you no
doubt have a little voice screaming at you in your
head , is that the refcount should never actually
reach zero unless the device has already been taken
off the list , in rfcomm_dev_del ( ) . And if that ' s not
true , we ' ll hit the BUG ( ) in rfcomm_dev_destruct ( )
anyway . */
if ( atomic_dec_and_test ( & dev - > refcnt ) )
rfcomm_dev_destruct ( dev ) ;
}
static struct rfcomm_dev * __rfcomm_dev_get ( int id )
{
struct rfcomm_dev * dev ;
struct list_head * p ;
list_for_each ( p , & rfcomm_dev_list ) {
dev = list_entry ( p , struct rfcomm_dev , list ) ;
if ( dev - > id = = id )
return dev ;
}
return NULL ;
}
static inline struct rfcomm_dev * rfcomm_dev_get ( int id )
{
struct rfcomm_dev * dev ;
read_lock ( & rfcomm_dev_lock ) ;
dev = __rfcomm_dev_get ( id ) ;
2007-07-11 11:23:41 +04:00
if ( dev ) {
if ( test_bit ( RFCOMM_TTY_RELEASED , & dev - > flags ) )
dev = NULL ;
else
rfcomm_dev_hold ( dev ) ;
}
2005-04-17 02:20:36 +04:00
read_unlock ( & rfcomm_dev_lock ) ;
return dev ;
}
2006-07-06 15:09:02 +04:00
static struct device * rfcomm_get_device ( struct rfcomm_dev * dev )
{
struct hci_dev * hdev ;
struct hci_conn * conn ;
hdev = hci_get_route ( & dev - > dst , & dev - > src ) ;
if ( ! hdev )
return NULL ;
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & dev - > dst ) ;
hci_dev_put ( hdev ) ;
2006-10-15 19:31:05 +04:00
return conn ? & conn - > dev : NULL ;
2006-07-06 15:09:02 +04:00
}
2007-10-20 16:52:38 +04:00
static ssize_t show_address ( struct device * tty_dev , struct device_attribute * attr , char * buf )
{
struct rfcomm_dev * dev = dev_get_drvdata ( tty_dev ) ;
bdaddr_t bdaddr ;
baswap ( & bdaddr , & dev - > dst ) ;
return sprintf ( buf , " %s \n " , batostr ( & bdaddr ) ) ;
}
static ssize_t show_channel ( struct device * tty_dev , struct device_attribute * attr , char * buf )
{
struct rfcomm_dev * dev = dev_get_drvdata ( tty_dev ) ;
return sprintf ( buf , " %d \n " , dev - > channel ) ;
}
static DEVICE_ATTR ( address , S_IRUGO , show_address , NULL ) ;
static DEVICE_ATTR ( channel , S_IRUGO , show_channel , NULL ) ;
2005-04-17 02:20:36 +04:00
static int rfcomm_dev_add ( struct rfcomm_dev_req * req , struct rfcomm_dlc * dlc )
{
struct rfcomm_dev * dev ;
struct list_head * head = & rfcomm_dev_list , * p ;
int err = 0 ;
BT_DBG ( " id %d channel %d " , req - > dev_id , req - > channel ) ;
2007-02-09 17:24:33 +03:00
2006-07-06 17:40:09 +04:00
dev = kzalloc ( sizeof ( struct rfcomm_dev ) , GFP_KERNEL ) ;
2005-04-17 02:20:36 +04:00
if ( ! dev )
return - ENOMEM ;
write_lock_bh ( & rfcomm_dev_lock ) ;
if ( req - > dev_id < 0 ) {
dev - > id = 0 ;
list_for_each ( p , & rfcomm_dev_list ) {
if ( list_entry ( p , struct rfcomm_dev , list ) - > id ! = dev - > id )
break ;
dev - > id + + ;
head = p ;
}
} else {
dev - > id = req - > dev_id ;
list_for_each ( p , & rfcomm_dev_list ) {
struct rfcomm_dev * entry = list_entry ( p , struct rfcomm_dev , list ) ;
if ( entry - > id = = dev - > id ) {
err = - EADDRINUSE ;
goto out ;
}
if ( entry - > id > dev - > id - 1 )
break ;
head = p ;
}
}
if ( ( dev - > id < 0 ) | | ( dev - > id > RFCOMM_MAX_DEV - 1 ) ) {
err = - ENFILE ;
goto out ;
}
sprintf ( dev - > name , " rfcomm%d " , dev - > id ) ;
list_add ( & dev - > list , head ) ;
atomic_set ( & dev - > refcnt , 1 ) ;
bacpy ( & dev - > src , & req - > src ) ;
bacpy ( & dev - > dst , & req - > dst ) ;
dev - > channel = req - > channel ;
2007-02-09 17:24:33 +03:00
dev - > flags = req - > flags &
2005-04-17 02:20:36 +04:00
( ( 1 < < RFCOMM_RELEASE_ONHUP ) | ( 1 < < RFCOMM_REUSE_DLC ) ) ;
2008-11-30 14:17:29 +03:00
atomic_set ( & dev - > opened , 0 ) ;
2005-04-17 02:20:36 +04:00
init_waitqueue_head ( & dev - > wait ) ;
tasklet_init ( & dev - > wakeup_task , rfcomm_tty_wakeup , ( unsigned long ) dev ) ;
2008-07-14 22:13:52 +04:00
skb_queue_head_init ( & dev - > pending ) ;
2005-04-17 02:20:36 +04:00
rfcomm_dlc_lock ( dlc ) ;
2008-07-14 22:13:52 +04:00
if ( req - > flags & ( 1 < < RFCOMM_REUSE_DLC ) ) {
struct sock * sk = dlc - > owner ;
struct sk_buff * skb ;
BUG_ON ( ! sk ) ;
rfcomm_dlc_throttle ( dlc ) ;
while ( ( skb = skb_dequeue ( & sk - > sk_receive_queue ) ) ) {
skb_orphan ( skb ) ;
skb_queue_tail ( & dev - > pending , skb ) ;
atomic_sub ( skb - > len , & sk - > sk_rmem_alloc ) ;
}
}
2005-04-17 02:20:36 +04:00
dlc - > data_ready = rfcomm_dev_data_ready ;
dlc - > state_change = rfcomm_dev_state_change ;
dlc - > modem_status = rfcomm_dev_modem_status ;
dlc - > owner = dev ;
dev - > dlc = dlc ;
2008-07-14 22:13:52 +04:00
rfcomm_dev_modem_status ( dlc , dlc - > remote_v24_sig ) ;
2005-04-17 02:20:36 +04:00
rfcomm_dlc_unlock ( dlc ) ;
2007-02-09 17:24:33 +03:00
/* It's safe to call __module_get() here because socket already
2005-04-17 02:20:36 +04:00
holds reference to this module . */
__module_get ( THIS_MODULE ) ;
out :
write_unlock_bh ( & rfcomm_dev_lock ) ;
2008-12-15 10:18:00 +03:00
if ( err < 0 )
goto free ;
2005-04-17 02:20:36 +04:00
2007-02-18 01:58:57 +03:00
dev - > tty_dev = tty_register_device ( rfcomm_tty_driver , dev - > id , NULL ) ;
2005-04-17 02:20:36 +04:00
2007-07-11 11:23:41 +04:00
if ( IS_ERR ( dev - > tty_dev ) ) {
2007-07-26 11:12:25 +04:00
err = PTR_ERR ( dev - > tty_dev ) ;
2007-07-11 11:23:41 +04:00
list_del ( & dev - > list ) ;
2008-12-15 10:18:00 +03:00
goto free ;
2007-07-11 11:23:41 +04:00
}
2007-10-20 16:52:38 +04:00
dev_set_drvdata ( dev - > tty_dev , dev ) ;
if ( device_create_file ( dev - > tty_dev , & dev_attr_address ) < 0 )
BT_ERR ( " Failed to create address attribute " ) ;
if ( device_create_file ( dev - > tty_dev , & dev_attr_channel ) < 0 )
BT_ERR ( " Failed to create channel attribute " ) ;
2005-04-17 02:20:36 +04:00
return dev - > id ;
2008-12-15 10:18:00 +03:00
free :
kfree ( dev ) ;
return err ;
2005-04-17 02:20:36 +04:00
}
static void rfcomm_dev_del ( struct rfcomm_dev * dev )
{
BT_DBG ( " dev %p " , dev ) ;
2008-11-30 14:17:29 +03:00
BUG_ON ( test_and_set_bit ( RFCOMM_TTY_RELEASED , & dev - > flags ) ) ;
if ( atomic_read ( & dev - > opened ) > 0 )
return ;
2008-01-11 09:22:52 +03:00
write_lock_bh ( & rfcomm_dev_lock ) ;
list_del_init ( & dev - > list ) ;
write_unlock_bh ( & rfcomm_dev_lock ) ;
2005-04-17 02:20:36 +04:00
rfcomm_dev_put ( dev ) ;
}
/* ---- Send buffer ---- */
static inline unsigned int rfcomm_room ( struct rfcomm_dlc * dlc )
{
/* We can't let it be zero, because we don't get a callback
when tx_credits becomes nonzero , hence we ' d never wake up */
return dlc - > mtu * ( dlc - > tx_credits ? : 1 ) ;
}
static void rfcomm_wfree ( struct sk_buff * skb )
{
struct rfcomm_dev * dev = ( void * ) skb - > sk ;
atomic_sub ( skb - > truesize , & dev - > wmem_alloc ) ;
if ( test_bit ( RFCOMM_TTY_ATTACHED , & dev - > flags ) )
tasklet_schedule ( & dev - > wakeup_task ) ;
rfcomm_dev_put ( dev ) ;
}
static inline void rfcomm_set_owner_w ( struct sk_buff * skb , struct rfcomm_dev * dev )
{
rfcomm_dev_hold ( dev ) ;
atomic_add ( skb - > truesize , & dev - > wmem_alloc ) ;
skb - > sk = ( void * ) dev ;
skb - > destructor = rfcomm_wfree ;
}
2005-10-07 10:46:04 +04:00
static struct sk_buff * rfcomm_wmalloc ( struct rfcomm_dev * dev , unsigned long size , gfp_t priority )
2005-04-17 02:20:36 +04:00
{
if ( atomic_read ( & dev - > wmem_alloc ) < rfcomm_room ( dev - > dlc ) ) {
struct sk_buff * skb = alloc_skb ( size , priority ) ;
if ( skb ) {
rfcomm_set_owner_w ( skb , dev ) ;
return skb ;
}
}
return NULL ;
}
/* ---- Device IOCTLs ---- */
# define NOCAP_FLAGS ((1 << RFCOMM_REUSE_DLC) | (1 << RFCOMM_RELEASE_ONHUP))
static int rfcomm_create_dev ( struct sock * sk , void __user * arg )
{
struct rfcomm_dev_req req ;
struct rfcomm_dlc * dlc ;
int id ;
if ( copy_from_user ( & req , arg , sizeof ( req ) ) )
return - EFAULT ;
2007-07-11 11:23:41 +04:00
BT_DBG ( " sk %p dev_id %d flags 0x%x " , sk , req . dev_id , req . flags ) ;
2005-04-17 02:20:36 +04:00
if ( req . flags ! = NOCAP_FLAGS & & ! capable ( CAP_NET_ADMIN ) )
return - EPERM ;
if ( req . flags & ( 1 < < RFCOMM_REUSE_DLC ) ) {
/* Socket must be connected */
if ( sk - > sk_state ! = BT_CONNECTED )
return - EBADFD ;
dlc = rfcomm_pi ( sk ) - > dlc ;
rfcomm_dlc_hold ( dlc ) ;
} else {
dlc = rfcomm_dlc_alloc ( GFP_KERNEL ) ;
if ( ! dlc )
return - ENOMEM ;
}
id = rfcomm_dev_add ( & req , dlc ) ;
if ( id < 0 ) {
rfcomm_dlc_put ( dlc ) ;
return id ;
}
if ( req . flags & ( 1 < < RFCOMM_REUSE_DLC ) ) {
/* DLC is now used by device.
* Socket must be disconnected */
sk - > sk_state = BT_CLOSED ;
}
return id ;
}
static int rfcomm_release_dev ( void __user * arg )
{
struct rfcomm_dev_req req ;
struct rfcomm_dev * dev ;
if ( copy_from_user ( & req , arg , sizeof ( req ) ) )
return - EFAULT ;
2007-07-11 11:23:41 +04:00
BT_DBG ( " dev_id %d flags 0x%x " , req . dev_id , req . flags ) ;
2005-04-17 02:20:36 +04:00
if ( ! ( dev = rfcomm_dev_get ( req . dev_id ) ) )
return - ENODEV ;
if ( dev - > flags ! = NOCAP_FLAGS & & ! capable ( CAP_NET_ADMIN ) ) {
rfcomm_dev_put ( dev ) ;
return - EPERM ;
}
if ( req . flags & ( 1 < < RFCOMM_HANGUP_NOW ) )
rfcomm_dlc_close ( dev - > dlc , 0 ) ;
2007-07-11 11:18:15 +04:00
/* Shut down TTY synchronously before freeing rfcomm_dev */
if ( dev - > tty )
tty_vhangup ( dev - > tty ) ;
2008-02-05 14:12:06 +03:00
if ( ! test_bit ( RFCOMM_RELEASE_ONHUP , & dev - > flags ) )
rfcomm_dev_del ( dev ) ;
2005-04-17 02:20:36 +04:00
rfcomm_dev_put ( dev ) ;
return 0 ;
}
static int rfcomm_get_dev_list ( void __user * arg )
{
struct rfcomm_dev_list_req * dl ;
struct rfcomm_dev_info * di ;
struct list_head * p ;
int n = 0 , size , err ;
u16 dev_num ;
BT_DBG ( " " ) ;
if ( get_user ( dev_num , ( u16 __user * ) arg ) )
return - EFAULT ;
if ( ! dev_num | | dev_num > ( PAGE_SIZE * 4 ) / sizeof ( * di ) )
return - EINVAL ;
size = sizeof ( * dl ) + dev_num * sizeof ( * di ) ;
if ( ! ( dl = kmalloc ( size , GFP_KERNEL ) ) )
return - ENOMEM ;
di = dl - > dev_info ;
read_lock_bh ( & rfcomm_dev_lock ) ;
list_for_each ( p , & rfcomm_dev_list ) {
struct rfcomm_dev * dev = list_entry ( p , struct rfcomm_dev , list ) ;
2007-07-11 11:23:41 +04:00
if ( test_bit ( RFCOMM_TTY_RELEASED , & dev - > flags ) )
continue ;
2005-04-17 02:20:36 +04:00
( di + n ) - > id = dev - > id ;
( di + n ) - > flags = dev - > flags ;
( di + n ) - > state = dev - > dlc - > state ;
( di + n ) - > channel = dev - > channel ;
bacpy ( & ( di + n ) - > src , & dev - > src ) ;
bacpy ( & ( di + n ) - > dst , & dev - > dst ) ;
if ( + + n > = dev_num )
break ;
}
read_unlock_bh ( & rfcomm_dev_lock ) ;
dl - > dev_num = n ;
size = sizeof ( * dl ) + n * sizeof ( * di ) ;
err = copy_to_user ( arg , dl , size ) ;
kfree ( dl ) ;
return err ? - EFAULT : 0 ;
}
static int rfcomm_get_dev_info ( void __user * arg )
{
struct rfcomm_dev * dev ;
struct rfcomm_dev_info di ;
int err = 0 ;
BT_DBG ( " " ) ;
if ( copy_from_user ( & di , arg , sizeof ( di ) ) )
return - EFAULT ;
if ( ! ( dev = rfcomm_dev_get ( di . id ) ) )
return - ENODEV ;
di . flags = dev - > flags ;
di . channel = dev - > channel ;
di . state = dev - > dlc - > state ;
bacpy ( & di . src , & dev - > src ) ;
bacpy ( & di . dst , & dev - > dst ) ;
if ( copy_to_user ( arg , & di , sizeof ( di ) ) )
err = - EFAULT ;
rfcomm_dev_put ( dev ) ;
return err ;
}
int rfcomm_dev_ioctl ( struct sock * sk , unsigned int cmd , void __user * arg )
{
BT_DBG ( " cmd %d arg %p " , cmd , arg ) ;
switch ( cmd ) {
case RFCOMMCREATEDEV :
return rfcomm_create_dev ( sk , arg ) ;
case RFCOMMRELEASEDEV :
return rfcomm_release_dev ( arg ) ;
case RFCOMMGETDEVLIST :
return rfcomm_get_dev_list ( arg ) ;
case RFCOMMGETDEVINFO :
return rfcomm_get_dev_info ( arg ) ;
}
return - EINVAL ;
}
/* ---- DLC callbacks ---- */
static void rfcomm_dev_data_ready ( struct rfcomm_dlc * dlc , struct sk_buff * skb )
{
struct rfcomm_dev * dev = dlc - > owner ;
struct tty_struct * tty ;
2007-02-09 17:24:33 +03:00
2008-07-14 22:13:52 +04:00
if ( ! dev ) {
2005-04-17 02:20:36 +04:00
kfree_skb ( skb ) ;
return ;
}
2008-07-14 22:13:52 +04:00
if ( ! ( tty = dev - > tty ) | | ! skb_queue_empty ( & dev - > pending ) ) {
skb_queue_tail ( & dev - > pending , skb ) ;
return ;
}
2005-04-17 02:20:36 +04:00
BT_DBG ( " dlc %p tty %p len %d " , dlc , tty , skb - > len ) ;
2006-06-28 15:26:47 +04:00
tty_insert_flip_string ( tty , skb - > data , skb - > len ) ;
tty_flip_buffer_push ( tty ) ;
2005-04-17 02:20:36 +04:00
kfree_skb ( skb ) ;
}
static void rfcomm_dev_state_change ( struct rfcomm_dlc * dlc , int err )
{
struct rfcomm_dev * dev = dlc - > owner ;
if ( ! dev )
return ;
2007-02-09 17:24:33 +03:00
2005-04-17 02:20:36 +04:00
BT_DBG ( " dlc %p dev %p err %d " , dlc , dev , err ) ;
dev - > err = err ;
wake_up_interruptible ( & dev - > wait ) ;
if ( dlc - > state = = BT_CLOSED ) {
if ( ! dev - > tty ) {
if ( test_bit ( RFCOMM_RELEASE_ONHUP , & dev - > flags ) ) {
2008-06-02 10:50:52 +04:00
/* Drop DLC lock here to avoid deadlock
* 1. rfcomm_dev_get will take rfcomm_dev_lock
* but in rfcomm_dev_add there ' s lock order :
* rfcomm_dev_lock - > dlc lock
* 2. rfcomm_dev_put will deadlock if it ' s
* the last reference
*/
rfcomm_dlc_unlock ( dlc ) ;
if ( rfcomm_dev_get ( dev - > id ) = = NULL ) {
rfcomm_dlc_lock ( dlc ) ;
2007-05-05 02:36:10 +04:00
return ;
2008-06-02 10:50:52 +04:00
}
2005-04-17 02:20:36 +04:00
2007-05-05 02:36:10 +04:00
rfcomm_dev_del ( dev ) ;
2005-04-17 02:20:36 +04:00
rfcomm_dev_put ( dev ) ;
2008-06-02 10:50:52 +04:00
rfcomm_dlc_lock ( dlc ) ;
2005-04-17 02:20:36 +04:00
}
2007-02-09 17:24:33 +03:00
} else
2005-04-17 02:20:36 +04:00
tty_hangup ( dev - > tty ) ;
}
}
static void rfcomm_dev_modem_status ( struct rfcomm_dlc * dlc , u8 v24_sig )
{
struct rfcomm_dev * dev = dlc - > owner ;
if ( ! dev )
return ;
2005-08-10 07:28:21 +04:00
2005-04-17 02:20:36 +04:00
BT_DBG ( " dlc %p dev %p v24_sig 0x%02x " , dlc , dev , v24_sig ) ;
2005-08-10 07:28:21 +04:00
if ( ( dev - > modem_status & TIOCM_CD ) & & ! ( v24_sig & RFCOMM_V24_DV ) ) {
if ( dev - > tty & & ! C_CLOCAL ( dev - > tty ) )
tty_hangup ( dev - > tty ) ;
}
2007-02-09 17:24:33 +03:00
dev - > modem_status =
2005-04-17 02:20:36 +04:00
( ( v24_sig & RFCOMM_V24_RTC ) ? ( TIOCM_DSR | TIOCM_DTR ) : 0 ) |
( ( v24_sig & RFCOMM_V24_RTR ) ? ( TIOCM_RTS | TIOCM_CTS ) : 0 ) |
( ( v24_sig & RFCOMM_V24_IC ) ? TIOCM_RI : 0 ) |
( ( v24_sig & RFCOMM_V24_DV ) ? TIOCM_CD : 0 ) ;
}
/* ---- TTY functions ---- */
static void rfcomm_tty_wakeup ( unsigned long arg )
{
struct rfcomm_dev * dev = ( void * ) arg ;
struct tty_struct * tty = dev - > tty ;
if ( ! tty )
return ;
BT_DBG ( " dev %p tty %p " , dev , tty ) ;
2008-07-17 00:53:12 +04:00
tty_wakeup ( tty ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-14 22:13:52 +04:00
static void rfcomm_tty_copy_pending ( struct rfcomm_dev * dev )
{
struct tty_struct * tty = dev - > tty ;
struct sk_buff * skb ;
int inserted = 0 ;
if ( ! tty )
return ;
BT_DBG ( " dev %p tty %p " , dev , tty ) ;
rfcomm_dlc_lock ( dev - > dlc ) ;
while ( ( skb = skb_dequeue ( & dev - > pending ) ) ) {
inserted + = tty_insert_flip_string ( tty , skb - > data , skb - > len ) ;
kfree_skb ( skb ) ;
}
rfcomm_dlc_unlock ( dev - > dlc ) ;
if ( inserted > 0 )
tty_flip_buffer_push ( tty ) ;
}
2005-04-17 02:20:36 +04:00
static int rfcomm_tty_open ( struct tty_struct * tty , struct file * filp )
{
DECLARE_WAITQUEUE ( wait , current ) ;
struct rfcomm_dev * dev ;
struct rfcomm_dlc * dlc ;
int err , id ;
2007-02-09 17:24:33 +03:00
id = tty - > index ;
2005-04-17 02:20:36 +04:00
BT_DBG ( " tty %p id %d " , tty , id ) ;
/* We don't leak this refcount. For reasons which are not entirely
clear , the TTY layer will call our - > close ( ) method even if the
open fails . We decrease the refcount there , and decreasing it
here too would cause breakage . */
dev = rfcomm_dev_get ( id ) ;
if ( ! dev )
return - ENODEV ;
2008-11-30 14:17:29 +03:00
BT_DBG ( " dev %p dst %s channel %d opened %d " , dev , batostr ( & dev - > dst ) ,
dev - > channel , atomic_read ( & dev - > opened ) ) ;
2005-04-17 02:20:36 +04:00
2008-11-30 14:17:29 +03:00
if ( atomic_inc_return ( & dev - > opened ) > 1 )
2005-04-17 02:20:36 +04:00
return 0 ;
dlc = dev - > dlc ;
/* Attach TTY and open DLC */
rfcomm_dlc_lock ( dlc ) ;
tty - > driver_data = dev ;
dev - > tty = tty ;
rfcomm_dlc_unlock ( dlc ) ;
set_bit ( RFCOMM_TTY_ATTACHED , & dev - > flags ) ;
err = rfcomm_dlc_open ( dlc , & dev - > src , & dev - > dst , dev - > channel ) ;
if ( err < 0 )
return err ;
/* Wait for DLC to connect */
add_wait_queue ( & dev - > wait , & wait ) ;
while ( 1 ) {
set_current_state ( TASK_INTERRUPTIBLE ) ;
if ( dlc - > state = = BT_CLOSED ) {
err = - dev - > err ;
break ;
}
if ( dlc - > state = = BT_CONNECTED )
break ;
if ( signal_pending ( current ) ) {
err = - EINTR ;
break ;
}
schedule ( ) ;
}
set_current_state ( TASK_RUNNING ) ;
remove_wait_queue ( & dev - > wait , & wait ) ;
2007-02-18 01:58:57 +03:00
if ( err = = 0 )
2009-03-04 14:44:00 +03:00
device_move ( dev - > tty_dev , rfcomm_get_device ( dev ) ,
DPM_ORDER_DEV_AFTER_PARENT ) ;
2007-02-18 01:58:57 +03:00
2008-07-14 22:13:52 +04:00
rfcomm_tty_copy_pending ( dev ) ;
rfcomm_dlc_unthrottle ( dev - > dlc ) ;
2005-04-17 02:20:36 +04:00
return err ;
}
static void rfcomm_tty_close ( struct tty_struct * tty , struct file * filp )
{
struct rfcomm_dev * dev = ( struct rfcomm_dev * ) tty - > driver_data ;
if ( ! dev )
return ;
2008-11-30 14:17:29 +03:00
BT_DBG ( " tty %p dev %p dlc %p opened %d " , tty , dev , dev - > dlc ,
atomic_read ( & dev - > opened ) ) ;
2005-04-17 02:20:36 +04:00
2008-11-30 14:17:29 +03:00
if ( atomic_dec_and_test ( & dev - > opened ) ) {
2008-01-22 09:35:21 +03:00
if ( dev - > tty_dev - > parent )
2009-03-04 14:44:00 +03:00
device_move ( dev - > tty_dev , NULL , DPM_ORDER_DEV_LAST ) ;
2007-02-18 01:58:57 +03:00
2005-04-17 02:20:36 +04:00
/* Close DLC and dettach TTY */
rfcomm_dlc_close ( dev - > dlc , 0 ) ;
clear_bit ( RFCOMM_TTY_ATTACHED , & dev - > flags ) ;
tasklet_kill ( & dev - > wakeup_task ) ;
rfcomm_dlc_lock ( dev - > dlc ) ;
tty - > driver_data = NULL ;
dev - > tty = NULL ;
rfcomm_dlc_unlock ( dev - > dlc ) ;
2008-11-30 14:17:29 +03:00
if ( test_bit ( RFCOMM_TTY_RELEASED , & dev - > flags ) ) {
write_lock_bh ( & rfcomm_dev_lock ) ;
list_del_init ( & dev - > list ) ;
write_unlock_bh ( & rfcomm_dev_lock ) ;
rfcomm_dev_put ( dev ) ;
}
2005-04-17 02:20:36 +04:00
}
rfcomm_dev_put ( dev ) ;
}
static int rfcomm_tty_write ( struct tty_struct * tty , const unsigned char * buf , int count )
{
struct rfcomm_dev * dev = ( struct rfcomm_dev * ) tty - > driver_data ;
struct rfcomm_dlc * dlc = dev - > dlc ;
struct sk_buff * skb ;
int err = 0 , sent = 0 , size ;
BT_DBG ( " tty %p count %d " , tty , count ) ;
while ( count ) {
size = min_t ( uint , count , dlc - > mtu ) ;
skb = rfcomm_wmalloc ( dev , size + RFCOMM_SKB_RESERVE , GFP_ATOMIC ) ;
2007-02-09 17:24:33 +03:00
2005-04-17 02:20:36 +04:00
if ( ! skb )
break ;
skb_reserve ( skb , RFCOMM_SKB_HEAD_RESERVE ) ;
memcpy ( skb_put ( skb , size ) , buf + sent , size ) ;
if ( ( err = rfcomm_dlc_send ( dlc , skb ) ) < 0 ) {
kfree_skb ( skb ) ;
break ;
}
sent + = size ;
count - = size ;
}
return sent ? sent : err ;
}
static int rfcomm_tty_write_room ( struct tty_struct * tty )
{
struct rfcomm_dev * dev = ( struct rfcomm_dev * ) tty - > driver_data ;
int room ;
BT_DBG ( " tty %p " , tty ) ;
2007-01-08 04:16:27 +03:00
if ( ! dev | | ! dev - > dlc )
return 0 ;
2005-04-17 02:20:36 +04:00
room = rfcomm_room ( dev - > dlc ) - atomic_read ( & dev - > wmem_alloc ) ;
if ( room < 0 )
room = 0 ;
2007-01-08 04:16:27 +03:00
2005-04-17 02:20:36 +04:00
return room ;
}
static int rfcomm_tty_ioctl ( struct tty_struct * tty , struct file * filp , unsigned int cmd , unsigned long arg )
{
BT_DBG ( " tty %p cmd 0x%02x " , tty , cmd ) ;
switch ( cmd ) {
case TCGETS :
BT_DBG ( " TCGETS is not supported " ) ;
return - ENOIOCTLCMD ;
case TCSETS :
BT_DBG ( " TCSETS is not supported " ) ;
return - ENOIOCTLCMD ;
case TIOCMIWAIT :
BT_DBG ( " TIOCMIWAIT " ) ;
break ;
case TIOCGICOUNT :
BT_DBG ( " TIOCGICOUNT " ) ;
break ;
case TIOCGSERIAL :
BT_ERR ( " TIOCGSERIAL is not supported " ) ;
return - ENOIOCTLCMD ;
case TIOCSSERIAL :
BT_ERR ( " TIOCSSERIAL is not supported " ) ;
return - ENOIOCTLCMD ;
case TIOCSERGSTRUCT :
BT_ERR ( " TIOCSERGSTRUCT is not supported " ) ;
return - ENOIOCTLCMD ;
case TIOCSERGETLSR :
BT_ERR ( " TIOCSERGETLSR is not supported " ) ;
return - ENOIOCTLCMD ;
case TIOCSERCONFIG :
BT_ERR ( " TIOCSERCONFIG is not supported " ) ;
return - ENOIOCTLCMD ;
default :
return - ENOIOCTLCMD ; /* ioctls which we must ignore */
}
return - ENOIOCTLCMD ;
}
2006-12-08 13:38:45 +03:00
static void rfcomm_tty_set_termios ( struct tty_struct * tty , struct ktermios * old )
2005-04-17 02:20:36 +04:00
{
2006-12-08 13:38:45 +03:00
struct ktermios * new = tty - > termios ;
2005-08-10 07:28:46 +04:00
int old_baud_rate = tty_termios_baud_rate ( old ) ;
int new_baud_rate = tty_termios_baud_rate ( new ) ;
2005-04-17 02:20:36 +04:00
2005-08-10 07:28:46 +04:00
u8 baud , data_bits , stop_bits , parity , x_on , x_off ;
u16 changes = 0 ;
struct rfcomm_dev * dev = ( struct rfcomm_dev * ) tty - > driver_data ;
BT_DBG ( " tty %p termios %p " , tty , old ) ;
2006-11-19 00:14:42 +03:00
if ( ! dev | | ! dev - > dlc | | ! dev - > dlc - > session )
2006-10-15 19:31:10 +04:00
return ;
2005-08-10 07:28:46 +04:00
/* Handle turning off CRTSCTS */
2007-02-09 17:24:33 +03:00
if ( ( old - > c_cflag & CRTSCTS ) & & ! ( new - > c_cflag & CRTSCTS ) )
2005-08-10 07:28:46 +04:00
BT_DBG ( " Turning off CRTSCTS unsupported " ) ;
/* Parity on/off and when on, odd/even */
if ( ( ( old - > c_cflag & PARENB ) ! = ( new - > c_cflag & PARENB ) ) | |
( ( old - > c_cflag & PARODD ) ! = ( new - > c_cflag & PARODD ) ) ) {
changes | = RFCOMM_RPN_PM_PARITY ;
BT_DBG ( " Parity change detected. " ) ;
}
/* Mark and space parity are not supported! */
if ( new - > c_cflag & PARENB ) {
if ( new - > c_cflag & PARODD ) {
BT_DBG ( " Parity is ODD " ) ;
parity = RFCOMM_RPN_PARITY_ODD ;
} else {
BT_DBG ( " Parity is EVEN " ) ;
parity = RFCOMM_RPN_PARITY_EVEN ;
}
} else {
BT_DBG ( " Parity is OFF " ) ;
parity = RFCOMM_RPN_PARITY_NONE ;
}
/* Setting the x_on / x_off characters */
if ( old - > c_cc [ VSTOP ] ! = new - > c_cc [ VSTOP ] ) {
BT_DBG ( " XOFF custom " ) ;
x_on = new - > c_cc [ VSTOP ] ;
changes | = RFCOMM_RPN_PM_XON ;
} else {
BT_DBG ( " XOFF default " ) ;
x_on = RFCOMM_RPN_XON_CHAR ;
}
if ( old - > c_cc [ VSTART ] ! = new - > c_cc [ VSTART ] ) {
BT_DBG ( " XON custom " ) ;
x_off = new - > c_cc [ VSTART ] ;
changes | = RFCOMM_RPN_PM_XOFF ;
} else {
BT_DBG ( " XON default " ) ;
x_off = RFCOMM_RPN_XOFF_CHAR ;
}
/* Handle setting of stop bits */
if ( ( old - > c_cflag & CSTOPB ) ! = ( new - > c_cflag & CSTOPB ) )
changes | = RFCOMM_RPN_PM_STOP ;
/* POSIX does not support 1.5 stop bits and RFCOMM does not
* support 2 stop bits . So a request for 2 stop bits gets
* translated to 1.5 stop bits */
if ( new - > c_cflag & CSTOPB ) {
stop_bits = RFCOMM_RPN_STOP_15 ;
} else {
stop_bits = RFCOMM_RPN_STOP_1 ;
}
2005-04-17 02:20:36 +04:00
2005-08-10 07:28:46 +04:00
/* Handle number of data bits [5-8] */
2007-02-09 17:24:33 +03:00
if ( ( old - > c_cflag & CSIZE ) ! = ( new - > c_cflag & CSIZE ) )
2005-08-10 07:28:46 +04:00
changes | = RFCOMM_RPN_PM_DATA ;
switch ( new - > c_cflag & CSIZE ) {
case CS5 :
data_bits = RFCOMM_RPN_DATA_5 ;
break ;
case CS6 :
data_bits = RFCOMM_RPN_DATA_6 ;
break ;
case CS7 :
data_bits = RFCOMM_RPN_DATA_7 ;
break ;
case CS8 :
data_bits = RFCOMM_RPN_DATA_8 ;
break ;
default :
data_bits = RFCOMM_RPN_DATA_8 ;
break ;
2005-04-17 02:20:36 +04:00
}
2005-08-10 07:28:46 +04:00
/* Handle baudrate settings */
if ( old_baud_rate ! = new_baud_rate )
changes | = RFCOMM_RPN_PM_BITRATE ;
switch ( new_baud_rate ) {
case 2400 :
baud = RFCOMM_RPN_BR_2400 ;
break ;
case 4800 :
baud = RFCOMM_RPN_BR_4800 ;
break ;
case 7200 :
baud = RFCOMM_RPN_BR_7200 ;
break ;
case 9600 :
baud = RFCOMM_RPN_BR_9600 ;
break ;
2007-02-09 17:24:33 +03:00
case 19200 :
2005-08-10 07:28:46 +04:00
baud = RFCOMM_RPN_BR_19200 ;
break ;
case 38400 :
baud = RFCOMM_RPN_BR_38400 ;
break ;
case 57600 :
baud = RFCOMM_RPN_BR_57600 ;
break ;
case 115200 :
baud = RFCOMM_RPN_BR_115200 ;
break ;
case 230400 :
baud = RFCOMM_RPN_BR_230400 ;
break ;
default :
/* 9600 is standard accordinag to the RFCOMM specification */
baud = RFCOMM_RPN_BR_9600 ;
break ;
2007-02-09 17:24:33 +03:00
2005-08-10 07:28:46 +04:00
}
if ( changes )
rfcomm_send_rpn ( dev - > dlc - > session , 1 , dev - > dlc - > dlci , baud ,
data_bits , stop_bits , parity ,
RFCOMM_RPN_FLOW_NONE , x_on , x_off , changes ) ;
2005-04-17 02:20:36 +04:00
}
static void rfcomm_tty_throttle ( struct tty_struct * tty )
{
struct rfcomm_dev * dev = ( struct rfcomm_dev * ) tty - > driver_data ;
BT_DBG ( " tty %p dev %p " , tty , dev ) ;
2005-08-10 07:28:46 +04:00
2005-04-17 02:20:36 +04:00
rfcomm_dlc_throttle ( dev - > dlc ) ;
}
static void rfcomm_tty_unthrottle ( struct tty_struct * tty )
{
struct rfcomm_dev * dev = ( struct rfcomm_dev * ) tty - > driver_data ;
BT_DBG ( " tty %p dev %p " , tty , dev ) ;
2005-08-10 07:28:46 +04:00
2005-04-17 02:20:36 +04:00
rfcomm_dlc_unthrottle ( dev - > dlc ) ;
}
static int rfcomm_tty_chars_in_buffer ( struct tty_struct * tty )
{
struct rfcomm_dev * dev = ( struct rfcomm_dev * ) tty - > driver_data ;
BT_DBG ( " tty %p dev %p " , tty , dev ) ;
2007-01-08 04:16:27 +03:00
if ( ! dev | | ! dev - > dlc )
return 0 ;
if ( ! skb_queue_empty ( & dev - > dlc - > tx_queue ) )
return dev - > dlc - > mtu ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
static void rfcomm_tty_flush_buffer ( struct tty_struct * tty )
{
struct rfcomm_dev * dev = ( struct rfcomm_dev * ) tty - > driver_data ;
BT_DBG ( " tty %p dev %p " , tty , dev ) ;
2007-01-08 04:16:27 +03:00
if ( ! dev | | ! dev - > dlc )
return ;
2005-04-17 02:20:36 +04:00
skb_queue_purge ( & dev - > dlc - > tx_queue ) ;
2008-07-17 00:53:12 +04:00
tty_wakeup ( tty ) ;
2005-04-17 02:20:36 +04:00
}
static void rfcomm_tty_send_xchar ( struct tty_struct * tty , char ch )
{
BT_DBG ( " tty %p ch %c " , tty , ch ) ;
}
static void rfcomm_tty_wait_until_sent ( struct tty_struct * tty , int timeout )
{
BT_DBG ( " tty %p timeout %d " , tty , timeout ) ;
}
static void rfcomm_tty_hangup ( struct tty_struct * tty )
{
struct rfcomm_dev * dev = ( struct rfcomm_dev * ) tty - > driver_data ;
BT_DBG ( " tty %p dev %p " , tty , dev ) ;
2007-01-08 04:16:27 +03:00
if ( ! dev )
return ;
2005-04-17 02:20:36 +04:00
rfcomm_tty_flush_buffer ( tty ) ;
2007-05-05 02:36:10 +04:00
if ( test_bit ( RFCOMM_RELEASE_ONHUP , & dev - > flags ) ) {
if ( rfcomm_dev_get ( dev - > id ) = = NULL )
return ;
2005-04-17 02:20:36 +04:00
rfcomm_dev_del ( dev ) ;
2007-05-05 02:36:10 +04:00
rfcomm_dev_put ( dev ) ;
}
2005-04-17 02:20:36 +04:00
}
static int rfcomm_tty_tiocmget ( struct tty_struct * tty , struct file * filp )
{
2007-02-09 17:24:33 +03:00
struct rfcomm_dev * dev = ( struct rfcomm_dev * ) tty - > driver_data ;
2005-04-17 02:20:36 +04:00
BT_DBG ( " tty %p dev %p " , tty , dev ) ;
2007-02-09 17:24:33 +03:00
return dev - > modem_status ;
2005-04-17 02:20:36 +04:00
}
static int rfcomm_tty_tiocmset ( struct tty_struct * tty , struct file * filp , unsigned int set , unsigned int clear )
{
2005-08-10 07:28:46 +04:00
struct rfcomm_dev * dev = ( struct rfcomm_dev * ) tty - > driver_data ;
struct rfcomm_dlc * dlc = dev - > dlc ;
u8 v24_sig ;
2005-04-17 02:20:36 +04:00
BT_DBG ( " tty %p dev %p set 0x%02x clear 0x%02x " , tty , dev , set , clear ) ;
2005-08-10 07:28:46 +04:00
rfcomm_dlc_get_modem_status ( dlc , & v24_sig ) ;
if ( set & TIOCM_DSR | | set & TIOCM_DTR )
v24_sig | = RFCOMM_V24_RTC ;
if ( set & TIOCM_RTS | | set & TIOCM_CTS )
v24_sig | = RFCOMM_V24_RTR ;
if ( set & TIOCM_RI )
v24_sig | = RFCOMM_V24_IC ;
if ( set & TIOCM_CD )
v24_sig | = RFCOMM_V24_DV ;
if ( clear & TIOCM_DSR | | clear & TIOCM_DTR )
v24_sig & = ~ RFCOMM_V24_RTC ;
if ( clear & TIOCM_RTS | | clear & TIOCM_CTS )
v24_sig & = ~ RFCOMM_V24_RTR ;
if ( clear & TIOCM_RI )
v24_sig & = ~ RFCOMM_V24_IC ;
if ( clear & TIOCM_CD )
v24_sig & = ~ RFCOMM_V24_DV ;
rfcomm_dlc_set_modem_status ( dlc , v24_sig ) ;
return 0 ;
2005-04-17 02:20:36 +04:00
}
/* ---- TTY structure ---- */
2006-10-02 13:17:18 +04:00
static const struct tty_operations rfcomm_ops = {
2005-04-17 02:20:36 +04:00
. open = rfcomm_tty_open ,
. close = rfcomm_tty_close ,
. write = rfcomm_tty_write ,
. write_room = rfcomm_tty_write_room ,
. chars_in_buffer = rfcomm_tty_chars_in_buffer ,
. flush_buffer = rfcomm_tty_flush_buffer ,
. ioctl = rfcomm_tty_ioctl ,
. throttle = rfcomm_tty_throttle ,
. unthrottle = rfcomm_tty_unthrottle ,
. set_termios = rfcomm_tty_set_termios ,
. send_xchar = rfcomm_tty_send_xchar ,
. hangup = rfcomm_tty_hangup ,
. wait_until_sent = rfcomm_tty_wait_until_sent ,
. tiocmget = rfcomm_tty_tiocmget ,
. tiocmset = rfcomm_tty_tiocmset ,
} ;
2010-07-24 09:04:45 +04:00
int __init rfcomm_init_ttys ( void )
2005-04-17 02:20:36 +04:00
{
rfcomm_tty_driver = alloc_tty_driver ( RFCOMM_TTY_PORTS ) ;
if ( ! rfcomm_tty_driver )
return - 1 ;
rfcomm_tty_driver - > owner = THIS_MODULE ;
rfcomm_tty_driver - > driver_name = " rfcomm " ;
rfcomm_tty_driver - > name = " rfcomm " ;
rfcomm_tty_driver - > major = RFCOMM_TTY_MAJOR ;
rfcomm_tty_driver - > minor_start = RFCOMM_TTY_MINOR ;
rfcomm_tty_driver - > type = TTY_DRIVER_TYPE_SERIAL ;
rfcomm_tty_driver - > subtype = SERIAL_TYPE_NORMAL ;
2005-06-21 08:15:16 +04:00
rfcomm_tty_driver - > flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_DYNAMIC_DEV ;
2005-04-17 02:20:36 +04:00
rfcomm_tty_driver - > init_termios = tty_std_termios ;
rfcomm_tty_driver - > init_termios . c_cflag = B9600 | CS8 | CREAD | HUPCL | CLOCAL ;
2008-07-14 22:13:52 +04:00
rfcomm_tty_driver - > init_termios . c_lflag & = ~ ICANON ;
2005-04-17 02:20:36 +04:00
tty_set_operations ( rfcomm_tty_driver , & rfcomm_ops ) ;
if ( tty_register_driver ( rfcomm_tty_driver ) ) {
BT_ERR ( " Can't register RFCOMM TTY driver " ) ;
put_tty_driver ( rfcomm_tty_driver ) ;
return - 1 ;
}
BT_INFO ( " RFCOMM TTY layer initialized " ) ;
return 0 ;
}
2010-07-24 09:04:45 +04:00
void __exit rfcomm_cleanup_ttys ( void )
2005-04-17 02:20:36 +04:00
{
tty_unregister_driver ( rfcomm_tty_driver ) ;
put_tty_driver ( rfcomm_tty_driver ) ;
}