mwifiex: Use function to check whether interface type change is allowed
Instead of bailing out in the function which is supposed to do the type change, detect invalid changes beforehand using a generic function and return an error if the change is not allowed. Signed-off-by: Jonas Dreßler <verdre@v0yd.nl> Signed-off-by: Kalle Valo <kvalo@codeaurora.org> Link: https://lore.kernel.org/r/20210914195909.36035-3-verdre@v0yd.nl
This commit is contained in:
		
							parent
							
								
									babe2a332d
								
							
						
					
					
						commit
						abe3a2c9ea
					
				| @ -939,6 +939,76 @@ mwifiex_init_new_priv_params(struct mwifiex_private *priv, | |||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static bool | ||||||
|  | is_vif_type_change_allowed(struct mwifiex_adapter *adapter, | ||||||
|  | 			   enum nl80211_iftype old_iftype, | ||||||
|  | 			   enum nl80211_iftype new_iftype) | ||||||
|  | { | ||||||
|  | 	switch (old_iftype) { | ||||||
|  | 	case NL80211_IFTYPE_ADHOC: | ||||||
|  | 		switch (new_iftype) { | ||||||
|  | 		case NL80211_IFTYPE_STATION: | ||||||
|  | 			return true; | ||||||
|  | 		case NL80211_IFTYPE_P2P_CLIENT: | ||||||
|  | 		case NL80211_IFTYPE_P2P_GO: | ||||||
|  | 			return adapter->curr_iface_comb.p2p_intf != | ||||||
|  | 			       adapter->iface_limit.p2p_intf; | ||||||
|  | 		case NL80211_IFTYPE_AP: | ||||||
|  | 			return adapter->curr_iface_comb.uap_intf != | ||||||
|  | 			       adapter->iface_limit.uap_intf; | ||||||
|  | 		default: | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 	case NL80211_IFTYPE_STATION: | ||||||
|  | 		switch (new_iftype) { | ||||||
|  | 		case NL80211_IFTYPE_ADHOC: | ||||||
|  | 			return true; | ||||||
|  | 		case NL80211_IFTYPE_P2P_CLIENT: | ||||||
|  | 		case NL80211_IFTYPE_P2P_GO: | ||||||
|  | 			return adapter->curr_iface_comb.p2p_intf != | ||||||
|  | 			       adapter->iface_limit.p2p_intf; | ||||||
|  | 		case NL80211_IFTYPE_AP: | ||||||
|  | 			return adapter->curr_iface_comb.uap_intf != | ||||||
|  | 			       adapter->iface_limit.uap_intf; | ||||||
|  | 		default: | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 	case NL80211_IFTYPE_AP: | ||||||
|  | 		switch (new_iftype) { | ||||||
|  | 		case NL80211_IFTYPE_ADHOC: | ||||||
|  | 		case NL80211_IFTYPE_STATION: | ||||||
|  | 			return adapter->curr_iface_comb.sta_intf != | ||||||
|  | 			       adapter->iface_limit.sta_intf; | ||||||
|  | 		case NL80211_IFTYPE_P2P_CLIENT: | ||||||
|  | 		case NL80211_IFTYPE_P2P_GO: | ||||||
|  | 			return adapter->curr_iface_comb.p2p_intf != | ||||||
|  | 			       adapter->iface_limit.p2p_intf; | ||||||
|  | 		default: | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 	case NL80211_IFTYPE_P2P_CLIENT: | ||||||
|  | 	case NL80211_IFTYPE_P2P_GO: | ||||||
|  | 		switch (new_iftype) { | ||||||
|  | 		case NL80211_IFTYPE_ADHOC: | ||||||
|  | 		case NL80211_IFTYPE_STATION: | ||||||
|  | 			return true; | ||||||
|  | 		case NL80211_IFTYPE_AP: | ||||||
|  | 			return adapter->curr_iface_comb.uap_intf != | ||||||
|  | 			       adapter->iface_limit.uap_intf; | ||||||
|  | 		default: | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 	default: | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static int | static int | ||||||
| mwifiex_change_vif_to_p2p(struct net_device *dev, | mwifiex_change_vif_to_p2p(struct net_device *dev, | ||||||
| 			  enum nl80211_iftype curr_iftype, | 			  enum nl80211_iftype curr_iftype, | ||||||
| @ -955,13 +1025,6 @@ mwifiex_change_vif_to_p2p(struct net_device *dev, | |||||||
| 
 | 
 | ||||||
| 	adapter = priv->adapter; | 	adapter = priv->adapter; | ||||||
| 
 | 
 | ||||||
| 	if (adapter->curr_iface_comb.p2p_intf == |  | ||||||
| 	    adapter->iface_limit.p2p_intf) { |  | ||||||
| 		mwifiex_dbg(adapter, ERROR, |  | ||||||
| 			    "cannot create multiple P2P ifaces\n"); |  | ||||||
| 		return -1; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	mwifiex_dbg(adapter, INFO, | 	mwifiex_dbg(adapter, INFO, | ||||||
| 		    "%s: changing role to p2p\n", dev->name); | 		    "%s: changing role to p2p\n", dev->name); | ||||||
| 
 | 
 | ||||||
| @ -1027,15 +1090,6 @@ mwifiex_change_vif_to_sta_adhoc(struct net_device *dev, | |||||||
| 
 | 
 | ||||||
| 	adapter = priv->adapter; | 	adapter = priv->adapter; | ||||||
| 
 | 
 | ||||||
| 	if ((curr_iftype != NL80211_IFTYPE_P2P_CLIENT && |  | ||||||
| 	     curr_iftype != NL80211_IFTYPE_P2P_GO) && |  | ||||||
| 	    (adapter->curr_iface_comb.sta_intf == |  | ||||||
| 	     adapter->iface_limit.sta_intf)) { |  | ||||||
| 		mwifiex_dbg(adapter, ERROR, |  | ||||||
| 			    "cannot create multiple station/adhoc ifaces\n"); |  | ||||||
| 		return -1; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (type == NL80211_IFTYPE_STATION) | 	if (type == NL80211_IFTYPE_STATION) | ||||||
| 		mwifiex_dbg(adapter, INFO, | 		mwifiex_dbg(adapter, INFO, | ||||||
| 			    "%s: changing role to station\n", dev->name); | 			    "%s: changing role to station\n", dev->name); | ||||||
| @ -1086,13 +1140,6 @@ mwifiex_change_vif_to_ap(struct net_device *dev, | |||||||
| 
 | 
 | ||||||
| 	adapter = priv->adapter; | 	adapter = priv->adapter; | ||||||
| 
 | 
 | ||||||
| 	if (adapter->curr_iface_comb.uap_intf == |  | ||||||
| 	    adapter->iface_limit.uap_intf) { |  | ||||||
| 		mwifiex_dbg(adapter, ERROR, |  | ||||||
| 			    "cannot create multiple AP ifaces\n"); |  | ||||||
| 		return -1; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	mwifiex_dbg(adapter, INFO, | 	mwifiex_dbg(adapter, INFO, | ||||||
| 		    "%s: changing role to AP\n", dev->name); | 		    "%s: changing role to AP\n", dev->name); | ||||||
| 
 | 
 | ||||||
| @ -1155,6 +1202,13 @@ mwifiex_cfg80211_change_virtual_intf(struct wiphy *wiphy, | |||||||
| 		return 0; | 		return 0; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if (!is_vif_type_change_allowed(priv->adapter, curr_iftype, type)) { | ||||||
|  | 		mwifiex_dbg(priv->adapter, ERROR, | ||||||
|  | 			    "%s: change from type %d to %d is not allowed\n", | ||||||
|  | 			    dev->name, curr_iftype, type); | ||||||
|  | 		return -EOPNOTSUPP; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	switch (curr_iftype) { | 	switch (curr_iftype) { | ||||||
| 	case NL80211_IFTYPE_ADHOC: | 	case NL80211_IFTYPE_ADHOC: | ||||||
| 		switch (type) { | 		switch (type) { | ||||||
| @ -1175,12 +1229,9 @@ mwifiex_cfg80211_change_virtual_intf(struct wiphy *wiphy, | |||||||
| 			return mwifiex_change_vif_to_ap(dev, curr_iftype, type, | 			return mwifiex_change_vif_to_ap(dev, curr_iftype, type, | ||||||
| 							params); | 							params); | ||||||
| 		default: | 		default: | ||||||
| 			mwifiex_dbg(priv->adapter, ERROR, | 			goto errnotsupp; | ||||||
| 				    "%s: changing to %d not supported\n", |  | ||||||
| 				    dev->name, type); |  | ||||||
| 			return -EOPNOTSUPP; |  | ||||||
| 		} | 		} | ||||||
| 		break; | 
 | ||||||
| 	case NL80211_IFTYPE_STATION: | 	case NL80211_IFTYPE_STATION: | ||||||
| 		switch (type) { | 		switch (type) { | ||||||
| 		case NL80211_IFTYPE_ADHOC: | 		case NL80211_IFTYPE_ADHOC: | ||||||
| @ -1200,12 +1251,9 @@ mwifiex_cfg80211_change_virtual_intf(struct wiphy *wiphy, | |||||||
| 			return mwifiex_change_vif_to_ap(dev, curr_iftype, type, | 			return mwifiex_change_vif_to_ap(dev, curr_iftype, type, | ||||||
| 							params); | 							params); | ||||||
| 		default: | 		default: | ||||||
| 			mwifiex_dbg(priv->adapter, ERROR, | 			goto errnotsupp; | ||||||
| 				    "%s: changing to %d not supported\n", |  | ||||||
| 				    dev->name, type); |  | ||||||
| 			return -EOPNOTSUPP; |  | ||||||
| 		} | 		} | ||||||
| 		break; | 
 | ||||||
| 	case NL80211_IFTYPE_AP: | 	case NL80211_IFTYPE_AP: | ||||||
| 		switch (type) { | 		switch (type) { | ||||||
| 		case NL80211_IFTYPE_ADHOC: | 		case NL80211_IFTYPE_ADHOC: | ||||||
| @ -1217,12 +1265,9 @@ mwifiex_cfg80211_change_virtual_intf(struct wiphy *wiphy, | |||||||
| 			return mwifiex_change_vif_to_p2p(dev, curr_iftype, | 			return mwifiex_change_vif_to_p2p(dev, curr_iftype, | ||||||
| 							 type, params); | 							 type, params); | ||||||
| 		default: | 		default: | ||||||
| 			mwifiex_dbg(priv->adapter, ERROR, | 			goto errnotsupp; | ||||||
| 				    "%s: changing to %d not supported\n", |  | ||||||
| 				    dev->name, type); |  | ||||||
| 			return -EOPNOTSUPP; |  | ||||||
| 		} | 		} | ||||||
| 		break; | 
 | ||||||
| 	case NL80211_IFTYPE_P2P_CLIENT: | 	case NL80211_IFTYPE_P2P_CLIENT: | ||||||
| 	case NL80211_IFTYPE_P2P_GO: | 	case NL80211_IFTYPE_P2P_GO: | ||||||
| 		switch (type) { | 		switch (type) { | ||||||
| @ -1251,21 +1296,21 @@ mwifiex_cfg80211_change_virtual_intf(struct wiphy *wiphy, | |||||||
| 			return mwifiex_change_vif_to_ap(dev, curr_iftype, type, | 			return mwifiex_change_vif_to_ap(dev, curr_iftype, type, | ||||||
| 							params); | 							params); | ||||||
| 		default: | 		default: | ||||||
| 			mwifiex_dbg(priv->adapter, ERROR, | 			goto errnotsupp; | ||||||
| 				    "%s: changing to %d not supported\n", |  | ||||||
| 				    dev->name, type); |  | ||||||
| 			return -EOPNOTSUPP; |  | ||||||
| 		} | 		} | ||||||
| 		break; | 
 | ||||||
| 	default: | 	default: | ||||||
| 		mwifiex_dbg(priv->adapter, ERROR, | 		goto errnotsupp; | ||||||
| 			    "%s: unknown iftype: %d\n", |  | ||||||
| 			    dev->name, dev->ieee80211_ptr->iftype); |  | ||||||
| 		return -EOPNOTSUPP; |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 	return 0; | 	return 0; | ||||||
|  | 
 | ||||||
|  | errnotsupp: | ||||||
|  | 	mwifiex_dbg(priv->adapter, ERROR, | ||||||
|  | 		    "unsupported interface type transition: %d to %d\n", | ||||||
|  | 		    curr_iftype, type); | ||||||
|  | 	return -EOPNOTSUPP; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void | static void | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user