1 // SPDX-License-Identifier: GPL-2.0+
2 /* Copyright (c) 2015-2016 Quantenna Communications. All rights reserved. */
4 #include <linux/nl80211.h>
6 #include "qlink_util.h"
8 u16
qlink_iface_type_to_nl_mask(u16 qlink_type
)
14 result
|= BIT(NL80211_IFTYPE_AP
);
16 case QLINK_IFTYPE_STATION
:
17 result
|= BIT(NL80211_IFTYPE_STATION
);
19 case QLINK_IFTYPE_ADHOC
:
20 result
|= BIT(NL80211_IFTYPE_ADHOC
);
22 case QLINK_IFTYPE_MONITOR
:
23 result
|= BIT(NL80211_IFTYPE_MONITOR
);
25 case QLINK_IFTYPE_WDS
:
26 result
|= BIT(NL80211_IFTYPE_WDS
);
28 case QLINK_IFTYPE_AP_VLAN
:
29 result
|= BIT(NL80211_IFTYPE_AP_VLAN
);
36 u8
qlink_chan_width_mask_to_nl(u16 qlink_mask
)
40 if (qlink_mask
& BIT(QLINK_CHAN_WIDTH_5
))
41 result
|= BIT(NL80211_CHAN_WIDTH_5
);
43 if (qlink_mask
& BIT(QLINK_CHAN_WIDTH_10
))
44 result
|= BIT(NL80211_CHAN_WIDTH_10
);
46 if (qlink_mask
& BIT(QLINK_CHAN_WIDTH_20_NOHT
))
47 result
|= BIT(NL80211_CHAN_WIDTH_20_NOHT
);
49 if (qlink_mask
& BIT(QLINK_CHAN_WIDTH_20
))
50 result
|= BIT(NL80211_CHAN_WIDTH_20
);
52 if (qlink_mask
& BIT(QLINK_CHAN_WIDTH_40
))
53 result
|= BIT(NL80211_CHAN_WIDTH_40
);
55 if (qlink_mask
& BIT(QLINK_CHAN_WIDTH_80
))
56 result
|= BIT(NL80211_CHAN_WIDTH_80
);
58 if (qlink_mask
& BIT(QLINK_CHAN_WIDTH_80P80
))
59 result
|= BIT(NL80211_CHAN_WIDTH_80P80
);
61 if (qlink_mask
& BIT(QLINK_CHAN_WIDTH_160
))
62 result
|= BIT(NL80211_CHAN_WIDTH_160
);
67 static enum nl80211_chan_width
qlink_chanwidth_to_nl(u8 qlw
)
70 case QLINK_CHAN_WIDTH_20_NOHT
:
71 return NL80211_CHAN_WIDTH_20_NOHT
;
72 case QLINK_CHAN_WIDTH_20
:
73 return NL80211_CHAN_WIDTH_20
;
74 case QLINK_CHAN_WIDTH_40
:
75 return NL80211_CHAN_WIDTH_40
;
76 case QLINK_CHAN_WIDTH_80
:
77 return NL80211_CHAN_WIDTH_80
;
78 case QLINK_CHAN_WIDTH_80P80
:
79 return NL80211_CHAN_WIDTH_80P80
;
80 case QLINK_CHAN_WIDTH_160
:
81 return NL80211_CHAN_WIDTH_160
;
82 case QLINK_CHAN_WIDTH_5
:
83 return NL80211_CHAN_WIDTH_5
;
84 case QLINK_CHAN_WIDTH_10
:
85 return NL80211_CHAN_WIDTH_10
;
91 static u8
qlink_chanwidth_nl_to_qlink(enum nl80211_chan_width nlwidth
)
94 case NL80211_CHAN_WIDTH_20_NOHT
:
95 return QLINK_CHAN_WIDTH_20_NOHT
;
96 case NL80211_CHAN_WIDTH_20
:
97 return QLINK_CHAN_WIDTH_20
;
98 case NL80211_CHAN_WIDTH_40
:
99 return QLINK_CHAN_WIDTH_40
;
100 case NL80211_CHAN_WIDTH_80
:
101 return QLINK_CHAN_WIDTH_80
;
102 case NL80211_CHAN_WIDTH_80P80
:
103 return QLINK_CHAN_WIDTH_80P80
;
104 case NL80211_CHAN_WIDTH_160
:
105 return QLINK_CHAN_WIDTH_160
;
106 case NL80211_CHAN_WIDTH_5
:
107 return QLINK_CHAN_WIDTH_5
;
108 case NL80211_CHAN_WIDTH_10
:
109 return QLINK_CHAN_WIDTH_10
;
115 void qlink_chandef_q2cfg(struct wiphy
*wiphy
,
116 const struct qlink_chandef
*qch
,
117 struct cfg80211_chan_def
*chdef
)
119 struct ieee80211_channel
*chan
;
121 chan
= ieee80211_get_channel(wiphy
, le16_to_cpu(qch
->chan
.center_freq
));
124 chdef
->center_freq1
= le16_to_cpu(qch
->center_freq1
);
125 chdef
->center_freq2
= le16_to_cpu(qch
->center_freq2
);
126 chdef
->width
= qlink_chanwidth_to_nl(qch
->width
);
127 chdef
->edmg
.bw_config
= 0;
128 chdef
->edmg
.channels
= 0;
131 void qlink_chandef_cfg2q(const struct cfg80211_chan_def
*chdef
,
132 struct qlink_chandef
*qch
)
134 struct ieee80211_channel
*chan
= chdef
->chan
;
136 qch
->chan
.hw_value
= cpu_to_le16(chan
->hw_value
);
137 qch
->chan
.center_freq
= cpu_to_le16(chan
->center_freq
);
138 qch
->chan
.flags
= cpu_to_le32(chan
->flags
);
140 qch
->center_freq1
= cpu_to_le16(chdef
->center_freq1
);
141 qch
->center_freq2
= cpu_to_le16(chdef
->center_freq2
);
142 qch
->width
= qlink_chanwidth_nl_to_qlink(chdef
->width
);
145 enum qlink_hidden_ssid
qlink_hidden_ssid_nl2q(enum nl80211_hidden_ssid nl_val
)
148 case NL80211_HIDDEN_SSID_ZERO_LEN
:
149 return QLINK_HIDDEN_SSID_ZERO_LEN
;
150 case NL80211_HIDDEN_SSID_ZERO_CONTENTS
:
151 return QLINK_HIDDEN_SSID_ZERO_CONTENTS
;
152 case NL80211_HIDDEN_SSID_NOT_IN_USE
:
154 return QLINK_HIDDEN_SSID_NOT_IN_USE
;
158 bool qtnf_utils_is_bit_set(const u8
*arr
, unsigned int bit
,
159 unsigned int arr_max_len
)
161 unsigned int idx
= bit
/ BITS_PER_BYTE
;
162 u8 mask
= 1 << (bit
- (idx
* BITS_PER_BYTE
));
164 if (idx
>= arr_max_len
)
167 return arr
[idx
] & mask
;
170 void qlink_acl_data_cfg2q(const struct cfg80211_acl_data
*acl
,
171 struct qlink_acl_data
*qacl
)
173 switch (acl
->acl_policy
) {
174 case NL80211_ACL_POLICY_ACCEPT_UNLESS_LISTED
:
176 cpu_to_le32(QLINK_ACL_POLICY_ACCEPT_UNLESS_LISTED
);
178 case NL80211_ACL_POLICY_DENY_UNLESS_LISTED
:
179 qacl
->policy
= cpu_to_le32(QLINK_ACL_POLICY_DENY_UNLESS_LISTED
);
183 qacl
->num_entries
= cpu_to_le32(acl
->n_acl_entries
);
184 memcpy(qacl
->mac_addrs
, acl
->mac_addrs
,
185 acl
->n_acl_entries
* sizeof(*qacl
->mac_addrs
));
188 enum qlink_band
qlink_utils_band_cfg2q(enum nl80211_band band
)
191 case NL80211_BAND_2GHZ
:
192 return QLINK_BAND_2GHZ
;
193 case NL80211_BAND_5GHZ
:
194 return QLINK_BAND_5GHZ
;
195 case NL80211_BAND_60GHZ
:
196 return QLINK_BAND_60GHZ
;
202 enum qlink_dfs_state
qlink_utils_dfs_state_cfg2q(enum nl80211_dfs_state state
)
205 case NL80211_DFS_USABLE
:
206 return QLINK_DFS_USABLE
;
207 case NL80211_DFS_AVAILABLE
:
208 return QLINK_DFS_AVAILABLE
;
209 case NL80211_DFS_UNAVAILABLE
:
211 return QLINK_DFS_UNAVAILABLE
;
215 u32
qlink_utils_chflags_cfg2q(u32 cfgflags
)
219 if (cfgflags
& IEEE80211_CHAN_DISABLED
)
220 flags
|= QLINK_CHAN_DISABLED
;
222 if (cfgflags
& IEEE80211_CHAN_NO_IR
)
223 flags
|= QLINK_CHAN_NO_IR
;
225 if (cfgflags
& IEEE80211_CHAN_RADAR
)
226 flags
|= QLINK_CHAN_RADAR
;
228 if (cfgflags
& IEEE80211_CHAN_NO_HT40PLUS
)
229 flags
|= QLINK_CHAN_NO_HT40PLUS
;
231 if (cfgflags
& IEEE80211_CHAN_NO_HT40MINUS
)
232 flags
|= QLINK_CHAN_NO_HT40MINUS
;
234 if (cfgflags
& IEEE80211_CHAN_NO_80MHZ
)
235 flags
|= QLINK_CHAN_NO_80MHZ
;
237 if (cfgflags
& IEEE80211_CHAN_NO_160MHZ
)
238 flags
|= QLINK_CHAN_NO_160MHZ
;
243 static u32
qtnf_reg_rule_flags_parse(u32 qflags
)
247 if (qflags
& QLINK_RRF_NO_OFDM
)
248 flags
|= NL80211_RRF_NO_OFDM
;
250 if (qflags
& QLINK_RRF_NO_CCK
)
251 flags
|= NL80211_RRF_NO_CCK
;
253 if (qflags
& QLINK_RRF_NO_INDOOR
)
254 flags
|= NL80211_RRF_NO_INDOOR
;
256 if (qflags
& QLINK_RRF_NO_OUTDOOR
)
257 flags
|= NL80211_RRF_NO_OUTDOOR
;
259 if (qflags
& QLINK_RRF_DFS
)
260 flags
|= NL80211_RRF_DFS
;
262 if (qflags
& QLINK_RRF_PTP_ONLY
)
263 flags
|= NL80211_RRF_PTP_ONLY
;
265 if (qflags
& QLINK_RRF_PTMP_ONLY
)
266 flags
|= NL80211_RRF_PTMP_ONLY
;
268 if (qflags
& QLINK_RRF_NO_IR
)
269 flags
|= NL80211_RRF_NO_IR
;
271 if (qflags
& QLINK_RRF_AUTO_BW
)
272 flags
|= NL80211_RRF_AUTO_BW
;
274 if (qflags
& QLINK_RRF_IR_CONCURRENT
)
275 flags
|= NL80211_RRF_IR_CONCURRENT
;
277 if (qflags
& QLINK_RRF_NO_HT40MINUS
)
278 flags
|= NL80211_RRF_NO_HT40MINUS
;
280 if (qflags
& QLINK_RRF_NO_HT40PLUS
)
281 flags
|= NL80211_RRF_NO_HT40PLUS
;
283 if (qflags
& QLINK_RRF_NO_80MHZ
)
284 flags
|= NL80211_RRF_NO_80MHZ
;
286 if (qflags
& QLINK_RRF_NO_160MHZ
)
287 flags
|= NL80211_RRF_NO_160MHZ
;
292 void qlink_utils_regrule_q2nl(struct ieee80211_reg_rule
*rule
,
293 const struct qlink_tlv_reg_rule
*tlv
)
295 rule
->freq_range
.start_freq_khz
= le32_to_cpu(tlv
->start_freq_khz
);
296 rule
->freq_range
.end_freq_khz
= le32_to_cpu(tlv
->end_freq_khz
);
297 rule
->freq_range
.max_bandwidth_khz
=
298 le32_to_cpu(tlv
->max_bandwidth_khz
);
299 rule
->power_rule
.max_antenna_gain
= le32_to_cpu(tlv
->max_antenna_gain
);
300 rule
->power_rule
.max_eirp
= le32_to_cpu(tlv
->max_eirp
);
301 rule
->dfs_cac_ms
= le32_to_cpu(tlv
->dfs_cac_ms
);
302 rule
->flags
= qtnf_reg_rule_flags_parse(le32_to_cpu(tlv
->flags
));