2019-05-27 09:55:01 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2014-01-17 10:57:49 +04:00
/* Sysfs attributes of bond slaves
*
* Copyright ( c ) 2014 Scott Feldman < sfeldma @ cumulusnetworks . com >
*/
# include <linux/capability.h>
# include <linux/kernel.h>
# include <linux/netdevice.h>
2014-11-10 21:27:49 +03:00
# include <net/bonding.h>
2014-01-17 10:57:49 +04:00
struct slave_attribute {
struct attribute attr ;
ssize_t ( * show ) ( struct slave * , char * ) ;
} ;
# define SLAVE_ATTR(_name, _mode, _show) \
const struct slave_attribute slave_attr_ # # _name = { \
. attr = { . name = __stringify ( _name ) , \
. mode = _mode } , \
. show = _show , \
} ;
2018-03-24 01:54:39 +03:00
# define SLAVE_ATTR_RO(_name) \
SLAVE_ATTR ( _name , 0444 , _name # # _show )
2014-01-17 10:57:49 +04:00
static ssize_t state_show ( struct slave * slave , char * buf )
{
switch ( bond_slave_state ( slave ) ) {
case BOND_STATE_ACTIVE :
return sprintf ( buf , " active \n " ) ;
case BOND_STATE_BACKUP :
return sprintf ( buf , " backup \n " ) ;
default :
2014-08-06 17:31:28 +04:00
return sprintf ( buf , " UNKNOWN \n " ) ;
2014-01-17 10:57:49 +04:00
}
}
static SLAVE_ATTR_RO ( state ) ;
static ssize_t mii_status_show ( struct slave * slave , char * buf )
{
return sprintf ( buf , " %s \n " , bond_slave_link_status ( slave - > link ) ) ;
}
static SLAVE_ATTR_RO ( mii_status ) ;
static ssize_t link_failure_count_show ( struct slave * slave , char * buf )
{
return sprintf ( buf , " %d \n " , slave - > link_failure_count ) ;
}
static SLAVE_ATTR_RO ( link_failure_count ) ;
static ssize_t perm_hwaddr_show ( struct slave * slave , char * buf )
{
2019-03-28 13:29:21 +03:00
return sprintf ( buf , " %*phC \n " ,
slave - > dev - > addr_len ,
slave - > perm_hwaddr ) ;
2014-01-17 10:57:49 +04:00
}
static SLAVE_ATTR_RO ( perm_hwaddr ) ;
static ssize_t queue_id_show ( struct slave * slave , char * buf )
{
return sprintf ( buf , " %d \n " , slave - > queue_id ) ;
}
static SLAVE_ATTR_RO ( queue_id ) ;
static ssize_t ad_aggregator_id_show ( struct slave * slave , char * buf )
{
const struct aggregator * agg ;
2014-05-15 23:39:55 +04:00
if ( BOND_MODE ( slave - > bond ) = = BOND_MODE_8023AD ) {
2014-05-12 11:08:43 +04:00
agg = SLAVE_AD_INFO ( slave ) - > port . aggregator ;
2014-01-17 10:57:49 +04:00
if ( agg )
return sprintf ( buf , " %d \n " ,
agg - > aggregator_identifier ) ;
}
return sprintf ( buf , " N/A \n " ) ;
}
static SLAVE_ATTR_RO ( ad_aggregator_id ) ;
2015-06-14 16:36:34 +03:00
static ssize_t ad_actor_oper_port_state_show ( struct slave * slave , char * buf )
{
const struct port * ad_port ;
if ( BOND_MODE ( slave - > bond ) = = BOND_MODE_8023AD ) {
ad_port = & SLAVE_AD_INFO ( slave ) - > port ;
if ( ad_port - > aggregator )
return sprintf ( buf , " %u \n " ,
ad_port - > actor_oper_port_state ) ;
}
return sprintf ( buf , " N/A \n " ) ;
}
static SLAVE_ATTR_RO ( ad_actor_oper_port_state ) ;
2015-06-14 16:36:35 +03:00
static ssize_t ad_partner_oper_port_state_show ( struct slave * slave , char * buf )
{
const struct port * ad_port ;
if ( BOND_MODE ( slave - > bond ) = = BOND_MODE_8023AD ) {
ad_port = & SLAVE_AD_INFO ( slave ) - > port ;
if ( ad_port - > aggregator )
return sprintf ( buf , " %u \n " ,
ad_port - > partner_oper . port_state ) ;
}
return sprintf ( buf , " N/A \n " ) ;
}
static SLAVE_ATTR_RO ( ad_partner_oper_port_state ) ;
2014-01-17 10:57:49 +04:00
static const struct slave_attribute * slave_attrs [ ] = {
& slave_attr_state ,
& slave_attr_mii_status ,
& slave_attr_link_failure_count ,
& slave_attr_perm_hwaddr ,
& slave_attr_queue_id ,
& slave_attr_ad_aggregator_id ,
2015-06-14 16:36:34 +03:00
& slave_attr_ad_actor_oper_port_state ,
2015-06-14 16:36:35 +03:00
& slave_attr_ad_partner_oper_port_state ,
2014-01-17 10:57:49 +04:00
NULL
} ;
# define to_slave_attr(_at) container_of(_at, struct slave_attribute, attr)
# define to_slave(obj) container_of(obj, struct slave, kobj)
static ssize_t slave_show ( struct kobject * kobj ,
struct attribute * attr , char * buf )
{
struct slave_attribute * slave_attr = to_slave_attr ( attr ) ;
struct slave * slave = to_slave ( kobj ) ;
return slave_attr - > show ( slave , buf ) ;
}
2014-01-19 02:54:18 +04:00
static const struct sysfs_ops slave_sysfs_ops = {
2014-01-17 10:57:49 +04:00
. show = slave_show ,
} ;
static struct kobj_type slave_ktype = {
# ifdef CONFIG_SYSFS
. sysfs_ops = & slave_sysfs_ops ,
# endif
} ;
int bond_sysfs_slave_add ( struct slave * slave )
{
const struct slave_attribute * * a ;
int err ;
err = kobject_init_and_add ( & slave - > kobj , & slave_ktype ,
2014-01-22 12:05:53 +04:00
& ( slave - > dev - > dev . kobj ) , " bonding_slave " ) ;
2014-01-17 10:57:49 +04:00
if ( err )
return err ;
for ( a = slave_attrs ; * a ; + + a ) {
err = sysfs_create_file ( & slave - > kobj , & ( ( * a ) - > attr ) ) ;
if ( err ) {
2014-07-29 15:27:43 +04:00
kobject_put ( & slave - > kobj ) ;
2014-01-17 10:57:49 +04:00
return err ;
}
}
return 0 ;
}
void bond_sysfs_slave_del ( struct slave * slave )
{
const struct slave_attribute * * a ;
for ( a = slave_attrs ; * a ; + + a )
sysfs_remove_file ( & slave - > kobj , & ( ( * a ) - > attr ) ) ;
2014-07-29 15:27:43 +04:00
kobject_put ( & slave - > kobj ) ;
2014-01-17 10:57:49 +04:00
}