Linux驱动(六)----Wifi基础

6350阅读 0评论2017-12-08 qq526665621
分类:LINUX

1. 内核函数说明
1subsys_initcall() 系统调用子系统初始化入口函数,你将入口函数加入到initd总,系统启动do_initcal时调用
2. WIFI 驱动学习
1. userspace nl80211 cfg80211 mac 80211结构图
2.1 基本概念
nl80211 cfg80211 mac80211由内核提供支持wifi配置的处理。
文件目录:linux-3.10.14-p112871/net/wireless/

1. nl80211: 用于对无线设备进行配置管理,它是一个基本Netlink的用户态协议(User态)
2. cfg80211:  与FullMAC, mac80211和nl80211一起工作。cfg80211是Linux 802.11配置API。 cfg80211用于取代Wireless-Extensions。 nl80211用来配置一个cfg80211设备,用于内核< - >用户空间之间的通信
3. mac80211: 是一个driver开发者可用于为SoftMAC无线设备写驱动的框架 (Kernel态)。mac80211为SoftMAC设备实现了cfg80211回调函数,且mac80211通过cfg80211实现了向网络子系统注册和配置。配置由cfg80211通过nl80211和wext实现。
2.2 nl80211驱动
开始文件:linux-3.10.14-p112871/net/wireless/nl80211.c
2.2.1 关键数据结构
1. nl消息头
struct genlmsghdr {
__u8 cmd;
__u8 version;
__u16 reserved;
};
2. 协议注册
static struct genl_family nl80211_fam = {
.id = GENL_ID_GENERATE, /* don't bother with a hardcoded ID */
.name = "nl80211", /* have users key off the name instead */
.hdrsize = 0, /* no private header */
.version = 1, /* no particular meaning now */
.maxattr = NL80211_ATTR_MAX,
.netnsok = true,
.pre_doit = nl80211_pre_doit,
.post_doit = nl80211_post_doit,
};
id:genl family的ID号,一般由内核进行分配,取值范围GENL_MIN_ID~GENL_MAX_ID(16~1023),其中GENL_ID_CTRL为控制器的family ID,不可另行分配,该familyID全局唯一并且在family_ht中的位置也由该值确定;
hdrsize:用户私有报头的长度,即可选的user msg header长度,若没有则为0;
name:genl family的名称,必须是独一无二且用户层已知的(用户通过它来向控制查找family id);
version:版本号;
maxattr:消息属性attr最大的类型数(即该genl family所支持的最大attr属性类型的种类个数);
netnsok:指示当前簇是否能够处理网络命名空间;
pre_doit:调用genl_ops结构中处理消息函数doit()前调用的钩子函数,一般用于执行一些前置的当前簇通用化处理,例如对临界区加锁等;
注释:主要通过用户态发送的消息中,通过cfg80211查找相应的dev信息,并填充到info当中。
post_doit:调用genl_ops结构中处理消息函数doit()后调用的钩子函数,一般执行pre_doit函数相反的操作;
mcast_bind/mcast_unbind:在绑定/解绑定socket到一个特定的genl netlink组播组中调用(目前内核中没有相关使用);
attrbuf:保存拷贝的attr属性缓存;
ops/n_ops:保存genl family命令处理结构即命令的个数,后面详细描述;
family_list:链表结构,用于将当前当前簇链入全局family_ht散列表中;
mcgrps/n_mcgrps:保存当前簇使用的组播组及组播地址的个数;
4. 命令 钩子函数
static struct genl_ops nl80211_ops[] = {
{
.cmd = NL80211_CMD_GET_WIPHY, //cmd
.doit = nl80211_get_wiphy, //对应的钩子函数
.dumpit = nl80211_dump_wiphy,
.policy = nl80211_policy, //策略
/* can be retrieved by unprivileged users */
.internal_flags = NL80211_FLAG_NEED_WIPHY,
},
}
policy:属性attr有效性策略,若该字段不为空,在genl执行消息处理函数前会对消息中的attr属性进行校验,否则则不做校验;
doit:标准命令回调函数,在当前族中收到数据时触发调用,函数的第一个入参skb中保存了用户下发的消息内容;
dumpit:转储回调函数,当genl_ops的flag标志被添加了NLM_F_DUMP以后会调用该回调函数,这里的第一个入参skb中不再有用户下发消息内容,而是要求函数能够在传入的skb中填入消息载荷并返回填入数据长度;
done:转储结束后执行的回调函数
支持的命令:
5. 接收信息结构
struct genl_info {
u32 snd_seq;
u32 snd_portid;
struct nlmsghdr * nlhdr;
struct genlmsghdr * genlhdr;
void * userhdr;
struct nlattr ** attrs;
#ifdef CONFIG_NET_NS
struct net * _net;
#endif
void * user_ptr[2];
};
内核在接收到用户的genetlink消息后,会对消息解析并封装成genl_info结构,便于命令回校函数进行处理,其中各字段含义如下:
snd_seq:消息的发送序号(不强制使用);
snd_portid:消息发送端socket所绑定的ID;
nlhdr:netlink消息头;
genlhdr:generic netlink消息头;
userhdr:用户私有报头;
attrs:netlink属性,包含了消息的实际载荷;
dst_sk:目的socket;
2.2.2 代码分析
2.2.2.1 nl80211_init()
主要完成两个任务:
1. 注册协议和命令回调函数
genl_register_family_with_ops();
2. 注册组播组(可以进行组播)
genl_register_mc_group();
2.2.2.2 支持的命令
nl80211接收来自用户态的netlink信息,通过回调函数nl80211_pre_doit, 查找cfg80211注册时的设备信息dev。 通过调用函数cfg80211函数接口或者dev钩子函数实现信息的获取或参数的设置。
enum nl80211_commands {
/* don't change the order or add anything between, this is ABI! */
NL80211_CMD_UNSPEC,
NL80211_CMD_GET_WIPHY, /* can dump */
NL80211_CMD_SET_WIPHY,
NL80211_CMD_NEW_WIPHY,
NL80211_CMD_DEL_WIPHY,

NL80211_CMD_GET_INTERFACE, /* can dump */
NL80211_CMD_SET_INTERFACE,
NL80211_CMD_NEW_INTERFACE,
NL80211_CMD_DEL_INTERFACE,
NL80211_CMD_GET_KEY,
NL80211_CMD_SET_KEY,
NL80211_CMD_NEW_KEY,
NL80211_CMD_DEL_KEY,
NL80211_CMD_GET_BEACON,
NL80211_CMD_SET_BEACON,
NL80211_CMD_START_AP,
NL80211_CMD_NEW_BEACON = NL80211_CMD_START_AP,
NL80211_CMD_STOP_AP,
NL80211_CMD_DEL_BEACON = NL80211_CMD_STOP_AP,
NL80211_CMD_GET_STATION,
NL80211_CMD_SET_STATION,
NL80211_CMD_NEW_STATION,
NL80211_CMD_DEL_STATION,
NL80211_CMD_GET_MPATH,
NL80211_CMD_SET_MPATH,
NL80211_CMD_NEW_MPATH,
NL80211_CMD_DEL_MPATH,
NL80211_CMD_SET_BSS,
NL80211_CMD_SET_REG,
NL80211_CMD_REQ_SET_REG,
NL80211_CMD_GET_MESH_CONFIG,
NL80211_CMD_SET_MESH_CONFIG,
NL80211_CMD_SET_MGMT_EXTRA_IE /* reserved; not used */,
NL80211_CMD_GET_REG,
NL80211_CMD_GET_SCAN,
NL80211_CMD_TRIGGER_SCAN,
NL80211_CMD_NEW_SCAN_RESULTS,
NL80211_CMD_SCAN_ABORTED,
NL80211_CMD_REG_CHANGE,
NL80211_CMD_AUTHENTICATE,
NL80211_CMD_ASSOCIATE,
NL80211_CMD_DEAUTHENTICATE,
NL80211_CMD_DISASSOCIATE,
NL80211_CMD_MICHAEL_MIC_FAILURE,
NL80211_CMD_REG_BEACON_HINT,
NL80211_CMD_JOIN_IBSS,
NL80211_CMD_LEAVE_IBSS,
NL80211_CMD_TESTMODE,
NL80211_CMD_CONNECT,
NL80211_CMD_ROAM,
NL80211_CMD_DISCONNECT,
NL80211_CMD_SET_WIPHY_NETNS,
NL80211_CMD_GET_SURVEY,
NL80211_CMD_NEW_SURVEY_RESULTS,
NL80211_CMD_SET_PMKSA,
NL80211_CMD_DEL_PMKSA,
NL80211_CMD_FLUSH_PMKSA,
NL80211_CMD_REMAIN_ON_CHANNEL,
NL80211_CMD_CANCEL_REMAIN_ON_CHANNEL,
NL80211_CMD_SET_TX_BITRATE_MASK,
NL80211_CMD_REGISTER_FRAME,
NL80211_CMD_REGISTER_ACTION = NL80211_CMD_REGISTER_FRAME,
NL80211_CMD_FRAME,
NL80211_CMD_ACTION = NL80211_CMD_FRAME,
NL80211_CMD_FRAME_TX_STATUS,
NL80211_CMD_ACTION_TX_STATUS = NL80211_CMD_FRAME_TX_STATUS,
NL80211_CMD_SET_POWER_SAVE,
NL80211_CMD_GET_POWER_SAVE,
NL80211_CMD_SET_CQM,
NL80211_CMD_NOTIFY_CQM,
NL80211_CMD_SET_CHANNEL,
NL80211_CMD_SET_WDS_PEER,
NL80211_CMD_FRAME_WAIT_CANCEL,
NL80211_CMD_JOIN_MESH,
NL80211_CMD_LEAVE_MESH,
NL80211_CMD_UNPROT_DEAUTHENTICATE,
NL80211_CMD_UNPROT_DISASSOCIATE,
NL80211_CMD_NEW_PEER_CANDIDATE,
NL80211_CMD_GET_WOWLAN,
NL80211_CMD_SET_WOWLAN,
NL80211_CMD_START_SCHED_SCAN,
NL80211_CMD_STOP_SCHED_SCAN,
NL80211_CMD_SCHED_SCAN_RESULTS,
NL80211_CMD_SCHED_SCAN_STOPPED,
NL80211_CMD_SET_REKEY_OFFLOAD,
NL80211_CMD_PMKSA_CANDIDATE,
NL80211_CMD_TDLS_OPER,
NL80211_CMD_TDLS_MGMT,
NL80211_CMD_UNEXPECTED_FRAME,
NL80211_CMD_PROBE_CLIENT,
NL80211_CMD_REGISTER_BEACONS,
NL80211_CMD_UNEXPECTED_4ADDR_FRAME,
NL80211_CMD_SET_NOACK_MAP,
NL80211_CMD_CH_SWITCH_NOTIFY,
NL80211_CMD_START_P2P_DEVICE,
NL80211_CMD_STOP_P2P_DEVICE,
NL80211_CMD_CONN_FAILED,
NL80211_CMD_SET_MCAST_RATE,
NL80211_CMD_SET_MAC_ACL,
NL80211_CMD_RADAR_DETECT,
NL80211_CMD_GET_PROTOCOL_FEATURES,
NL80211_CMD_UPDATE_FT_IES,
NL80211_CMD_FT_EVENT,
NL80211_CMD_CRIT_PROTOCOL_START,
NL80211_CMD_CRIT_PROTOCOL_STOP,
__NL80211_CMD_AFTER_LAST,
NL80211_CMD_MAX = __NL80211_CMD_AFTER_LAST - 1
};
2.3 cfg80211驱动
开始文件:linux-3.10.14-p112871/net/wireless/core.c, 文件主要提供一些相关接口函数,实现设备在cfg80211中设备的注册,在链表cfg80211_rdev_list 中添加设备信息。
2.3.1 关键数据结构
1.
struct pernet_operations {
struct list_head list;
int (*init)(struct net *net);
void (*exit)(struct net *net);
void (*exit_batch)(struct list_head *net_exit_list);
int *id;
size_t size;
};

