2005-04-17 02:20:36 +04:00
/** -*- linux-c -*- ***********************************************************
* Linux PPP over Ethernet ( PPPoX / PPPoE ) Sockets
*
* PPPoX - - - Generic PPP encapsulation socket family
* PPPoE - - - PPP over Ethernet ( RFC 2516 )
*
*
* Version : 0.7 .0
*
2007-03-03 00:16:56 +03:00
* 07022 8 : Fix to allow multiple sessions with same remote MAC and same
* session id by including the local device ifindex in the
* tuple identifying a session . This also ensures packets can ' t
* be injected into a session from interfaces other than the one
* specified by userspace . Florian Zumbiehl < florz @ florz . de >
* ( Oh , BTW , this one is YYMMDD , in case you were wondering . . . )
2005-04-17 02:20:36 +04:00
* 220102 : Fix module use count on failure in pppoe_create , pppox_sk - acme
* 030700 : Fixed connect logic to allow for disconnect .
* 270700 : Fixed potential SMP problems ; we must protect against
* simultaneous invocation of ppp_input
* and ppp_unregister_channel .
* 040 800 : Respect reference count mechanisms on net - devices .
* 200800 : fix kfree ( skb ) in pppoe_rcv ( acme )
* Module reference count is decremented in the right spot now ,
* guards against sock_put not actually freeing the sk
* in pppoe_release .
* 051000 : Initialization cleanup .
* 111100 : Fix recvmsg .
* 050101 : Fix PADT procesing .
* 140501 : Use pppoe_rcv_core to handle all backlog . ( Alexey )
* 170701 : Do not lock_sock with rwlock held . ( DaveM )
* Ignore discovery frames if user has socket
* locked . ( DaveM )
* Ignore return value of dev_queue_xmit in __pppoe_xmit
* or else we may kfree an SKB twice . ( DaveM )
* 190701 : When doing copies of skb ' s in __pppoe_xmit , always delete
* the original skb that was passed in on success , never on
* failure . Delete the copy of the skb on failure to avoid
* a memory leak .
* 081001 : Misc . cleanup ( licence string , non - blocking , prevent
* reference of device on close ) .
* 121301 : New ppp channels interface ; cannot unregister a channel
* from interrupts . Thus , we mark the socket as a ZOMBIE
* and do the unregistration later .
* 081002 : seq_file support for proc stuff - acme
* 111602 : Merge all 2.4 fixes into 2.5 / 2.6 tree . Label 2.5 / 2.6
* as version 0.7 . Spacing cleanup .
* Author : Michal Ostrowski < mostrows @ speakeasy . net >
* Contributors :
* Arnaldo Carvalho de Melo < acme @ conectiva . com . br >
* David S . Miller ( davem @ redhat . com )
*
* License :
* 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 .
*
*/
# include <linux/string.h>
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/slab.h>
# include <linux/errno.h>
# include <linux/netdevice.h>
# include <linux/net.h>
# include <linux/inetdevice.h>
# include <linux/etherdevice.h>
# include <linux/skbuff.h>
# include <linux/init.h>
# include <linux/if_ether.h>
# include <linux/if_pppox.h>
# include <linux/ppp_channel.h>
# include <linux/ppp_defs.h>
# include <linux/if_ppp.h>
# include <linux/notifier.h>
# include <linux/file.h>
# include <linux/proc_fs.h>
# include <linux/seq_file.h>
# include <net/sock.h>
# include <asm/uaccess.h>
# define PPPOE_HASH_BITS 4
# define PPPOE_HASH_SIZE (1<<PPPOE_HASH_BITS)
static struct ppp_channel_ops pppoe_chan_ops ;
static int pppoe_ioctl ( struct socket * sock , unsigned int cmd , unsigned long arg ) ;
static int pppoe_xmit ( struct ppp_channel * chan , struct sk_buff * skb ) ;
static int __pppoe_xmit ( struct sock * sk , struct sk_buff * skb ) ;
2005-12-28 07:57:40 +03:00
static const struct proto_ops pppoe_ops ;
2005-04-17 02:20:36 +04:00
static DEFINE_RWLOCK ( pppoe_hash_lock ) ;
static struct ppp_channel_ops pppoe_chan_ops ;
static inline int cmp_2_addr ( struct pppoe_addr * a , struct pppoe_addr * b )
{
return ( a - > sid = = b - > sid & &
( memcmp ( a - > remote , b - > remote , ETH_ALEN ) = = 0 ) ) ;
}
static inline int cmp_addr ( struct pppoe_addr * a , unsigned long sid , char * addr )
{
return ( a - > sid = = sid & &
( memcmp ( a - > remote , addr , ETH_ALEN ) = = 0 ) ) ;
}
static int hash_item ( unsigned long sid , unsigned char * addr )
{
char hash = 0 ;
int i , j ;
for ( i = 0 ; i < ETH_ALEN ; + + i ) {
for ( j = 0 ; j < 8 / PPPOE_HASH_BITS ; + + j ) {
hash ^ = addr [ i ] > > ( j * PPPOE_HASH_BITS ) ;
}
}
for ( i = 0 ; i < ( sizeof ( unsigned long ) * 8 ) / PPPOE_HASH_BITS ; + + i )
hash ^ = sid > > ( i * PPPOE_HASH_BITS ) ;
return hash & ( PPPOE_HASH_SIZE - 1 ) ;
}
/* zeroed because its in .bss */
static struct pppox_sock * item_hash_table [ PPPOE_HASH_SIZE ] ;
/**********************************************************************
*
* Set / get / delete / rehash items ( internal versions )
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-03-03 00:16:56 +03:00
static struct pppox_sock * __get_item ( unsigned long sid , unsigned char * addr , int ifindex )
2005-04-17 02:20:36 +04:00
{
int hash = hash_item ( sid , addr ) ;
struct pppox_sock * ret ;
ret = item_hash_table [ hash ] ;
2007-03-05 03:03:22 +03:00
while ( ret & & ! ( cmp_addr ( & ret - > pppoe_pa , sid , addr ) & & ret - > pppoe_ifindex = = ifindex ) )
2005-04-17 02:20:36 +04:00
ret = ret - > next ;
return ret ;
}
static int __set_item ( struct pppox_sock * po )
{
int hash = hash_item ( po - > pppoe_pa . sid , po - > pppoe_pa . remote ) ;
struct pppox_sock * ret ;
ret = item_hash_table [ hash ] ;
while ( ret ) {
2007-03-05 03:03:22 +03:00
if ( cmp_2_addr ( & ret - > pppoe_pa , & po - > pppoe_pa ) & & ret - > pppoe_ifindex = = po - > pppoe_ifindex )
2005-04-17 02:20:36 +04:00
return - EALREADY ;
ret = ret - > next ;
}
2007-03-03 00:16:56 +03:00
po - > next = item_hash_table [ hash ] ;
item_hash_table [ hash ] = po ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2007-03-03 00:16:56 +03:00
static struct pppox_sock * __delete_item ( unsigned long sid , char * addr , int ifindex )
2005-04-17 02:20:36 +04:00
{
int hash = hash_item ( sid , addr ) ;
struct pppox_sock * ret , * * src ;
ret = item_hash_table [ hash ] ;
src = & item_hash_table [ hash ] ;
while ( ret ) {
2007-03-05 03:03:22 +03:00
if ( cmp_addr ( & ret - > pppoe_pa , sid , addr ) & & ret - > pppoe_ifindex = = ifindex ) {
2005-04-17 02:20:36 +04:00
* src = ret - > next ;
break ;
}
src = & ret - > next ;
ret = ret - > next ;
}
return ret ;
}
/**********************************************************************
*
* Set / get / delete / rehash items
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static inline struct pppox_sock * get_item ( unsigned long sid ,
2007-03-03 00:16:56 +03:00
unsigned char * addr , int ifindex )
2005-04-17 02:20:36 +04:00
{
struct pppox_sock * po ;
read_lock_bh ( & pppoe_hash_lock ) ;
2007-03-03 00:16:56 +03:00
po = __get_item ( sid , addr , ifindex ) ;
2005-04-17 02:20:36 +04:00
if ( po )
sock_hold ( sk_pppox ( po ) ) ;
read_unlock_bh ( & pppoe_hash_lock ) ;
return po ;
}
static inline struct pppox_sock * get_item_by_addr ( struct sockaddr_pppox * sp )
{
2007-04-21 03:56:31 +04:00
struct net_device * dev ;
2007-03-03 00:16:56 +03:00
int ifindex ;
dev = dev_get_by_name ( sp - > sa_addr . pppoe . dev ) ;
if ( ! dev )
return NULL ;
ifindex = dev - > ifindex ;
dev_put ( dev ) ;
return get_item ( sp - > sa_addr . pppoe . sid , sp - > sa_addr . pppoe . remote , ifindex ) ;
2005-04-17 02:20:36 +04:00
}
2007-03-03 00:16:56 +03:00
static inline struct pppox_sock * delete_item ( unsigned long sid , char * addr , int ifindex )
2005-04-17 02:20:36 +04:00
{
struct pppox_sock * ret ;
write_lock_bh ( & pppoe_hash_lock ) ;
2007-03-03 00:16:56 +03:00
ret = __delete_item ( sid , addr , ifindex ) ;
2005-04-17 02:20:36 +04:00
write_unlock_bh ( & pppoe_hash_lock ) ;
return ret ;
}
/***************************************************************************
*
* Handler for device events .
* Certain device events require that sockets be unconnected .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void pppoe_flush_dev ( struct net_device * dev )
{
int hash ;
BUG_ON ( dev = = NULL ) ;
2007-04-21 03:59:24 +04:00
write_lock_bh ( & pppoe_hash_lock ) ;
2005-04-17 02:20:36 +04:00
for ( hash = 0 ; hash < PPPOE_HASH_SIZE ; hash + + ) {
struct pppox_sock * po = item_hash_table [ hash ] ;
while ( po ! = NULL ) {
2007-04-21 03:59:24 +04:00
struct sock * sk = sk_pppox ( po ) ;
if ( po - > pppoe_dev ! = dev ) {
po = po - > next ;
continue ;
}
po - > pppoe_dev = NULL ;
dev_put ( dev ) ;
2005-04-17 02:20:36 +04:00
2007-04-21 03:59:24 +04:00
/* We always grab the socket lock, followed by the
* pppoe_hash_lock , in that order . Since we should
* hold the sock lock while doing any unbinding ,
* we need to release the lock we ' re holding .
* Hold a reference to the sock so it doesn ' t disappear
* as we ' re jumping between locks .
*/
2005-04-17 02:20:36 +04:00
2007-04-21 03:59:24 +04:00
sock_hold ( sk ) ;
2005-04-17 02:20:36 +04:00
2007-04-21 03:59:24 +04:00
write_unlock_bh ( & pppoe_hash_lock ) ;
lock_sock ( sk ) ;
2005-04-17 02:20:36 +04:00
2007-04-21 03:59:24 +04:00
if ( sk - > sk_state & ( PPPOX_CONNECTED | PPPOX_BOUND ) ) {
pppox_unbind_sock ( sk ) ;
sk - > sk_state = PPPOX_ZOMBIE ;
sk - > sk_state_change ( sk ) ;
}
2005-04-17 02:20:36 +04:00
2007-04-21 03:59:24 +04:00
release_sock ( sk ) ;
sock_put ( sk ) ;
2005-04-17 02:20:36 +04:00
2007-04-21 03:59:24 +04:00
/* Restart scan at the beginning of this hash chain.
* While the lock was dropped the chain contents may
* have changed .
*/
write_lock_bh ( & pppoe_hash_lock ) ;
po = item_hash_table [ hash ] ;
2005-04-17 02:20:36 +04:00
}
}
2007-04-21 03:59:24 +04:00
write_unlock_bh ( & pppoe_hash_lock ) ;
2005-04-17 02:20:36 +04:00
}
static int pppoe_device_event ( struct notifier_block * this ,
unsigned long event , void * ptr )
{
struct net_device * dev = ( struct net_device * ) ptr ;
/* Only look at sockets that are using this specific device. */
switch ( event ) {
case NETDEV_CHANGEMTU :
/* A change in mtu is a bad thing, requiring
* LCP re - negotiation .
*/
case NETDEV_GOING_DOWN :
case NETDEV_DOWN :
/* Find every socket on this device and kill it. */
pppoe_flush_dev ( dev ) ;
break ;
default :
break ;
} ;
return NOTIFY_DONE ;
}
static struct notifier_block pppoe_notifier = {
. notifier_call = pppoe_device_event ,
} ;
/************************************************************************
*
* Do the real work of receiving a PPPoE Session frame .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int pppoe_rcv_core ( struct sock * sk , struct sk_buff * skb )
{
struct pppox_sock * po = pppox_sk ( sk ) ;
2007-04-21 03:56:31 +04:00
struct pppox_sock * relay_po ;
2005-04-17 02:20:36 +04:00
if ( sk - > sk_state & PPPOX_BOUND ) {
2007-03-10 21:56:08 +03:00
struct pppoe_hdr * ph = pppoe_hdr ( skb ) ;
2005-04-17 02:20:36 +04:00
int len = ntohs ( ph - > length ) ;
2006-03-21 09:43:56 +03:00
skb_pull_rcsum ( skb , sizeof ( struct pppoe_hdr ) ) ;
2005-04-17 02:20:36 +04:00
if ( pskb_trim_rcsum ( skb , len ) )
goto abort_kfree ;
ppp_input ( & po - > chan , skb ) ;
} else if ( sk - > sk_state & PPPOX_RELAY ) {
relay_po = get_item_by_addr ( & po - > pppoe_relay ) ;
if ( relay_po = = NULL )
goto abort_kfree ;
if ( ( sk_pppox ( relay_po ) - > sk_state & PPPOX_CONNECTED ) = = 0 )
goto abort_put ;
skb_pull ( skb , sizeof ( struct pppoe_hdr ) ) ;
if ( ! __pppoe_xmit ( sk_pppox ( relay_po ) , skb ) )
goto abort_put ;
} else {
if ( sock_queue_rcv_skb ( sk , skb ) )
goto abort_kfree ;
}
return NET_RX_SUCCESS ;
abort_put :
sock_put ( sk_pppox ( relay_po ) ) ;
abort_kfree :
kfree_skb ( skb ) ;
return NET_RX_DROP ;
}
/************************************************************************
*
* Receive wrapper called in BH context .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int pppoe_rcv ( struct sk_buff * skb ,
struct net_device * dev ,
2005-08-10 06:34:12 +04:00
struct packet_type * pt ,
struct net_device * orig_dev )
2005-04-17 02:20:36 +04:00
{
struct pppoe_hdr * ph ;
struct pppox_sock * po ;
if ( ! pskb_may_pull ( skb , sizeof ( struct pppoe_hdr ) ) )
goto drop ;
2006-09-13 21:24:59 +04:00
if ( ! ( skb = skb_share_check ( skb , GFP_ATOMIC ) ) )
2005-04-17 02:20:36 +04:00
goto out ;
2007-03-10 21:56:08 +03:00
ph = pppoe_hdr ( skb ) ;
2005-04-17 02:20:36 +04:00
2007-03-03 00:16:56 +03:00
po = get_item ( ( unsigned long ) ph - > sid , eth_hdr ( skb ) - > h_source , dev - > ifindex ) ;
2006-09-13 21:24:59 +04:00
if ( po ! = NULL )
2006-11-16 19:06:06 +03:00
return sk_receive_skb ( sk_pppox ( po ) , skb , 0 ) ;
2005-04-17 02:20:36 +04:00
drop :
kfree_skb ( skb ) ;
out :
return NET_RX_DROP ;
}
/************************************************************************
*
* Receive a PPPoE Discovery frame .
* This is solely for detection of PADT frames
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int pppoe_disc_rcv ( struct sk_buff * skb ,
struct net_device * dev ,
2005-08-10 06:34:12 +04:00
struct packet_type * pt ,
struct net_device * orig_dev )
2005-04-17 02:20:36 +04:00
{
struct pppoe_hdr * ph ;
struct pppox_sock * po ;
if ( ! pskb_may_pull ( skb , sizeof ( struct pppoe_hdr ) ) )
goto abort ;
2006-09-13 21:24:59 +04:00
if ( ! ( skb = skb_share_check ( skb , GFP_ATOMIC ) ) )
2005-04-17 02:20:36 +04:00
goto out ;
2007-03-10 21:56:08 +03:00
ph = pppoe_hdr ( skb ) ;
2005-04-17 02:20:36 +04:00
if ( ph - > code ! = PADT_CODE )
goto abort ;
2007-03-03 00:16:56 +03:00
po = get_item ( ( unsigned long ) ph - > sid , eth_hdr ( skb ) - > h_source , dev - > ifindex ) ;
2005-04-17 02:20:36 +04:00
if ( po ) {
struct sock * sk = sk_pppox ( po ) ;
bh_lock_sock ( sk ) ;
/* If the user has locked the socket, just ignore
* the packet . With the way two rcv protocols hook into
* one socket family type , we cannot ( easily ) distinguish
* what kind of SKB it is during backlog rcv .
*/
if ( sock_owned_by_user ( sk ) = = 0 ) {
/* We're no longer connect at the PPPOE layer,
* and must wait for ppp channel to disconnect us .
*/
sk - > sk_state = PPPOX_ZOMBIE ;
}
bh_unlock_sock ( sk ) ;
sock_put ( sk ) ;
}
abort :
kfree_skb ( skb ) ;
out :
return NET_RX_SUCCESS ; /* Lies... :-) */
}
static struct packet_type pppoes_ptype = {
. type = __constant_htons ( ETH_P_PPP_SES ) ,
. func = pppoe_rcv ,
} ;
static struct packet_type pppoed_ptype = {
. type = __constant_htons ( ETH_P_PPP_DISC ) ,
. func = pppoe_disc_rcv ,
} ;
static struct proto pppoe_sk_proto = {
. name = " PPPOE " ,
. owner = THIS_MODULE ,
. obj_size = sizeof ( struct pppox_sock ) ,
} ;
/***********************************************************************
*
* Initialize a new struct sock .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int pppoe_create ( struct socket * sock )
{
int error = - ENOMEM ;
struct sock * sk ;
sk = sk_alloc ( PF_PPPOX , GFP_KERNEL , & pppoe_sk_proto , 1 ) ;
if ( ! sk )
goto out ;
sock_init_data ( sock , sk ) ;
sock - > state = SS_UNCONNECTED ;
sock - > ops = & pppoe_ops ;
sk - > sk_backlog_rcv = pppoe_rcv_core ;
sk - > sk_state = PPPOX_NONE ;
sk - > sk_type = SOCK_STREAM ;
sk - > sk_family = PF_PPPOX ;
sk - > sk_protocol = PX_PROTO_OE ;
error = 0 ;
out : return error ;
}
static int pppoe_release ( struct socket * sock )
{
struct sock * sk = sock - > sk ;
struct pppox_sock * po ;
if ( ! sk )
return 0 ;
2007-04-21 03:59:24 +04:00
lock_sock ( sk ) ;
if ( sock_flag ( sk , SOCK_DEAD ) ) {
release_sock ( sk ) ;
2005-04-17 02:20:36 +04:00
return - EBADF ;
2007-04-21 03:59:24 +04:00
}
2005-04-17 02:20:36 +04:00
pppox_unbind_sock ( sk ) ;
/* Signal the death of the socket. */
sk - > sk_state = PPPOX_DEAD ;
2007-04-21 03:59:24 +04:00
/* Write lock on hash lock protects the entire "po" struct from
* concurrent updates via pppoe_flush_dev . The " po " struct should
* be considered part of the hash table contents , thus protected
* by the hash table lock */
write_lock_bh ( & pppoe_hash_lock ) ;
2005-04-17 02:20:36 +04:00
po = pppox_sk ( sk ) ;
if ( po - > pppoe_pa . sid ) {
2007-04-21 03:59:24 +04:00
__delete_item ( po - > pppoe_pa . sid ,
po - > pppoe_pa . remote , po - > pppoe_ifindex ) ;
2005-04-17 02:20:36 +04:00
}
2007-04-21 03:59:24 +04:00
if ( po - > pppoe_dev ) {
2005-04-17 02:20:36 +04:00
dev_put ( po - > pppoe_dev ) ;
2007-04-21 03:59:24 +04:00
po - > pppoe_dev = NULL ;
}
2005-04-17 02:20:36 +04:00
2007-04-21 03:59:24 +04:00
write_unlock_bh ( & pppoe_hash_lock ) ;
2005-04-17 02:20:36 +04:00
sock_orphan ( sk ) ;
sock - > sk = NULL ;
skb_queue_purge ( & sk - > sk_receive_queue ) ;
2007-04-21 03:59:24 +04:00
release_sock ( sk ) ;
2005-04-17 02:20:36 +04:00
sock_put ( sk ) ;
2007-04-21 03:56:31 +04:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
static int pppoe_connect ( struct socket * sock , struct sockaddr * uservaddr ,
int sockaddr_len , int flags )
{
struct sock * sk = sock - > sk ;
2007-03-03 00:16:56 +03:00
struct net_device * dev ;
2005-04-17 02:20:36 +04:00
struct sockaddr_pppox * sp = ( struct sockaddr_pppox * ) uservaddr ;
struct pppox_sock * po = pppox_sk ( sk ) ;
int error ;
lock_sock ( sk ) ;
error = - EINVAL ;
if ( sp - > sa_protocol ! = PX_PROTO_OE )
goto end ;
/* Check for already bound sockets */
error = - EBUSY ;
if ( ( sk - > sk_state & PPPOX_CONNECTED ) & & sp - > sa_addr . pppoe . sid )
goto end ;
/* Check for already disconnected sockets, on attempts to disconnect */
error = - EALREADY ;
if ( ( sk - > sk_state & PPPOX_DEAD ) & & ! sp - > sa_addr . pppoe . sid )
goto end ;
error = 0 ;
if ( po - > pppoe_pa . sid ) {
pppox_unbind_sock ( sk ) ;
/* Delete the old binding */
2007-03-05 03:03:22 +03:00
delete_item ( po - > pppoe_pa . sid , po - > pppoe_pa . remote , po - > pppoe_ifindex ) ;
2005-04-17 02:20:36 +04:00
if ( po - > pppoe_dev )
dev_put ( po - > pppoe_dev ) ;
memset ( sk_pppox ( po ) + 1 , 0 ,
sizeof ( struct pppox_sock ) - sizeof ( struct sock ) ) ;
sk - > sk_state = PPPOX_NONE ;
}
/* Don't re-bind if sid==0 */
if ( sp - > sa_addr . pppoe . sid ! = 0 ) {
dev = dev_get_by_name ( sp - > sa_addr . pppoe . dev ) ;
error = - ENODEV ;
if ( ! dev )
goto end ;
po - > pppoe_dev = dev ;
2007-03-05 03:03:22 +03:00
po - > pppoe_ifindex = dev - > ifindex ;
2005-04-17 02:20:36 +04:00
2007-04-21 03:57:27 +04:00
write_lock_bh ( & pppoe_hash_lock ) ;
if ( ! ( dev - > flags & IFF_UP ) ) {
write_unlock_bh ( & pppoe_hash_lock ) ;
2005-04-17 02:20:36 +04:00
goto err_put ;
2007-04-21 03:57:27 +04:00
}
2005-04-17 02:20:36 +04:00
memcpy ( & po - > pppoe_pa ,
& sp - > sa_addr . pppoe ,
sizeof ( struct pppoe_addr ) ) ;
2007-04-21 03:57:27 +04:00
error = __set_item ( po ) ;
write_unlock_bh ( & pppoe_hash_lock ) ;
2005-04-17 02:20:36 +04:00
if ( error < 0 )
goto err_put ;
po - > chan . hdrlen = ( sizeof ( struct pppoe_hdr ) +
dev - > hard_header_len ) ;
2006-09-28 03:11:25 +04:00
po - > chan . mtu = dev - > mtu - sizeof ( struct pppoe_hdr ) ;
2005-04-17 02:20:36 +04:00
po - > chan . private = sk ;
po - > chan . ops = & pppoe_chan_ops ;
error = ppp_register_channel ( & po - > chan ) ;
if ( error )
goto err_put ;
sk - > sk_state = PPPOX_CONNECTED ;
}
po - > num = sp - > sa_addr . pppoe . sid ;
end :
release_sock ( sk ) ;
return error ;
err_put :
if ( po - > pppoe_dev ) {
dev_put ( po - > pppoe_dev ) ;
po - > pppoe_dev = NULL ;
}
goto end ;
}
static int pppoe_getname ( struct socket * sock , struct sockaddr * uaddr ,
int * usockaddr_len , int peer )
{
int len = sizeof ( struct sockaddr_pppox ) ;
struct sockaddr_pppox sp ;
sp . sa_family = AF_PPPOX ;
sp . sa_protocol = PX_PROTO_OE ;
memcpy ( & sp . sa_addr . pppoe , & pppox_sk ( sock - > sk ) - > pppoe_pa ,
sizeof ( struct pppoe_addr ) ) ;
memcpy ( uaddr , & sp , len ) ;
* usockaddr_len = len ;
return 0 ;
}
static int pppoe_ioctl ( struct socket * sock , unsigned int cmd ,
unsigned long arg )
{
struct sock * sk = sock - > sk ;
struct pppox_sock * po = pppox_sk ( sk ) ;
int val = 0 ;
int err = 0 ;
switch ( cmd ) {
case PPPIOCGMRU :
err = - ENXIO ;
if ( ! ( sk - > sk_state & PPPOX_CONNECTED ) )
break ;
err = - EFAULT ;
if ( put_user ( po - > pppoe_dev - > mtu -
sizeof ( struct pppoe_hdr ) -
PPP_HDRLEN ,
( int __user * ) arg ) )
break ;
err = 0 ;
break ;
case PPPIOCSMRU :
err = - ENXIO ;
if ( ! ( sk - > sk_state & PPPOX_CONNECTED ) )
break ;
err = - EFAULT ;
if ( get_user ( val , ( int __user * ) arg ) )
break ;
if ( val < ( po - > pppoe_dev - > mtu
- sizeof ( struct pppoe_hdr )
- PPP_HDRLEN ) )
err = 0 ;
else
err = - EINVAL ;
break ;
case PPPIOCSFLAGS :
err = - EFAULT ;
if ( get_user ( val , ( int __user * ) arg ) )
break ;
err = 0 ;
break ;
case PPPOEIOCSFWD :
{
struct pppox_sock * relay_po ;
err = - EBUSY ;
if ( sk - > sk_state & ( PPPOX_BOUND | PPPOX_ZOMBIE | PPPOX_DEAD ) )
break ;
err = - ENOTCONN ;
if ( ! ( sk - > sk_state & PPPOX_CONNECTED ) )
break ;
/* PPPoE address from the user specifies an outbound
2007-03-03 00:16:56 +03:00
PPPoE address which frames are forwarded to */
2005-04-17 02:20:36 +04:00
err = - EFAULT ;
if ( copy_from_user ( & po - > pppoe_relay ,
( void __user * ) arg ,
sizeof ( struct sockaddr_pppox ) ) )
break ;
err = - EINVAL ;
if ( po - > pppoe_relay . sa_family ! = AF_PPPOX | |
po - > pppoe_relay . sa_protocol ! = PX_PROTO_OE )
break ;
/* Check that the socket referenced by the address
actually exists . */
relay_po = get_item_by_addr ( & po - > pppoe_relay ) ;
if ( ! relay_po )
break ;
sock_put ( sk_pppox ( relay_po ) ) ;
sk - > sk_state | = PPPOX_RELAY ;
err = 0 ;
break ;
}
case PPPOEIOCDFWD :
err = - EALREADY ;
if ( ! ( sk - > sk_state & PPPOX_RELAY ) )
break ;
sk - > sk_state & = ~ PPPOX_RELAY ;
err = 0 ;
break ;
default : ;
} ;
return err ;
}
2006-09-13 21:24:59 +04:00
static int pppoe_sendmsg ( struct kiocb * iocb , struct socket * sock ,
2005-04-17 02:20:36 +04:00
struct msghdr * m , size_t total_len )
{
2007-04-21 03:56:31 +04:00
struct sk_buff * skb ;
2005-04-17 02:20:36 +04:00
struct sock * sk = sock - > sk ;
struct pppox_sock * po = pppox_sk ( sk ) ;
2007-04-21 03:56:31 +04:00
int error ;
2005-04-17 02:20:36 +04:00
struct pppoe_hdr hdr ;
struct pppoe_hdr * ph ;
struct net_device * dev ;
char * start ;
if ( sock_flag ( sk , SOCK_DEAD ) | | ! ( sk - > sk_state & PPPOX_CONNECTED ) ) {
error = - ENOTCONN ;
goto end ;
}
hdr . ver = 1 ;
hdr . type = 1 ;
hdr . code = 0 ;
hdr . sid = po - > num ;
lock_sock ( sk ) ;
dev = po - > pppoe_dev ;
error = - EMSGSIZE ;
if ( total_len > ( dev - > mtu + dev - > hard_header_len ) )
goto end ;
skb = sock_wmalloc ( sk , total_len + dev - > hard_header_len + 32 ,
0 , GFP_KERNEL ) ;
if ( ! skb ) {
error = - ENOMEM ;
goto end ;
}
/* Reserve space for headers. */
skb_reserve ( skb , dev - > hard_header_len ) ;
2007-04-11 07:45:18 +04:00
skb_reset_network_header ( skb ) ;
2005-04-17 02:20:36 +04:00
skb - > dev = dev ;
skb - > priority = sk - > sk_priority ;
skb - > protocol = __constant_htons ( ETH_P_PPP_SES ) ;
ph = ( struct pppoe_hdr * ) skb_put ( skb , total_len + sizeof ( struct pppoe_hdr ) ) ;
start = ( char * ) & ph - > tag [ 0 ] ;
error = memcpy_fromiovec ( start , m - > msg_iov , total_len ) ;
if ( error < 0 ) {
kfree_skb ( skb ) ;
goto end ;
}
error = total_len ;
dev - > hard_header ( skb , dev , ETH_P_PPP_SES ,
po - > pppoe_pa . remote , NULL , total_len ) ;
memcpy ( ph , & hdr , sizeof ( struct pppoe_hdr ) ) ;
ph - > length = htons ( total_len ) ;
dev_queue_xmit ( skb ) ;
end :
release_sock ( sk ) ;
return error ;
}
/************************************************************************
*
* xmit function for internal use .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int __pppoe_xmit ( struct sock * sk , struct sk_buff * skb )
{
struct pppox_sock * po = pppox_sk ( sk ) ;
struct net_device * dev = po - > pppoe_dev ;
struct pppoe_hdr hdr ;
struct pppoe_hdr * ph ;
int headroom = skb_headroom ( skb ) ;
int data_len = skb - > len ;
struct sk_buff * skb2 ;
if ( sock_flag ( sk , SOCK_DEAD ) | | ! ( sk - > sk_state & PPPOX_CONNECTED ) )
goto abort ;
hdr . ver = 1 ;
hdr . type = 1 ;
hdr . code = 0 ;
hdr . sid = po - > num ;
hdr . length = htons ( skb - > len ) ;
if ( ! dev )
goto abort ;
/* Copy the skb if there is no space for the header. */
if ( headroom < ( sizeof ( struct pppoe_hdr ) + dev - > hard_header_len ) ) {
skb2 = dev_alloc_skb ( 32 + skb - > len +
sizeof ( struct pppoe_hdr ) +
dev - > hard_header_len ) ;
if ( skb2 = = NULL )
goto abort ;
skb_reserve ( skb2 , dev - > hard_header_len + sizeof ( struct pppoe_hdr ) ) ;
2007-03-28 01:55:52 +04:00
skb_copy_from_linear_data ( skb , skb_put ( skb2 , skb - > len ) ,
skb - > len ) ;
2005-04-17 02:20:36 +04:00
} else {
/* Make a clone so as to not disturb the original skb,
* give dev_queue_xmit something it can free .
*/
skb2 = skb_clone ( skb , GFP_ATOMIC ) ;
2006-06-06 02:34:33 +04:00
if ( skb2 = = NULL )
goto abort ;
2005-04-17 02:20:36 +04:00
}
ph = ( struct pppoe_hdr * ) skb_push ( skb2 , sizeof ( struct pppoe_hdr ) ) ;
memcpy ( ph , & hdr , sizeof ( struct pppoe_hdr ) ) ;
skb2 - > protocol = __constant_htons ( ETH_P_PPP_SES ) ;
2007-04-11 07:45:18 +04:00
skb_reset_network_header ( skb2 ) ;
2005-04-17 02:20:36 +04:00
skb2 - > dev = dev ;
dev - > hard_header ( skb2 , dev , ETH_P_PPP_SES ,
po - > pppoe_pa . remote , NULL , data_len ) ;
/* We're transmitting skb2, and assuming that dev_queue_xmit
* will free it . The generic ppp layer however , is expecting
* that we give back ' skb ' ( not ' skb2 ' ) in case of failure ,
* but free it in case of success .
*/
if ( dev_queue_xmit ( skb2 ) < 0 )
goto abort ;
kfree_skb ( skb ) ;
return 1 ;
abort :
return 0 ;
}
/************************************************************************
*
* xmit function called by generic PPP driver
* sends PPP frame over PPPoE socket
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int pppoe_xmit ( struct ppp_channel * chan , struct sk_buff * skb )
{
struct sock * sk = ( struct sock * ) chan - > private ;
return __pppoe_xmit ( sk , skb ) ;
}
2006-09-13 21:24:59 +04:00
static struct ppp_channel_ops pppoe_chan_ops = {
. start_xmit = pppoe_xmit ,
2005-04-17 02:20:36 +04:00
} ;
static int pppoe_recvmsg ( struct kiocb * iocb , struct socket * sock ,
struct msghdr * m , size_t total_len , int flags )
{
struct sock * sk = sock - > sk ;
2007-04-21 03:56:31 +04:00
struct sk_buff * skb ;
2005-04-17 02:20:36 +04:00
int error = 0 ;
if ( sk - > sk_state & PPPOX_BOUND ) {
error = - EIO ;
goto end ;
}
skb = skb_recv_datagram ( sk , flags & ~ MSG_DONTWAIT ,
flags & MSG_DONTWAIT , & error ) ;
2007-04-21 03:56:31 +04:00
if ( error < 0 )
2005-04-17 02:20:36 +04:00
goto end ;
m - > msg_namelen = 0 ;
if ( skb ) {
2007-03-10 21:56:08 +03:00
struct pppoe_hdr * ph = pppoe_hdr ( skb ) ;
const int len = ntohs ( ph - > length ) ;
2005-04-17 02:20:36 +04:00
error = memcpy_toiovec ( m - > msg_iov , ( unsigned char * ) & ph - > tag [ 0 ] , len ) ;
2007-03-10 21:56:08 +03:00
if ( error = = 0 )
error = len ;
2005-04-17 02:20:36 +04:00
}
2007-03-10 21:56:08 +03:00
kfree_skb ( skb ) ;
2005-04-17 02:20:36 +04:00
end :
return error ;
}
# ifdef CONFIG_PROC_FS
static int pppoe_seq_show ( struct seq_file * seq , void * v )
{
struct pppox_sock * po ;
char * dev_name ;
if ( v = = SEQ_START_TOKEN ) {
seq_puts ( seq , " Id Address Device \n " ) ;
goto out ;
}
po = v ;
dev_name = po - > pppoe_pa . dev ;
seq_printf ( seq , " %08X %02X:%02X:%02X:%02X:%02X:%02X %8s \n " ,
po - > pppoe_pa . sid ,
po - > pppoe_pa . remote [ 0 ] , po - > pppoe_pa . remote [ 1 ] ,
po - > pppoe_pa . remote [ 2 ] , po - > pppoe_pa . remote [ 3 ] ,
po - > pppoe_pa . remote [ 4 ] , po - > pppoe_pa . remote [ 5 ] , dev_name ) ;
out :
return 0 ;
}
static __inline__ struct pppox_sock * pppoe_get_idx ( loff_t pos )
{
2007-04-21 03:56:31 +04:00
struct pppox_sock * po ;
2005-04-17 02:20:36 +04:00
int i = 0 ;
for ( ; i < PPPOE_HASH_SIZE ; i + + ) {
po = item_hash_table [ i ] ;
while ( po ) {
if ( ! pos - - )
goto out ;
po = po - > next ;
}
}
out :
return po ;
}
static void * pppoe_seq_start ( struct seq_file * seq , loff_t * pos )
{
loff_t l = * pos ;
read_lock_bh ( & pppoe_hash_lock ) ;
return l ? pppoe_get_idx ( - - l ) : SEQ_START_TOKEN ;
}
static void * pppoe_seq_next ( struct seq_file * seq , void * v , loff_t * pos )
{
struct pppox_sock * po ;
+ + * pos ;
if ( v = = SEQ_START_TOKEN ) {
po = pppoe_get_idx ( 0 ) ;
goto out ;
}
po = v ;
2006-09-13 21:24:59 +04:00
if ( po - > next )
2005-04-17 02:20:36 +04:00
po = po - > next ;
else {
int hash = hash_item ( po - > pppoe_pa . sid , po - > pppoe_pa . remote ) ;
while ( + + hash < PPPOE_HASH_SIZE ) {
po = item_hash_table [ hash ] ;
if ( po )
break ;
}
}
out :
return po ;
}
static void pppoe_seq_stop ( struct seq_file * seq , void * v )
{
read_unlock_bh ( & pppoe_hash_lock ) ;
}
static struct seq_operations pppoe_seq_ops = {
. start = pppoe_seq_start ,
. next = pppoe_seq_next ,
. stop = pppoe_seq_stop ,
. show = pppoe_seq_show ,
} ;
static int pppoe_seq_open ( struct inode * inode , struct file * file )
{
return seq_open ( file , & pppoe_seq_ops ) ;
}
2007-02-12 11:55:34 +03:00
static const struct file_operations pppoe_seq_fops = {
2005-04-17 02:20:36 +04:00
. owner = THIS_MODULE ,
. open = pppoe_seq_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = seq_release ,
} ;
static int __init pppoe_proc_init ( void )
{
struct proc_dir_entry * p ;
2005-09-29 01:32:57 +04:00
p = create_proc_entry ( " net/pppoe " , S_IRUGO , NULL ) ;
2005-04-17 02:20:36 +04:00
if ( ! p )
return - ENOMEM ;
p - > proc_fops = & pppoe_seq_fops ;
return 0 ;
}
# else /* CONFIG_PROC_FS */
static inline int pppoe_proc_init ( void ) { return 0 ; }
# endif /* CONFIG_PROC_FS */
2005-12-28 07:57:40 +03:00
static const struct proto_ops pppoe_ops = {
2005-04-17 02:20:36 +04:00
. family = AF_PPPOX ,
. owner = THIS_MODULE ,
. release = pppoe_release ,
. bind = sock_no_bind ,
. connect = pppoe_connect ,
. socketpair = sock_no_socketpair ,
. accept = sock_no_accept ,
. getname = pppoe_getname ,
. poll = datagram_poll ,
. listen = sock_no_listen ,
. shutdown = sock_no_shutdown ,
. setsockopt = sock_no_setsockopt ,
. getsockopt = sock_no_getsockopt ,
. sendmsg = pppoe_sendmsg ,
. recvmsg = pppoe_recvmsg ,
2005-12-28 07:57:40 +03:00
. mmap = sock_no_mmap ,
. ioctl = pppox_ioctl ,
2005-04-17 02:20:36 +04:00
} ;
static struct pppox_proto pppoe_proto = {
. create = pppoe_create ,
. ioctl = pppoe_ioctl ,
. owner = THIS_MODULE ,
} ;
static int __init pppoe_init ( void )
{
int err = proto_register ( & pppoe_sk_proto , 0 ) ;
if ( err )
goto out ;
err = register_pppox_proto ( PX_PROTO_OE , & pppoe_proto ) ;
if ( err )
goto out_unregister_pppoe_proto ;
err = pppoe_proc_init ( ) ;
if ( err )
goto out_unregister_pppox_proto ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
dev_add_pack ( & pppoes_ptype ) ;
dev_add_pack ( & pppoed_ptype ) ;
register_netdevice_notifier ( & pppoe_notifier ) ;
out :
return err ;
out_unregister_pppox_proto :
unregister_pppox_proto ( PX_PROTO_OE ) ;
out_unregister_pppoe_proto :
proto_unregister ( & pppoe_sk_proto ) ;
goto out ;
}
static void __exit pppoe_exit ( void )
{
unregister_pppox_proto ( PX_PROTO_OE ) ;
dev_remove_pack ( & pppoes_ptype ) ;
dev_remove_pack ( & pppoed_ptype ) ;
unregister_netdevice_notifier ( & pppoe_notifier ) ;
2005-09-29 01:32:57 +04:00
remove_proc_entry ( " net/pppoe " , NULL ) ;
2005-04-17 02:20:36 +04:00
proto_unregister ( & pppoe_sk_proto ) ;
}
module_init ( pppoe_init ) ;
module_exit ( pppoe_exit ) ;
MODULE_AUTHOR ( " Michal Ostrowski <mostrows@speakeasy.net> " ) ;
MODULE_DESCRIPTION ( " PPP over Ethernet driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS_NETPROTO ( PF_PPPOX ) ;