| // SPDX-License-Identifier: GPL-2.0 | 
 | /* | 
 |  * Parts of this file are | 
 |  * Copyright (C) 2022 Intel Corporation | 
 |  */ | 
 | #include <linux/ieee80211.h> | 
 | #include <linux/export.h> | 
 | #include <net/cfg80211.h> | 
 | #include "nl80211.h" | 
 | #include "core.h" | 
 | #include "rdev-ops.h" | 
 |  | 
 |  | 
 | static int ___cfg80211_stop_ap(struct cfg80211_registered_device *rdev, | 
 | 			       struct net_device *dev, unsigned int link_id, | 
 | 			       bool notify) | 
 | { | 
 | 	struct wireless_dev *wdev = dev->ieee80211_ptr; | 
 | 	int err; | 
 |  | 
 | 	ASSERT_WDEV_LOCK(wdev); | 
 |  | 
 | 	if (!rdev->ops->stop_ap) | 
 | 		return -EOPNOTSUPP; | 
 |  | 
 | 	if (dev->ieee80211_ptr->iftype != NL80211_IFTYPE_AP && | 
 | 	    dev->ieee80211_ptr->iftype != NL80211_IFTYPE_P2P_GO) | 
 | 		return -EOPNOTSUPP; | 
 |  | 
 | 	if (!wdev->links[link_id].ap.beacon_interval) | 
 | 		return -ENOENT; | 
 |  | 
 | 	err = rdev_stop_ap(rdev, dev, link_id); | 
 | 	if (!err) { | 
 | 		wdev->conn_owner_nlportid = 0; | 
 | 		wdev->links[link_id].ap.beacon_interval = 0; | 
 | 		memset(&wdev->links[link_id].ap.chandef, 0, | 
 | 		       sizeof(wdev->links[link_id].ap.chandef)); | 
 | 		wdev->u.ap.ssid_len = 0; | 
 | 		rdev_set_qos_map(rdev, dev, NULL); | 
 | 		if (notify) | 
 | 			nl80211_send_ap_stopped(wdev, link_id); | 
 |  | 
 | 		/* Should we apply the grace period during beaconing interface | 
 | 		 * shutdown also? | 
 | 		 */ | 
 | 		cfg80211_sched_dfs_chan_update(rdev); | 
 | 	} | 
 |  | 
 | 	schedule_work(&cfg80211_disconnect_work); | 
 |  | 
 | 	return err; | 
 | } | 
 |  | 
 | int __cfg80211_stop_ap(struct cfg80211_registered_device *rdev, | 
 | 		       struct net_device *dev, int link_id, | 
 | 		       bool notify) | 
 | { | 
 | 	unsigned int link; | 
 | 	int ret = 0; | 
 |  | 
 | 	if (link_id >= 0) | 
 | 		return ___cfg80211_stop_ap(rdev, dev, link_id, notify); | 
 |  | 
 | 	for_each_valid_link(dev->ieee80211_ptr, link) { | 
 | 		int ret1 = ___cfg80211_stop_ap(rdev, dev, link, notify); | 
 |  | 
 | 		if (ret1) | 
 | 			ret = ret1; | 
 | 		/* try the next one also if one errored */ | 
 | 	} | 
 |  | 
 | 	return ret; | 
 | } | 
 |  | 
 | int cfg80211_stop_ap(struct cfg80211_registered_device *rdev, | 
 | 		     struct net_device *dev, int link_id, | 
 | 		     bool notify) | 
 | { | 
 | 	struct wireless_dev *wdev = dev->ieee80211_ptr; | 
 | 	int err; | 
 |  | 
 | 	wdev_lock(wdev); | 
 | 	err = __cfg80211_stop_ap(rdev, dev, link_id, notify); | 
 | 	wdev_unlock(wdev); | 
 |  | 
 | 	return err; | 
 | } |