1 /*
2  * Copyright (c) 2023 Nordic Semiconductor ASA.
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #include <zephyr/logging/log.h>
8 LOG_MODULE_REGISTER(wifi_nm, CONFIG_WIFI_NM_LOG_LEVEL);
9 
10 #include <zephyr/kernel.h>
11 #include <zephyr/net/wifi_nm.h>
12 
13 /* Used to protect nm data */
14 static K_MUTEX_DEFINE(wifi_nm_lock);
15 
wifi_nm_get_instance(const char * name)16 struct wifi_nm_instance *wifi_nm_get_instance(const char *name)
17 {
18 	STRUCT_SECTION_FOREACH(wifi_nm_instance, nm) {
19 		if (!strcmp(nm->name, name)) {
20 			return nm;
21 		}
22 	}
23 
24 	return NULL;
25 }
26 
wifi_nm_get_instance_iface(struct net_if * iface)27 struct wifi_nm_instance *wifi_nm_get_instance_iface(struct net_if *iface)
28 {
29 	if (!iface || !net_if_is_wifi(iface)) {
30 		return NULL;
31 	}
32 
33 	k_mutex_lock(&wifi_nm_lock, K_FOREVER);
34 	STRUCT_SECTION_FOREACH(wifi_nm_instance, nm) {
35 		for (int i = 0; i < CONFIG_WIFI_NM_MAX_MANAGED_INTERFACES; i++) {
36 			if (nm->mgd_ifaces[i].iface == iface) {
37 				k_mutex_unlock(&wifi_nm_lock);
38 				return nm;
39 			}
40 		}
41 	}
42 	k_mutex_unlock(&wifi_nm_lock);
43 
44 	return NULL;
45 }
46 
wifi_nm_get_type_iface(struct net_if * iface)47 unsigned char wifi_nm_get_type_iface(struct net_if *iface)
48 {
49 	if (!iface || !net_if_is_wifi(iface)) {
50 		return 0;
51 	}
52 
53 	k_mutex_lock(&wifi_nm_lock, K_FOREVER);
54 	STRUCT_SECTION_FOREACH(wifi_nm_instance, nm) {
55 		for (int i = 0; i < CONFIG_WIFI_NM_MAX_MANAGED_INTERFACES; i++) {
56 			if (nm->mgd_ifaces[i].iface == iface) {
57 				k_mutex_unlock(&wifi_nm_lock);
58 				return nm->mgd_ifaces[i].type;
59 			}
60 		}
61 	}
62 
63 	k_mutex_unlock(&wifi_nm_lock);
64 	return 0;
65 }
66 
wifi_nm_iface_is_sta(struct net_if * iface)67 bool wifi_nm_iface_is_sta(struct net_if *iface)
68 {
69 	return wifi_nm_get_type_iface(iface) & BIT(WIFI_TYPE_STA);
70 }
71 
wifi_nm_iface_is_sap(struct net_if * iface)72 bool wifi_nm_iface_is_sap(struct net_if *iface)
73 {
74 	return wifi_nm_get_type_iface(iface) & BIT(WIFI_TYPE_SAP);
75 }
76 
wifi_nm_register_mgd_iface(struct wifi_nm_instance * nm,struct net_if * iface)77 int wifi_nm_register_mgd_iface(struct wifi_nm_instance *nm, struct net_if *iface)
78 {
79 	if (!nm || !iface) {
80 		return -EINVAL;
81 	}
82 
83 	if (!net_if_is_wifi(iface)) {
84 		return -ENOTSUP;
85 	}
86 
87 	k_mutex_lock(&wifi_nm_lock, K_FOREVER);
88 	for (int i = 0; i < CONFIG_WIFI_NM_MAX_MANAGED_INTERFACES; i++) {
89 		if (nm->mgd_ifaces[i].iface == iface) {
90 			k_mutex_unlock(&wifi_nm_lock);
91 			return 0;
92 		}
93 
94 		if (!nm->mgd_ifaces[i].iface) {
95 			nm->mgd_ifaces[i].iface = iface;
96 			k_mutex_unlock(&wifi_nm_lock);
97 			return 0;
98 		}
99 	}
100 	k_mutex_unlock(&wifi_nm_lock);
101 
102 	return -ENOMEM;
103 }
104 
wifi_nm_register_mgd_type_iface(struct wifi_nm_instance * nm,enum wifi_nm_iface_type type,struct net_if * iface)105 int wifi_nm_register_mgd_type_iface(struct wifi_nm_instance *nm,
106 		enum wifi_nm_iface_type type, struct net_if *iface)
107 {
108 	if (!nm || !iface) {
109 		return -EINVAL;
110 	}
111 
112 	if (!net_if_is_wifi(iface)) {
113 		return -ENOTSUP;
114 	}
115 
116 	k_mutex_lock(&wifi_nm_lock, K_FOREVER);
117 	for (int i = 0; i < CONFIG_WIFI_NM_MAX_MANAGED_INTERFACES; i++) {
118 		if ((nm->mgd_ifaces[i].iface == iface)
119 			&& (nm->mgd_ifaces[i].type == BIT(type))) {
120 			k_mutex_unlock(&wifi_nm_lock);
121 			return 0;
122 		}
123 
124 		if (!nm->mgd_ifaces[i].iface) {
125 			nm->mgd_ifaces[i].iface = iface;
126 			nm->mgd_ifaces[i].type = BIT(type);
127 			k_mutex_unlock(&wifi_nm_lock);
128 			return 0;
129 		}
130 	}
131 	k_mutex_unlock(&wifi_nm_lock);
132 
133 	return -ENOMEM;
134 }
135 
wifi_nm_unregister_mgd_iface(struct wifi_nm_instance * nm,struct net_if * iface)136 int wifi_nm_unregister_mgd_iface(struct wifi_nm_instance *nm, struct net_if *iface)
137 {
138 	if (!nm || !iface) {
139 		return -EINVAL;
140 	}
141 
142 	k_mutex_lock(&wifi_nm_lock, K_FOREVER);
143 	for (int i = 0; i < CONFIG_WIFI_NM_MAX_MANAGED_INTERFACES; i++) {
144 		if (nm->mgd_ifaces[i].iface == iface) {
145 			nm->mgd_ifaces[i].iface = NULL;
146 			k_mutex_unlock(&wifi_nm_lock);
147 			return 0;
148 		}
149 	}
150 	k_mutex_unlock(&wifi_nm_lock);
151 
152 	return -ENOENT;
153 }
154