162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Parts of this file are
462306a36Sopenharmony_ci * Copyright (C) 2022 Intel Corporation
562306a36Sopenharmony_ci */
662306a36Sopenharmony_ci#include <linux/ieee80211.h>
762306a36Sopenharmony_ci#include <linux/export.h>
862306a36Sopenharmony_ci#include <net/cfg80211.h>
962306a36Sopenharmony_ci#include "nl80211.h"
1062306a36Sopenharmony_ci#include "core.h"
1162306a36Sopenharmony_ci#include "rdev-ops.h"
1262306a36Sopenharmony_ci
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_cistatic int ___cfg80211_stop_ap(struct cfg80211_registered_device *rdev,
1562306a36Sopenharmony_ci			       struct net_device *dev, unsigned int link_id,
1662306a36Sopenharmony_ci			       bool notify)
1762306a36Sopenharmony_ci{
1862306a36Sopenharmony_ci	struct wireless_dev *wdev = dev->ieee80211_ptr;
1962306a36Sopenharmony_ci	int err;
2062306a36Sopenharmony_ci
2162306a36Sopenharmony_ci	ASSERT_WDEV_LOCK(wdev);
2262306a36Sopenharmony_ci
2362306a36Sopenharmony_ci	if (!rdev->ops->stop_ap)
2462306a36Sopenharmony_ci		return -EOPNOTSUPP;
2562306a36Sopenharmony_ci
2662306a36Sopenharmony_ci	if (dev->ieee80211_ptr->iftype != NL80211_IFTYPE_AP &&
2762306a36Sopenharmony_ci	    dev->ieee80211_ptr->iftype != NL80211_IFTYPE_P2P_GO)
2862306a36Sopenharmony_ci		return -EOPNOTSUPP;
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_ci	if (!wdev->links[link_id].ap.beacon_interval)
3162306a36Sopenharmony_ci		return -ENOENT;
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_ci	err = rdev_stop_ap(rdev, dev, link_id);
3462306a36Sopenharmony_ci	if (!err) {
3562306a36Sopenharmony_ci		wdev->conn_owner_nlportid = 0;
3662306a36Sopenharmony_ci		wdev->links[link_id].ap.beacon_interval = 0;
3762306a36Sopenharmony_ci		memset(&wdev->links[link_id].ap.chandef, 0,
3862306a36Sopenharmony_ci		       sizeof(wdev->links[link_id].ap.chandef));
3962306a36Sopenharmony_ci		wdev->u.ap.ssid_len = 0;
4062306a36Sopenharmony_ci		rdev_set_qos_map(rdev, dev, NULL);
4162306a36Sopenharmony_ci		if (notify)
4262306a36Sopenharmony_ci			nl80211_send_ap_stopped(wdev, link_id);
4362306a36Sopenharmony_ci
4462306a36Sopenharmony_ci		/* Should we apply the grace period during beaconing interface
4562306a36Sopenharmony_ci		 * shutdown also?
4662306a36Sopenharmony_ci		 */
4762306a36Sopenharmony_ci		cfg80211_sched_dfs_chan_update(rdev);
4862306a36Sopenharmony_ci	}
4962306a36Sopenharmony_ci
5062306a36Sopenharmony_ci	schedule_work(&cfg80211_disconnect_work);
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_ci	return err;
5362306a36Sopenharmony_ci}
5462306a36Sopenharmony_ci
5562306a36Sopenharmony_ciint __cfg80211_stop_ap(struct cfg80211_registered_device *rdev,
5662306a36Sopenharmony_ci		       struct net_device *dev, int link_id,
5762306a36Sopenharmony_ci		       bool notify)
5862306a36Sopenharmony_ci{
5962306a36Sopenharmony_ci	unsigned int link;
6062306a36Sopenharmony_ci	int ret = 0;
6162306a36Sopenharmony_ci
6262306a36Sopenharmony_ci	if (link_id >= 0)
6362306a36Sopenharmony_ci		return ___cfg80211_stop_ap(rdev, dev, link_id, notify);
6462306a36Sopenharmony_ci
6562306a36Sopenharmony_ci	for_each_valid_link(dev->ieee80211_ptr, link) {
6662306a36Sopenharmony_ci		int ret1 = ___cfg80211_stop_ap(rdev, dev, link, notify);
6762306a36Sopenharmony_ci
6862306a36Sopenharmony_ci		if (ret1)
6962306a36Sopenharmony_ci			ret = ret1;
7062306a36Sopenharmony_ci		/* try the next one also if one errored */
7162306a36Sopenharmony_ci	}
7262306a36Sopenharmony_ci
7362306a36Sopenharmony_ci	return ret;
7462306a36Sopenharmony_ci}
7562306a36Sopenharmony_ci
7662306a36Sopenharmony_ciint cfg80211_stop_ap(struct cfg80211_registered_device *rdev,
7762306a36Sopenharmony_ci		     struct net_device *dev, int link_id,
7862306a36Sopenharmony_ci		     bool notify)
7962306a36Sopenharmony_ci{
8062306a36Sopenharmony_ci	struct wireless_dev *wdev = dev->ieee80211_ptr;
8162306a36Sopenharmony_ci	int err;
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_ci	wdev_lock(wdev);
8462306a36Sopenharmony_ci	err = __cfg80211_stop_ap(rdev, dev, link_id, notify);
8562306a36Sopenharmony_ci	wdev_unlock(wdev);
8662306a36Sopenharmony_ci
8762306a36Sopenharmony_ci	return err;
8862306a36Sopenharmony_ci}
89