1 /* Implement 802.11d. */
2 
3 #include "dot11d.h"
4 
5 void Dot11d_Init(struct ieee80211_device *ieee)
6 {
7 	PRT_DOT11D_INFO pDot11dInfo = GET_DOT11D_INFO(ieee);
8 
9 	pDot11dInfo->bEnabled = false;
10 
11 	pDot11dInfo->State = DOT11D_STATE_NONE;
12 	pDot11dInfo->CountryIeLen = 0;
13 	memset(pDot11dInfo->channel_map, 0, MAX_CHANNEL_NUMBER+1);
14 	memset(pDot11dInfo->MaxTxPwrDbmList, 0xFF, MAX_CHANNEL_NUMBER+1);
15 	RESET_CIE_WATCHDOG(ieee);
16 
17 	netdev_info(ieee->dev, "Dot11d_Init()\n");
18 }
19 EXPORT_SYMBOL(Dot11d_Init);
20 
21 /* Reset to the state as we are just entering a regulatory domain. */
22 void Dot11d_Reset(struct ieee80211_device *ieee)
23 {
24 	u32 i;
25 	PRT_DOT11D_INFO pDot11dInfo = GET_DOT11D_INFO(ieee);
26 	/* Clear old channel map */
27 	memset(pDot11dInfo->channel_map, 0, MAX_CHANNEL_NUMBER+1);
28 	memset(pDot11dInfo->MaxTxPwrDbmList, 0xFF, MAX_CHANNEL_NUMBER+1);
29 	/* Set new channel map */
30 	for (i = 1; i <= 11; i++)
31 		(pDot11dInfo->channel_map)[i] = 1;
32 
33 	for (i = 12; i <= 14; i++)
34 		(pDot11dInfo->channel_map)[i] = 2;
35 
36 	pDot11dInfo->State = DOT11D_STATE_NONE;
37 	pDot11dInfo->CountryIeLen = 0;
38 	RESET_CIE_WATCHDOG(ieee);
39 }
40 EXPORT_SYMBOL(Dot11d_Reset);
41 
42 /*
43  * Update country IE from Beacon or Probe Resopnse and configure PHY for
44  * operation in the regulatory domain.
45  *
46  * TODO: Configure Tx power.
47  * Assumption:
48  * 1. IS_DOT11D_ENABLE() is TRUE.
49  * 2. Input IE is an valid one.
50  */
51 void Dot11d_UpdateCountryIe(struct ieee80211_device *dev, u8 *pTaddr,
52 			    u16 CoutryIeLen, u8 *pCoutryIe)
53 {
54 	PRT_DOT11D_INFO pDot11dInfo = GET_DOT11D_INFO(dev);
55 	u8 i, j, NumTriples, MaxChnlNum;
56 	PCHNL_TXPOWER_TRIPLE pTriple;
57 
58 	memset(pDot11dInfo->channel_map, 0, MAX_CHANNEL_NUMBER+1);
59 	memset(pDot11dInfo->MaxTxPwrDbmList, 0xFF, MAX_CHANNEL_NUMBER+1);
60 	MaxChnlNum = 0;
61 	NumTriples = (CoutryIeLen - 3) / 3; /* skip 3-byte country string. */
62 	pTriple = (PCHNL_TXPOWER_TRIPLE)(pCoutryIe + 3);
63 	for (i = 0; i < NumTriples; i++) {
64 		if (MaxChnlNum >= pTriple->FirstChnl) {
65 			/* It is not in a monotonically increasing order, so
66 			 * stop processing.
67 			 */
68 			netdev_err(dev->dev, "Dot11d_UpdateCountryIe(): Invalid country IE, skip it........1\n");
69 			return;
70 		}
71 		if (MAX_CHANNEL_NUMBER < (pTriple->FirstChnl + pTriple->NumChnls)) {
72 			/* It is not a valid set of channel id, so stop
73 			 * processing.
74 			 */
75 			netdev_err(dev->dev, "Dot11d_UpdateCountryIe(): Invalid country IE, skip it........2\n");
76 			return;
77 		}
78 
79 		for (j = 0; j < pTriple->NumChnls; j++) {
80 			pDot11dInfo->channel_map[pTriple->FirstChnl + j] = 1;
81 			pDot11dInfo->MaxTxPwrDbmList[pTriple->FirstChnl + j] = pTriple->MaxTxPowerInDbm;
82 			MaxChnlNum = pTriple->FirstChnl + j;
83 		}
84 
85 		pTriple = (PCHNL_TXPOWER_TRIPLE)((u8 *)pTriple + 3);
86 	}
87 	netdev_info(dev->dev, "Channel List:");
88 	for (i = 1; i <= MAX_CHANNEL_NUMBER; i++)
89 		if (pDot11dInfo->channel_map[i] > 0)
90 			netdev_info(dev->dev, " %d", i);
91 	netdev_info(dev->dev, "\n");
92 
93 	UPDATE_CIE_SRC(dev, pTaddr);
94 
95 	pDot11dInfo->CountryIeLen = CoutryIeLen;
96 	memcpy(pDot11dInfo->CountryIeBuf, pCoutryIe, CoutryIeLen);
97 	pDot11dInfo->State = DOT11D_STATE_LEARNED;
98 }
99 EXPORT_SYMBOL(Dot11d_UpdateCountryIe);
100 
101 u8 DOT11D_GetMaxTxPwrInDbm(struct ieee80211_device *dev, u8 Channel)
102 {
103 	PRT_DOT11D_INFO pDot11dInfo = GET_DOT11D_INFO(dev);
104 	u8 MaxTxPwrInDbm = 255;
105 
106 	if (Channel > MAX_CHANNEL_NUMBER) {
107 		netdev_err(dev->dev, "DOT11D_GetMaxTxPwrInDbm(): Invalid Channel\n");
108 		return MaxTxPwrInDbm;
109 	}
110 	if (pDot11dInfo->channel_map[Channel])
111 		MaxTxPwrInDbm = pDot11dInfo->MaxTxPwrDbmList[Channel];
112 
113 	return MaxTxPwrInDbm;
114 }
115 EXPORT_SYMBOL(DOT11D_GetMaxTxPwrInDbm);
116 
117 void DOT11D_ScanComplete(struct ieee80211_device *dev)
118 {
119 	PRT_DOT11D_INFO pDot11dInfo = GET_DOT11D_INFO(dev);
120 
121 	switch (pDot11dInfo->State) {
122 	case DOT11D_STATE_LEARNED:
123 		pDot11dInfo->State = DOT11D_STATE_DONE;
124 		break;
125 
126 	case DOT11D_STATE_DONE:
127 		if (GET_CIE_WATCHDOG(dev) == 0) {
128 			/* Reset country IE if previous one is gone. */
129 			Dot11d_Reset(dev);
130 		}
131 		break;
132 	case DOT11D_STATE_NONE:
133 		break;
134 	}
135 }
136 EXPORT_SYMBOL(DOT11D_ScanComplete);
137 
138 int IsLegalChannel(struct ieee80211_device *dev, u8 channel)
139 {
140 	PRT_DOT11D_INFO pDot11dInfo = GET_DOT11D_INFO(dev);
141 
142 	if (channel > MAX_CHANNEL_NUMBER) {
143 		netdev_err(dev->dev, "IsLegalChannel(): Invalid Channel\n");
144 		return 0;
145 	}
146 	if (pDot11dInfo->channel_map[channel] > 0)
147 		return 1;
148 	return 0;
149 }
150 EXPORT_SYMBOL(IsLegalChannel);
151 
152 int ToLegalChannel(struct ieee80211_device *dev, u8 channel)
153 {
154 	PRT_DOT11D_INFO pDot11dInfo = GET_DOT11D_INFO(dev);
155 	u8 default_chn = 0;
156 	u32 i = 0;
157 
158 	for (i = 1; i <= MAX_CHANNEL_NUMBER; i++) {
159 		if (pDot11dInfo->channel_map[i] > 0) {
160 			default_chn = i;
161 			break;
162 		}
163 	}
164 
165 	if (channel > MAX_CHANNEL_NUMBER) {
166 		netdev_err(dev->dev, "IsLegalChannel(): Invalid Channel\n");
167 		return default_chn;
168 	}
169 
170 	if (pDot11dInfo->channel_map[channel] > 0)
171 		return channel;
172 
173 	return default_chn;
174 }
175 EXPORT_SYMBOL(ToLegalChannel);
176