| // SPDX-License-Identifier: GPL-2.0-only | 
 | /* | 
 |  * Copyright 2015 Intel Deutschland GmbH | 
 |  * Copyright (C) 2022-2025 Intel Corporation | 
 |  */ | 
 | #include <net/mac80211.h> | 
 | #include "ieee80211_i.h" | 
 | #include "trace.h" | 
 | #include "driver-ops.h" | 
 | #include "debugfs_sta.h" | 
 | #include "debugfs_netdev.h" | 
 |  | 
 | int drv_start(struct ieee80211_local *local) | 
 | { | 
 | 	int ret; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (WARN_ON(local->started)) | 
 | 		return -EALREADY; | 
 |  | 
 | 	trace_drv_start(local); | 
 | 	local->started = true; | 
 | 	/* allow rx frames */ | 
 | 	smp_mb(); | 
 | 	ret = local->ops->start(&local->hw); | 
 | 	trace_drv_return_int(local, ret); | 
 |  | 
 | 	if (ret) | 
 | 		local->started = false; | 
 |  | 
 | 	return ret; | 
 | } | 
 |  | 
 | void drv_stop(struct ieee80211_local *local, bool suspend) | 
 | { | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (WARN_ON(!local->started)) | 
 | 		return; | 
 |  | 
 | 	trace_drv_stop(local, suspend); | 
 | 	local->ops->stop(&local->hw, suspend); | 
 | 	trace_drv_return_void(local); | 
 |  | 
 | 	/* sync away all work on the tasklet before clearing started */ | 
 | 	tasklet_disable(&local->tasklet); | 
 | 	tasklet_enable(&local->tasklet); | 
 |  | 
 | 	barrier(); | 
 |  | 
 | 	local->started = false; | 
 | } | 
 |  | 
 | int drv_add_interface(struct ieee80211_local *local, | 
 | 		      struct ieee80211_sub_if_data *sdata) | 
 | { | 
 | 	int ret; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (WARN_ON(sdata->vif.type == NL80211_IFTYPE_AP_VLAN || | 
 | 		    (sdata->vif.type == NL80211_IFTYPE_MONITOR && | 
 | 		     !ieee80211_hw_check(&local->hw, WANT_MONITOR_VIF) && | 
 | 		     !ieee80211_hw_check(&local->hw, NO_VIRTUAL_MONITOR) && | 
 | 		     !(sdata->u.mntr.flags & MONITOR_FLAG_ACTIVE)))) | 
 | 		return -EINVAL; | 
 |  | 
 | 	trace_drv_add_interface(local, sdata); | 
 | 	ret = local->ops->add_interface(&local->hw, &sdata->vif); | 
 | 	trace_drv_return_int(local, ret); | 
 |  | 
 | 	if (ret) | 
 | 		return ret; | 
 |  | 
 | 	if (!(sdata->flags & IEEE80211_SDATA_IN_DRIVER)) { | 
 | 		sdata->flags |= IEEE80211_SDATA_IN_DRIVER; | 
 |  | 
 | 		drv_vif_add_debugfs(local, sdata); | 
 | 		/* initially vif is not MLD */ | 
 | 		ieee80211_link_debugfs_drv_add(&sdata->deflink); | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | int drv_change_interface(struct ieee80211_local *local, | 
 | 			 struct ieee80211_sub_if_data *sdata, | 
 | 			 enum nl80211_iftype type, bool p2p) | 
 | { | 
 | 	int ret; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return -EIO; | 
 |  | 
 | 	trace_drv_change_interface(local, sdata, type, p2p); | 
 | 	ret = local->ops->change_interface(&local->hw, &sdata->vif, type, p2p); | 
 | 	trace_drv_return_int(local, ret); | 
 | 	return ret; | 
 | } | 
 |  | 
 | void drv_remove_interface(struct ieee80211_local *local, | 
 | 			  struct ieee80211_sub_if_data *sdata) | 
 | { | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return; | 
 |  | 
 | 	sdata->flags &= ~IEEE80211_SDATA_IN_DRIVER; | 
 |  | 
 | 	/* | 
 | 	 * Remove driver debugfs entries. | 
 | 	 * The virtual monitor interface doesn't get a debugfs | 
 | 	 * entry, so it's exempt here. | 
 | 	 */ | 
 | 	if (sdata != rcu_access_pointer(local->monitor_sdata)) | 
 | 		ieee80211_debugfs_recreate_netdev(sdata, | 
 | 						  sdata->vif.valid_links); | 
 |  | 
 | 	trace_drv_remove_interface(local, sdata); | 
 | 	local->ops->remove_interface(&local->hw, &sdata->vif); | 
 | 	trace_drv_return_void(local); | 
 | } | 
 |  | 
 | __must_check | 
 | int drv_sta_state(struct ieee80211_local *local, | 
 | 		  struct ieee80211_sub_if_data *sdata, | 
 | 		  struct sta_info *sta, | 
 | 		  enum ieee80211_sta_state old_state, | 
 | 		  enum ieee80211_sta_state new_state) | 
 | { | 
 | 	int ret = 0; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	sdata = get_bss_sdata(sdata); | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return -EIO; | 
 |  | 
 | 	trace_drv_sta_state(local, sdata, &sta->sta, old_state, new_state); | 
 | 	if (local->ops->sta_state) { | 
 | 		ret = local->ops->sta_state(&local->hw, &sdata->vif, &sta->sta, | 
 | 					    old_state, new_state); | 
 | 	} else if (old_state == IEEE80211_STA_AUTH && | 
 | 		   new_state == IEEE80211_STA_ASSOC) { | 
 | 		ret = drv_sta_add(local, sdata, &sta->sta); | 
 | 		if (ret == 0) { | 
 | 			sta->uploaded = true; | 
 | 			if (rcu_access_pointer(sta->sta.rates)) | 
 | 				drv_sta_rate_tbl_update(local, sdata, &sta->sta); | 
 | 		} | 
 | 	} else if (old_state == IEEE80211_STA_ASSOC && | 
 | 		   new_state == IEEE80211_STA_AUTH) { | 
 | 		drv_sta_remove(local, sdata, &sta->sta); | 
 | 	} | 
 | 	trace_drv_return_int(local, ret); | 
 | 	return ret; | 
 | } | 
 |  | 
 | __must_check | 
 | int drv_sta_set_txpwr(struct ieee80211_local *local, | 
 | 		      struct ieee80211_sub_if_data *sdata, | 
 | 		      struct sta_info *sta) | 
 | { | 
 | 	int ret = -EOPNOTSUPP; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	sdata = get_bss_sdata(sdata); | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return -EIO; | 
 |  | 
 | 	trace_drv_sta_set_txpwr(local, sdata, &sta->sta); | 
 | 	if (local->ops->sta_set_txpwr) | 
 | 		ret = local->ops->sta_set_txpwr(&local->hw, &sdata->vif, | 
 | 						&sta->sta); | 
 | 	trace_drv_return_int(local, ret); | 
 | 	return ret; | 
 | } | 
 |  | 
 | void drv_link_sta_rc_update(struct ieee80211_local *local, | 
 | 			    struct ieee80211_sub_if_data *sdata, | 
 | 			    struct ieee80211_link_sta *link_sta, | 
 | 			    u32 changed) | 
 | { | 
 | 	sdata = get_bss_sdata(sdata); | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return; | 
 |  | 
 | 	WARN_ON(changed & IEEE80211_RC_SUPP_RATES_CHANGED && | 
 | 		(sdata->vif.type != NL80211_IFTYPE_ADHOC && | 
 | 		 sdata->vif.type != NL80211_IFTYPE_MESH_POINT)); | 
 |  | 
 | 	trace_drv_link_sta_rc_update(local, sdata, link_sta, changed); | 
 | 	if (local->ops->link_sta_rc_update) | 
 | 		local->ops->link_sta_rc_update(&local->hw, &sdata->vif, | 
 | 					       link_sta, changed); | 
 |  | 
 | 	trace_drv_return_void(local); | 
 | } | 
 |  | 
 | int drv_conf_tx(struct ieee80211_local *local, | 
 | 		struct ieee80211_link_data *link, u16 ac, | 
 | 		const struct ieee80211_tx_queue_params *params) | 
 | { | 
 | 	struct ieee80211_sub_if_data *sdata = link->sdata; | 
 | 	int ret = -EOPNOTSUPP; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return -EIO; | 
 |  | 
 | 	if (!ieee80211_vif_link_active(&sdata->vif, link->link_id)) | 
 | 		return 0; | 
 |  | 
 | 	if (params->cw_min == 0 || params->cw_min > params->cw_max) { | 
 | 		/* | 
 | 		 * If we can't configure hardware anyway, don't warn. We may | 
 | 		 * never have initialized the CW parameters. | 
 | 		 */ | 
 | 		WARN_ONCE(local->ops->conf_tx, | 
 | 			  "%s: invalid CW_min/CW_max: %d/%d\n", | 
 | 			  sdata->name, params->cw_min, params->cw_max); | 
 | 		return -EINVAL; | 
 | 	} | 
 |  | 
 | 	trace_drv_conf_tx(local, sdata, link->link_id, ac, params); | 
 | 	if (local->ops->conf_tx) | 
 | 		ret = local->ops->conf_tx(&local->hw, &sdata->vif, | 
 | 					  link->link_id, ac, params); | 
 | 	trace_drv_return_int(local, ret); | 
 | 	return ret; | 
 | } | 
 |  | 
 | u64 drv_get_tsf(struct ieee80211_local *local, | 
 | 		struct ieee80211_sub_if_data *sdata) | 
 | { | 
 | 	u64 ret = -1ULL; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return ret; | 
 |  | 
 | 	trace_drv_get_tsf(local, sdata); | 
 | 	if (local->ops->get_tsf) | 
 | 		ret = local->ops->get_tsf(&local->hw, &sdata->vif); | 
 | 	trace_drv_return_u64(local, ret); | 
 | 	return ret; | 
 | } | 
 |  | 
 | void drv_set_tsf(struct ieee80211_local *local, | 
 | 		 struct ieee80211_sub_if_data *sdata, | 
 | 		 u64 tsf) | 
 | { | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return; | 
 |  | 
 | 	trace_drv_set_tsf(local, sdata, tsf); | 
 | 	if (local->ops->set_tsf) | 
 | 		local->ops->set_tsf(&local->hw, &sdata->vif, tsf); | 
 | 	trace_drv_return_void(local); | 
 | } | 
 |  | 
 | void drv_offset_tsf(struct ieee80211_local *local, | 
 | 		    struct ieee80211_sub_if_data *sdata, | 
 | 		    s64 offset) | 
 | { | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return; | 
 |  | 
 | 	trace_drv_offset_tsf(local, sdata, offset); | 
 | 	if (local->ops->offset_tsf) | 
 | 		local->ops->offset_tsf(&local->hw, &sdata->vif, offset); | 
 | 	trace_drv_return_void(local); | 
 | } | 
 |  | 
 | void drv_reset_tsf(struct ieee80211_local *local, | 
 | 		   struct ieee80211_sub_if_data *sdata) | 
 | { | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return; | 
 |  | 
 | 	trace_drv_reset_tsf(local, sdata); | 
 | 	if (local->ops->reset_tsf) | 
 | 		local->ops->reset_tsf(&local->hw, &sdata->vif); | 
 | 	trace_drv_return_void(local); | 
 | } | 
 |  | 
 | int drv_assign_vif_chanctx(struct ieee80211_local *local, | 
 | 			   struct ieee80211_sub_if_data *sdata, | 
 | 			   struct ieee80211_bss_conf *link_conf, | 
 | 			   struct ieee80211_chanctx *ctx) | 
 | { | 
 | 	int ret = 0; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	/* | 
 | 	 * We should perhaps push emulate chanctx down and only | 
 | 	 * make it call ->config() when the chanctx is actually | 
 | 	 * assigned here (and unassigned below), but that's yet | 
 | 	 * another change to all drivers to add assign/unassign | 
 | 	 * emulation callbacks. Maybe later. | 
 | 	 */ | 
 | 	if (sdata->vif.type == NL80211_IFTYPE_MONITOR && | 
 | 	    local->emulate_chanctx && | 
 | 	    !ieee80211_hw_check(&local->hw, WANT_MONITOR_VIF)) | 
 | 		return 0; | 
 |  | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return -EIO; | 
 |  | 
 | 	if (!ieee80211_vif_link_active(&sdata->vif, link_conf->link_id)) | 
 | 		return 0; | 
 |  | 
 | 	trace_drv_assign_vif_chanctx(local, sdata, link_conf, ctx); | 
 | 	if (local->ops->assign_vif_chanctx) { | 
 | 		WARN_ON_ONCE(!ctx->driver_present); | 
 | 		ret = local->ops->assign_vif_chanctx(&local->hw, | 
 | 						     &sdata->vif, | 
 | 						     link_conf, | 
 | 						     &ctx->conf); | 
 | 	} | 
 | 	trace_drv_return_int(local, ret); | 
 |  | 
 | 	return ret; | 
 | } | 
 |  | 
 | void drv_unassign_vif_chanctx(struct ieee80211_local *local, | 
 | 			      struct ieee80211_sub_if_data *sdata, | 
 | 			      struct ieee80211_bss_conf *link_conf, | 
 | 			      struct ieee80211_chanctx *ctx) | 
 | { | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (sdata->vif.type == NL80211_IFTYPE_MONITOR && | 
 | 	    local->emulate_chanctx && | 
 | 	    !ieee80211_hw_check(&local->hw, WANT_MONITOR_VIF)) | 
 | 		return; | 
 |  | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return; | 
 |  | 
 | 	if (!ieee80211_vif_link_active(&sdata->vif, link_conf->link_id)) | 
 | 		return; | 
 |  | 
 | 	trace_drv_unassign_vif_chanctx(local, sdata, link_conf, ctx); | 
 | 	if (local->ops->unassign_vif_chanctx) { | 
 | 		WARN_ON_ONCE(!ctx->driver_present); | 
 | 		local->ops->unassign_vif_chanctx(&local->hw, | 
 | 						 &sdata->vif, | 
 | 						 link_conf, | 
 | 						 &ctx->conf); | 
 | 	} | 
 | 	trace_drv_return_void(local); | 
 | } | 
 |  | 
 | int drv_switch_vif_chanctx(struct ieee80211_local *local, | 
 | 			   struct ieee80211_vif_chanctx_switch *vifs, | 
 | 			   int n_vifs, enum ieee80211_chanctx_switch_mode mode) | 
 | { | 
 | 	int ret = 0; | 
 | 	int i; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (!local->ops->switch_vif_chanctx) | 
 | 		return -EOPNOTSUPP; | 
 |  | 
 | 	for (i = 0; i < n_vifs; i++) { | 
 | 		struct ieee80211_chanctx *new_ctx = | 
 | 			container_of(vifs[i].new_ctx, | 
 | 				     struct ieee80211_chanctx, | 
 | 				     conf); | 
 | 		struct ieee80211_chanctx *old_ctx = | 
 | 			container_of(vifs[i].old_ctx, | 
 | 				     struct ieee80211_chanctx, | 
 | 				     conf); | 
 |  | 
 | 		WARN_ON_ONCE(!old_ctx->driver_present); | 
 | 		WARN_ON_ONCE((mode == CHANCTX_SWMODE_SWAP_CONTEXTS && | 
 | 			      new_ctx->driver_present) || | 
 | 			     (mode == CHANCTX_SWMODE_REASSIGN_VIF && | 
 | 			      !new_ctx->driver_present)); | 
 | 	} | 
 |  | 
 | 	trace_drv_switch_vif_chanctx(local, vifs, n_vifs, mode); | 
 | 	ret = local->ops->switch_vif_chanctx(&local->hw, | 
 | 					     vifs, n_vifs, mode); | 
 | 	trace_drv_return_int(local, ret); | 
 |  | 
 | 	if (!ret && mode == CHANCTX_SWMODE_SWAP_CONTEXTS) { | 
 | 		for (i = 0; i < n_vifs; i++) { | 
 | 			struct ieee80211_chanctx *new_ctx = | 
 | 				container_of(vifs[i].new_ctx, | 
 | 					     struct ieee80211_chanctx, | 
 | 					     conf); | 
 | 			struct ieee80211_chanctx *old_ctx = | 
 | 				container_of(vifs[i].old_ctx, | 
 | 					     struct ieee80211_chanctx, | 
 | 					     conf); | 
 |  | 
 | 			new_ctx->driver_present = true; | 
 | 			old_ctx->driver_present = false; | 
 | 		} | 
 | 	} | 
 |  | 
 | 	return ret; | 
 | } | 
 |  | 
 | int drv_ampdu_action(struct ieee80211_local *local, | 
 | 		     struct ieee80211_sub_if_data *sdata, | 
 | 		     struct ieee80211_ampdu_params *params) | 
 | { | 
 | 	int ret = -EOPNOTSUPP; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	sdata = get_bss_sdata(sdata); | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return -EIO; | 
 |  | 
 | 	trace_drv_ampdu_action(local, sdata, params); | 
 |  | 
 | 	if (local->ops->ampdu_action) | 
 | 		ret = local->ops->ampdu_action(&local->hw, &sdata->vif, params); | 
 |  | 
 | 	trace_drv_return_int(local, ret); | 
 |  | 
 | 	return ret; | 
 | } | 
 |  | 
 | void drv_link_info_changed(struct ieee80211_local *local, | 
 | 			   struct ieee80211_sub_if_data *sdata, | 
 | 			   struct ieee80211_bss_conf *info, | 
 | 			   int link_id, u64 changed) | 
 | { | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (WARN_ON_ONCE(changed & (BSS_CHANGED_BEACON | | 
 | 				    BSS_CHANGED_BEACON_ENABLED) && | 
 | 			 sdata->vif.type != NL80211_IFTYPE_AP && | 
 | 			 sdata->vif.type != NL80211_IFTYPE_ADHOC && | 
 | 			 sdata->vif.type != NL80211_IFTYPE_MESH_POINT && | 
 | 			 sdata->vif.type != NL80211_IFTYPE_OCB)) | 
 | 		return; | 
 |  | 
 | 	if (WARN_ON_ONCE(sdata->vif.type == NL80211_IFTYPE_P2P_DEVICE || | 
 | 			 sdata->vif.type == NL80211_IFTYPE_NAN || | 
 | 			 (sdata->vif.type == NL80211_IFTYPE_MONITOR && | 
 | 			  !sdata->vif.bss_conf.mu_mimo_owner && | 
 | 			  !(changed & BSS_CHANGED_TXPOWER)))) | 
 | 		return; | 
 |  | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return; | 
 |  | 
 | 	if (!ieee80211_vif_link_active(&sdata->vif, link_id)) | 
 | 		return; | 
 |  | 
 | 	trace_drv_link_info_changed(local, sdata, info, changed); | 
 | 	if (local->ops->link_info_changed) | 
 | 		local->ops->link_info_changed(&local->hw, &sdata->vif, | 
 | 					      info, changed); | 
 | 	else if (local->ops->bss_info_changed) | 
 | 		local->ops->bss_info_changed(&local->hw, &sdata->vif, | 
 | 					     info, changed); | 
 | 	trace_drv_return_void(local); | 
 | } | 
 |  | 
 | int drv_set_key(struct ieee80211_local *local, | 
 | 		enum set_key_cmd cmd, | 
 | 		struct ieee80211_sub_if_data *sdata, | 
 | 		struct ieee80211_sta *sta, | 
 | 		struct ieee80211_key_conf *key) | 
 | { | 
 | 	int ret; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	sdata = get_bss_sdata(sdata); | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return -EIO; | 
 |  | 
 | 	if (WARN_ON(key->link_id >= 0 && sdata->vif.active_links && | 
 | 		    !(sdata->vif.active_links & BIT(key->link_id)))) | 
 | 		return -ENOLINK; | 
 |  | 
 | 	if (fips_enabled) | 
 | 		return -EOPNOTSUPP; | 
 |  | 
 | 	trace_drv_set_key(local, cmd, sdata, sta, key); | 
 | 	ret = local->ops->set_key(&local->hw, cmd, &sdata->vif, sta, key); | 
 | 	trace_drv_return_int(local, ret); | 
 | 	return ret; | 
 | } | 
 |  | 
 | int drv_change_vif_links(struct ieee80211_local *local, | 
 | 			 struct ieee80211_sub_if_data *sdata, | 
 | 			 u16 old_links, u16 new_links, | 
 | 			 struct ieee80211_bss_conf *old[IEEE80211_MLD_MAX_NUM_LINKS]) | 
 | { | 
 | 	struct ieee80211_link_data *link; | 
 | 	unsigned long links_to_add; | 
 | 	unsigned long links_to_rem; | 
 | 	unsigned int link_id; | 
 | 	int ret = -EOPNOTSUPP; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return -EIO; | 
 |  | 
 | 	if (old_links == new_links) | 
 | 		return 0; | 
 |  | 
 | 	links_to_add = ~old_links & new_links; | 
 | 	links_to_rem = old_links & ~new_links; | 
 |  | 
 | 	for_each_set_bit(link_id, &links_to_rem, IEEE80211_MLD_MAX_NUM_LINKS) { | 
 | 		link = rcu_access_pointer(sdata->link[link_id]); | 
 |  | 
 | 		ieee80211_link_debugfs_drv_remove(link); | 
 | 	} | 
 |  | 
 | 	trace_drv_change_vif_links(local, sdata, old_links, new_links); | 
 | 	if (local->ops->change_vif_links) | 
 | 		ret = local->ops->change_vif_links(&local->hw, &sdata->vif, | 
 | 						   old_links, new_links, old); | 
 | 	trace_drv_return_int(local, ret); | 
 |  | 
 | 	if (ret) | 
 | 		return ret; | 
 |  | 
 | 	if (!local->in_reconfig && !local->resuming) { | 
 | 		for_each_set_bit(link_id, &links_to_add, | 
 | 				 IEEE80211_MLD_MAX_NUM_LINKS) { | 
 | 			link = rcu_access_pointer(sdata->link[link_id]); | 
 |  | 
 | 			ieee80211_link_debugfs_drv_add(link); | 
 | 		} | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | int drv_change_sta_links(struct ieee80211_local *local, | 
 | 			 struct ieee80211_sub_if_data *sdata, | 
 | 			 struct ieee80211_sta *sta, | 
 | 			 u16 old_links, u16 new_links) | 
 | { | 
 | 	struct sta_info *info = container_of(sta, struct sta_info, sta); | 
 | 	struct link_sta_info *link_sta; | 
 | 	unsigned long links_to_add; | 
 | 	unsigned long links_to_rem; | 
 | 	unsigned int link_id; | 
 | 	int ret = -EOPNOTSUPP; | 
 |  | 
 | 	might_sleep(); | 
 | 	lockdep_assert_wiphy(local->hw.wiphy); | 
 |  | 
 | 	if (!check_sdata_in_driver(sdata)) | 
 | 		return -EIO; | 
 |  | 
 | 	old_links &= sdata->vif.active_links; | 
 | 	new_links &= sdata->vif.active_links; | 
 |  | 
 | 	if (old_links == new_links) | 
 | 		return 0; | 
 |  | 
 | 	links_to_add = ~old_links & new_links; | 
 | 	links_to_rem = old_links & ~new_links; | 
 |  | 
 | 	for_each_set_bit(link_id, &links_to_rem, IEEE80211_MLD_MAX_NUM_LINKS) { | 
 | 		link_sta = rcu_dereference_protected(info->link[link_id], | 
 | 						     lockdep_is_held(&local->hw.wiphy->mtx)); | 
 |  | 
 | 		ieee80211_link_sta_debugfs_drv_remove(link_sta); | 
 | 	} | 
 |  | 
 | 	trace_drv_change_sta_links(local, sdata, sta, old_links, new_links); | 
 | 	if (local->ops->change_sta_links) | 
 | 		ret = local->ops->change_sta_links(&local->hw, &sdata->vif, sta, | 
 | 						   old_links, new_links); | 
 | 	trace_drv_return_int(local, ret); | 
 |  | 
 | 	if (ret) | 
 | 		return ret; | 
 |  | 
 | 	/* during reconfig don't add it to debugfs again */ | 
 | 	if (local->in_reconfig || local->resuming) | 
 | 		return 0; | 
 |  | 
 | 	for_each_set_bit(link_id, &links_to_add, IEEE80211_MLD_MAX_NUM_LINKS) { | 
 | 		link_sta = rcu_dereference_protected(info->link[link_id], | 
 | 						     lockdep_is_held(&local->hw.wiphy->mtx)); | 
 | 		ieee80211_link_sta_debugfs_drv_add(link_sta); | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | } |