blob: 0a582506bdc765c02d63cc394c183ac76922a3a3 [file] [log] [blame]
Nathan Harold1a371532017-01-30 12:30:48 -08001/*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16#ifndef _XFRM_CONTROLLER_H
17#define _XFRM_CONTROLLER_H
18
19#include <atomic>
20#include <list>
21#include <map>
22#include <string>
23#include <utility> // for pair
24
25#include <linux/netlink.h>
Nathan Harold420ceac2017-04-05 19:36:59 -070026#include <linux/udp.h>
Nathan Harold1a371532017-01-30 12:30:48 -080027#include <linux/xfrm.h>
28#include <sysutils/SocketClient.h>
29#include <utils/RWLock.h>
30
31#include "NetdConstants.h"
ludi6e8eccd2017-08-14 14:40:37 -070032#include "netdutils/Status.h"
Nathan Harold1a371532017-01-30 12:30:48 -080033
34namespace android {
35namespace net {
36
Nathan Harold39b5df42017-08-02 18:45:25 -070037// Exposed for testing
38extern const uint32_t ALGO_MASK_AUTH_ALL;
39// Exposed for testing
40extern const uint32_t ALGO_MASK_CRYPT_ALL;
41// Exposed for testing
42extern const uint8_t REPLAY_WINDOW_SIZE;
43
Nathan Harold1a371532017-01-30 12:30:48 -080044// Suggest we avoid the smallest and largest ints
45class XfrmMessage;
46class TransportModeSecurityAssociation;
47
48class XfrmSocket {
49public:
50 virtual void close() {
Nathan Harold420ceac2017-04-05 19:36:59 -070051 if (mSock >= 0) {
Nathan Harold1a371532017-01-30 12:30:48 -080052 ::close(mSock);
53 }
54 mSock = -1;
55 }
56
ludi6e8eccd2017-08-14 14:40:37 -070057 virtual netdutils::Status open() = 0;
Nathan Harold1a371532017-01-30 12:30:48 -080058
59 virtual ~XfrmSocket() { close(); }
60
ludie51e3862017-08-15 19:28:12 -070061 // Sends the netlink message contained in iovecs. This populates iovecs[0] with
62 // a valid netlink message header.
ludi6e8eccd2017-08-14 14:40:37 -070063 virtual netdutils::Status sendMessage(uint16_t nlMsgType, uint16_t nlMsgFlags,
64 uint16_t nlMsgSeqNum,
65 std::vector<iovec>* iovecs) const = 0;
Nathan Harold1a371532017-01-30 12:30:48 -080066
67protected:
68 int mSock;
69};
70
71enum struct XfrmDirection : uint8_t {
72 IN = XFRM_POLICY_IN,
73 OUT = XFRM_POLICY_OUT,
74 FORWARD = XFRM_POLICY_FWD,
75 MASK = XFRM_POLICY_MASK,
76};
77
78enum struct XfrmMode : uint8_t {
79 TRANSPORT = XFRM_MODE_TRANSPORT,
80 TUNNEL = XFRM_MODE_TUNNEL,
81};
82
Nathan Harold420ceac2017-04-05 19:36:59 -070083enum struct XfrmEncapType : uint16_t {
84 NONE = 0,
85 ESPINUDP_NON_IKE = UDP_ENCAP_ESPINUDP_NON_IKE,
86 ESPINUDP = UDP_ENCAP_ESPINUDP
87};
88
Nathan Harold1a371532017-01-30 12:30:48 -080089struct XfrmAlgo {
90 std::string name;
91 std::vector<uint8_t> key;
92 uint16_t truncLenBits;
93};
94
Nathan Harold420ceac2017-04-05 19:36:59 -070095struct XfrmEncap {
96 XfrmEncapType type;
97 uint16_t srcPort;
98 uint16_t dstPort;
99};
100
Nathan Harold1a371532017-01-30 12:30:48 -0800101struct XfrmSaId {
102 XfrmDirection direction;
103 xfrm_address_t dstAddr; // network order
104 xfrm_address_t srcAddr;
105 int addrFamily; // AF_INET or AF_INET6
106 int transformId; // requestId
107 int spi;
108};
109
110struct XfrmSaInfo : XfrmSaId {
111 XfrmAlgo auth;
112 XfrmAlgo crypt;
113 int netId;
114 XfrmMode mode;
Nathan Harold420ceac2017-04-05 19:36:59 -0700115 XfrmEncap encap;
Nathan Harold1a371532017-01-30 12:30:48 -0800116};
117
118class XfrmController {
119public:
120 XfrmController();
121
ludi6e8eccd2017-08-14 14:40:37 -0700122 netdutils::Status ipSecAllocateSpi(int32_t transformId, int32_t direction,
123 const std::string& localAddress,
124 const std::string& remoteAddress, int32_t inSpi,
125 int32_t* outSpi);
Nathan Harold1a371532017-01-30 12:30:48 -0800126
ludi6e8eccd2017-08-14 14:40:37 -0700127 netdutils::Status ipSecAddSecurityAssociation(
Nathan Harold1a371532017-01-30 12:30:48 -0800128 int32_t transformId, int32_t mode, int32_t direction, const std::string& localAddress,
129 const std::string& remoteAddress, int64_t underlyingNetworkHandle, int32_t spi,
130 const std::string& authAlgo, const std::vector<uint8_t>& authKey, int32_t authTruncBits,
131 const std::string& cryptAlgo, const std::vector<uint8_t>& cryptKey, int32_t cryptTruncBits,
ludiec836052017-05-20 14:17:05 -0700132 int32_t encapType, int32_t encapLocalPort, int32_t encapRemotePort);
Nathan Harold1a371532017-01-30 12:30:48 -0800133
ludi6e8eccd2017-08-14 14:40:37 -0700134 netdutils::Status ipSecDeleteSecurityAssociation(int32_t transformId, int32_t direction,
135 const std::string& localAddress,
136 const std::string& remoteAddress, int32_t spi);
Nathan Harold1a371532017-01-30 12:30:48 -0800137
ludi6e8eccd2017-08-14 14:40:37 -0700138 netdutils::Status ipSecApplyTransportModeTransform(const android::base::unique_fd& socket,
139 int32_t transformId, int32_t direction,
140 const std::string& localAddress,
141 const std::string& remoteAddress,
142 int32_t spi);
Nathan Harold1a371532017-01-30 12:30:48 -0800143
ludi6e8eccd2017-08-14 14:40:37 -0700144 netdutils::Status ipSecRemoveTransportModeTransform(const android::base::unique_fd& socket);
Nathan Harold1a371532017-01-30 12:30:48 -0800145
Nathan Harold39b5df42017-08-02 18:45:25 -0700146 // Exposed for testing
147 static constexpr size_t MAX_ALGO_LENGTH = 128;
148
149 // Exposed for testing
150 struct nlattr_algo_crypt {
151 nlattr hdr;
152 xfrm_algo crypt;
153 uint8_t key[MAX_ALGO_LENGTH];
154 };
155
156 // Exposed for testing
157 struct nlattr_algo_auth {
158 nlattr hdr;
159 xfrm_algo_auth auth;
160 uint8_t key[MAX_ALGO_LENGTH];
161 };
162
163 // Exposed for testing
164 struct nlattr_user_tmpl {
165 nlattr hdr;
166 xfrm_user_tmpl tmpl;
167 };
168
169 // Exposed for testing
170 struct nlattr_encap_tmpl {
171 nlattr hdr;
172 xfrm_encap_tmpl tmpl;
173 };
174
Nathan Harold1a371532017-01-30 12:30:48 -0800175private:
176 // prevent concurrent modification of XFRM
177 android::RWLock mLock;
178
Nathan Harolde2dd4c72017-04-19 11:09:11 -0700179/*
180 * Below is a redefinition of the xfrm_usersa_info struct that is part
181 * of the Linux uapi <linux/xfrm.h> to align the structures to a 64-bit
182 * boundary.
183 */
184#ifdef NETLINK_COMPAT32
185 // Shadow the kernel definition of xfrm_usersa_info with a 64-bit aligned version
186 struct xfrm_usersa_info : ::xfrm_usersa_info {
187 } __attribute__((aligned(8)));
188 // Shadow the kernel's version, using the aligned version of xfrm_usersa_info
189 struct xfrm_userspi_info {
190 struct xfrm_usersa_info info;
191 __u32 min;
192 __u32 max;
193 };
194
195 /*
196 * Anyone who encounters a failure when sending netlink messages should look here
197 * first. Hitting the static_assert() below should be a strong hint that Android
198 * IPsec will probably not work with your current settings.
199 *
200 * Again, experimentally determined, the "flags" field should be the first byte in
201 * the final word of the xfrm_usersa_info struct. The check validates the size of
202 * the padding to be 7.
203 *
204 * This padding is verified to be correct on gcc/x86_64 kernel, and clang/x86 userspace.
205 */
206 static_assert(sizeof(::xfrm_usersa_info) % 8 != 0, "struct xfrm_usersa_info has changed "
207 "alignment. Please consider whether this "
208 "patch is needed.");
209 static_assert(sizeof(xfrm_usersa_info) - offsetof(xfrm_usersa_info, flags) == 8,
210 "struct xfrm_usersa_info probably misaligned with kernel struct.");
211 static_assert(sizeof(xfrm_usersa_info) % 8 == 0, "struct xfrm_usersa_info_t is not 64-bit "
212 "aligned. Please consider whether this patch "
213 "is needed.");
214 static_assert(sizeof(::xfrm_userspi_info) - sizeof(::xfrm_usersa_info) ==
215 sizeof(xfrm_userspi_info) - sizeof(xfrm_usersa_info),
216 "struct xfrm_userspi_info has changed and does not match the kernel struct.");
217#endif
218
Nathan Harold1a371532017-01-30 12:30:48 -0800219 // helper function for filling in the XfrmSaInfo structure
ludi6e8eccd2017-08-14 14:40:37 -0700220 static netdutils::Status fillXfrmSaId(int32_t direction, const std::string& localAddress,
221 const std::string& remoteAddress, int32_t spi,
222 XfrmSaId* xfrmId);
Nathan Harold1a371532017-01-30 12:30:48 -0800223
224 // Top level functions for managing a Transport Mode Transform
ludi6e8eccd2017-08-14 14:40:37 -0700225 static netdutils::Status addTransportModeTransform(const XfrmSaInfo& record);
Nathan Harold1a371532017-01-30 12:30:48 -0800226 static int removeTransportModeTransform(const XfrmSaInfo& record);
227
228 // TODO(messagerefactor): FACTOR OUT ALL MESSAGE BUILDING CODE BELOW HERE
229 // Shared between SA and SP
230 static void fillTransportModeSelector(const XfrmSaInfo& record, xfrm_selector* selector);
231
232 // Shared between Transport and Tunnel Mode
233 static int fillNlAttrXfrmAlgoEnc(const XfrmAlgo& in_algo, nlattr_algo_crypt* algo);
234 static int fillNlAttrXfrmAlgoAuth(const XfrmAlgo& in_algo, nlattr_algo_auth* algo);
Nathan Harold420ceac2017-04-05 19:36:59 -0700235 static int fillNlAttrXfrmEncapTmpl(const XfrmSaInfo& record, nlattr_encap_tmpl* tmpl);
Nathan Harold1a371532017-01-30 12:30:48 -0800236
237 // Functions for Creating a Transport Mode SA
ludi6e8eccd2017-08-14 14:40:37 -0700238 static netdutils::Status createTransportModeSecurityAssociation(const XfrmSaInfo& record,
239 const XfrmSocket& sock);
Nathan Harold1a371532017-01-30 12:30:48 -0800240 static int fillUserSaInfo(const XfrmSaInfo& record, xfrm_usersa_info* usersa);
241
242 // Functions for deleting a Transport Mode SA
ludi6e8eccd2017-08-14 14:40:37 -0700243 static netdutils::Status deleteSecurityAssociation(const XfrmSaId& record,
244 const XfrmSocket& sock);
Nathan Harold1a371532017-01-30 12:30:48 -0800245 static int fillUserSaId(const XfrmSaId& record, xfrm_usersa_id* said);
246 static int fillUserTemplate(const XfrmSaInfo& record, xfrm_user_tmpl* tmpl);
247 static int fillTransportModeUserSpInfo(const XfrmSaInfo& record, xfrm_userpolicy_info* usersp);
248
ludi6e8eccd2017-08-14 14:40:37 -0700249 static netdutils::Status allocateSpi(const XfrmSaInfo& record, uint32_t minSpi, uint32_t maxSpi,
250 uint32_t* outSpi, const XfrmSocket& sock);
Nathan Harold1a371532017-01-30 12:30:48 -0800251
252 // END TODO(messagerefactor)
253};
254
255} // namespace net
256} // namespace android
257
258#endif /* !defined(XFRM_CONTROLLER_H) */