2005-04-16 15:20:36 -07:00
/** -*- linux-c -*- ***********************************************************
* Linux PPP over X / Ethernet ( PPPoX / PPPoE ) Sockets
*
* PPPoX - - - Generic PPP encapsulation socket family
* PPPoE - - - PPP over Ethernet ( RFC 2516 )
*
*
* Version : 0.5 .2
*
* Author : Michal Ostrowski < mostrows @ speakeasy . net >
*
* 051000 : Initialization cleanup
*
* 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/errno.h>
# include <linux/netdevice.h>
# include <linux/net.h>
# include <linux/init.h>
# include <linux/if_pppox.h>
# include <linux/ppp_defs.h>
2012-03-04 12:56:55 +00:00
# include <linux/ppp-ioctl.h>
2005-04-16 15:20:36 -07:00
# include <linux/ppp_channel.h>
2007-04-30 00:21:02 -07:00
# include <linux/kmod.h>
2005-04-16 15:20:36 -07:00
# include <net/sock.h>
2016-12-24 11:46:01 -08:00
# include <linux/uaccess.h>
2005-04-16 15:20:36 -07:00
2010-09-21 06:43:54 +00:00
static const struct pppox_proto * pppox_protos [ PX_MAX_PROTO + 1 ] ;
2005-04-16 15:20:36 -07:00
2010-09-21 06:43:54 +00:00
int register_pppox_proto ( int proto_num , const struct pppox_proto * pp )
2005-04-16 15:20:36 -07:00
{
if ( proto_num < 0 | | proto_num > PX_MAX_PROTO )
return - EINVAL ;
if ( pppox_protos [ proto_num ] )
return - EALREADY ;
pppox_protos [ proto_num ] = pp ;
return 0 ;
}
void unregister_pppox_proto ( int proto_num )
{
if ( proto_num > = 0 & & proto_num < = PX_MAX_PROTO )
pppox_protos [ proto_num ] = NULL ;
}
void pppox_unbind_sock ( struct sock * sk )
{
/* Clear connection to ppp device, if attached. */
2015-11-19 12:53:21 +01:00
if ( sk - > sk_state & ( PPPOX_BOUND | PPPOX_CONNECTED ) ) {
2005-04-16 15:20:36 -07:00
ppp_unregister_channel ( & pppox_sk ( sk ) - > chan ) ;
sk - > sk_state = PPPOX_DEAD ;
}
}
EXPORT_SYMBOL ( register_pppox_proto ) ;
EXPORT_SYMBOL ( unregister_pppox_proto ) ;
EXPORT_SYMBOL ( pppox_unbind_sock ) ;
2005-12-27 20:57:40 -08:00
int pppox_ioctl ( struct socket * sock , unsigned int cmd , unsigned long arg )
2005-04-16 15:20:36 -07:00
{
struct sock * sk = sock - > sk ;
struct pppox_sock * po = pppox_sk ( sk ) ;
2007-07-30 17:48:23 -07:00
int rc ;
2005-04-16 15:20:36 -07:00
lock_sock ( sk ) ;
switch ( cmd ) {
case PPPIOCGCHAN : {
int index ;
rc = - ENOTCONN ;
if ( ! ( sk - > sk_state & PPPOX_CONNECTED ) )
break ;
rc = - EINVAL ;
index = ppp_channel_index ( & po - > chan ) ;
if ( put_user ( index , ( int __user * ) arg ) )
break ;
rc = 0 ;
sk - > sk_state | = PPPOX_BOUND ;
break ;
}
default :
2007-07-30 17:48:23 -07:00
rc = pppox_protos [ sk - > sk_protocol ] - > ioctl ?
pppox_protos [ sk - > sk_protocol ] - > ioctl ( sock , cmd , arg ) : - ENOTTY ;
}
2005-04-16 15:20:36 -07:00
release_sock ( sk ) ;
return rc ;
}
2005-12-27 20:57:40 -08:00
EXPORT_SYMBOL ( pppox_ioctl ) ;
2005-04-16 15:20:36 -07:00
2009-11-05 22:18:14 -08:00
static int pppox_create ( struct net * net , struct socket * sock , int protocol ,
int kern )
2005-04-16 15:20:36 -07:00
{
int rc = - EPROTOTYPE ;
if ( protocol < 0 | | protocol > PX_MAX_PROTO )
goto out ;
rc = - EPROTONOSUPPORT ;
2008-07-09 10:28:38 +02:00
if ( ! pppox_protos [ protocol ] )
2015-12-02 16:27:39 +01:00
request_module ( " net-pf-%d-proto-%d " , PF_PPPOX , protocol ) ;
2005-04-16 15:20:36 -07:00
if ( ! pppox_protos [ protocol ] | |
! try_module_get ( pppox_protos [ protocol ] - > owner ) )
goto out ;
2015-05-08 21:09:13 -05:00
rc = pppox_protos [ protocol ] - > create ( net , sock , kern ) ;
2005-12-27 20:57:40 -08:00
2005-04-16 15:20:36 -07:00
module_put ( pppox_protos [ protocol ] - > owner ) ;
out :
return rc ;
}
2009-10-05 05:58:39 +00:00
static const struct net_proto_family pppox_proto_family = {
2005-04-16 15:20:36 -07:00
. family = PF_PPPOX ,
. create = pppox_create ,
. owner = THIS_MODULE ,
} ;
static int __init pppox_init ( void )
{
return sock_register ( & pppox_proto_family ) ;
}
static void __exit pppox_exit ( void )
{
sock_unregister ( PF_PPPOX ) ;
}
module_init ( pppox_init ) ;
module_exit ( pppox_exit ) ;
MODULE_AUTHOR ( " Michal Ostrowski <mostrows@speakeasy.net> " ) ;
MODULE_DESCRIPTION ( " PPP over Ethernet driver (generic socket layer) " ) ;
MODULE_LICENSE ( " GPL " ) ;
2015-12-02 16:27:39 +01:00
MODULE_ALIAS_NETPROTO ( PF_PPPOX ) ;