| 
									
										
										
										
											2012-05-15 20:50:25 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * MAC commands interface | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright 2007-2012 Siemens AG | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU General Public License version 2 | 
					
						
							|  |  |  |  * as published by the Free Software Foundation. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Written by: | 
					
						
							|  |  |  |  * Sergey Lapin <slapin@ossfans.org> | 
					
						
							|  |  |  |  * Dmitry Eremin-Solenikov <dbaryshkov@gmail.com> | 
					
						
							|  |  |  |  * Alexander Smirnov <alex.bluesman.smirnov@gmail.com> | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/skbuff.h>
 | 
					
						
							|  |  |  | #include <linux/if_arp.h>
 | 
					
						
							| 
									
										
										
										
											2014-10-25 09:41:04 +02:00
										 |  |  | #include <linux/ieee802154.h>
 | 
					
						
							| 
									
										
										
										
											2012-05-15 20:50:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <net/ieee802154_netdev.h>
 | 
					
						
							| 
									
										
										
										
											2014-10-25 09:41:02 +02:00
										 |  |  | #include <net/cfg802154.h>
 | 
					
						
							| 
									
										
										
										
											2012-05-15 20:50:25 +00:00
										 |  |  | #include <net/mac802154.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-25 09:41:00 +02:00
										 |  |  | #include "ieee802154_i.h"
 | 
					
						
							| 
									
										
										
										
											2014-11-02 04:18:44 +01:00
										 |  |  | #include "driver-ops.h"
 | 
					
						
							| 
									
										
										
										
											2012-05-15 20:50:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-25 23:24:52 +00:00
										 |  |  | static int mac802154_mlme_start_req(struct net_device *dev, | 
					
						
							| 
									
										
										
										
											2014-03-14 21:24:02 +01:00
										 |  |  | 				    struct ieee802154_addr *addr, | 
					
						
							| 
									
										
										
										
											2012-06-25 23:24:52 +00:00
										 |  |  | 				    u8 channel, u8 page, | 
					
						
							|  |  |  | 				    u8 bcn_ord, u8 sf_ord, | 
					
						
							|  |  |  | 				    u8 pan_coord, u8 blx, | 
					
						
							|  |  |  | 				    u8 coord_realign) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-05-16 17:46:43 +02:00
										 |  |  | 	struct ieee802154_mlme_ops *ops = ieee802154_mlme_ops(dev); | 
					
						
							|  |  |  | 	int rc = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-09 08:36:57 +01:00
										 |  |  | 	ASSERT_RTNL(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-14 21:24:02 +01:00
										 |  |  | 	BUG_ON(addr->mode != IEEE802154_ADDR_SHORT); | 
					
						
							| 
									
										
										
										
											2012-06-25 23:24:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-14 21:24:02 +01:00
										 |  |  | 	mac802154_dev_set_pan_id(dev, addr->pan_id); | 
					
						
							|  |  |  | 	mac802154_dev_set_short_addr(dev, addr->short_addr); | 
					
						
							| 
									
										
										
										
											2012-06-25 23:24:52 +00:00
										 |  |  | 	mac802154_dev_set_page_channel(dev, page, channel); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-16 17:46:43 +02:00
										 |  |  | 	if (ops->llsec) { | 
					
						
							|  |  |  | 		struct ieee802154_llsec_params params; | 
					
						
							|  |  |  | 		int changed = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		params.coord_shortaddr = addr->short_addr; | 
					
						
							|  |  |  | 		changed |= IEEE802154_LLSEC_PARAM_COORD_SHORTADDR; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		params.pan_id = addr->pan_id; | 
					
						
							|  |  |  | 		changed |= IEEE802154_LLSEC_PARAM_PAN_ID; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		params.hwaddr = ieee802154_devaddr_from_raw(dev->dev_addr); | 
					
						
							|  |  |  | 		changed |= IEEE802154_LLSEC_PARAM_HWADDR; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		params.coord_hwaddr = params.hwaddr; | 
					
						
							|  |  |  | 		changed |= IEEE802154_LLSEC_PARAM_COORD_HWADDR; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		rc = ops->llsec->set_params(dev, ¶ms, changed); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return rc; | 
					
						
							| 
									
										
										
										
											2012-06-25 23:24:52 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-28 18:21:30 +01:00
										 |  |  | static int mac802154_set_mac_params(struct net_device *dev, | 
					
						
							|  |  |  | 				    const struct ieee802154_mac_params *params) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ieee802154_sub_if_data *sdata = IEEE802154_DEV_TO_SUB_IF(dev); | 
					
						
							| 
									
										
										
										
											2014-11-02 04:18:44 +01:00
										 |  |  | 	struct ieee802154_local *local = sdata->local; | 
					
						
							| 
									
										
										
										
											2014-11-09 08:36:46 +01:00
										 |  |  | 	struct wpan_dev *wpan_dev = &sdata->wpan_dev; | 
					
						
							| 
									
										
										
										
											2014-11-02 04:18:44 +01:00
										 |  |  | 	int ret; | 
					
						
							| 
									
										
										
										
											2014-10-28 18:21:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-09 08:36:57 +01:00
										 |  |  | 	ASSERT_RTNL(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-09 08:36:46 +01:00
										 |  |  | 	/* PHY */ | 
					
						
							|  |  |  | 	wpan_dev->wpan_phy->transmit_power = params->transmit_power; | 
					
						
							|  |  |  | 	wpan_dev->wpan_phy->cca_mode = params->cca_mode; | 
					
						
							|  |  |  | 	wpan_dev->wpan_phy->cca_ed_level = params->cca_ed_level; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* MAC */ | 
					
						
							|  |  |  | 	wpan_dev->min_be = params->min_be; | 
					
						
							|  |  |  | 	wpan_dev->max_be = params->max_be; | 
					
						
							|  |  |  | 	wpan_dev->csma_retries = params->csma_retries; | 
					
						
							|  |  |  | 	wpan_dev->frame_retries = params->frame_retries; | 
					
						
							|  |  |  | 	wpan_dev->lbt = params->lbt; | 
					
						
							| 
									
										
										
										
											2014-10-28 18:21:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-02 04:18:44 +01:00
										 |  |  | 	if (local->hw.flags & IEEE802154_HW_TXPOWER) { | 
					
						
							|  |  |  | 		ret = drv_set_tx_power(local, params->transmit_power); | 
					
						
							|  |  |  | 		if (ret < 0) | 
					
						
							|  |  |  | 			return ret; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (local->hw.flags & IEEE802154_HW_CCA_MODE) { | 
					
						
							|  |  |  | 		ret = drv_set_cca_mode(local, params->cca_mode); | 
					
						
							|  |  |  | 		if (ret < 0) | 
					
						
							|  |  |  | 			return ret; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (local->hw.flags & IEEE802154_HW_CCA_ED_LEVEL) { | 
					
						
							|  |  |  | 		ret = drv_set_cca_ed_level(local, params->cca_ed_level); | 
					
						
							|  |  |  | 		if (ret < 0) | 
					
						
							|  |  |  | 			return ret; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-28 18:21:30 +01:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void mac802154_get_mac_params(struct net_device *dev, | 
					
						
							|  |  |  | 				     struct ieee802154_mac_params *params) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ieee802154_sub_if_data *sdata = IEEE802154_DEV_TO_SUB_IF(dev); | 
					
						
							| 
									
										
										
										
											2014-11-09 08:36:46 +01:00
										 |  |  | 	struct wpan_dev *wpan_dev = &sdata->wpan_dev; | 
					
						
							| 
									
										
										
										
											2014-10-28 18:21:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-09 08:36:57 +01:00
										 |  |  | 	ASSERT_RTNL(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-09 08:36:46 +01:00
										 |  |  | 	/* PHY */ | 
					
						
							|  |  |  | 	params->transmit_power = wpan_dev->wpan_phy->transmit_power; | 
					
						
							|  |  |  | 	params->cca_mode = wpan_dev->wpan_phy->cca_mode; | 
					
						
							|  |  |  | 	params->cca_ed_level = wpan_dev->wpan_phy->cca_ed_level; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* MAC */ | 
					
						
							|  |  |  | 	params->min_be = wpan_dev->min_be; | 
					
						
							|  |  |  | 	params->max_be = wpan_dev->max_be; | 
					
						
							|  |  |  | 	params->csma_retries = wpan_dev->csma_retries; | 
					
						
							|  |  |  | 	params->frame_retries = wpan_dev->frame_retries; | 
					
						
							|  |  |  | 	params->lbt = wpan_dev->lbt; | 
					
						
							| 
									
										
										
										
											2014-10-28 18:21:30 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-16 17:46:42 +02:00
										 |  |  | static struct ieee802154_llsec_ops mac802154_llsec_ops = { | 
					
						
							|  |  |  | 	.get_params = mac802154_get_params, | 
					
						
							|  |  |  | 	.set_params = mac802154_set_params, | 
					
						
							|  |  |  | 	.add_key = mac802154_add_key, | 
					
						
							|  |  |  | 	.del_key = mac802154_del_key, | 
					
						
							|  |  |  | 	.add_dev = mac802154_add_dev, | 
					
						
							|  |  |  | 	.del_dev = mac802154_del_dev, | 
					
						
							|  |  |  | 	.add_devkey = mac802154_add_devkey, | 
					
						
							|  |  |  | 	.del_devkey = mac802154_del_devkey, | 
					
						
							|  |  |  | 	.add_seclevel = mac802154_add_seclevel, | 
					
						
							|  |  |  | 	.del_seclevel = mac802154_del_seclevel, | 
					
						
							|  |  |  | 	.lock_table = mac802154_lock_table, | 
					
						
							|  |  |  | 	.get_table = mac802154_get_table, | 
					
						
							|  |  |  | 	.unlock_table = mac802154_unlock_table, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-25 23:24:48 +00:00
										 |  |  | struct ieee802154_mlme_ops mac802154_mlme_wpan = { | 
					
						
							| 
									
										
										
										
											2012-06-25 23:24:52 +00:00
										 |  |  | 	.start_req = mac802154_mlme_start_req, | 
					
						
							| 
									
										
										
										
											2012-07-10 21:22:44 +00:00
										 |  |  | 	.get_pan_id = mac802154_dev_get_pan_id, | 
					
						
							|  |  |  | 	.get_short_addr = mac802154_dev_get_short_addr, | 
					
						
							| 
									
										
										
										
											2013-03-25 17:59:29 +00:00
										 |  |  | 	.get_dsn = mac802154_dev_get_dsn, | 
					
						
							| 
									
										
										
										
											2014-03-31 21:37:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-16 17:46:42 +02:00
										 |  |  | 	.llsec = &mac802154_llsec_ops, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-31 21:37:46 +02:00
										 |  |  | 	.set_mac_params = mac802154_set_mac_params, | 
					
						
							|  |  |  | 	.get_mac_params = mac802154_get_mac_params, | 
					
						
							| 
									
										
										
										
											2012-06-25 23:24:48 +00:00
										 |  |  | }; |