2.3.2 代码分析
2.3.2.1 cfg80211_init(void)
1. register_pernet_device(&cfg80211_pernet_ops);(无初始化函数)
注册网络设备操作,该函数是register_pernet_operations的包裹函数,其中first_device的定义如下:
static struct list_head *first_device = &pernet_list; 所有要添加到网络命名空间的网络协议模块都会添加到链表first_device中。当kernel支持网络命令空间时,register_pernet_operations的定义如下:
当支持多网络命令空间时,会调用该函数,该函数会遍历目前已存在的所有网络命名空间,将网络协议模块添加到每一个网络命令空间中;并执行init操作,在每一个网络命名空间中,执行协议初始化相关的东东(生成proc相关的文件或者为协议申请缓存等)。
2. wiphy_sysfs_init();
在/net/wireless/sysfs.c中,该文件的作用是:当cfg80211.ko驱动模块被安装后,在终端设备的sys/class/下会出现ieee80211文件夹.
3. register_netdevice_notifier(&cfg80211_netdev_notifier);
实现设备cfg80211中的初始化、注册cfg80211以及相关参数的设置
4.nl80211_init();
nl80211 初始化函数
5. regulatory_init();

2.4 mac80211驱动
mac80211:是一个Linux内核子系统,是驱动开发者可用于为SoftMAC无线设备写驱动的框架。 mac80211在内核空间实现STA模式,在用户空间实现AP模式(hostapd)。
2.4.1 代码结构
1. ieee80211_i.h(主要数据结构)
2. main.c(主函数入口)
3. iface.c(虚拟接口处理)
4. key.c,key.h(密钥管理)
5. sta_info.c,sta_info.h(用户管理)
6. pm.c(功率管理)
7. rate.c,rate.h(速率控制函数)
8. rc80211*(速率控制算法)
9. rx.c(帧接收路径代码)
10. tx.c(帧发送路径代码)
11. scan.c(软件扫描代码)
12. mlme.c(station/managed模式MLME)
13. ibss.c(IBSS MLME)
14. cfg.c,cfg.h,wext.c(配置入口代码)
15. aes*,tkip*,wep*,michael*,wpa*(WPA/RSN/WEP代码)
16. wme.c,wme.h(QoS代码)
17. util.c(公共函数)

