2008-10-05 22:16:16 +04:00
/*
* File : pep - gprs . c
*
* GPRS over Phonet pipe end point socket
*
* Copyright ( C ) 2008 Nokia Corporation .
*
* Author : Rémi Denis - Courmont < remi . denis - courmont @ nokia . com >
*
* 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 .
*
* 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 . , 51 Franklin St , Fifth Floor , Boston , MA
* 02110 - 1301 USA
*/
# include <linux/kernel.h>
# include <linux/netdevice.h>
# include <linux/if_ether.h>
# include <linux/if_arp.h>
# include <net/sock.h>
# include <linux/if_phonet.h>
# include <net/tcp_states.h>
# include <net/phonet/gprs.h>
# define GPRS_DEFAULT_MTU 1400
struct gprs_dev {
struct sock * sk ;
void ( * old_state_change ) ( struct sock * ) ;
void ( * old_data_ready ) ( struct sock * , int ) ;
void ( * old_write_space ) ( struct sock * ) ;
2008-12-16 12:18:31 +03:00
struct net_device * dev ;
2008-10-05 22:16:16 +04:00
struct sk_buff_head tx_queue ;
struct work_struct tx_work ;
spinlock_t tx_lock ;
unsigned tx_max ;
} ;
2008-11-07 10:10:50 +03:00
static __be16 gprs_type_trans ( struct sk_buff * skb )
2008-10-05 22:16:16 +04:00
{
const u8 * pvfc ;
u8 buf ;
pvfc = skb_header_pointer ( skb , 0 , 1 , & buf ) ;
if ( ! pvfc )
2008-11-07 10:10:50 +03:00
return htons ( 0 ) ;
2008-10-05 22:16:16 +04:00
/* Look at IP version field */
switch ( * pvfc > > 4 ) {
case 4 :
return htons ( ETH_P_IP ) ;
case 6 :
return htons ( ETH_P_IPV6 ) ;
}
2008-11-07 10:10:50 +03:00
return htons ( 0 ) ;
2008-10-05 22:16:16 +04:00
}
/*
* Socket callbacks
*/
static void gprs_state_change ( struct sock * sk )
{
2008-12-16 12:18:31 +03:00
struct gprs_dev * gp = sk - > sk_user_data ;
2008-10-05 22:16:16 +04:00
if ( sk - > sk_state = = TCP_CLOSE_WAIT ) {
2008-12-16 12:18:31 +03:00
struct net_device * dev = gp - > dev ;
netif_stop_queue ( dev ) ;
netif_carrier_off ( dev ) ;
2008-10-05 22:16:16 +04:00
}
}
2008-12-16 12:18:31 +03:00
static int gprs_recv ( struct gprs_dev * gp , struct sk_buff * skb )
2008-10-05 22:16:16 +04:00
{
2008-12-16 12:18:31 +03:00
struct net_device * dev = gp - > dev ;
2008-10-05 22:16:16 +04:00
int err = 0 ;
2008-11-07 10:10:50 +03:00
__be16 protocol = gprs_type_trans ( skb ) ;
2008-10-05 22:16:16 +04:00
if ( ! protocol ) {
err = - EINVAL ;
goto drop ;
}
if ( likely ( skb_headroom ( skb ) & 3 ) ) {
struct sk_buff * rskb , * fs ;
int flen = 0 ;
/* Phonet Pipe data header is misaligned (3 bytes),
* so wrap the IP packet as a single fragment of an head - less
* socket buffer . The network stack will pull what it needs ,
* but at least , the whole IP payload is not memcpy ' d . */
2008-12-16 12:18:31 +03:00
rskb = netdev_alloc_skb ( dev , 0 ) ;
2008-10-05 22:16:16 +04:00
if ( ! rskb ) {
err = - ENOBUFS ;
goto drop ;
}
skb_shinfo ( rskb ) - > frag_list = skb ;
rskb - > len + = skb - > len ;
rskb - > data_len + = rskb - > len ;
rskb - > truesize + = rskb - > len ;
/* Avoid nested fragments */
for ( fs = skb_shinfo ( skb ) - > frag_list ; fs ; fs = fs - > next )
flen + = fs - > len ;
skb - > next = skb_shinfo ( skb ) - > frag_list ;
skb_shinfo ( skb ) - > frag_list = NULL ;
skb - > len - = flen ;
skb - > data_len - = flen ;
skb - > truesize - = flen ;
skb = rskb ;
}
skb - > protocol = protocol ;
skb_reset_mac_header ( skb ) ;
2008-12-16 12:18:31 +03:00
skb - > dev = dev ;
2008-10-05 22:16:16 +04:00
2008-12-16 12:18:31 +03:00
if ( likely ( dev - > flags & IFF_UP ) ) {
dev - > stats . rx_packets + + ;
dev - > stats . rx_bytes + = skb - > len ;
2008-10-05 22:16:16 +04:00
netif_rx ( skb ) ;
skb = NULL ;
} else
err = - ENODEV ;
drop :
if ( skb ) {
dev_kfree_skb ( skb ) ;
2008-12-16 12:18:31 +03:00
dev - > stats . rx_dropped + + ;
2008-10-05 22:16:16 +04:00
}
return err ;
}
static void gprs_data_ready ( struct sock * sk , int len )
{
2008-12-16 12:18:31 +03:00
struct gprs_dev * gp = sk - > sk_user_data ;
2008-10-05 22:16:16 +04:00
struct sk_buff * skb ;
while ( ( skb = pep_read ( sk ) ) ! = NULL ) {
skb_orphan ( skb ) ;
2008-12-16 12:18:31 +03:00
gprs_recv ( gp , skb ) ;
2008-10-05 22:16:16 +04:00
}
}
static void gprs_write_space ( struct sock * sk )
{
2008-12-16 12:18:31 +03:00
struct gprs_dev * gp = sk - > sk_user_data ;
struct net_device * dev = gp - > dev ;
2008-10-05 22:16:16 +04:00
unsigned credits = pep_writeable ( sk ) ;
2008-12-16 12:18:31 +03:00
spin_lock_bh ( & gp - > tx_lock ) ;
gp - > tx_max = credits ;
if ( credits > skb_queue_len ( & gp - > tx_queue ) & & netif_running ( dev ) )
netif_wake_queue ( dev ) ;
spin_unlock_bh ( & gp - > tx_lock ) ;
2008-10-05 22:16:16 +04:00
}
/*
* Network device callbacks
*/
2008-12-15 11:53:57 +03:00
static int gprs_open ( struct net_device * dev )
{
struct gprs_dev * gp = netdev_priv ( dev ) ;
gprs_write_space ( gp - > sk ) ;
return 0 ;
}
static int gprs_close ( struct net_device * dev )
{
struct gprs_dev * gp = netdev_priv ( dev ) ;
netif_stop_queue ( dev ) ;
flush_work ( & gp - > tx_work ) ;
return 0 ;
}
2008-12-16 12:18:31 +03:00
static int gprs_xmit ( struct sk_buff * skb , struct net_device * dev )
2008-10-05 22:16:16 +04:00
{
2008-12-16 12:18:31 +03:00
struct gprs_dev * gp = netdev_priv ( dev ) ;
2008-10-05 22:16:16 +04:00
switch ( skb - > protocol ) {
case htons ( ETH_P_IP ) :
case htons ( ETH_P_IPV6 ) :
break ;
default :
dev_kfree_skb ( skb ) ;
return 0 ;
}
2008-12-16 12:18:31 +03:00
spin_lock ( & gp - > tx_lock ) ;
if ( likely ( skb_queue_len ( & gp - > tx_queue ) < gp - > tx_max ) ) {
skb_queue_tail ( & gp - > tx_queue , skb ) ;
2008-10-05 22:16:16 +04:00
skb = NULL ;
}
2008-12-16 12:18:31 +03:00
if ( skb_queue_len ( & gp - > tx_queue ) > = gp - > tx_max )
netif_stop_queue ( dev ) ;
spin_unlock ( & gp - > tx_lock ) ;
2008-10-05 22:16:16 +04:00
2008-12-16 12:18:31 +03:00
schedule_work ( & gp - > tx_work ) ;
2008-10-05 22:16:16 +04:00
if ( unlikely ( skb ) )
dev_kfree_skb ( skb ) ;
return 0 ;
}
static void gprs_tx ( struct work_struct * work )
{
2008-12-16 12:18:31 +03:00
struct gprs_dev * gp = container_of ( work , struct gprs_dev , tx_work ) ;
struct net_device * dev = gp - > dev ;
struct sock * sk = gp - > sk ;
2008-10-05 22:16:16 +04:00
struct sk_buff * skb ;
2008-12-16 12:18:31 +03:00
while ( ( skb = skb_dequeue ( & gp - > tx_queue ) ) ! = NULL ) {
2008-10-05 22:16:16 +04:00
int err ;
2008-12-16 12:18:31 +03:00
dev - > stats . tx_bytes + = skb - > len ;
dev - > stats . tx_packets + + ;
2008-10-05 22:16:16 +04:00
skb_orphan ( skb ) ;
skb_set_owner_w ( skb , sk ) ;
lock_sock ( sk ) ;
err = pep_write ( sk , skb ) ;
if ( err ) {
LIMIT_NETDEBUG ( KERN_WARNING " %s: TX error (%d) \n " ,
2008-12-16 12:18:31 +03:00
dev - > name , err ) ;
dev - > stats . tx_aborted_errors + + ;
dev - > stats . tx_errors + + ;
2008-10-05 22:16:16 +04:00
}
release_sock ( sk ) ;
}
lock_sock ( sk ) ;
gprs_write_space ( sk ) ;
release_sock ( sk ) ;
}
2008-12-16 12:18:31 +03:00
static int gprs_set_mtu ( struct net_device * dev , int new_mtu )
2008-10-05 22:16:16 +04:00
{
if ( ( new_mtu < 576 ) | | ( new_mtu > ( PHONET_MAX_MTU - 11 ) ) )
return - EINVAL ;
2008-12-16 12:18:31 +03:00
dev - > mtu = new_mtu ;
2008-10-05 22:16:16 +04:00
return 0 ;
}
2008-12-16 12:18:31 +03:00
static void gprs_setup ( struct net_device * dev )
2008-10-05 22:16:16 +04:00
{
2008-12-16 12:18:31 +03:00
dev - > features = NETIF_F_FRAGLIST ;
dev - > type = ARPHRD_NONE ;
dev - > flags = IFF_POINTOPOINT | IFF_NOARP ;
dev - > mtu = GPRS_DEFAULT_MTU ;
dev - > hard_header_len = 0 ;
dev - > addr_len = 0 ;
dev - > tx_queue_len = 10 ;
dev - > destructor = free_netdev ;
dev - > open = gprs_open ;
dev - > stop = gprs_close ;
dev - > hard_start_xmit = gprs_xmit ; /* mandatory */
dev - > change_mtu = gprs_set_mtu ;
2008-10-05 22:16:16 +04:00
}
/*
* External interface
*/
/*
* Attach a GPRS interface to a datagram socket .
* Returns the interface index on success , negative error code on error .
*/
int gprs_attach ( struct sock * sk )
{
static const char ifname [ ] = " gprs%d " ;
2008-12-16 12:18:31 +03:00
struct gprs_dev * gp ;
struct net_device * dev ;
2008-10-05 22:16:16 +04:00
int err ;
if ( unlikely ( sk - > sk_type = = SOCK_STREAM ) )
return - EINVAL ; /* need packet boundaries */
/* Create net device */
2008-12-16 12:18:31 +03:00
dev = alloc_netdev ( sizeof ( * gp ) , ifname , gprs_setup ) ;
if ( ! dev )
2008-10-05 22:16:16 +04:00
return - ENOMEM ;
2008-12-16 12:18:31 +03:00
gp = netdev_priv ( dev ) ;
gp - > dev = dev ;
gp - > tx_max = 0 ;
spin_lock_init ( & gp - > tx_lock ) ;
skb_queue_head_init ( & gp - > tx_queue ) ;
INIT_WORK ( & gp - > tx_work , gprs_tx ) ;
netif_stop_queue ( dev ) ;
err = register_netdev ( dev ) ;
2008-10-05 22:16:16 +04:00
if ( err ) {
2008-12-16 12:18:31 +03:00
free_netdev ( dev ) ;
2008-10-05 22:16:16 +04:00
return err ;
}
lock_sock ( sk ) ;
if ( unlikely ( sk - > sk_user_data ) ) {
err = - EBUSY ;
goto out_rel ;
}
if ( unlikely ( ( 1 < < sk - > sk_state & ( TCPF_CLOSE | TCPF_LISTEN ) ) | |
sock_flag ( sk , SOCK_DEAD ) ) ) {
err = - EINVAL ;
goto out_rel ;
}
2008-12-16 12:18:31 +03:00
sk - > sk_user_data = gp ;
gp - > old_state_change = sk - > sk_state_change ;
gp - > old_data_ready = sk - > sk_data_ready ;
gp - > old_write_space = sk - > sk_write_space ;
2008-10-05 22:16:16 +04:00
sk - > sk_state_change = gprs_state_change ;
sk - > sk_data_ready = gprs_data_ready ;
sk - > sk_write_space = gprs_write_space ;
release_sock ( sk ) ;
sock_hold ( sk ) ;
2008-12-16 12:18:31 +03:00
gp - > sk = sk ;
2008-10-05 22:16:16 +04:00
2008-12-16 12:18:31 +03:00
printk ( KERN_DEBUG " %s: attached \n " , dev - > name ) ;
return dev - > ifindex ;
2008-10-05 22:16:16 +04:00
out_rel :
release_sock ( sk ) ;
2008-12-16 12:18:31 +03:00
unregister_netdev ( dev ) ;
2008-10-05 22:16:16 +04:00
return err ;
}
void gprs_detach ( struct sock * sk )
{
2008-12-16 12:18:31 +03:00
struct gprs_dev * gp = sk - > sk_user_data ;
struct net_device * dev = gp - > dev ;
2008-10-05 22:16:16 +04:00
lock_sock ( sk ) ;
sk - > sk_user_data = NULL ;
2008-12-16 12:18:31 +03:00
sk - > sk_state_change = gp - > old_state_change ;
sk - > sk_data_ready = gp - > old_data_ready ;
sk - > sk_write_space = gp - > old_write_space ;
2008-10-05 22:16:16 +04:00
release_sock ( sk ) ;
2008-12-16 12:18:31 +03:00
printk ( KERN_DEBUG " %s: detached \n " , dev - > name ) ;
unregister_netdev ( dev ) ;
2008-10-05 22:16:16 +04:00
sock_put ( sk ) ;
}