blob: ee42dec669f51eacd49930d1d8aab9af58e48b8d [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>
Nathan Harold1a371532017-01-30 12:30:48 -080029
30#include "NetdConstants.h"
ludi6e8eccd2017-08-14 14:40:37 -070031#include "netdutils/Status.h"
Nathan Harold1a371532017-01-30 12:30:48 -080032
33namespace android {
34namespace net {
35
Nathan Harold39b5df42017-08-02 18:45:25 -070036// Exposed for testing
37extern const uint32_t ALGO_MASK_AUTH_ALL;
38// Exposed for testing
39extern const uint32_t ALGO_MASK_CRYPT_ALL;
40// Exposed for testing
Benedict Wongbe65b432017-08-22 21:43:14 -070041extern const uint32_t ALGO_MASK_AEAD_ALL;
42// Exposed for testing
Nathan Harold39b5df42017-08-02 18:45:25 -070043extern const uint8_t REPLAY_WINDOW_SIZE;
44
Nathan Harold1a371532017-01-30 12:30:48 -080045// Suggest we avoid the smallest and largest ints
46class XfrmMessage;
47class TransportModeSecurityAssociation;
48
49class XfrmSocket {
50public:
51 virtual void close() {
Nathan Harold420ceac2017-04-05 19:36:59 -070052 if (mSock >= 0) {
Nathan Harold1a371532017-01-30 12:30:48 -080053 ::close(mSock);
54 }
55 mSock = -1;
56 }
57
ludi6e8eccd2017-08-14 14:40:37 -070058 virtual netdutils::Status open() = 0;
Nathan Harold1a371532017-01-30 12:30:48 -080059
60 virtual ~XfrmSocket() { close(); }
61
ludie51e3862017-08-15 19:28:12 -070062 // Sends the netlink message contained in iovecs. This populates iovecs[0] with
63 // a valid netlink message header.
ludi6e8eccd2017-08-14 14:40:37 -070064 virtual netdutils::Status sendMessage(uint16_t nlMsgType, uint16_t nlMsgFlags,
65 uint16_t nlMsgSeqNum,
66 std::vector<iovec>* iovecs) const = 0;
Nathan Harold1a371532017-01-30 12:30:48 -080067
68protected:
69 int mSock;
70};
71
72enum struct XfrmDirection : uint8_t {
73 IN = XFRM_POLICY_IN,
74 OUT = XFRM_POLICY_OUT,
75 FORWARD = XFRM_POLICY_FWD,
76 MASK = XFRM_POLICY_MASK,
77};
78
79enum struct XfrmMode : uint8_t {
80 TRANSPORT = XFRM_MODE_TRANSPORT,
81 TUNNEL = XFRM_MODE_TUNNEL,
82};
83
Nathan Harold420ceac2017-04-05 19:36:59 -070084enum struct XfrmEncapType : uint16_t {
85 NONE = 0,
86 ESPINUDP_NON_IKE = UDP_ENCAP_ESPINUDP_NON_IKE,
87 ESPINUDP = UDP_ENCAP_ESPINUDP
88};
89
Nathan Harold1a371532017-01-30 12:30:48 -080090struct XfrmAlgo {
91 std::string name;
92 std::vector<uint8_t> key;
93 uint16_t truncLenBits;
94};
95
Nathan Harold420ceac2017-04-05 19:36:59 -070096struct XfrmEncap {
97 XfrmEncapType type;
98 uint16_t srcPort;
99 uint16_t dstPort;
100};
101
Nathan Harold7441c692017-12-12 21:25:01 -0800102// minimally sufficient structure to match either an SA or a Policy
103struct XfrmId {
Nathan Harold1a371532017-01-30 12:30:48 -0800104 xfrm_address_t dstAddr; // network order
105 xfrm_address_t srcAddr;
106 int addrFamily; // AF_INET or AF_INET6
107 int transformId; // requestId
108 int spi;
109};
110
Nathan Harold7441c692017-12-12 21:25:01 -0800111struct XfrmSaInfo : XfrmId {
Nathan Harold1a371532017-01-30 12:30:48 -0800112 XfrmAlgo auth;
113 XfrmAlgo crypt;
Benedict Wongbe65b432017-08-22 21:43:14 -0700114 XfrmAlgo aead;
Nathan Harold1a371532017-01-30 12:30:48 -0800115 int netId;
116 XfrmMode mode;
Nathan Harold420ceac2017-04-05 19:36:59 -0700117 XfrmEncap encap;
Nathan Harold1a371532017-01-30 12:30:48 -0800118};
119
120class XfrmController {
121public:
122 XfrmController();
123
Benedict Wongb2daefb2017-12-06 22:05:46 -0800124 netdutils::Status ipSecSetEncapSocketOwner(const android::base::unique_fd& socket, int newUid,
125 uid_t callerUid);
126
Nathan Haroldda54f122018-01-09 16:42:57 -0800127 netdutils::Status ipSecAllocateSpi(int32_t transformId, const std::string& localAddress,
ludi6e8eccd2017-08-14 14:40:37 -0700128 const std::string& remoteAddress, int32_t inSpi,
129 int32_t* outSpi);
Nathan Harold1a371532017-01-30 12:30:48 -0800130
ludi6e8eccd2017-08-14 14:40:37 -0700131 netdutils::Status ipSecAddSecurityAssociation(
Nathan Haroldda54f122018-01-09 16:42:57 -0800132 int32_t transformId, int32_t mode, const std::string& sourceAddress,
133 const std::string& destinationAddress, int64_t underlyingNetworkHandle, int32_t spi,
Nathan Harold1a371532017-01-30 12:30:48 -0800134 const std::string& authAlgo, const std::vector<uint8_t>& authKey, int32_t authTruncBits,
135 const std::string& cryptAlgo, const std::vector<uint8_t>& cryptKey, int32_t cryptTruncBits,
Benedict Wongbe65b432017-08-22 21:43:14 -0700136 const std::string& aeadAlgo, const std::vector<uint8_t>& aeadKey, int32_t aeadIcvBits,
ludiec836052017-05-20 14:17:05 -0700137 int32_t encapType, int32_t encapLocalPort, int32_t encapRemotePort);
Nathan Harold1a371532017-01-30 12:30:48 -0800138
Nathan Haroldda54f122018-01-09 16:42:57 -0800139 netdutils::Status ipSecDeleteSecurityAssociation(int32_t transformId,
ludi6e8eccd2017-08-14 14:40:37 -0700140 const std::string& localAddress,
141 const std::string& remoteAddress, int32_t spi);
Nathan Harold1a371532017-01-30 12:30:48 -0800142
ludi6e8eccd2017-08-14 14:40:37 -0700143 netdutils::Status ipSecApplyTransportModeTransform(const android::base::unique_fd& socket,
144 int32_t transformId, int32_t direction,
145 const std::string& localAddress,
146 const std::string& remoteAddress,
147 int32_t spi);
Nathan Harold1a371532017-01-30 12:30:48 -0800148
ludi6e8eccd2017-08-14 14:40:37 -0700149 netdutils::Status ipSecRemoveTransportModeTransform(const android::base::unique_fd& socket);
Nathan Harold1a371532017-01-30 12:30:48 -0800150
Jonathan Basseric6461a62017-08-31 14:47:33 -0700151 // Some XFRM netlink attributes comprise a header, a struct, and some data
152 // after the struct. We wrap all of those in one struct for easier
153 // marshalling. The structs below must be ABI compatible with the kernel and
154 // are composed from kernel structures; thus, they use the kernel naming
155 // convention.
156
Nathan Harold39b5df42017-08-02 18:45:25 -0700157 // Exposed for testing
Benedict Wong4f60ee12017-10-10 12:27:25 -0700158 static constexpr size_t MAX_KEY_LENGTH = 128;
Nathan Harold39b5df42017-08-02 18:45:25 -0700159
Jonathan Basseric6461a62017-08-31 14:47:33 -0700160 // Container for the content of an XFRMA_ALG_CRYPT netlink attribute.
Nathan Harold39b5df42017-08-02 18:45:25 -0700161 // Exposed for testing
162 struct nlattr_algo_crypt {
163 nlattr hdr;
164 xfrm_algo crypt;
Benedict Wong4f60ee12017-10-10 12:27:25 -0700165 uint8_t key[MAX_KEY_LENGTH];
Nathan Harold39b5df42017-08-02 18:45:25 -0700166 };
167
Jonathan Basseric6461a62017-08-31 14:47:33 -0700168 // Container for the content of an XFRMA_ALG_AUTH_TRUNC netlink attribute.
Nathan Harold39b5df42017-08-02 18:45:25 -0700169 // Exposed for testing
170 struct nlattr_algo_auth {
171 nlattr hdr;
172 xfrm_algo_auth auth;
Benedict Wong4f60ee12017-10-10 12:27:25 -0700173 uint8_t key[MAX_KEY_LENGTH];
Nathan Harold39b5df42017-08-02 18:45:25 -0700174 };
175
Jonathan Basseric6461a62017-08-31 14:47:33 -0700176 // Container for the content of an XFRMA_TMPL netlink attribute.
Nathan Harold39b5df42017-08-02 18:45:25 -0700177 // Exposed for testing
Benedict Wongbe65b432017-08-22 21:43:14 -0700178 struct nlattr_algo_aead {
179 nlattr hdr;
180 xfrm_algo_aead aead;
Benedict Wong4f60ee12017-10-10 12:27:25 -0700181 uint8_t key[MAX_KEY_LENGTH];
Benedict Wongbe65b432017-08-22 21:43:14 -0700182 };
183
184 // Exposed for testing
Nathan Harold39b5df42017-08-02 18:45:25 -0700185 struct nlattr_user_tmpl {
186 nlattr hdr;
187 xfrm_user_tmpl tmpl;
188 };
189
Jonathan Basseric6461a62017-08-31 14:47:33 -0700190 // Container for the content of an XFRMA_ENCAP netlink attribute.
Nathan Harold39b5df42017-08-02 18:45:25 -0700191 // Exposed for testing
192 struct nlattr_encap_tmpl {
193 nlattr hdr;
194 xfrm_encap_tmpl tmpl;
195 };
196
Nathan Harold1a371532017-01-30 12:30:48 -0800197private:
Nathan Harolde2dd4c72017-04-19 11:09:11 -0700198/*
199 * Below is a redefinition of the xfrm_usersa_info struct that is part
200 * of the Linux uapi <linux/xfrm.h> to align the structures to a 64-bit
201 * boundary.
202 */
203#ifdef NETLINK_COMPAT32
204 // Shadow the kernel definition of xfrm_usersa_info with a 64-bit aligned version
205 struct xfrm_usersa_info : ::xfrm_usersa_info {
206 } __attribute__((aligned(8)));
207 // Shadow the kernel's version, using the aligned version of xfrm_usersa_info
208 struct xfrm_userspi_info {
209 struct xfrm_usersa_info info;
210 __u32 min;
211 __u32 max;
212 };
213
214 /*
215 * Anyone who encounters a failure when sending netlink messages should look here
216 * first. Hitting the static_assert() below should be a strong hint that Android
217 * IPsec will probably not work with your current settings.
218 *
219 * Again, experimentally determined, the "flags" field should be the first byte in
220 * the final word of the xfrm_usersa_info struct. The check validates the size of
221 * the padding to be 7.
222 *
223 * This padding is verified to be correct on gcc/x86_64 kernel, and clang/x86 userspace.
224 */
225 static_assert(sizeof(::xfrm_usersa_info) % 8 != 0, "struct xfrm_usersa_info has changed "
226 "alignment. Please consider whether this "
227 "patch is needed.");
228 static_assert(sizeof(xfrm_usersa_info) - offsetof(xfrm_usersa_info, flags) == 8,
229 "struct xfrm_usersa_info probably misaligned with kernel struct.");
230 static_assert(sizeof(xfrm_usersa_info) % 8 == 0, "struct xfrm_usersa_info_t is not 64-bit "
231 "aligned. Please consider whether this patch "
232 "is needed.");
233 static_assert(sizeof(::xfrm_userspi_info) - sizeof(::xfrm_usersa_info) ==
234 sizeof(xfrm_userspi_info) - sizeof(xfrm_usersa_info),
235 "struct xfrm_userspi_info has changed and does not match the kernel struct.");
236#endif
237
Nathan Harold7441c692017-12-12 21:25:01 -0800238 // helper function for filling in the XfrmId (and XfrmSaInfo) structure
Nathan Haroldda54f122018-01-09 16:42:57 -0800239 static netdutils::Status fillXfrmId(const std::string& sourceAddress,
240 const std::string& destinationAddress, int32_t spi,
241 int32_t transformId, XfrmId* xfrmId);
Nathan Harold1a371532017-01-30 12:30:48 -0800242
243 // Top level functions for managing a Transport Mode Transform
ludi6e8eccd2017-08-14 14:40:37 -0700244 static netdutils::Status addTransportModeTransform(const XfrmSaInfo& record);
Nathan Harold1a371532017-01-30 12:30:48 -0800245 static int removeTransportModeTransform(const XfrmSaInfo& record);
246
247 // TODO(messagerefactor): FACTOR OUT ALL MESSAGE BUILDING CODE BELOW HERE
248 // Shared between SA and SP
249 static void fillTransportModeSelector(const XfrmSaInfo& record, xfrm_selector* selector);
250
251 // Shared between Transport and Tunnel Mode
252 static int fillNlAttrXfrmAlgoEnc(const XfrmAlgo& in_algo, nlattr_algo_crypt* algo);
253 static int fillNlAttrXfrmAlgoAuth(const XfrmAlgo& in_algo, nlattr_algo_auth* algo);
Benedict Wongbe65b432017-08-22 21:43:14 -0700254 static int fillNlAttrXfrmAlgoAead(const XfrmAlgo& in_algo, nlattr_algo_aead* algo);
Nathan Harold420ceac2017-04-05 19:36:59 -0700255 static int fillNlAttrXfrmEncapTmpl(const XfrmSaInfo& record, nlattr_encap_tmpl* tmpl);
Nathan Harold1a371532017-01-30 12:30:48 -0800256
Di Lu0e16b5e2018-01-12 10:37:53 -0800257 // Functions for updating a Transport Mode SA
258 static netdutils::Status updateSecurityAssociation(const XfrmSaInfo& record,
manojboopathi4d2d6f12017-12-06 11:11:31 -0800259 const XfrmSocket& sock);
Nathan Harold1a371532017-01-30 12:30:48 -0800260 static int fillUserSaInfo(const XfrmSaInfo& record, xfrm_usersa_info* usersa);
261
262 // Functions for deleting a Transport Mode SA
Nathan Harold7441c692017-12-12 21:25:01 -0800263 static netdutils::Status deleteSecurityAssociation(const XfrmId& record,
ludi6e8eccd2017-08-14 14:40:37 -0700264 const XfrmSocket& sock);
Nathan Harold7441c692017-12-12 21:25:01 -0800265 static int fillUserSaId(const XfrmId& record, xfrm_usersa_id* said);
Nathan Harold1a371532017-01-30 12:30:48 -0800266 static int fillUserTemplate(const XfrmSaInfo& record, xfrm_user_tmpl* tmpl);
Nathan Haroldda54f122018-01-09 16:42:57 -0800267 static int fillTransportModeUserSpInfo(const XfrmSaInfo& record, XfrmDirection direction,
268 xfrm_userpolicy_info* usersp);
Nathan Harold1a371532017-01-30 12:30:48 -0800269
ludi6e8eccd2017-08-14 14:40:37 -0700270 static netdutils::Status allocateSpi(const XfrmSaInfo& record, uint32_t minSpi, uint32_t maxSpi,
271 uint32_t* outSpi, const XfrmSocket& sock);
Nathan Harold1a371532017-01-30 12:30:48 -0800272
273 // END TODO(messagerefactor)
274};
275
276} // namespace net
277} // namespace android
278
279#endif /* !defined(XFRM_CONTROLLER_H) */