1 /*
2 Copyright (c) 2013-2016, The Linux Foundation. All rights reserved.
3 
4 Redistribution and use in source and binary forms, with or without
5 modification, are permitted provided that the following conditions are
6 met:
7     * Redistributions of source code must retain the above copyright
8       notice, this list of conditions and the following disclaimer.
9     * Redistributions in binary form must reproduce the above
10       copyright notice, this list of conditions and the following
11       disclaimer in the documentation and/or other materials provided
12       with the distribution.
13     * Neither the name of The Linux Foundation nor the names of its
14       contributors may be used to endorse or promote products derived
15       from this software without specific prior written permission.
16 
17 THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
18 WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
20 ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
21 BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22 CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23 SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
24 BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
25 WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
26 OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27 IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 */
29 /*!
30 		@file
31 		IPACM_Wlan.h
32 
33 		@brief
34 		This file implements the WLAN iface functionality.
35 
36 		@Author
37 		Skylar Chang
38 
39 */
40 #ifndef IPACM_WLAN_H
41 #define IPACM_WLAN_H
42 
43 #include <stdio.h>
44 #include <IPACM_CmdQueue.h>
45 #include <linux/msm_ipa.h>
46 #include "IPACM_Routing.h"
47 #include "IPACM_Filtering.h"
48 #include "IPACM_Lan.h"
49 #include "IPACM_Iface.h"
50 #include "IPACM_Conntrack_NATApp.h"
51 
52 typedef struct _wlan_client_rt_hdl
53 {
54 	uint32_t wifi_rt_rule_hdl_v4;
55 	uint32_t wifi_rt_rule_hdl_v6[IPV6_NUM_ADDR];
56 	uint32_t wifi_rt_rule_hdl_v6_wan[IPV6_NUM_ADDR];
57 }wlan_client_rt_hdl;
58 
59 typedef struct _ipa_wlan_client
60 {
61 	ipacm_event_data_wlan_ex* p_hdr_info;
62 	uint8_t mac[IPA_MAC_ADDR_SIZE];
63 	uint32_t v4_addr;
64 	uint32_t v6_addr[IPV6_NUM_ADDR][4];
65 	uint32_t hdr_hdl_v4;
66 	uint32_t hdr_hdl_v6;
67 	bool route_rule_set_v4;
68 	int route_rule_set_v6;
69 	bool ipv4_set;
70 	int ipv6_set;
71 	bool ipv4_header_set;
72 	bool ipv6_header_set;
73 	bool power_save_set;
74 	wlan_client_rt_hdl wifi_rt_hdl[0]; /* depends on number of tx properties */
75 }ipa_wlan_client;
76 
77 /* wlan iface */
78 class IPACM_Wlan : public IPACM_Lan
79 {
80 
81 public:
82 
83 	IPACM_Wlan(int iface_index);
84 	virtual ~IPACM_Wlan(void);
85 
86 	static int total_num_wifi_clients;
87 
88 	void event_callback(ipa_cm_event_id event, void *data);
89 
90 	bool is_guest_ap();
91 
92 private:
93 
94 	bool m_is_guest_ap;
95 
96 	/* handle wlan access mode switch in ethernet bridging*/
97 	void eth_bridge_handle_wlan_mode_switch();
98 
99 
100 	int wlan_client_len;
101 	ipa_wlan_client *wlan_client;
102 
103 	int header_name_count;
104 	uint32_t num_wifi_client;
105 
106 	int wlan_ap_index;
107 
108 	static int num_wlan_ap_iface;
109 
110 	NatApp *Nat_App;
111 
112 	inline ipa_wlan_client* get_client_memptr(ipa_wlan_client *param, int cnt)
113 	{
114 	    char *ret = ((char *)param) + (wlan_client_len * cnt);
115 		return (ipa_wlan_client *)ret;
116 	}
117 
118 	inline int get_wlan_client_index(uint8_t *mac_addr)
119 	{
120 		int cnt;
121 		int num_wifi_client_tmp = num_wifi_client;
122 
123 		IPACMDBG_H("Passed MAC %02x:%02x:%02x:%02x:%02x:%02x\n",
124 						 mac_addr[0], mac_addr[1], mac_addr[2],
125 						 mac_addr[3], mac_addr[4], mac_addr[5]);
126 
127 		for(cnt = 0; cnt < num_wifi_client_tmp; cnt++)
128 		{
129 			IPACMDBG_H("stored MAC %02x:%02x:%02x:%02x:%02x:%02x\n",
130 							 get_client_memptr(wlan_client, cnt)->mac[0],
131 							 get_client_memptr(wlan_client, cnt)->mac[1],
132 							 get_client_memptr(wlan_client, cnt)->mac[2],
133 							 get_client_memptr(wlan_client, cnt)->mac[3],
134 							 get_client_memptr(wlan_client, cnt)->mac[4],
135 							 get_client_memptr(wlan_client, cnt)->mac[5]);
136 
137 			if(memcmp(get_client_memptr(wlan_client, cnt)->mac,
138 								mac_addr,
139 								sizeof(get_client_memptr(wlan_client, cnt)->mac)) == 0)
140 			{
141 				IPACMDBG_H("Matched client index: %d\n", cnt);
142 				return cnt;
143 			}
144 		}
145 
146 		return IPACM_INVALID_INDEX;
147 	}
148 
149 	inline int delete_default_qos_rtrules(int clt_indx, ipa_ip_type iptype)
150 	{
151 		uint32_t tx_index;
152 		uint32_t rt_hdl;
153 		int num_v6;
154 
155 		if(iptype == IPA_IP_v4)
156 		{
157 		     for(tx_index = 0; tx_index < iface_query->num_tx_props; tx_index++)
158 		     {
159 		        if((tx_prop->tx[tx_index].ip == IPA_IP_v4) && (get_client_memptr(wlan_client, clt_indx)->route_rule_set_v4==true)) /* for ipv4 */
160 			{
161 				IPACMDBG_H("Delete client index %d ipv4 Qos rules for tx:%d \n",clt_indx,tx_index);
162 				rt_hdl = get_client_memptr(wlan_client, clt_indx)->wifi_rt_hdl[tx_index].wifi_rt_rule_hdl_v4;
163 
164 				if(m_routing.DeleteRoutingHdl(rt_hdl, IPA_IP_v4) == false)
165 				{
166 					return IPACM_FAILURE;
167 				}
168 			}
169 		     } /* end of for loop */
170 
171 		     /* clean the 4 Qos ipv4 RT rules for client:clt_indx */
172 		     if(get_client_memptr(wlan_client, clt_indx)->route_rule_set_v4==true) /* for ipv4 */
173 		     {
174 				get_client_memptr(wlan_client, clt_indx)->route_rule_set_v4 = false;
175 		     }
176 		}
177 
178 		if(iptype == IPA_IP_v6)
179 		{
180 		    for(tx_index = 0; tx_index < iface_query->num_tx_props; tx_index++)
181 		    {
182 
183 				if((tx_prop->tx[tx_index].ip == IPA_IP_v6) && (get_client_memptr(wlan_client, clt_indx)->route_rule_set_v6 != 0)) /* for ipv6 */
184 				{
185 					for(num_v6 =0;num_v6 < get_client_memptr(wlan_client, clt_indx)->route_rule_set_v6;num_v6++)
186 					{
187 						IPACMDBG_H("Delete client index %d ipv6 Qos rules for %d-st ipv6 for tx:%d\n", clt_indx,num_v6,tx_index);
188 						rt_hdl = get_client_memptr(wlan_client, clt_indx)->wifi_rt_hdl[tx_index].wifi_rt_rule_hdl_v6[num_v6];
189 						if(m_routing.DeleteRoutingHdl(rt_hdl, IPA_IP_v6) == false)
190 						{
191 							return IPACM_FAILURE;
192 						}
193 
194 						rt_hdl = get_client_memptr(wlan_client, clt_indx)->wifi_rt_hdl[tx_index].wifi_rt_rule_hdl_v6_wan[num_v6];
195 						if(m_routing.DeleteRoutingHdl(rt_hdl, IPA_IP_v6) == false)
196 						{
197 							return IPACM_FAILURE;
198 						}
199 					}
200 
201 				}
202 			} /* end of for loop */
203 
204 		    /* clean the 4 Qos ipv6 RT rules for client:clt_indx */
205 		    if(get_client_memptr(wlan_client, clt_indx)->route_rule_set_v6 != 0) /* for ipv6 */
206 		    {
207 		                 get_client_memptr(wlan_client, clt_indx)->route_rule_set_v6 = 0;
208                     }
209 		}
210 
211 		return IPACM_SUCCESS;
212 	}
213 
214 	/* for handle wifi client initial,copy all partial headers (tx property) */
215 	int handle_wlan_client_init_ex(ipacm_event_data_wlan_ex *data);
216 
217 	/*handle wifi client */
218 	int handle_wlan_client_ipaddr(ipacm_event_data_all *data);
219 
220 	/*handle wifi client routing rule*/
221 	int handle_wlan_client_route_rule(uint8_t *mac_addr, ipa_ip_type iptype);
222 
223 	/*handle wifi client power-save mode*/
224 	int handle_wlan_client_pwrsave(uint8_t *mac_addr);
225 
226 	/*handle wifi client del mode*/
227 	int handle_wlan_client_down_evt(uint8_t *mac_addr);
228 
229 	/*handle wlan iface down event*/
230 	int handle_down_evt();
231 
232 	/*handle reset wifi-client rt-rules */
233 	int handle_wlan_client_reset_rt(ipa_ip_type iptype);
234 
235 	void handle_SCC_MCC_switch(ipa_ip_type);
236 
237 };
238 
239 
240 #endif /* IPACM_WLAN_H */
241