2006-08-03 16:48:59 -07:00
/*
* NetLabel Unlabeled Support
*
* This file defines functions for dealing with unlabeled packets for the
* NetLabel system . The NetLabel system manages static and dynamic label
* mappings for network protocols such as CIPSO and RIPSO .
*
* Author : Paul Moore < paul . moore @ hp . com >
*
*/
/*
2008-10-10 10:16:32 -04:00
* ( c ) Copyright Hewlett - Packard Development Company , L . P . , 2006 - 2008
2006-08-03 16:48:59 -07: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/types.h>
2008-01-29 08:44:21 -05:00
# include <linux/rcupdate.h>
2006-08-03 16:48:59 -07:00
# include <linux/list.h>
# include <linux/spinlock.h>
# include <linux/socket.h>
# include <linux/string.h>
# include <linux/skbuff.h>
2006-11-17 17:38:55 -05:00
# include <linux/audit.h>
2008-01-29 08:44:21 -05:00
# include <linux/in.h>
# include <linux/in6.h>
# include <linux/ip.h>
# include <linux/ipv6.h>
# include <linux/notifier.h>
# include <linux/netdevice.h>
# include <linux/security.h>
2006-08-03 16:48:59 -07:00
# include <net/sock.h>
# include <net/netlink.h>
# include <net/genetlink.h>
2008-01-29 08:44:21 -05:00
# include <net/ip.h>
# include <net/ipv6.h>
# include <net/net_namespace.h>
2006-08-03 16:48:59 -07:00
# include <net/netlabel.h>
# include <asm/bug.h>
2008-01-29 08:44:21 -05:00
# include <asm/atomic.h>
2006-08-03 16:48:59 -07:00
# include "netlabel_user.h"
2008-10-10 10:16:32 -04:00
# include "netlabel_addrlist.h"
2006-08-03 16:48:59 -07:00
# include "netlabel_domainhash.h"
# include "netlabel_unlabeled.h"
2008-01-29 08:44:21 -05:00
# include "netlabel_mgmt.h"
/* NOTE: at present we always use init's network namespace since we don't
* presently support different namespaces even though the majority of
* the functions in this file are " namespace safe " */
/* The unlabeled connection hash table which we use to map network interfaces
* and addresses of unlabeled packets to a user specified secid value for the
* LSM . The hash table is used to lookup the network interface entry
* ( struct netlbl_unlhsh_iface ) and then the interface entry is used to
* lookup an IP address match from an ordered list . If a network interface
* match can not be found in the hash table then the default entry
* ( netlbl_unlhsh_def ) is used . The IP address entry list
* ( struct netlbl_unlhsh_addr ) is ordered such that the entries with a
* larger netmask come first .
*/
struct netlbl_unlhsh_tbl {
struct list_head * tbl ;
u32 size ;
} ;
2008-10-10 10:16:32 -04:00
# define netlbl_unlhsh_addr4_entry(iter) \
container_of ( iter , struct netlbl_unlhsh_addr4 , list )
2008-01-29 08:44:21 -05:00
struct netlbl_unlhsh_addr4 {
u32 secid ;
2008-10-10 10:16:32 -04:00
struct netlbl_af4list list ;
2008-01-29 08:44:21 -05:00
struct rcu_head rcu ;
} ;
2008-10-10 10:16:32 -04:00
# define netlbl_unlhsh_addr6_entry(iter) \
container_of ( iter , struct netlbl_unlhsh_addr6 , list )
2008-01-29 08:44:21 -05:00
struct netlbl_unlhsh_addr6 {
u32 secid ;
2008-10-10 10:16:32 -04:00
struct netlbl_af6list list ;
2008-01-29 08:44:21 -05:00
struct rcu_head rcu ;
} ;
struct netlbl_unlhsh_iface {
int ifindex ;
struct list_head addr4_list ;
struct list_head addr6_list ;
u32 valid ;
struct list_head list ;
struct rcu_head rcu ;
} ;
/* Argument struct for netlbl_unlhsh_walk() */
struct netlbl_unlhsh_walk_arg {
struct netlink_callback * nl_cb ;
struct sk_buff * skb ;
u32 seq ;
} ;
/* Unlabeled connection hash table */
/* updates should be so rare that having one spinlock for the entire
* hash table should be okay */
static DEFINE_SPINLOCK ( netlbl_unlhsh_lock ) ;
static struct netlbl_unlhsh_tbl * netlbl_unlhsh = NULL ;
static struct netlbl_unlhsh_iface * netlbl_unlhsh_def = NULL ;
2006-08-03 16:48:59 -07:00
/* Accept unlabeled packets flag */
2006-11-17 17:38:44 -05:00
static u8 netlabel_unlabel_acceptflg = 0 ;
2006-08-03 16:48:59 -07:00
2008-01-29 08:44:21 -05:00
/* NetLabel Generic NETLINK unlabeled family */
2006-08-03 16:48:59 -07:00
static struct genl_family netlbl_unlabel_gnl_family = {
. id = GENL_ID_GENERATE ,
. hdrsize = 0 ,
. name = NETLBL_NLTYPE_UNLABELED_NAME ,
. version = NETLBL_PROTO_VERSION ,
2006-09-25 15:56:37 -07:00
. maxattr = NLBL_UNLABEL_A_MAX ,
2006-08-03 16:48:59 -07:00
} ;
2006-09-25 15:56:37 -07:00
/* NetLabel Netlink attribute policy */
2007-06-05 12:38:30 -07:00
static const struct nla_policy netlbl_unlabel_genl_policy [ NLBL_UNLABEL_A_MAX + 1 ] = {
2006-09-25 15:56:37 -07:00
[ NLBL_UNLABEL_A_ACPTFLG ] = { . type = NLA_U8 } ,
2008-01-29 08:44:21 -05:00
[ NLBL_UNLABEL_A_IPV6ADDR ] = { . type = NLA_BINARY ,
. len = sizeof ( struct in6_addr ) } ,
[ NLBL_UNLABEL_A_IPV6MASK ] = { . type = NLA_BINARY ,
. len = sizeof ( struct in6_addr ) } ,
[ NLBL_UNLABEL_A_IPV4ADDR ] = { . type = NLA_BINARY ,
. len = sizeof ( struct in_addr ) } ,
[ NLBL_UNLABEL_A_IPV4MASK ] = { . type = NLA_BINARY ,
. len = sizeof ( struct in_addr ) } ,
[ NLBL_UNLABEL_A_IFACE ] = { . type = NLA_NUL_STRING ,
. len = IFNAMSIZ - 1 } ,
[ NLBL_UNLABEL_A_SECCTX ] = { . type = NLA_BINARY }
2006-09-25 15:56:37 -07:00
} ;
2006-08-03 16:48:59 -07:00
2006-09-28 14:51:47 -07:00
/*
2008-01-29 08:44:21 -05:00
* Unlabeled Connection Hash Table Functions
*/
/**
* netlbl_unlhsh_free_addr4 - Frees an IPv4 address entry from the hash table
* @ entry : the entry ' s RCU field
*
* Description :
* This function is designed to be used as a callback to the call_rcu ( )
* function so that memory allocated to a hash table address entry can be
* released safely .
*
*/
static void netlbl_unlhsh_free_addr4 ( struct rcu_head * entry )
{
struct netlbl_unlhsh_addr4 * ptr ;
ptr = container_of ( entry , struct netlbl_unlhsh_addr4 , rcu ) ;
kfree ( ptr ) ;
}
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
/**
* netlbl_unlhsh_free_addr6 - Frees an IPv6 address entry from the hash table
* @ entry : the entry ' s RCU field
*
* Description :
* This function is designed to be used as a callback to the call_rcu ( )
* function so that memory allocated to a hash table address entry can be
* released safely .
*
*/
static void netlbl_unlhsh_free_addr6 ( struct rcu_head * entry )
{
struct netlbl_unlhsh_addr6 * ptr ;
ptr = container_of ( entry , struct netlbl_unlhsh_addr6 , rcu ) ;
kfree ( ptr ) ;
}
# endif /* IPv6 */
/**
* netlbl_unlhsh_free_iface - Frees an interface entry from the hash table
* @ entry : the entry ' s RCU field
*
* Description :
* This function is designed to be used as a callback to the call_rcu ( )
* function so that memory allocated to a hash table interface entry can be
* released safely . It is important to note that this function does not free
* the IPv4 and IPv6 address lists contained as part of an interface entry . It
* is up to the rest of the code to make sure an interface entry is only freed
* once it ' s address lists are empty .
*
*/
static void netlbl_unlhsh_free_iface ( struct rcu_head * entry )
{
struct netlbl_unlhsh_iface * iface ;
2008-10-10 10:16:32 -04:00
struct netlbl_af4list * iter4 ;
struct netlbl_af4list * tmp4 ;
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
struct netlbl_af6list * iter6 ;
struct netlbl_af6list * tmp6 ;
# endif /* IPv6 */
2008-01-29 08:44:21 -05:00
iface = container_of ( entry , struct netlbl_unlhsh_iface , rcu ) ;
/* no need for locks here since we are the only one with access to this
* structure */
2008-10-10 10:16:32 -04:00
netlbl_af4list_foreach_safe ( iter4 , tmp4 , & iface - > addr4_list ) {
netlbl_af4list_remove_entry ( iter4 ) ;
kfree ( netlbl_unlhsh_addr4_entry ( iter4 ) ) ;
}
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
netlbl_af6list_foreach_safe ( iter6 , tmp6 , & iface - > addr6_list ) {
netlbl_af6list_remove_entry ( iter6 ) ;
kfree ( netlbl_unlhsh_addr6_entry ( iter6 ) ) ;
}
# endif /* IPv6 */
2008-01-29 08:44:21 -05:00
kfree ( iface ) ;
}
/**
* netlbl_unlhsh_hash - Hashing function for the hash table
* @ ifindex : the network interface / device to hash
*
* Description :
* This is the hashing function for the unlabeled hash table , it returns the
* bucket number for the given device / interface . The caller is responsible for
* calling the rcu_read_ [ un ] lock ( ) functions .
*
*/
static u32 netlbl_unlhsh_hash ( int ifindex )
{
/* this is taken _almost_ directly from
* security / selinux / netif . c : sel_netif_hasfn ( ) as they do pretty much
* the same thing */
return ifindex & ( rcu_dereference ( netlbl_unlhsh ) - > size - 1 ) ;
}
/**
* netlbl_unlhsh_search_iface - Search for a matching interface entry
* @ ifindex : the network interface
*
* Description :
* Searches the unlabeled connection hash table and returns a pointer to the
* interface entry which matches @ ifindex , otherwise NULL is returned . The
* caller is responsible for calling the rcu_read_ [ un ] lock ( ) functions .
*
*/
static struct netlbl_unlhsh_iface * netlbl_unlhsh_search_iface ( int ifindex )
{
u32 bkt ;
2008-10-10 10:16:29 -04:00
struct list_head * bkt_list ;
2008-01-29 08:44:21 -05:00
struct netlbl_unlhsh_iface * iter ;
bkt = netlbl_unlhsh_hash ( ifindex ) ;
2008-10-10 10:16:29 -04:00
bkt_list = & rcu_dereference ( netlbl_unlhsh ) - > tbl [ bkt ] ;
list_for_each_entry_rcu ( iter , bkt_list , list )
2008-01-29 08:44:21 -05:00
if ( iter - > valid & & iter - > ifindex = = ifindex )
return iter ;
return NULL ;
}
/**
* netlbl_unlhsh_search_iface_def - Search for a matching interface entry
* @ ifindex : the network interface
*
* Description :
* Searches the unlabeled connection hash table and returns a pointer to the
* interface entry which matches @ ifindex . If an exact match can not be found
* and there is a valid default entry , the default entry is returned , otherwise
* NULL is returned . The caller is responsible for calling the
* rcu_read_ [ un ] lock ( ) functions .
*
*/
static struct netlbl_unlhsh_iface * netlbl_unlhsh_search_iface_def ( int ifindex )
{
struct netlbl_unlhsh_iface * entry ;
entry = netlbl_unlhsh_search_iface ( ifindex ) ;
if ( entry ! = NULL )
return entry ;
entry = rcu_dereference ( netlbl_unlhsh_def ) ;
if ( entry ! = NULL & & entry - > valid )
return entry ;
return NULL ;
}
/**
* netlbl_unlhsh_add_addr4 - Add a new IPv4 address entry to the hash table
* @ iface : the associated interface entry
* @ addr : IPv4 address in network byte order
* @ mask : IPv4 address mask in network byte order
* @ secid : LSM secid value for entry
*
* Description :
* Add a new address entry into the unlabeled connection hash table using the
* interface entry specified by @ iface . On success zero is returned , otherwise
* a negative value is returned . The caller is responsible for calling the
* rcu_read_ [ un ] lock ( ) functions .
*
2006-09-28 14:51:47 -07:00
*/
2008-01-29 08:44:21 -05:00
static int netlbl_unlhsh_add_addr4 ( struct netlbl_unlhsh_iface * iface ,
const struct in_addr * addr ,
const struct in_addr * mask ,
u32 secid )
{
2008-10-10 10:16:32 -04:00
int ret_val ;
2008-01-29 08:44:21 -05:00
struct netlbl_unlhsh_addr4 * entry ;
entry = kzalloc ( sizeof ( * entry ) , GFP_ATOMIC ) ;
if ( entry = = NULL )
return - ENOMEM ;
2008-10-10 10:16:32 -04:00
entry - > list . addr = addr - > s_addr & mask - > s_addr ;
entry - > list . mask = mask - > s_addr ;
entry - > list . valid = 1 ;
2008-01-29 08:44:21 -05:00
INIT_RCU_HEAD ( & entry - > rcu ) ;
2008-10-10 10:16:32 -04:00
entry - > secid = secid ;
2008-01-29 08:44:21 -05:00
spin_lock ( & netlbl_unlhsh_lock ) ;
2008-10-10 10:16:32 -04:00
ret_val = netlbl_af4list_add ( & entry - > list , & iface - > addr4_list ) ;
2008-01-29 08:44:21 -05:00
spin_unlock ( & netlbl_unlhsh_lock ) ;
2008-10-10 10:16:32 -04:00
if ( ret_val ! = 0 )
kfree ( entry ) ;
return ret_val ;
2008-01-29 08:44:21 -05:00
}
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
/**
* netlbl_unlhsh_add_addr6 - Add a new IPv6 address entry to the hash table
* @ iface : the associated interface entry
* @ addr : IPv6 address in network byte order
* @ mask : IPv6 address mask in network byte order
* @ secid : LSM secid value for entry
*
* Description :
* Add a new address entry into the unlabeled connection hash table using the
* interface entry specified by @ iface . On success zero is returned , otherwise
* a negative value is returned . The caller is responsible for calling the
* rcu_read_ [ un ] lock ( ) functions .
*
*/
static int netlbl_unlhsh_add_addr6 ( struct netlbl_unlhsh_iface * iface ,
const struct in6_addr * addr ,
const struct in6_addr * mask ,
u32 secid )
{
2008-10-10 10:16:32 -04:00
int ret_val ;
2008-01-29 08:44:21 -05:00
struct netlbl_unlhsh_addr6 * entry ;
entry = kzalloc ( sizeof ( * entry ) , GFP_ATOMIC ) ;
if ( entry = = NULL )
return - ENOMEM ;
2008-10-10 10:16:32 -04:00
ipv6_addr_copy ( & entry - > list . addr , addr ) ;
entry - > list . addr . s6_addr32 [ 0 ] & = mask - > s6_addr32 [ 0 ] ;
entry - > list . addr . s6_addr32 [ 1 ] & = mask - > s6_addr32 [ 1 ] ;
entry - > list . addr . s6_addr32 [ 2 ] & = mask - > s6_addr32 [ 2 ] ;
entry - > list . addr . s6_addr32 [ 3 ] & = mask - > s6_addr32 [ 3 ] ;
ipv6_addr_copy ( & entry - > list . mask , mask ) ;
entry - > list . valid = 1 ;
2008-01-29 08:44:21 -05:00
INIT_RCU_HEAD ( & entry - > rcu ) ;
2008-10-10 10:16:32 -04:00
entry - > secid = secid ;
2008-01-29 08:44:21 -05:00
spin_lock ( & netlbl_unlhsh_lock ) ;
2008-10-10 10:16:32 -04:00
ret_val = netlbl_af6list_add ( & entry - > list , & iface - > addr6_list ) ;
2008-01-29 08:44:21 -05:00
spin_unlock ( & netlbl_unlhsh_lock ) ;
2008-10-10 10:16:32 -04:00
if ( ret_val ! = 0 )
kfree ( entry ) ;
2008-01-29 08:44:21 -05:00
return 0 ;
}
# endif /* IPv6 */
/**
* netlbl_unlhsh_add_iface - Adds a new interface entry to the hash table
* @ ifindex : network interface
*
* Description :
* Add a new , empty , interface entry into the unlabeled connection hash table .
* On success a pointer to the new interface entry is returned , on failure NULL
* is returned . The caller is responsible for calling the rcu_read_ [ un ] lock ( )
* functions .
*
*/
static struct netlbl_unlhsh_iface * netlbl_unlhsh_add_iface ( int ifindex )
{
u32 bkt ;
struct netlbl_unlhsh_iface * iface ;
iface = kzalloc ( sizeof ( * iface ) , GFP_ATOMIC ) ;
if ( iface = = NULL )
return NULL ;
iface - > ifindex = ifindex ;
INIT_LIST_HEAD ( & iface - > addr4_list ) ;
INIT_LIST_HEAD ( & iface - > addr6_list ) ;
iface - > valid = 1 ;
INIT_RCU_HEAD ( & iface - > rcu ) ;
spin_lock ( & netlbl_unlhsh_lock ) ;
if ( ifindex > 0 ) {
bkt = netlbl_unlhsh_hash ( ifindex ) ;
if ( netlbl_unlhsh_search_iface ( ifindex ) ! = NULL )
goto add_iface_failure ;
list_add_tail_rcu ( & iface - > list ,
& rcu_dereference ( netlbl_unlhsh ) - > tbl [ bkt ] ) ;
} else {
INIT_LIST_HEAD ( & iface - > list ) ;
if ( rcu_dereference ( netlbl_unlhsh_def ) ! = NULL )
goto add_iface_failure ;
rcu_assign_pointer ( netlbl_unlhsh_def , iface ) ;
}
spin_unlock ( & netlbl_unlhsh_lock ) ;
return iface ;
add_iface_failure :
spin_unlock ( & netlbl_unlhsh_lock ) ;
kfree ( iface ) ;
return NULL ;
}
/**
* netlbl_unlhsh_add - Adds a new entry to the unlabeled connection hash table
* @ net : network namespace
* @ dev_name : interface name
* @ addr : IP address in network byte order
* @ mask : address mask in network byte order
* @ addr_len : length of address / mask ( 4 for IPv4 , 16 for IPv6 )
* @ secid : LSM secid value for the entry
2008-01-29 08:44:23 -05:00
* @ audit_info : NetLabel audit information
2008-01-29 08:44:21 -05:00
*
* Description :
* Adds a new entry to the unlabeled connection hash table . Returns zero on
* success , negative values on failure .
*
*/
2008-12-31 12:54:11 -05:00
int netlbl_unlhsh_add ( struct net * net ,
const char * dev_name ,
const void * addr ,
const void * mask ,
u32 addr_len ,
u32 secid ,
struct netlbl_audit * audit_info )
2008-01-29 08:44:21 -05:00
{
int ret_val ;
int ifindex ;
struct net_device * dev ;
struct netlbl_unlhsh_iface * iface ;
2008-01-29 08:44:23 -05:00
struct audit_buffer * audit_buf = NULL ;
char * secctx = NULL ;
u32 secctx_len ;
2008-01-29 08:44:21 -05:00
if ( addr_len ! = sizeof ( struct in_addr ) & &
addr_len ! = sizeof ( struct in6_addr ) )
return - EINVAL ;
rcu_read_lock ( ) ;
if ( dev_name ! = NULL ) {
dev = dev_get_by_name ( net , dev_name ) ;
if ( dev = = NULL ) {
ret_val = - ENODEV ;
goto unlhsh_add_return ;
}
ifindex = dev - > ifindex ;
dev_put ( dev ) ;
iface = netlbl_unlhsh_search_iface ( ifindex ) ;
} else {
ifindex = 0 ;
iface = rcu_dereference ( netlbl_unlhsh_def ) ;
}
if ( iface = = NULL ) {
iface = netlbl_unlhsh_add_iface ( ifindex ) ;
if ( iface = = NULL ) {
ret_val = - ENOMEM ;
goto unlhsh_add_return ;
}
}
2008-01-29 08:44:23 -05:00
audit_buf = netlbl_audit_start_common ( AUDIT_MAC_UNLBL_STCADD ,
audit_info ) ;
2008-01-29 08:44:21 -05:00
switch ( addr_len ) {
2008-02-12 22:37:19 -08:00
case sizeof ( struct in_addr ) : {
struct in_addr * addr4 , * mask4 ;
2008-01-29 08:44:23 -05:00
addr4 = ( struct in_addr * ) addr ;
mask4 = ( struct in_addr * ) mask ;
ret_val = netlbl_unlhsh_add_addr4 ( iface , addr4 , mask4 , secid ) ;
if ( audit_buf ! = NULL )
2008-10-10 10:16:32 -04:00
netlbl_af4list_audit_addr ( audit_buf , 1 ,
dev_name ,
addr4 - > s_addr ,
mask4 - > s_addr ) ;
2008-01-29 08:44:21 -05:00
break ;
2008-02-12 22:37:19 -08:00
}
2008-01-29 08:44:21 -05:00
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
2008-02-12 22:37:19 -08:00
case sizeof ( struct in6_addr ) : {
struct in6_addr * addr6 , * mask6 ;
2008-01-29 08:44:23 -05:00
addr6 = ( struct in6_addr * ) addr ;
mask6 = ( struct in6_addr * ) mask ;
ret_val = netlbl_unlhsh_add_addr6 ( iface , addr6 , mask6 , secid ) ;
if ( audit_buf ! = NULL )
2008-10-10 10:16:32 -04:00
netlbl_af6list_audit_addr ( audit_buf , 1 ,
dev_name ,
addr6 , mask6 ) ;
2008-01-29 08:44:21 -05:00
break ;
2008-02-12 22:37:19 -08:00
}
2008-01-29 08:44:21 -05:00
# endif /* IPv6 */
default :
ret_val = - EINVAL ;
}
if ( ret_val = = 0 )
atomic_inc ( & netlabel_mgmt_protocount ) ;
unlhsh_add_return :
rcu_read_unlock ( ) ;
2008-01-29 08:44:23 -05:00
if ( audit_buf ! = NULL ) {
if ( security_secid_to_secctx ( secid ,
& secctx ,
& secctx_len ) = = 0 ) {
audit_log_format ( audit_buf , " sec_obj=%s " , secctx ) ;
security_release_secctx ( secctx , secctx_len ) ;
}
audit_log_format ( audit_buf , " res=%u " , ret_val = = 0 ? 1 : 0 ) ;
audit_log_end ( audit_buf ) ;
}
2008-01-29 08:44:21 -05:00
return ret_val ;
}
/**
* netlbl_unlhsh_remove_addr4 - Remove an IPv4 address entry
2008-01-29 08:44:23 -05:00
* @ net : network namespace
2008-01-29 08:44:21 -05:00
* @ iface : interface entry
* @ addr : IP address
* @ mask : IP address mask
2008-01-29 08:44:23 -05:00
* @ audit_info : NetLabel audit information
2008-01-29 08:44:21 -05:00
*
* Description :
* Remove an IP address entry from the unlabeled connection hash table .
* Returns zero on success , negative values on failure . The caller is
* responsible for calling the rcu_read_ [ un ] lock ( ) functions .
*
*/
2008-01-29 08:44:23 -05:00
static int netlbl_unlhsh_remove_addr4 ( struct net * net ,
struct netlbl_unlhsh_iface * iface ,
2008-01-29 08:44:21 -05:00
const struct in_addr * addr ,
2008-01-29 08:44:23 -05:00
const struct in_addr * mask ,
struct netlbl_audit * audit_info )
2008-01-29 08:44:21 -05:00
{
2008-10-10 10:16:32 -04:00
struct netlbl_af4list * list_entry ;
2008-01-29 08:44:21 -05:00
struct netlbl_unlhsh_addr4 * entry ;
2008-10-10 10:16:32 -04:00
struct audit_buffer * audit_buf ;
2008-01-29 08:44:23 -05:00
struct net_device * dev ;
2008-10-10 10:16:32 -04:00
char * secctx ;
2008-01-29 08:44:23 -05:00
u32 secctx_len ;
2008-01-29 08:44:21 -05:00
spin_lock ( & netlbl_unlhsh_lock ) ;
2008-10-10 10:16:32 -04:00
list_entry = netlbl_af4list_remove ( addr - > s_addr , mask - > s_addr ,
& iface - > addr4_list ) ;
2008-01-29 08:44:21 -05:00
spin_unlock ( & netlbl_unlhsh_lock ) ;
2008-12-03 00:37:04 -08:00
if ( list_entry ! = NULL )
entry = netlbl_unlhsh_addr4_entry ( list_entry ) ;
else
2008-12-11 21:31:50 -08:00
entry = NULL ;
2008-01-29 08:44:21 -05:00
2008-01-29 08:44:23 -05:00
audit_buf = netlbl_audit_start_common ( AUDIT_MAC_UNLBL_STCDEL ,
audit_info ) ;
if ( audit_buf ! = NULL ) {
dev = dev_get_by_index ( net , iface - > ifindex ) ;
2008-10-10 10:16:32 -04:00
netlbl_af4list_audit_addr ( audit_buf , 1 ,
( dev ! = NULL ? dev - > name : NULL ) ,
addr - > s_addr , mask - > s_addr ) ;
2008-01-29 08:44:23 -05:00
if ( dev ! = NULL )
dev_put ( dev ) ;
2008-12-11 21:31:50 -08:00
if ( entry ! = NULL & &
security_secid_to_secctx ( entry - > secid ,
& secctx , & secctx_len ) = = 0 ) {
2008-01-29 08:44:23 -05:00
audit_log_format ( audit_buf , " sec_obj=%s " , secctx ) ;
security_release_secctx ( secctx , secctx_len ) ;
}
2008-12-11 21:31:50 -08:00
audit_log_format ( audit_buf , " res=%u " , entry ! = NULL ? 1 : 0 ) ;
2008-01-29 08:44:23 -05:00
audit_log_end ( audit_buf ) ;
}
2008-12-11 21:31:50 -08:00
if ( entry = = NULL )
return - ENOENT ;
call_rcu ( & entry - > rcu , netlbl_unlhsh_free_addr4 ) ;
return 0 ;
2008-01-29 08:44:21 -05:00
}
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
/**
* netlbl_unlhsh_remove_addr6 - Remove an IPv6 address entry
2008-01-29 08:44:23 -05:00
* @ net : network namespace
2008-01-29 08:44:21 -05:00
* @ iface : interface entry
* @ addr : IP address
* @ mask : IP address mask
2008-01-29 08:44:23 -05:00
* @ audit_info : NetLabel audit information
2008-01-29 08:44:21 -05:00
*
* Description :
* Remove an IP address entry from the unlabeled connection hash table .
* Returns zero on success , negative values on failure . The caller is
* responsible for calling the rcu_read_ [ un ] lock ( ) functions .
*
*/
2008-01-29 08:44:23 -05:00
static int netlbl_unlhsh_remove_addr6 ( struct net * net ,
struct netlbl_unlhsh_iface * iface ,
2008-01-29 08:44:21 -05:00
const struct in6_addr * addr ,
2008-01-29 08:44:23 -05:00
const struct in6_addr * mask ,
struct netlbl_audit * audit_info )
2008-01-29 08:44:21 -05:00
{
2008-10-10 10:16:32 -04:00
struct netlbl_af6list * list_entry ;
2008-01-29 08:44:21 -05:00
struct netlbl_unlhsh_addr6 * entry ;
2008-10-10 10:16:32 -04:00
struct audit_buffer * audit_buf ;
2008-01-29 08:44:23 -05:00
struct net_device * dev ;
2008-10-10 10:16:32 -04:00
char * secctx ;
2008-01-29 08:44:23 -05:00
u32 secctx_len ;
2008-01-29 08:44:21 -05:00
spin_lock ( & netlbl_unlhsh_lock ) ;
2008-10-10 10:16:32 -04:00
list_entry = netlbl_af6list_remove ( addr , mask , & iface - > addr6_list ) ;
2008-01-29 08:44:21 -05:00
spin_unlock ( & netlbl_unlhsh_lock ) ;
2008-12-03 00:37:04 -08:00
if ( list_entry ! = NULL )
entry = netlbl_unlhsh_addr6_entry ( list_entry ) ;
else
2008-12-11 21:31:50 -08:00
entry = NULL ;
2008-01-29 08:44:21 -05:00
2008-01-29 08:44:23 -05:00
audit_buf = netlbl_audit_start_common ( AUDIT_MAC_UNLBL_STCDEL ,
audit_info ) ;
if ( audit_buf ! = NULL ) {
dev = dev_get_by_index ( net , iface - > ifindex ) ;
2008-10-10 10:16:32 -04:00
netlbl_af6list_audit_addr ( audit_buf , 1 ,
( dev ! = NULL ? dev - > name : NULL ) ,
addr , mask ) ;
2008-01-29 08:44:23 -05:00
if ( dev ! = NULL )
dev_put ( dev ) ;
2008-12-11 21:31:50 -08:00
if ( entry ! = NULL & &
security_secid_to_secctx ( entry - > secid ,
& secctx , & secctx_len ) = = 0 ) {
2008-01-29 08:44:23 -05:00
audit_log_format ( audit_buf , " sec_obj=%s " , secctx ) ;
security_release_secctx ( secctx , secctx_len ) ;
}
2008-12-11 21:31:50 -08:00
audit_log_format ( audit_buf , " res=%u " , entry ! = NULL ? 1 : 0 ) ;
2008-01-29 08:44:23 -05:00
audit_log_end ( audit_buf ) ;
}
2008-12-11 21:31:50 -08:00
if ( entry = = NULL )
return - ENOENT ;
call_rcu ( & entry - > rcu , netlbl_unlhsh_free_addr6 ) ;
return 0 ;
2008-01-29 08:44:21 -05:00
}
# endif /* IPv6 */
/**
* netlbl_unlhsh_condremove_iface - Remove an interface entry
* @ iface : the interface entry
*
* Description :
* Remove an interface entry from the unlabeled connection hash table if it is
* empty . An interface entry is considered to be empty if there are no
* address entries assigned to it .
*
*/
static void netlbl_unlhsh_condremove_iface ( struct netlbl_unlhsh_iface * iface )
{
2008-10-10 10:16:32 -04:00
struct netlbl_af4list * iter4 ;
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
struct netlbl_af6list * iter6 ;
# endif /* IPv6 */
2008-01-29 08:44:21 -05:00
spin_lock ( & netlbl_unlhsh_lock ) ;
2008-10-10 10:16:32 -04:00
netlbl_af4list_foreach_rcu ( iter4 , & iface - > addr4_list )
goto unlhsh_condremove_failure ;
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
netlbl_af6list_foreach_rcu ( iter6 , & iface - > addr6_list )
goto unlhsh_condremove_failure ;
# endif /* IPv6 */
2008-01-29 08:44:21 -05:00
iface - > valid = 0 ;
if ( iface - > ifindex > 0 )
list_del_rcu ( & iface - > list ) ;
else
rcu_assign_pointer ( netlbl_unlhsh_def , NULL ) ;
spin_unlock ( & netlbl_unlhsh_lock ) ;
call_rcu ( & iface - > rcu , netlbl_unlhsh_free_iface ) ;
return ;
unlhsh_condremove_failure :
spin_unlock ( & netlbl_unlhsh_lock ) ;
return ;
}
/**
* netlbl_unlhsh_remove - Remove an entry from the unlabeled hash table
* @ net : network namespace
* @ dev_name : interface name
* @ addr : IP address in network byte order
* @ mask : address mask in network byte order
* @ addr_len : length of address / mask ( 4 for IPv4 , 16 for IPv6 )
2008-01-29 08:44:23 -05:00
* @ audit_info : NetLabel audit information
2008-01-29 08:44:21 -05:00
*
* Description :
* Removes and existing entry from the unlabeled connection hash table .
* Returns zero on success , negative values on failure .
*
*/
2008-12-31 12:54:11 -05:00
int netlbl_unlhsh_remove ( struct net * net ,
const char * dev_name ,
const void * addr ,
const void * mask ,
u32 addr_len ,
struct netlbl_audit * audit_info )
2008-01-29 08:44:21 -05:00
{
int ret_val ;
struct net_device * dev ;
struct netlbl_unlhsh_iface * iface ;
if ( addr_len ! = sizeof ( struct in_addr ) & &
addr_len ! = sizeof ( struct in6_addr ) )
return - EINVAL ;
rcu_read_lock ( ) ;
if ( dev_name ! = NULL ) {
dev = dev_get_by_name ( net , dev_name ) ;
if ( dev = = NULL ) {
ret_val = - ENODEV ;
goto unlhsh_remove_return ;
}
iface = netlbl_unlhsh_search_iface ( dev - > ifindex ) ;
dev_put ( dev ) ;
} else
iface = rcu_dereference ( netlbl_unlhsh_def ) ;
if ( iface = = NULL ) {
ret_val = - ENOENT ;
goto unlhsh_remove_return ;
}
switch ( addr_len ) {
case sizeof ( struct in_addr ) :
2008-01-29 08:44:23 -05:00
ret_val = netlbl_unlhsh_remove_addr4 ( net ,
iface , addr , mask ,
audit_info ) ;
2008-01-29 08:44:21 -05:00
break ;
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
case sizeof ( struct in6_addr ) :
2008-01-29 08:44:23 -05:00
ret_val = netlbl_unlhsh_remove_addr6 ( net ,
iface , addr , mask ,
audit_info ) ;
2008-01-29 08:44:21 -05:00
break ;
# endif /* IPv6 */
default :
ret_val = - EINVAL ;
}
if ( ret_val = = 0 ) {
netlbl_unlhsh_condremove_iface ( iface ) ;
atomic_dec ( & netlabel_mgmt_protocount ) ;
}
unlhsh_remove_return :
rcu_read_unlock ( ) ;
return ret_val ;
}
/*
* General Helper Functions
*/
/**
* netlbl_unlhsh_netdev_handler - Network device notification handler
* @ this : notifier block
* @ event : the event
* @ ptr : the network device ( cast to void )
*
* Description :
* Handle network device events , although at present all we care about is a
* network device going away . In the case of a device going away we clear any
* related entries from the unlabeled connection hash table .
*
*/
static int netlbl_unlhsh_netdev_handler ( struct notifier_block * this ,
unsigned long event ,
void * ptr )
{
struct net_device * dev = ptr ;
struct netlbl_unlhsh_iface * iface = NULL ;
2008-07-19 22:34:43 -07:00
if ( ! net_eq ( dev_net ( dev ) , & init_net ) )
2008-01-29 08:44:21 -05:00
return NOTIFY_DONE ;
/* XXX - should this be a check for NETDEV_DOWN or _UNREGISTER? */
if ( event = = NETDEV_DOWN ) {
spin_lock ( & netlbl_unlhsh_lock ) ;
iface = netlbl_unlhsh_search_iface ( dev - > ifindex ) ;
if ( iface ! = NULL & & iface - > valid ) {
iface - > valid = 0 ;
list_del_rcu ( & iface - > list ) ;
} else
iface = NULL ;
spin_unlock ( & netlbl_unlhsh_lock ) ;
}
if ( iface ! = NULL )
call_rcu ( & iface - > rcu , netlbl_unlhsh_free_iface ) ;
return NOTIFY_DONE ;
}
2006-09-28 14:51:47 -07:00
/**
* netlbl_unlabel_acceptflg_set - Set the unlabeled accept flag
* @ value : desired value
2006-09-29 17:05:05 -07:00
* @ audit_info : NetLabel audit information
2006-09-28 14:51:47 -07:00
*
* Description :
* Set the value of the unlabeled accept flag to @ value .
*
*/
2006-09-29 17:05:05 -07:00
static void netlbl_unlabel_acceptflg_set ( u8 value ,
struct netlbl_audit * audit_info )
2006-09-28 14:51:47 -07:00
{
2006-09-29 17:05:05 -07:00
struct audit_buffer * audit_buf ;
u8 old_val ;
2007-10-26 04:29:08 -07:00
old_val = netlabel_unlabel_acceptflg ;
2006-11-17 17:38:44 -05:00
netlabel_unlabel_acceptflg = value ;
2006-09-29 17:05:05 -07:00
audit_buf = netlbl_audit_start_common ( AUDIT_MAC_UNLBL_ALLOW ,
audit_info ) ;
2006-11-17 17:38:55 -05:00
if ( audit_buf ! = NULL ) {
audit_log_format ( audit_buf ,
" unlbl_accept=%u old=%u " , value , old_val ) ;
audit_log_end ( audit_buf ) ;
}
2006-09-28 14:51:47 -07:00
}
2008-01-29 08:44:21 -05:00
/**
* netlbl_unlabel_addrinfo_get - Get the IPv4 / 6 address information
* @ info : the Generic NETLINK info block
* @ addr : the IP address
* @ mask : the IP address mask
* @ len : the address length
*
* Description :
* Examine the Generic NETLINK message and extract the IP address information .
* Returns zero on success , negative values on failure .
*
*/
static int netlbl_unlabel_addrinfo_get ( struct genl_info * info ,
void * * addr ,
void * * mask ,
u32 * len )
{
u32 addr_len ;
if ( info - > attrs [ NLBL_UNLABEL_A_IPV4ADDR ] ) {
addr_len = nla_len ( info - > attrs [ NLBL_UNLABEL_A_IPV4ADDR ] ) ;
if ( addr_len ! = sizeof ( struct in_addr ) & &
addr_len ! = nla_len ( info - > attrs [ NLBL_UNLABEL_A_IPV4MASK ] ) )
return - EINVAL ;
* len = addr_len ;
* addr = nla_data ( info - > attrs [ NLBL_UNLABEL_A_IPV4ADDR ] ) ;
* mask = nla_data ( info - > attrs [ NLBL_UNLABEL_A_IPV4MASK ] ) ;
return 0 ;
} else if ( info - > attrs [ NLBL_UNLABEL_A_IPV6ADDR ] ) {
addr_len = nla_len ( info - > attrs [ NLBL_UNLABEL_A_IPV6ADDR ] ) ;
if ( addr_len ! = sizeof ( struct in6_addr ) & &
addr_len ! = nla_len ( info - > attrs [ NLBL_UNLABEL_A_IPV6MASK ] ) )
return - EINVAL ;
* len = addr_len ;
* addr = nla_data ( info - > attrs [ NLBL_UNLABEL_A_IPV6ADDR ] ) ;
* mask = nla_data ( info - > attrs [ NLBL_UNLABEL_A_IPV6MASK ] ) ;
return 0 ;
}
return - EINVAL ;
}
2006-08-03 16:48:59 -07:00
/*
* NetLabel Command Handlers
*/
/**
* netlbl_unlabel_accept - Handle an ACCEPT message
* @ skb : the NETLINK buffer
* @ info : the Generic NETLINK info block
*
* Description :
* Process a user generated ACCEPT message and set the accept flag accordingly .
* Returns zero on success , negative values on failure .
*
*/
static int netlbl_unlabel_accept ( struct sk_buff * skb , struct genl_info * info )
{
2006-09-25 15:56:37 -07:00
u8 value ;
2006-09-29 17:05:05 -07:00
struct netlbl_audit audit_info ;
2006-08-03 16:48:59 -07:00
2006-09-25 15:56:37 -07:00
if ( info - > attrs [ NLBL_UNLABEL_A_ACPTFLG ] ) {
value = nla_get_u8 ( info - > attrs [ NLBL_UNLABEL_A_ACPTFLG ] ) ;
2006-08-03 16:48:59 -07:00
if ( value = = 1 | | value = = 0 ) {
2006-09-29 17:05:05 -07:00
netlbl_netlink_auditinfo ( skb , & audit_info ) ;
netlbl_unlabel_acceptflg_set ( value , & audit_info ) ;
2006-09-28 14:51:47 -07:00
return 0 ;
2006-08-03 16:48:59 -07:00
}
}
2006-09-28 14:51:47 -07:00
return - EINVAL ;
2006-08-03 16:48:59 -07:00
}
/**
* netlbl_unlabel_list - Handle a LIST message
* @ skb : the NETLINK buffer
* @ info : the Generic NETLINK info block
*
* Description :
* Process a user generated LIST message and respond with the current status .
* Returns zero on success , negative values on failure .
*
*/
static int netlbl_unlabel_list ( struct sk_buff * skb , struct genl_info * info )
{
2006-09-25 15:56:37 -07:00
int ret_val = - EINVAL ;
2006-08-03 16:48:59 -07:00
struct sk_buff * ans_skb ;
2006-09-25 15:56:37 -07:00
void * data ;
2006-08-03 16:48:59 -07:00
2006-11-10 14:10:15 -08:00
ans_skb = nlmsg_new ( NLMSG_DEFAULT_SIZE , GFP_KERNEL ) ;
2006-08-03 16:48:59 -07:00
if ( ans_skb = = NULL )
goto list_failure ;
2006-11-14 19:46:02 -08:00
data = genlmsg_put_reply ( ans_skb , info , & netlbl_unlabel_gnl_family ,
0 , NLBL_UNLABEL_C_LIST ) ;
2006-09-25 15:56:37 -07:00
if ( data = = NULL ) {
ret_val = - ENOMEM ;
2006-08-03 16:48:59 -07:00
goto list_failure ;
2006-09-25 15:56:37 -07:00
}
2006-08-03 16:48:59 -07:00
2006-09-25 15:56:37 -07:00
ret_val = nla_put_u8 ( ans_skb ,
NLBL_UNLABEL_A_ACPTFLG ,
2006-11-17 17:38:44 -05:00
netlabel_unlabel_acceptflg ) ;
2006-08-03 16:48:59 -07:00
if ( ret_val ! = 0 )
goto list_failure ;
2006-09-25 15:56:37 -07:00
genlmsg_end ( ans_skb , data ) ;
2008-07-10 16:53:39 -07:00
return genlmsg_reply ( ans_skb , info ) ;
2006-08-03 16:48:59 -07:00
list_failure :
2007-02-27 09:57:37 -08:00
kfree_skb ( ans_skb ) ;
2006-08-03 16:48:59 -07:00
return ret_val ;
}
2008-01-29 08:44:21 -05:00
/**
* netlbl_unlabel_staticadd - Handle a STATICADD message
* @ skb : the NETLINK buffer
* @ info : the Generic NETLINK info block
*
* Description :
* Process a user generated STATICADD message and add a new unlabeled
* connection entry to the hash table . Returns zero on success , negative
* values on failure .
*
*/
static int netlbl_unlabel_staticadd ( struct sk_buff * skb ,
struct genl_info * info )
{
int ret_val ;
char * dev_name ;
void * addr ;
void * mask ;
u32 addr_len ;
u32 secid ;
2008-01-29 08:44:23 -05:00
struct netlbl_audit audit_info ;
2008-01-29 08:44:21 -05:00
/* Don't allow users to add both IPv4 and IPv6 addresses for a
* single entry . However , allow users to create two entries , one each
* for IPv4 and IPv4 , with the same LSM security context which should
* achieve the same result . */
if ( ! info - > attrs [ NLBL_UNLABEL_A_SECCTX ] | |
! info - > attrs [ NLBL_UNLABEL_A_IFACE ] | |
! ( ( ! info - > attrs [ NLBL_UNLABEL_A_IPV4ADDR ] | |
! info - > attrs [ NLBL_UNLABEL_A_IPV4MASK ] ) ^
( ! info - > attrs [ NLBL_UNLABEL_A_IPV6ADDR ] | |
! info - > attrs [ NLBL_UNLABEL_A_IPV6MASK ] ) ) )
return - EINVAL ;
2008-01-29 08:44:23 -05:00
netlbl_netlink_auditinfo ( skb , & audit_info ) ;
2008-01-29 08:44:21 -05:00
ret_val = netlbl_unlabel_addrinfo_get ( info , & addr , & mask , & addr_len ) ;
if ( ret_val ! = 0 )
return ret_val ;
dev_name = nla_data ( info - > attrs [ NLBL_UNLABEL_A_IFACE ] ) ;
ret_val = security_secctx_to_secid (
nla_data ( info - > attrs [ NLBL_UNLABEL_A_SECCTX ] ) ,
nla_len ( info - > attrs [ NLBL_UNLABEL_A_SECCTX ] ) ,
& secid ) ;
if ( ret_val ! = 0 )
return ret_val ;
return netlbl_unlhsh_add ( & init_net ,
2008-01-29 08:44:23 -05:00
dev_name , addr , mask , addr_len , secid ,
& audit_info ) ;
2008-01-29 08:44:21 -05:00
}
/**
* netlbl_unlabel_staticadddef - Handle a STATICADDDEF message
* @ skb : the NETLINK buffer
* @ info : the Generic NETLINK info block
*
* Description :
* Process a user generated STATICADDDEF message and add a new default
* unlabeled connection entry . Returns zero on success , negative values on
* failure .
*
*/
static int netlbl_unlabel_staticadddef ( struct sk_buff * skb ,
struct genl_info * info )
{
int ret_val ;
void * addr ;
void * mask ;
u32 addr_len ;
u32 secid ;
2008-01-29 08:44:23 -05:00
struct netlbl_audit audit_info ;
2008-01-29 08:44:21 -05:00
/* Don't allow users to add both IPv4 and IPv6 addresses for a
* single entry . However , allow users to create two entries , one each
* for IPv4 and IPv6 , with the same LSM security context which should
* achieve the same result . */
if ( ! info - > attrs [ NLBL_UNLABEL_A_SECCTX ] | |
! ( ( ! info - > attrs [ NLBL_UNLABEL_A_IPV4ADDR ] | |
! info - > attrs [ NLBL_UNLABEL_A_IPV4MASK ] ) ^
( ! info - > attrs [ NLBL_UNLABEL_A_IPV6ADDR ] | |
! info - > attrs [ NLBL_UNLABEL_A_IPV6MASK ] ) ) )
return - EINVAL ;
2008-01-29 08:44:23 -05:00
netlbl_netlink_auditinfo ( skb , & audit_info ) ;
2008-01-29 08:44:21 -05:00
ret_val = netlbl_unlabel_addrinfo_get ( info , & addr , & mask , & addr_len ) ;
if ( ret_val ! = 0 )
return ret_val ;
ret_val = security_secctx_to_secid (
nla_data ( info - > attrs [ NLBL_UNLABEL_A_SECCTX ] ) ,
nla_len ( info - > attrs [ NLBL_UNLABEL_A_SECCTX ] ) ,
& secid ) ;
if ( ret_val ! = 0 )
return ret_val ;
2008-01-29 08:44:23 -05:00
return netlbl_unlhsh_add ( & init_net ,
NULL , addr , mask , addr_len , secid ,
& audit_info ) ;
2008-01-29 08:44:21 -05:00
}
/**
* netlbl_unlabel_staticremove - Handle a STATICREMOVE message
* @ skb : the NETLINK buffer
* @ info : the Generic NETLINK info block
*
* Description :
* Process a user generated STATICREMOVE message and remove the specified
* unlabeled connection entry . Returns zero on success , negative values on
* failure .
*
*/
static int netlbl_unlabel_staticremove ( struct sk_buff * skb ,
struct genl_info * info )
{
int ret_val ;
char * dev_name ;
void * addr ;
void * mask ;
u32 addr_len ;
2008-01-29 08:44:23 -05:00
struct netlbl_audit audit_info ;
2008-01-29 08:44:21 -05:00
/* See the note in netlbl_unlabel_staticadd() about not allowing both
* IPv4 and IPv6 in the same entry . */
if ( ! info - > attrs [ NLBL_UNLABEL_A_IFACE ] | |
! ( ( ! info - > attrs [ NLBL_UNLABEL_A_IPV4ADDR ] | |
! info - > attrs [ NLBL_UNLABEL_A_IPV4MASK ] ) ^
( ! info - > attrs [ NLBL_UNLABEL_A_IPV6ADDR ] | |
! info - > attrs [ NLBL_UNLABEL_A_IPV6MASK ] ) ) )
return - EINVAL ;
2008-01-29 08:44:23 -05:00
netlbl_netlink_auditinfo ( skb , & audit_info ) ;
2008-01-29 08:44:21 -05:00
ret_val = netlbl_unlabel_addrinfo_get ( info , & addr , & mask , & addr_len ) ;
if ( ret_val ! = 0 )
return ret_val ;
dev_name = nla_data ( info - > attrs [ NLBL_UNLABEL_A_IFACE ] ) ;
2008-01-29 08:44:23 -05:00
return netlbl_unlhsh_remove ( & init_net ,
dev_name , addr , mask , addr_len ,
& audit_info ) ;
2008-01-29 08:44:21 -05:00
}
/**
* netlbl_unlabel_staticremovedef - Handle a STATICREMOVEDEF message
* @ skb : the NETLINK buffer
* @ info : the Generic NETLINK info block
*
* Description :
* Process a user generated STATICREMOVEDEF message and remove the default
* unlabeled connection entry . Returns zero on success , negative values on
* failure .
*
*/
static int netlbl_unlabel_staticremovedef ( struct sk_buff * skb ,
struct genl_info * info )
{
int ret_val ;
void * addr ;
void * mask ;
u32 addr_len ;
2008-01-29 08:44:23 -05:00
struct netlbl_audit audit_info ;
2008-01-29 08:44:21 -05:00
/* See the note in netlbl_unlabel_staticadd() about not allowing both
* IPv4 and IPv6 in the same entry . */
if ( ! ( ( ! info - > attrs [ NLBL_UNLABEL_A_IPV4ADDR ] | |
! info - > attrs [ NLBL_UNLABEL_A_IPV4MASK ] ) ^
( ! info - > attrs [ NLBL_UNLABEL_A_IPV6ADDR ] | |
! info - > attrs [ NLBL_UNLABEL_A_IPV6MASK ] ) ) )
return - EINVAL ;
2008-01-29 08:44:23 -05:00
netlbl_netlink_auditinfo ( skb , & audit_info ) ;
2008-01-29 08:44:21 -05:00
ret_val = netlbl_unlabel_addrinfo_get ( info , & addr , & mask , & addr_len ) ;
if ( ret_val ! = 0 )
return ret_val ;
2008-01-29 08:44:23 -05:00
return netlbl_unlhsh_remove ( & init_net ,
NULL , addr , mask , addr_len ,
& audit_info ) ;
2008-01-29 08:44:21 -05:00
}
/**
* netlbl_unlabel_staticlist_gen - Generate messages for STATICLIST [ DEF ]
* @ cmd : command / message
* @ iface : the interface entry
* @ addr4 : the IPv4 address entry
* @ addr6 : the IPv6 address entry
* @ arg : the netlbl_unlhsh_walk_arg structure
*
* Description :
* This function is designed to be used to generate a response for a
* STATICLIST or STATICLISTDEF message . When called either @ addr4 or @ addr6
* can be specified , not both , the other unspecified entry should be set to
* NULL by the caller . Returns the size of the message on success , negative
* values on failure .
*
*/
static int netlbl_unlabel_staticlist_gen ( u32 cmd ,
const struct netlbl_unlhsh_iface * iface ,
const struct netlbl_unlhsh_addr4 * addr4 ,
const struct netlbl_unlhsh_addr6 * addr6 ,
void * arg )
{
int ret_val = - ENOMEM ;
struct netlbl_unlhsh_walk_arg * cb_arg = arg ;
struct net_device * dev ;
void * data ;
u32 secid ;
char * secctx ;
u32 secctx_len ;
data = genlmsg_put ( cb_arg - > skb , NETLINK_CB ( cb_arg - > nl_cb - > skb ) . pid ,
cb_arg - > seq , & netlbl_unlabel_gnl_family ,
NLM_F_MULTI , cmd ) ;
if ( data = = NULL )
goto list_cb_failure ;
if ( iface - > ifindex > 0 ) {
dev = dev_get_by_index ( & init_net , iface - > ifindex ) ;
2008-04-17 23:22:54 -07:00
if ( ! dev ) {
ret_val = - ENODEV ;
goto list_cb_failure ;
}
2008-01-29 08:44:21 -05:00
ret_val = nla_put_string ( cb_arg - > skb ,
NLBL_UNLABEL_A_IFACE , dev - > name ) ;
dev_put ( dev ) ;
if ( ret_val ! = 0 )
goto list_cb_failure ;
}
if ( addr4 ) {
struct in_addr addr_struct ;
2008-10-10 10:16:32 -04:00
addr_struct . s_addr = addr4 - > list . addr ;
2008-01-29 08:44:21 -05:00
ret_val = nla_put ( cb_arg - > skb ,
NLBL_UNLABEL_A_IPV4ADDR ,
sizeof ( struct in_addr ) ,
& addr_struct ) ;
if ( ret_val ! = 0 )
goto list_cb_failure ;
2008-10-10 10:16:32 -04:00
addr_struct . s_addr = addr4 - > list . mask ;
2008-01-29 08:44:21 -05:00
ret_val = nla_put ( cb_arg - > skb ,
NLBL_UNLABEL_A_IPV4MASK ,
sizeof ( struct in_addr ) ,
& addr_struct ) ;
if ( ret_val ! = 0 )
goto list_cb_failure ;
secid = addr4 - > secid ;
} else {
ret_val = nla_put ( cb_arg - > skb ,
NLBL_UNLABEL_A_IPV6ADDR ,
sizeof ( struct in6_addr ) ,
2008-10-10 10:16:32 -04:00
& addr6 - > list . addr ) ;
2008-01-29 08:44:21 -05:00
if ( ret_val ! = 0 )
goto list_cb_failure ;
ret_val = nla_put ( cb_arg - > skb ,
NLBL_UNLABEL_A_IPV6MASK ,
sizeof ( struct in6_addr ) ,
2008-10-10 10:16:32 -04:00
& addr6 - > list . mask ) ;
2008-01-29 08:44:21 -05:00
if ( ret_val ! = 0 )
goto list_cb_failure ;
secid = addr6 - > secid ;
}
ret_val = security_secid_to_secctx ( secid , & secctx , & secctx_len ) ;
if ( ret_val ! = 0 )
goto list_cb_failure ;
ret_val = nla_put ( cb_arg - > skb ,
NLBL_UNLABEL_A_SECCTX ,
secctx_len ,
secctx ) ;
security_release_secctx ( secctx , secctx_len ) ;
if ( ret_val ! = 0 )
goto list_cb_failure ;
cb_arg - > seq + + ;
return genlmsg_end ( cb_arg - > skb , data ) ;
list_cb_failure :
genlmsg_cancel ( cb_arg - > skb , data ) ;
return ret_val ;
}
/**
* netlbl_unlabel_staticlist - Handle a STATICLIST message
* @ skb : the NETLINK buffer
* @ cb : the NETLINK callback
*
* Description :
* Process a user generated STATICLIST message and dump the unlabeled
* connection hash table in a form suitable for use in a kernel generated
* STATICLIST message . Returns the length of @ skb .
*
*/
static int netlbl_unlabel_staticlist ( struct sk_buff * skb ,
struct netlink_callback * cb )
{
struct netlbl_unlhsh_walk_arg cb_arg ;
u32 skip_bkt = cb - > args [ 0 ] ;
u32 skip_chain = cb - > args [ 1 ] ;
u32 skip_addr4 = cb - > args [ 2 ] ;
u32 skip_addr6 = cb - > args [ 3 ] ;
u32 iter_bkt ;
u32 iter_chain = 0 , iter_addr4 = 0 , iter_addr6 = 0 ;
struct netlbl_unlhsh_iface * iface ;
2008-10-10 10:16:29 -04:00
struct list_head * iter_list ;
2008-10-10 10:16:32 -04:00
struct netlbl_af4list * addr4 ;
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
struct netlbl_af6list * addr6 ;
# endif
2008-01-29 08:44:21 -05:00
cb_arg . nl_cb = cb ;
cb_arg . skb = skb ;
cb_arg . seq = cb - > nlh - > nlmsg_seq ;
rcu_read_lock ( ) ;
for ( iter_bkt = skip_bkt ;
iter_bkt < rcu_dereference ( netlbl_unlhsh ) - > size ;
iter_bkt + + , iter_chain = 0 , iter_addr4 = 0 , iter_addr6 = 0 ) {
2008-10-10 10:16:29 -04:00
iter_list = & rcu_dereference ( netlbl_unlhsh ) - > tbl [ iter_bkt ] ;
list_for_each_entry_rcu ( iface , iter_list , list ) {
2008-01-29 08:44:21 -05:00
if ( ! iface - > valid | |
iter_chain + + < skip_chain )
continue ;
2008-10-10 10:16:32 -04:00
netlbl_af4list_foreach_rcu ( addr4 ,
& iface - > addr4_list ) {
if ( iter_addr4 + + < skip_addr4 )
2008-01-29 08:44:21 -05:00
continue ;
if ( netlbl_unlabel_staticlist_gen (
2008-10-10 10:16:32 -04:00
NLBL_UNLABEL_C_STATICLIST ,
iface ,
netlbl_unlhsh_addr4_entry ( addr4 ) ,
NULL ,
& cb_arg ) < 0 ) {
2008-01-29 08:44:21 -05:00
iter_addr4 - - ;
iter_chain - - ;
goto unlabel_staticlist_return ;
}
}
2008-10-10 10:16:32 -04:00
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
netlbl_af6list_foreach_rcu ( addr6 ,
& iface - > addr6_list ) {
if ( iter_addr6 + + < skip_addr6 )
2008-01-29 08:44:21 -05:00
continue ;
if ( netlbl_unlabel_staticlist_gen (
2008-10-10 10:16:32 -04:00
NLBL_UNLABEL_C_STATICLIST ,
iface ,
NULL ,
netlbl_unlhsh_addr6_entry ( addr6 ) ,
& cb_arg ) < 0 ) {
2008-01-29 08:44:21 -05:00
iter_addr6 - - ;
iter_chain - - ;
goto unlabel_staticlist_return ;
}
}
2008-10-10 10:16:32 -04:00
# endif /* IPv6 */
2008-01-29 08:44:21 -05:00
}
}
unlabel_staticlist_return :
rcu_read_unlock ( ) ;
cb - > args [ 0 ] = skip_bkt ;
cb - > args [ 1 ] = skip_chain ;
cb - > args [ 2 ] = skip_addr4 ;
cb - > args [ 3 ] = skip_addr6 ;
return skb - > len ;
}
/**
* netlbl_unlabel_staticlistdef - Handle a STATICLISTDEF message
* @ skb : the NETLINK buffer
* @ cb : the NETLINK callback
*
* Description :
* Process a user generated STATICLISTDEF message and dump the default
* unlabeled connection entry in a form suitable for use in a kernel generated
* STATICLISTDEF message . Returns the length of @ skb .
*
*/
static int netlbl_unlabel_staticlistdef ( struct sk_buff * skb ,
struct netlink_callback * cb )
{
struct netlbl_unlhsh_walk_arg cb_arg ;
struct netlbl_unlhsh_iface * iface ;
u32 skip_addr4 = cb - > args [ 0 ] ;
u32 skip_addr6 = cb - > args [ 1 ] ;
2008-10-10 10:16:32 -04:00
u32 iter_addr4 = 0 ;
struct netlbl_af4list * addr4 ;
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
u32 iter_addr6 = 0 ;
struct netlbl_af6list * addr6 ;
# endif
2008-01-29 08:44:21 -05:00
cb_arg . nl_cb = cb ;
cb_arg . skb = skb ;
cb_arg . seq = cb - > nlh - > nlmsg_seq ;
rcu_read_lock ( ) ;
iface = rcu_dereference ( netlbl_unlhsh_def ) ;
if ( iface = = NULL | | ! iface - > valid )
goto unlabel_staticlistdef_return ;
2008-10-10 10:16:32 -04:00
netlbl_af4list_foreach_rcu ( addr4 , & iface - > addr4_list ) {
if ( iter_addr4 + + < skip_addr4 )
2008-01-29 08:44:21 -05:00
continue ;
if ( netlbl_unlabel_staticlist_gen ( NLBL_UNLABEL_C_STATICLISTDEF ,
2008-10-10 10:16:32 -04:00
iface ,
netlbl_unlhsh_addr4_entry ( addr4 ) ,
NULL ,
& cb_arg ) < 0 ) {
2008-01-29 08:44:21 -05:00
iter_addr4 - - ;
goto unlabel_staticlistdef_return ;
}
}
2008-10-10 10:16:32 -04:00
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
netlbl_af6list_foreach_rcu ( addr6 , & iface - > addr6_list ) {
if ( iter_addr6 + + < skip_addr6 )
2008-01-29 08:44:21 -05:00
continue ;
if ( netlbl_unlabel_staticlist_gen ( NLBL_UNLABEL_C_STATICLISTDEF ,
2008-10-10 10:16:32 -04:00
iface ,
NULL ,
netlbl_unlhsh_addr6_entry ( addr6 ) ,
& cb_arg ) < 0 ) {
2008-01-29 08:44:21 -05:00
iter_addr6 - - ;
goto unlabel_staticlistdef_return ;
}
}
2008-10-10 10:16:32 -04:00
# endif /* IPv6 */
2008-01-29 08:44:21 -05:00
unlabel_staticlistdef_return :
rcu_read_unlock ( ) ;
cb - > args [ 0 ] = skip_addr4 ;
cb - > args [ 1 ] = skip_addr6 ;
return skb - > len ;
}
2006-08-03 16:48:59 -07:00
/*
* NetLabel Generic NETLINK Command Definitions
*/
2008-02-17 22:33:16 -08:00
static struct genl_ops netlbl_unlabel_genl_ops [ ] = {
{
2008-01-29 08:44:21 -05:00
. cmd = NLBL_UNLABEL_C_STATICADD ,
. flags = GENL_ADMIN_PERM ,
. policy = netlbl_unlabel_genl_policy ,
. doit = netlbl_unlabel_staticadd ,
. dumpit = NULL ,
2008-02-17 22:33:16 -08:00
} ,
{
2008-01-29 08:44:21 -05:00
. cmd = NLBL_UNLABEL_C_STATICREMOVE ,
. flags = GENL_ADMIN_PERM ,
. policy = netlbl_unlabel_genl_policy ,
. doit = netlbl_unlabel_staticremove ,
. dumpit = NULL ,
2008-02-17 22:33:16 -08:00
} ,
{
2008-01-29 08:44:21 -05:00
. cmd = NLBL_UNLABEL_C_STATICLIST ,
. flags = 0 ,
. policy = netlbl_unlabel_genl_policy ,
. doit = NULL ,
. dumpit = netlbl_unlabel_staticlist ,
2008-02-17 22:33:16 -08:00
} ,
{
2008-01-29 08:44:21 -05:00
. cmd = NLBL_UNLABEL_C_STATICADDDEF ,
. flags = GENL_ADMIN_PERM ,
. policy = netlbl_unlabel_genl_policy ,
. doit = netlbl_unlabel_staticadddef ,
. dumpit = NULL ,
2008-02-17 22:33:16 -08:00
} ,
{
2008-01-29 08:44:21 -05:00
. cmd = NLBL_UNLABEL_C_STATICREMOVEDEF ,
. flags = GENL_ADMIN_PERM ,
. policy = netlbl_unlabel_genl_policy ,
. doit = netlbl_unlabel_staticremovedef ,
. dumpit = NULL ,
2008-02-17 22:33:16 -08:00
} ,
{
2008-01-29 08:44:21 -05:00
. cmd = NLBL_UNLABEL_C_STATICLISTDEF ,
. flags = 0 ,
. policy = netlbl_unlabel_genl_policy ,
. doit = NULL ,
. dumpit = netlbl_unlabel_staticlistdef ,
2008-02-17 22:33:16 -08:00
} ,
{
2006-08-03 16:48:59 -07:00
. cmd = NLBL_UNLABEL_C_ACCEPT ,
2006-09-25 15:56:37 -07:00
. flags = GENL_ADMIN_PERM ,
. policy = netlbl_unlabel_genl_policy ,
2006-08-03 16:48:59 -07:00
. doit = netlbl_unlabel_accept ,
. dumpit = NULL ,
2008-02-17 22:33:16 -08:00
} ,
{
2006-08-03 16:48:59 -07:00
. cmd = NLBL_UNLABEL_C_LIST ,
. flags = 0 ,
2006-09-25 15:56:37 -07:00
. policy = netlbl_unlabel_genl_policy ,
2006-08-03 16:48:59 -07:00
. doit = netlbl_unlabel_list ,
. dumpit = NULL ,
2008-02-17 22:33:16 -08:00
} ,
2006-08-03 16:48:59 -07:00
} ;
/*
* NetLabel Generic NETLINK Protocol Functions
*/
/**
* netlbl_unlabel_genl_init - Register the Unlabeled NetLabel component
*
* Description :
* Register the unlabeled packet NetLabel component with the Generic NETLINK
* mechanism . Returns zero on success , negative values on failure .
*
*/
2008-02-17 22:33:57 -08:00
int __init netlbl_unlabel_genl_init ( void )
2006-08-03 16:48:59 -07:00
{
2009-05-21 10:34:05 +00:00
return genl_register_family_with_ops ( & netlbl_unlabel_gnl_family ,
netlbl_unlabel_genl_ops , ARRAY_SIZE ( netlbl_unlabel_genl_ops ) ) ;
2006-08-03 16:48:59 -07:00
}
/*
* NetLabel KAPI Hooks
*/
2008-01-29 08:44:21 -05:00
static struct notifier_block netlbl_unlhsh_netdev_notifier = {
. notifier_call = netlbl_unlhsh_netdev_handler ,
} ;
/**
* netlbl_unlabel_init - Initialize the unlabeled connection hash table
* @ size : the number of bits to use for the hash buckets
*
* Description :
* Initializes the unlabeled connection hash table and registers a network
* device notification handler . This function should only be called by the
* NetLabel subsystem itself during initialization . Returns zero on success ,
* non - zero values on error .
*
*/
2008-02-17 22:33:57 -08:00
int __init netlbl_unlabel_init ( u32 size )
2008-01-29 08:44:21 -05:00
{
u32 iter ;
struct netlbl_unlhsh_tbl * hsh_tbl ;
if ( size = = 0 )
return - EINVAL ;
hsh_tbl = kmalloc ( sizeof ( * hsh_tbl ) , GFP_KERNEL ) ;
if ( hsh_tbl = = NULL )
return - ENOMEM ;
hsh_tbl - > size = 1 < < size ;
hsh_tbl - > tbl = kcalloc ( hsh_tbl - > size ,
sizeof ( struct list_head ) ,
GFP_KERNEL ) ;
if ( hsh_tbl - > tbl = = NULL ) {
kfree ( hsh_tbl ) ;
return - ENOMEM ;
}
for ( iter = 0 ; iter < hsh_tbl - > size ; iter + + )
INIT_LIST_HEAD ( & hsh_tbl - > tbl [ iter ] ) ;
rcu_read_lock ( ) ;
spin_lock ( & netlbl_unlhsh_lock ) ;
rcu_assign_pointer ( netlbl_unlhsh , hsh_tbl ) ;
spin_unlock ( & netlbl_unlhsh_lock ) ;
rcu_read_unlock ( ) ;
register_netdevice_notifier ( & netlbl_unlhsh_netdev_notifier ) ;
return 0 ;
}
2006-08-03 16:48:59 -07:00
/**
* netlbl_unlabel_getattr - Get the security attributes for an unlabled packet
2008-01-29 08:44:21 -05:00
* @ skb : the packet
* @ family : protocol family
2006-08-03 16:48:59 -07:00
* @ secattr : the security attributes
*
* Description :
* Determine the security attributes , if any , for an unlabled packet and return
* them in @ secattr . Returns zero on success and negative values on failure .
*
*/
2008-01-29 08:44:21 -05:00
int netlbl_unlabel_getattr ( const struct sk_buff * skb ,
u16 family ,
struct netlbl_lsm_secattr * secattr )
2006-08-03 16:48:59 -07:00
{
2008-01-29 08:44:21 -05:00
struct netlbl_unlhsh_iface * iface ;
rcu_read_lock ( ) ;
iface = netlbl_unlhsh_search_iface_def ( skb - > iif ) ;
if ( iface = = NULL )
goto unlabel_getattr_nolabel ;
switch ( family ) {
2008-02-12 22:37:19 -08:00
case PF_INET : {
struct iphdr * hdr4 ;
2008-10-10 10:16:32 -04:00
struct netlbl_af4list * addr4 ;
2008-02-12 22:37:19 -08:00
2008-01-29 08:44:21 -05:00
hdr4 = ip_hdr ( skb ) ;
2008-10-10 10:16:32 -04:00
addr4 = netlbl_af4list_search ( hdr4 - > saddr ,
& iface - > addr4_list ) ;
2008-01-29 08:44:21 -05:00
if ( addr4 = = NULL )
goto unlabel_getattr_nolabel ;
2008-10-10 10:16:32 -04:00
secattr - > attr . secid = netlbl_unlhsh_addr4_entry ( addr4 ) - > secid ;
2008-01-29 08:44:21 -05:00
break ;
2008-02-12 22:37:19 -08:00
}
2008-01-29 08:44:21 -05:00
# if defined(CONFIG_IPV6) || defined(CONFIG_IPV6_MODULE)
2008-02-12 22:37:19 -08:00
case PF_INET6 : {
struct ipv6hdr * hdr6 ;
2008-10-10 10:16:32 -04:00
struct netlbl_af6list * addr6 ;
2008-02-12 22:37:19 -08:00
2008-01-29 08:44:21 -05:00
hdr6 = ipv6_hdr ( skb ) ;
2008-10-10 10:16:32 -04:00
addr6 = netlbl_af6list_search ( & hdr6 - > saddr ,
& iface - > addr6_list ) ;
2008-01-29 08:44:21 -05:00
if ( addr6 = = NULL )
goto unlabel_getattr_nolabel ;
2008-10-10 10:16:32 -04:00
secattr - > attr . secid = netlbl_unlhsh_addr6_entry ( addr6 ) - > secid ;
2008-01-29 08:44:21 -05:00
break ;
2008-02-12 22:37:19 -08:00
}
2008-01-29 08:44:21 -05:00
# endif /* IPv6 */
default :
goto unlabel_getattr_nolabel ;
}
rcu_read_unlock ( ) ;
secattr - > flags | = NETLBL_SECATTR_SECID ;
secattr - > type = NETLBL_NLTYPE_UNLABELED ;
return 0 ;
unlabel_getattr_nolabel :
rcu_read_unlock ( ) ;
2008-01-29 08:37:52 -05:00
if ( netlabel_unlabel_acceptflg = = 0 )
return - ENOMSG ;
2008-01-29 08:37:59 -05:00
secattr - > type = NETLBL_NLTYPE_UNLABELED ;
2008-01-29 08:37:52 -05:00
return 0 ;
2006-08-03 16:48:59 -07:00
}
/**
* netlbl_unlabel_defconf - Set the default config to allow unlabeled packets
*
* Description :
* Set the default NetLabel configuration to allow incoming unlabeled packets
* and to send unlabeled network traffic by default .
*
*/
2008-02-17 22:33:57 -08:00
int __init netlbl_unlabel_defconf ( void )
2006-08-03 16:48:59 -07:00
{
int ret_val ;
struct netlbl_dom_map * entry ;
2006-09-29 17:05:05 -07:00
struct netlbl_audit audit_info ;
2006-09-28 14:51:47 -07:00
2006-09-29 17:05:05 -07:00
/* Only the kernel is allowed to call this function and the only time
* it is called is at bootup before the audit subsystem is reporting
* messages so don ' t worry to much about these values . */
security_task_getsecid ( current , & audit_info . secid ) ;
audit_info . loginuid = 0 ;
2008-04-18 10:09:25 -04:00
audit_info . sessionid = 0 ;
2006-08-03 16:48:59 -07:00
entry = kzalloc ( sizeof ( * entry ) , GFP_KERNEL ) ;
if ( entry = = NULL )
return - ENOMEM ;
entry - > type = NETLBL_NLTYPE_UNLABELED ;
2006-09-29 17:05:05 -07:00
ret_val = netlbl_domhsh_add_default ( entry , & audit_info ) ;
2006-08-03 16:48:59 -07:00
if ( ret_val ! = 0 )
return ret_val ;
2006-09-29 17:05:05 -07:00
netlbl_unlabel_acceptflg_set ( 1 , & audit_info ) ;
2006-08-03 16:48:59 -07:00
return 0 ;
}