2.4.2主要的数据结构
1 ieee80211_local/ieee80211_hw
位置:ieee80211_i.h
2 sta_info/ieee80211_sta
位置:sta_info.h
3 ieee80211_conf
位置:mac80211.h
4 ieee80211_bss_conf
位置:mac80211.h
5 ieee80211_key/ieee80211_key_conf
位置:key.h
6 ieee80211_tx_info
位置:mac80211.h
7 ieee80211_rx_status
位置:mac80211.h
8 ieee80211_sub_if_data/ieee80211_vif
位置:ieee80211_i.h
2.4.3 主要流程
1 配置
2 接收路径
3 接收处理钩子(invoke_rx_handlers)
4 发送路径
5 发送处理钩子(invoke_tx_handlers)
6 mangement/MLME
7 IBSS
8 创建接口路径
9 删除接口路径
10 创建station路径
11 删除station路径
12 扫描请求路径
13 扫描状态机路径
2.4.4 切换点
1 配置
2 wext
3 cfg80211
4 mac80211到速率控制
5 mac80211到驱动
2.4.5 主要函数
1 ieee80211_alloc_hw()
2 ieee80211_register_hw()
3 ieee80211_rx()
4 ieee80211_xmit()
2.4.6 代码分析
MAC80211/main.c
2.4.6.1 ieee80211_init
1 rc80211_minstrel_init()
注册速率控制ops
位置:rc80211_minstrel.c
struct rate_control_ops mac80211_minstrel = {
.name = "minstrel",
.tx_status = minstrel_tx_status,
.get_rate = minstrel_get_rate,
.rate_init = minstrel_rate_init,
.alloc = minstrel_alloc,
.free = minstrel_free,
.alloc_sta = minstrel_alloc_sta,
.free_sta = minstrel_free_sta,
#ifdef CONFIG_MAC80211_DEBUGFS
.add_sta_debugfs = minstrel_add_sta_debugfs,
.remove_sta_debugfs = minstrel_remove_sta_debugfs,
#endif
};
2 rc80211_minstrel_ht_init();
注册80211 ht 速率控制ops
位置:rc80211_minstrel_ht.c
static struct rate_control_ops mac80211_minstrel_ht = {
.name = "minstrel_ht",
.tx_status = minstrel_ht_tx_status,
.get_rate = minstrel_ht_get_rate,
.rate_init = minstrel_ht_rate_init,
.rate_update = minstrel_ht_rate_update,
.alloc_sta = minstrel_ht_alloc_sta,
.free_sta = minstrel_ht_free_sta,
.alloc = minstrel_ht_alloc,
.free = minstrel_ht_free,
#ifdef CONFIG_MAC80211_DEBUGFS
.add_sta_debugfs = minstrel_ht_add_sta_debugfs,
.remove_sta_debugfs = minstrel_ht_remove_sta_debugfs,
#endif
};
3 rc80211_pid_init()
注册发送速率控制算法ops
位置:rc80211_pid_algo.c
static struct rate_control_ops mac80211_rcpid = {
.name = "pid",
.tx_status = rate_control_pid_tx_status,
.get_rate = rate_control_pid_get_rate,
.rate_init = rate_control_pid_rate_init,
.alloc = rate_control_pid_alloc,
.free = rate_control_pid_free,
.alloc_sta = rate_control_pid_alloc_sta,
.free_sta = rate_control_pid_free_sta,
#ifdef CONFIG_MAC80211_DEBUGFS
.add_sta_debugfs = rate_control_pid_add_sta_debugfs,
.remove_sta_debugfs = rate_control_pid_remove_sta_debugfs,
#endif
};
4 ieee80211_iface_init();
注册设备接口回调函数,用于更改相应结构体设备名称
位置:iface.c
static struct notifier_block mac80211_netdev_notifier = {
.notifier_call = netdev_notify,
};

2.4.6.2 ieee80211_restart_hw()
硬件重启初始化相关结构
2.4.6.3 ieee80211_alloc_hw
为硬件分配内存, 初始化ops,以及相关结构初始化
2.4.6.4 ieee80211_register_hw()
完成硬件的注册,以及相关初始化工作。
上一篇:Linux驱动(三) Makefile
下一篇:Ubuntu64位支持32编译工具