1 /* 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 17 package com.android.server; 18 19 import static android.system.OsConstants.AF_INET; 20 import static android.system.OsConstants.EADDRINUSE; 21 import static android.system.OsConstants.IPPROTO_UDP; 22 import static android.system.OsConstants.SOCK_DGRAM; 23 24 import static org.junit.Assert.assertEquals; 25 import static org.junit.Assert.assertNotEquals; 26 import static org.junit.Assert.assertNotNull; 27 import static org.junit.Assert.assertTrue; 28 import static org.junit.Assert.fail; 29 import static org.mockito.Matchers.anyInt; 30 import static org.mockito.Matchers.anyString; 31 import static org.mockito.Matchers.argThat; 32 import static org.mockito.Matchers.eq; 33 import static org.mockito.Mockito.mock; 34 import static org.mockito.Mockito.verify; 35 import static org.mockito.Mockito.when; 36 37 import android.content.Context; 38 import android.net.ConnectivityManager; 39 import android.net.INetd; 40 import android.net.IpSecAlgorithm; 41 import android.net.IpSecConfig; 42 import android.net.IpSecManager; 43 import android.net.IpSecSpiResponse; 44 import android.net.IpSecUdpEncapResponse; 45 import android.os.Binder; 46 import android.os.Build; 47 import android.os.ParcelFileDescriptor; 48 import android.os.Process; 49 import android.system.ErrnoException; 50 import android.system.Os; 51 import android.system.StructStat; 52 import android.util.Range; 53 54 import androidx.test.filters.SmallTest; 55 56 import com.android.testutils.DevSdkIgnoreRule; 57 import com.android.testutils.DevSdkIgnoreRunner; 58 59 import dalvik.system.SocketTagger; 60 61 import org.junit.Before; 62 import org.junit.Test; 63 import org.junit.runner.RunWith; 64 import org.mockito.ArgumentMatcher; 65 66 import java.io.FileDescriptor; 67 import java.net.InetAddress; 68 import java.net.ServerSocket; 69 import java.net.Socket; 70 import java.net.UnknownHostException; 71 import java.util.ArrayList; 72 import java.util.List; 73 74 /** Unit tests for {@link IpSecService}. */ 75 @SmallTest 76 @RunWith(DevSdkIgnoreRunner.class) 77 @DevSdkIgnoreRule.IgnoreUpTo(Build.VERSION_CODES.R) 78 public class IpSecServiceTest { 79 80 private static final int DROID_SPI = 0xD1201D; 81 private static final int MAX_NUM_ENCAP_SOCKETS = 100; 82 private static final int MAX_NUM_SPIS = 100; 83 private static final int TEST_UDP_ENCAP_INVALID_PORT = 100; 84 private static final int TEST_UDP_ENCAP_PORT_OUT_RANGE = 100000; 85 86 private static final InetAddress INADDR_ANY; 87 88 private static final byte[] AEAD_KEY = { 89 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 90 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 91 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 92 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F, 93 0x73, 0x61, 0x6C, 0x74 94 }; 95 private static final byte[] CRYPT_KEY = { 96 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 97 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 98 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 99 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F 100 }; 101 private static final byte[] AUTH_KEY = { 102 0x7A, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 103 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F, 104 0x7A, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 105 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7F 106 }; 107 108 private static final IpSecAlgorithm AUTH_ALGO = 109 new IpSecAlgorithm(IpSecAlgorithm.AUTH_HMAC_SHA256, AUTH_KEY, AUTH_KEY.length * 4); 110 private static final IpSecAlgorithm CRYPT_ALGO = 111 new IpSecAlgorithm(IpSecAlgorithm.CRYPT_AES_CBC, CRYPT_KEY); 112 private static final IpSecAlgorithm AEAD_ALGO = 113 new IpSecAlgorithm(IpSecAlgorithm.AUTH_CRYPT_AES_GCM, AEAD_KEY, 128); 114 115 static { 116 try { 117 INADDR_ANY = InetAddress.getByAddress(new byte[] {0, 0, 0, 0}); 118 } catch (UnknownHostException e) { 119 throw new RuntimeException(e); 120 } 121 } 122 123 Context mMockContext; 124 INetd mMockNetd; 125 IpSecService.IpSecServiceConfiguration mMockIpSecSrvConfig; 126 IpSecService mIpSecService; 127 128 @Before 129 public void setUp() throws Exception { 130 mMockContext = mock(Context.class); 131 mMockNetd = mock(INetd.class); 132 mMockIpSecSrvConfig = mock(IpSecService.IpSecServiceConfiguration.class); 133 mIpSecService = new IpSecService(mMockContext, mMockIpSecSrvConfig); 134 135 // Injecting mock netd 136 when(mMockIpSecSrvConfig.getNetdInstance()).thenReturn(mMockNetd); 137 } 138 139 @Test 140 public void testIpSecServiceCreate() throws InterruptedException { 141 IpSecService ipSecSrv = IpSecService.create(mMockContext); 142 assertNotNull(ipSecSrv); 143 } 144 145 @Test 146 public void testReleaseInvalidSecurityParameterIndex() throws Exception { 147 try { 148 mIpSecService.releaseSecurityParameterIndex(1); 149 fail("IllegalArgumentException not thrown"); 150 } catch (IllegalArgumentException e) { 151 } 152 } 153 154 /** This function finds an available port */ 155 int findUnusedPort() throws Exception { 156 // Get an available port. 157 ServerSocket s = new ServerSocket(0); 158 int port = s.getLocalPort(); 159 s.close(); 160 return port; 161 } 162 163 @Test 164 public void testOpenAndCloseUdpEncapsulationSocket() throws Exception { 165 int localport = -1; 166 IpSecUdpEncapResponse udpEncapResp = null; 167 168 for (int i = 0; i < IpSecService.MAX_PORT_BIND_ATTEMPTS; i++) { 169 localport = findUnusedPort(); 170 171 udpEncapResp = mIpSecService.openUdpEncapsulationSocket(localport, new Binder()); 172 assertNotNull(udpEncapResp); 173 if (udpEncapResp.status == IpSecManager.Status.OK) { 174 break; 175 } 176 177 // Else retry to reduce possibility for port-bind failures. 178 } 179 180 assertNotNull(udpEncapResp); 181 assertEquals(IpSecManager.Status.OK, udpEncapResp.status); 182 assertEquals(localport, udpEncapResp.port); 183 184 mIpSecService.closeUdpEncapsulationSocket(udpEncapResp.resourceId); 185 udpEncapResp.fileDescriptor.close(); 186 187 // Verify quota and RefcountedResource objects cleaned up 188 IpSecService.UserRecord userRecord = 189 mIpSecService.mUserResourceTracker.getUserRecord(Os.getuid()); 190 assertEquals(0, userRecord.mSocketQuotaTracker.mCurrent); 191 try { 192 userRecord.mEncapSocketRecords.getRefcountedResourceOrThrow(udpEncapResp.resourceId); 193 fail("Expected IllegalArgumentException on attempt to access deleted resource"); 194 } catch (IllegalArgumentException expected) { 195 196 } 197 } 198 199 @Test 200 public void testUdpEncapsulationSocketBinderDeath() throws Exception { 201 IpSecUdpEncapResponse udpEncapResp = 202 mIpSecService.openUdpEncapsulationSocket(0, new Binder()); 203 204 IpSecService.UserRecord userRecord = 205 mIpSecService.mUserResourceTracker.getUserRecord(Os.getuid()); 206 IpSecService.RefcountedResource refcountedRecord = 207 userRecord.mEncapSocketRecords.getRefcountedResourceOrThrow( 208 udpEncapResp.resourceId); 209 210 refcountedRecord.binderDied(); 211 212 // Verify quota and RefcountedResource objects cleaned up 213 assertEquals(0, userRecord.mSocketQuotaTracker.mCurrent); 214 try { 215 userRecord.mEncapSocketRecords.getRefcountedResourceOrThrow(udpEncapResp.resourceId); 216 fail("Expected IllegalArgumentException on attempt to access deleted resource"); 217 } catch (IllegalArgumentException expected) { 218 219 } 220 } 221 222 @Test 223 public void testOpenUdpEncapsulationSocketAfterClose() throws Exception { 224 IpSecUdpEncapResponse udpEncapResp = 225 mIpSecService.openUdpEncapsulationSocket(0, new Binder()); 226 assertNotNull(udpEncapResp); 227 assertEquals(IpSecManager.Status.OK, udpEncapResp.status); 228 int localport = udpEncapResp.port; 229 230 mIpSecService.closeUdpEncapsulationSocket(udpEncapResp.resourceId); 231 udpEncapResp.fileDescriptor.close(); 232 233 /** Check if localport is available. */ 234 FileDescriptor newSocket = Os.socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP); 235 Os.bind(newSocket, INADDR_ANY, localport); 236 Os.close(newSocket); 237 } 238 239 /** 240 * This function checks if the IpSecService holds the reserved port. If 241 * closeUdpEncapsulationSocket is not called, the socket cleanup should not be complete. 242 */ 243 @Test 244 public void testUdpEncapPortNotReleased() throws Exception { 245 IpSecUdpEncapResponse udpEncapResp = 246 mIpSecService.openUdpEncapsulationSocket(0, new Binder()); 247 assertNotNull(udpEncapResp); 248 assertEquals(IpSecManager.Status.OK, udpEncapResp.status); 249 int localport = udpEncapResp.port; 250 251 udpEncapResp.fileDescriptor.close(); 252 253 FileDescriptor newSocket = Os.socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP); 254 try { 255 Os.bind(newSocket, INADDR_ANY, localport); 256 fail("ErrnoException not thrown"); 257 } catch (ErrnoException e) { 258 assertEquals(EADDRINUSE, e.errno); 259 } 260 mIpSecService.closeUdpEncapsulationSocket(udpEncapResp.resourceId); 261 } 262 263 @Test 264 public void testOpenUdpEncapsulationSocketOnRandomPort() throws Exception { 265 IpSecUdpEncapResponse udpEncapResp = 266 mIpSecService.openUdpEncapsulationSocket(0, new Binder()); 267 assertNotNull(udpEncapResp); 268 assertEquals(IpSecManager.Status.OK, udpEncapResp.status); 269 assertNotEquals(0, udpEncapResp.port); 270 mIpSecService.closeUdpEncapsulationSocket(udpEncapResp.resourceId); 271 udpEncapResp.fileDescriptor.close(); 272 } 273 274 @Test 275 public void testOpenUdpEncapsulationSocketPortRange() throws Exception { 276 try { 277 mIpSecService.openUdpEncapsulationSocket(TEST_UDP_ENCAP_INVALID_PORT, new Binder()); 278 fail("IllegalArgumentException not thrown"); 279 } catch (IllegalArgumentException e) { 280 } 281 282 try { 283 mIpSecService.openUdpEncapsulationSocket(TEST_UDP_ENCAP_PORT_OUT_RANGE, new Binder()); 284 fail("IllegalArgumentException not thrown"); 285 } catch (IllegalArgumentException e) { 286 } 287 } 288 289 @Test 290 public void testOpenUdpEncapsulationSocketTwice() throws Exception { 291 IpSecUdpEncapResponse udpEncapResp = 292 mIpSecService.openUdpEncapsulationSocket(0, new Binder()); 293 assertNotNull(udpEncapResp); 294 assertEquals(IpSecManager.Status.OK, udpEncapResp.status); 295 int localport = udpEncapResp.port; 296 297 IpSecUdpEncapResponse testUdpEncapResp = 298 mIpSecService.openUdpEncapsulationSocket(localport, new Binder()); 299 assertEquals(IpSecManager.Status.RESOURCE_UNAVAILABLE, testUdpEncapResp.status); 300 301 mIpSecService.closeUdpEncapsulationSocket(udpEncapResp.resourceId); 302 udpEncapResp.fileDescriptor.close(); 303 } 304 305 @Test 306 public void testCloseInvalidUdpEncapsulationSocket() throws Exception { 307 try { 308 mIpSecService.closeUdpEncapsulationSocket(1); 309 fail("IllegalArgumentException not thrown"); 310 } catch (IllegalArgumentException e) { 311 } 312 } 313 314 @Test 315 public void testValidateAlgorithmsAuth() { 316 // Validate that correct algorithm type succeeds 317 IpSecConfig config = new IpSecConfig(); 318 config.setAuthentication(AUTH_ALGO); 319 mIpSecService.validateAlgorithms(config); 320 321 // Validate that incorrect algorithm types fails 322 for (IpSecAlgorithm algo : new IpSecAlgorithm[] {CRYPT_ALGO, AEAD_ALGO}) { 323 try { 324 config = new IpSecConfig(); 325 config.setAuthentication(algo); 326 mIpSecService.validateAlgorithms(config); 327 fail("Did not throw exception on invalid algorithm type"); 328 } catch (IllegalArgumentException expected) { 329 } 330 } 331 } 332 333 @Test 334 public void testValidateAlgorithmsCrypt() { 335 // Validate that correct algorithm type succeeds 336 IpSecConfig config = new IpSecConfig(); 337 config.setEncryption(CRYPT_ALGO); 338 mIpSecService.validateAlgorithms(config); 339 340 // Validate that incorrect algorithm types fails 341 for (IpSecAlgorithm algo : new IpSecAlgorithm[] {AUTH_ALGO, AEAD_ALGO}) { 342 try { 343 config = new IpSecConfig(); 344 config.setEncryption(algo); 345 mIpSecService.validateAlgorithms(config); 346 fail("Did not throw exception on invalid algorithm type"); 347 } catch (IllegalArgumentException expected) { 348 } 349 } 350 } 351 352 @Test 353 public void testValidateAlgorithmsAead() { 354 // Validate that correct algorithm type succeeds 355 IpSecConfig config = new IpSecConfig(); 356 config.setAuthenticatedEncryption(AEAD_ALGO); 357 mIpSecService.validateAlgorithms(config); 358 359 // Validate that incorrect algorithm types fails 360 for (IpSecAlgorithm algo : new IpSecAlgorithm[] {AUTH_ALGO, CRYPT_ALGO}) { 361 try { 362 config = new IpSecConfig(); 363 config.setAuthenticatedEncryption(algo); 364 mIpSecService.validateAlgorithms(config); 365 fail("Did not throw exception on invalid algorithm type"); 366 } catch (IllegalArgumentException expected) { 367 } 368 } 369 } 370 371 @Test 372 public void testValidateAlgorithmsAuthCrypt() { 373 // Validate that correct algorithm type succeeds 374 IpSecConfig config = new IpSecConfig(); 375 config.setAuthentication(AUTH_ALGO); 376 config.setEncryption(CRYPT_ALGO); 377 mIpSecService.validateAlgorithms(config); 378 } 379 380 @Test 381 public void testValidateAlgorithmsNoAlgorithms() { 382 IpSecConfig config = new IpSecConfig(); 383 try { 384 mIpSecService.validateAlgorithms(config); 385 fail("Expected exception; no algorithms specified"); 386 } catch (IllegalArgumentException expected) { 387 } 388 } 389 390 @Test 391 public void testValidateAlgorithmsAeadWithAuth() { 392 IpSecConfig config = new IpSecConfig(); 393 config.setAuthenticatedEncryption(AEAD_ALGO); 394 config.setAuthentication(AUTH_ALGO); 395 try { 396 mIpSecService.validateAlgorithms(config); 397 fail("Expected exception; both AEAD and auth algorithm specified"); 398 } catch (IllegalArgumentException expected) { 399 } 400 } 401 402 @Test 403 public void testValidateAlgorithmsAeadWithCrypt() { 404 IpSecConfig config = new IpSecConfig(); 405 config.setAuthenticatedEncryption(AEAD_ALGO); 406 config.setEncryption(CRYPT_ALGO); 407 try { 408 mIpSecService.validateAlgorithms(config); 409 fail("Expected exception; both AEAD and crypt algorithm specified"); 410 } catch (IllegalArgumentException expected) { 411 } 412 } 413 414 @Test 415 public void testValidateAlgorithmsAeadWithAuthAndCrypt() { 416 IpSecConfig config = new IpSecConfig(); 417 config.setAuthenticatedEncryption(AEAD_ALGO); 418 config.setAuthentication(AUTH_ALGO); 419 config.setEncryption(CRYPT_ALGO); 420 try { 421 mIpSecService.validateAlgorithms(config); 422 fail("Expected exception; AEAD, auth and crypt algorithm specified"); 423 } catch (IllegalArgumentException expected) { 424 } 425 } 426 427 @Test 428 public void testDeleteInvalidTransform() throws Exception { 429 try { 430 mIpSecService.deleteTransform(1); 431 fail("IllegalArgumentException not thrown"); 432 } catch (IllegalArgumentException e) { 433 } 434 } 435 436 @Test 437 public void testRemoveTransportModeTransform() throws Exception { 438 Socket socket = new Socket(); 439 socket.bind(null); 440 ParcelFileDescriptor pfd = ParcelFileDescriptor.fromSocket(socket); 441 mIpSecService.removeTransportModeTransforms(pfd); 442 443 verify(mMockNetd).ipSecRemoveTransportModeTransform(pfd); 444 } 445 446 @Test 447 public void testValidateIpAddresses() throws Exception { 448 String[] invalidAddresses = 449 new String[] {"www.google.com", "::", "2001::/64", "0.0.0.0", ""}; 450 for (String address : invalidAddresses) { 451 try { 452 IpSecSpiResponse spiResp = 453 mIpSecService.allocateSecurityParameterIndex( 454 address, DROID_SPI, new Binder()); 455 fail("Invalid address was passed through IpSecService validation: " + address); 456 } catch (IllegalArgumentException e) { 457 } catch (Exception e) { 458 fail( 459 "Invalid InetAddress was not caught in validation: " 460 + address 461 + ", Exception: " 462 + e); 463 } 464 } 465 } 466 467 /** 468 * This function checks if the number of encap UDP socket that one UID can reserve has a 469 * reasonable limit. 470 */ 471 @Test 472 public void testSocketResourceTrackerLimitation() throws Exception { 473 List<IpSecUdpEncapResponse> openUdpEncapSockets = new ArrayList<IpSecUdpEncapResponse>(); 474 // Reserve sockets until it fails. 475 for (int i = 0; i < MAX_NUM_ENCAP_SOCKETS; i++) { 476 IpSecUdpEncapResponse newUdpEncapSocket = 477 mIpSecService.openUdpEncapsulationSocket(0, new Binder()); 478 assertNotNull(newUdpEncapSocket); 479 if (IpSecManager.Status.OK != newUdpEncapSocket.status) { 480 break; 481 } 482 openUdpEncapSockets.add(newUdpEncapSocket); 483 } 484 // Assert that the total sockets quota has a reasonable limit. 485 assertTrue("No UDP encap socket was open", !openUdpEncapSockets.isEmpty()); 486 assertTrue( 487 "Number of open UDP encap sockets is out of bound", 488 openUdpEncapSockets.size() < MAX_NUM_ENCAP_SOCKETS); 489 490 // Try to reserve one more UDP encapsulation socket, and should fail. 491 IpSecUdpEncapResponse extraUdpEncapSocket = 492 mIpSecService.openUdpEncapsulationSocket(0, new Binder()); 493 assertNotNull(extraUdpEncapSocket); 494 assertEquals(IpSecManager.Status.RESOURCE_UNAVAILABLE, extraUdpEncapSocket.status); 495 496 // Close one of the open UDP encapsulation sockets. 497 mIpSecService.closeUdpEncapsulationSocket(openUdpEncapSockets.get(0).resourceId); 498 openUdpEncapSockets.get(0).fileDescriptor.close(); 499 openUdpEncapSockets.remove(0); 500 501 // Try to reserve one more UDP encapsulation socket, and should be successful. 502 extraUdpEncapSocket = mIpSecService.openUdpEncapsulationSocket(0, new Binder()); 503 assertNotNull(extraUdpEncapSocket); 504 assertEquals(IpSecManager.Status.OK, extraUdpEncapSocket.status); 505 openUdpEncapSockets.add(extraUdpEncapSocket); 506 507 // Close open UDP sockets. 508 for (IpSecUdpEncapResponse openSocket : openUdpEncapSockets) { 509 mIpSecService.closeUdpEncapsulationSocket(openSocket.resourceId); 510 openSocket.fileDescriptor.close(); 511 } 512 } 513 514 /** 515 * This function checks if the number of SPI that one UID can reserve has a reasonable limit. 516 * This test does not test for both address families or duplicate SPIs because resource tracking 517 * code does not depend on them. 518 */ 519 @Test 520 public void testSpiResourceTrackerLimitation() throws Exception { 521 List<IpSecSpiResponse> reservedSpis = new ArrayList<IpSecSpiResponse>(); 522 // Return the same SPI for all SPI allocation since IpSecService only 523 // tracks the resource ID. 524 when(mMockNetd.ipSecAllocateSpi( 525 anyInt(), 526 anyString(), 527 eq(InetAddress.getLoopbackAddress().getHostAddress()), 528 anyInt())) 529 .thenReturn(DROID_SPI); 530 // Reserve spis until it fails. 531 for (int i = 0; i < MAX_NUM_SPIS; i++) { 532 IpSecSpiResponse newSpi = 533 mIpSecService.allocateSecurityParameterIndex( 534 InetAddress.getLoopbackAddress().getHostAddress(), 535 DROID_SPI + i, 536 new Binder()); 537 assertNotNull(newSpi); 538 if (IpSecManager.Status.OK != newSpi.status) { 539 break; 540 } 541 reservedSpis.add(newSpi); 542 } 543 // Assert that the SPI quota has a reasonable limit. 544 assertTrue(reservedSpis.size() > 0 && reservedSpis.size() < MAX_NUM_SPIS); 545 546 // Try to reserve one more SPI, and should fail. 547 IpSecSpiResponse extraSpi = 548 mIpSecService.allocateSecurityParameterIndex( 549 InetAddress.getLoopbackAddress().getHostAddress(), 550 DROID_SPI + MAX_NUM_SPIS, 551 new Binder()); 552 assertNotNull(extraSpi); 553 assertEquals(IpSecManager.Status.RESOURCE_UNAVAILABLE, extraSpi.status); 554 555 // Release one reserved spi. 556 mIpSecService.releaseSecurityParameterIndex(reservedSpis.get(0).resourceId); 557 reservedSpis.remove(0); 558 559 // Should successfully reserve one more spi. 560 extraSpi = 561 mIpSecService.allocateSecurityParameterIndex( 562 InetAddress.getLoopbackAddress().getHostAddress(), 563 DROID_SPI + MAX_NUM_SPIS, 564 new Binder()); 565 assertNotNull(extraSpi); 566 assertEquals(IpSecManager.Status.OK, extraSpi.status); 567 568 // Release reserved SPIs. 569 for (IpSecSpiResponse spiResp : reservedSpis) { 570 mIpSecService.releaseSecurityParameterIndex(spiResp.resourceId); 571 } 572 } 573 574 @Test 575 public void testUidFdtagger() throws Exception { 576 SocketTagger actualSocketTagger = SocketTagger.get(); 577 578 try { 579 FileDescriptor sockFd = Os.socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP); 580 581 // Has to be done after socket creation because BlockGuardOS calls tag on new sockets 582 SocketTagger mockSocketTagger = mock(SocketTagger.class); 583 SocketTagger.set(mockSocketTagger); 584 585 mIpSecService.mUidFdTagger.tag(sockFd, Process.LAST_APPLICATION_UID); 586 verify(mockSocketTagger).tag(eq(sockFd)); 587 } finally { 588 SocketTagger.set(actualSocketTagger); 589 } 590 } 591 592 /** 593 * Checks if two file descriptors point to the same file. 594 * 595 * <p>According to stat.h documentation, the correct way to check for equivalent or duplicated 596 * file descriptors is to check their inode and device. These two entries uniquely identify any 597 * file. 598 */ 599 private boolean fileDescriptorsEqual(FileDescriptor fd1, FileDescriptor fd2) { 600 try { 601 StructStat fd1Stat = Os.fstat(fd1); 602 StructStat fd2Stat = Os.fstat(fd2); 603 604 return fd1Stat.st_ino == fd2Stat.st_ino && fd1Stat.st_dev == fd2Stat.st_dev; 605 } catch (ErrnoException e) { 606 return false; 607 } 608 } 609 610 @Test 611 public void testOpenUdpEncapSocketTagsSocket() throws Exception { 612 IpSecService.UidFdTagger mockTagger = mock(IpSecService.UidFdTagger.class); 613 IpSecService testIpSecService = new IpSecService( 614 mMockContext, mMockIpSecSrvConfig, mockTagger); 615 616 IpSecUdpEncapResponse udpEncapResp = 617 testIpSecService.openUdpEncapsulationSocket(0, new Binder()); 618 assertNotNull(udpEncapResp); 619 assertEquals(IpSecManager.Status.OK, udpEncapResp.status); 620 621 FileDescriptor sockFd = udpEncapResp.fileDescriptor.getFileDescriptor(); 622 ArgumentMatcher<FileDescriptor> fdMatcher = 623 (argFd) -> { 624 return fileDescriptorsEqual(sockFd, argFd); 625 }; 626 verify(mockTagger).tag(argThat(fdMatcher), eq(Os.getuid())); 627 628 testIpSecService.closeUdpEncapsulationSocket(udpEncapResp.resourceId); 629 udpEncapResp.fileDescriptor.close(); 630 } 631 632 @Test 633 public void testOpenUdpEncapsulationSocketCallsSetEncapSocketOwner() throws Exception { 634 IpSecUdpEncapResponse udpEncapResp = 635 mIpSecService.openUdpEncapsulationSocket(0, new Binder()); 636 637 FileDescriptor sockFd = udpEncapResp.fileDescriptor.getFileDescriptor(); 638 ArgumentMatcher<ParcelFileDescriptor> fdMatcher = (arg) -> { 639 try { 640 StructStat sockStat = Os.fstat(sockFd); 641 StructStat argStat = Os.fstat(arg.getFileDescriptor()); 642 643 return sockStat.st_ino == argStat.st_ino 644 && sockStat.st_dev == argStat.st_dev; 645 } catch (ErrnoException e) { 646 return false; 647 } 648 }; 649 650 verify(mMockNetd).ipSecSetEncapSocketOwner(argThat(fdMatcher), eq(Os.getuid())); 651 mIpSecService.closeUdpEncapsulationSocket(udpEncapResp.resourceId); 652 } 653 654 @Test 655 public void testReserveNetId() { 656 final Range<Integer> netIdRange = ConnectivityManager.getIpSecNetIdRange(); 657 for (int netId = netIdRange.getLower(); netId <= netIdRange.getUpper(); netId++) { 658 assertEquals(netId, mIpSecService.reserveNetId()); 659 } 660 661 // Check that resource exhaustion triggers an exception 662 try { 663 mIpSecService.reserveNetId(); 664 fail("Did not throw error for all netIds reserved"); 665 } catch (IllegalStateException expected) { 666 } 667 668 // Now release one and try again 669 int releasedNetId = 670 netIdRange.getLower() + (netIdRange.getUpper() - netIdRange.getLower()) / 2; 671 mIpSecService.releaseNetId(releasedNetId); 672 assertEquals(releasedNetId, mIpSecService.reserveNetId()); 673 } 674 } 675