2007-04-23 12:20:05 -07:00
/*
* This file provides / sys / class / ieee80211 / < wiphy name > /
* and some default attributes .
*
* Copyright 2005 - 2006 Jiri Benc < jbenc @ suse . cz >
* Copyright 2006 Johannes Berg < johannes @ sipsolutions . net >
*
* This file is GPLv2 as found in COPYING .
*/
# include <linux/device.h>
# include <linux/module.h>
# include <linux/netdevice.h>
# include <linux/nl80211.h>
# include <linux/rtnetlink.h>
# include <net/cfg80211.h>
# include "sysfs.h"
# include "core.h"
2012-06-27 17:19:42 +03:00
# include "rdev-ops.h"
2007-04-23 12:20:05 -07:00
static inline struct cfg80211_registered_device * dev_to_rdev (
struct device * dev )
{
return container_of ( dev , struct cfg80211_registered_device , wiphy . dev ) ;
}
2008-10-28 14:00:54 +01:00
# define SHOW_FMT(name, fmt, member) \
static ssize_t name # # _show ( struct device * dev , \
struct device_attribute * attr , \
char * buf ) \
{ \
return sprintf ( buf , fmt " \n " , dev_to_rdev ( dev ) - > member ) ; \
2013-07-24 15:05:35 -07:00
} \
static DEVICE_ATTR_RO ( name )
2007-04-23 12:20:05 -07:00
2009-02-21 00:04:19 -05:00
SHOW_FMT ( index , " %d " , wiphy_idx ) ;
2008-10-28 14:00:54 +01:00
SHOW_FMT ( macaddress , " %pM " , wiphy . perm_addr ) ;
2010-01-20 12:02:33 +01:00
SHOW_FMT ( address_mask , " %pM " , wiphy . addr_mask ) ;
2010-10-11 10:28:59 -07:00
static ssize_t name_show ( struct device * dev ,
struct device_attribute * attr ,
char * buf ) {
struct wiphy * wiphy = & dev_to_rdev ( dev ) - > wiphy ;
return sprintf ( buf , " %s \n " , dev_name ( & wiphy - > dev ) ) ;
}
2013-07-24 15:05:35 -07:00
static DEVICE_ATTR_RO ( name ) ;
2010-10-11 10:28:59 -07:00
2010-01-20 12:02:33 +01:00
static ssize_t addresses_show ( struct device * dev ,
struct device_attribute * attr ,
char * buf )
{
struct wiphy * wiphy = & dev_to_rdev ( dev ) - > wiphy ;
char * start = buf ;
int i ;
if ( ! wiphy - > addresses )
return sprintf ( buf , " %pM \n " , wiphy - > perm_addr ) ;
for ( i = 0 ; i < wiphy - > n_addresses ; i + + )
buf + = sprintf ( buf , " %pM \n " , & wiphy - > addresses [ i ] . addr ) ;
return buf - start ;
}
2013-07-24 15:05:35 -07:00
static DEVICE_ATTR_RO ( addresses ) ;
static struct attribute * ieee80211_attrs [ ] = {
& dev_attr_index . attr ,
& dev_attr_macaddress . attr ,
& dev_attr_address_mask . attr ,
& dev_attr_addresses . attr ,
& dev_attr_name . attr ,
NULL ,
2007-04-23 12:20:05 -07:00
} ;
2013-07-24 15:05:35 -07:00
ATTRIBUTE_GROUPS ( ieee80211 ) ;
2007-04-23 12:20:05 -07:00
static void wiphy_dev_release ( struct device * dev )
{
struct cfg80211_registered_device * rdev = dev_to_rdev ( dev ) ;
cfg80211_dev_free ( rdev ) ;
}
2007-08-14 15:15:12 +02:00
static int wiphy_uevent ( struct device * dev , struct kobj_uevent_env * env )
2007-04-23 12:20:05 -07:00
{
/* TODO, we probably need stuff here */
return 0 ;
}
2015-05-16 14:35:54 +02:00
# ifdef CONFIG_PM_SLEEP
2013-02-28 10:55:25 +01:00
static void cfg80211_leave_all ( struct cfg80211_registered_device * rdev )
{
struct wireless_dev * wdev ;
2016-05-03 16:52:04 +03:00
list_for_each_entry ( wdev , & rdev - > wiphy . wdev_list , list )
2013-02-28 10:55:25 +01:00
cfg80211_leave ( rdev , wdev ) ;
}
2015-05-16 14:35:54 +02:00
static int wiphy_suspend ( struct device * dev )
2009-01-19 11:20:52 -05:00
{
struct cfg80211_registered_device * rdev = dev_to_rdev ( dev ) ;
int ret = 0 ;
2009-02-11 17:14:43 -05:00
rdev - > suspend_at = get_seconds ( ) ;
2013-02-28 10:55:25 +01:00
rtnl_lock ( ) ;
if ( rdev - > wiphy . registered ) {
2013-05-15 09:30:07 +02:00
if ( ! rdev - > wiphy . wowlan_config )
2013-02-28 10:55:25 +01:00
cfg80211_leave_all ( rdev ) ;
if ( rdev - > ops - > suspend )
2013-05-15 09:30:07 +02:00
ret = rdev_suspend ( rdev , rdev - > wiphy . wowlan_config ) ;
2013-02-28 10:55:25 +01:00
if ( ret = = 1 ) {
/* Driver refuse to configure wowlan */
cfg80211_leave_all ( rdev ) ;
ret = rdev_suspend ( rdev , NULL ) ;
}
2009-01-19 11:20:52 -05:00
}
2013-02-28 10:55:25 +01:00
rtnl_unlock ( ) ;
2009-01-19 11:20:52 -05:00
return ret ;
}
static int wiphy_resume ( struct device * dev )
{
struct cfg80211_registered_device * rdev = dev_to_rdev ( dev ) ;
int ret = 0 ;
2009-02-11 17:14:43 -05:00
/* Age scan results with time spent in suspend */
cfg80211_bss_age ( rdev , get_seconds ( ) - rdev - > suspend_at ) ;
2009-01-19 11:20:52 -05:00
if ( rdev - > ops - > resume ) {
rtnl_lock ( ) ;
2011-08-12 14:00:59 +02:00
if ( rdev - > wiphy . registered )
2012-06-27 17:19:42 +03:00
ret = rdev_resume ( rdev ) ;
2009-01-19 11:20:52 -05:00
rtnl_unlock ( ) ;
}
return ret ;
}
2015-05-16 14:35:54 +02:00
static SIMPLE_DEV_PM_OPS ( wiphy_pm_ops , wiphy_suspend , wiphy_resume ) ;
# define WIPHY_PM_OPS (&wiphy_pm_ops)
# else
# define WIPHY_PM_OPS NULL
2013-05-15 09:30:07 +02:00
# endif
2009-01-19 11:20:52 -05:00
2010-08-05 17:45:15 +02:00
static const void * wiphy_namespace ( struct device * d )
{
struct wiphy * wiphy = container_of ( d , struct wiphy , dev ) ;
return wiphy_net ( wiphy ) ;
}
2007-04-23 12:20:05 -07:00
struct class ieee80211_class = {
. name = " ieee80211 " ,
. owner = THIS_MODULE ,
. dev_release = wiphy_dev_release ,
2013-07-24 15:05:35 -07:00
. dev_groups = ieee80211_groups ,
2007-04-23 12:20:05 -07:00
. dev_uevent = wiphy_uevent ,
2015-05-16 14:35:54 +02:00
. pm = WIPHY_PM_OPS ,
2010-08-05 17:45:15 +02:00
. ns_type = & net_ns_type_operations ,
. namespace = wiphy_namespace ,
2007-04-23 12:20:05 -07:00
} ;
int wiphy_sysfs_init ( void )
{
return class_register ( & ieee80211_class ) ;
}
void wiphy_sysfs_exit ( void )
{
class_unregister ( & ieee80211_class ) ;
}