1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 package io.netty.handler.ssl;
17
18 import io.netty.buffer.ByteBuf;
19 import io.netty.buffer.ByteBufAllocator;
20 import io.netty.handler.ssl.util.LazyJavaxX509Certificate;
21 import io.netty.handler.ssl.util.LazyX509Certificate;
22 import io.netty.internal.tcnative.AsyncTask;
23 import io.netty.internal.tcnative.Buffer;
24 import io.netty.internal.tcnative.SSL;
25 import io.netty.util.AbstractReferenceCounted;
26 import io.netty.util.CharsetUtil;
27 import io.netty.util.ReferenceCounted;
28 import io.netty.util.ResourceLeakDetector;
29 import io.netty.util.ResourceLeakDetectorFactory;
30 import io.netty.util.ResourceLeakTracker;
31 import io.netty.util.internal.EmptyArrays;
32 import io.netty.util.internal.PlatformDependent;
33 import io.netty.util.internal.StringUtil;
34 import io.netty.util.internal.SuppressJava6Requirement;
35 import io.netty.util.internal.ThrowableUtil;
36 import io.netty.util.internal.UnstableApi;
37 import io.netty.util.internal.logging.InternalLogger;
38 import io.netty.util.internal.logging.InternalLoggerFactory;
39
40 import java.nio.ByteBuffer;
41 import java.nio.ReadOnlyBufferException;
42 import java.security.Principal;
43 import java.security.cert.Certificate;
44 import java.util.Arrays;
45 import java.util.Collection;
46 import java.util.Collections;
47 import java.util.HashSet;
48 import java.util.LinkedHashSet;
49 import java.util.List;
50 import java.util.Map;
51 import java.util.Set;
52 import java.util.concurrent.ConcurrentHashMap;
53 import java.util.concurrent.locks.Lock;
54
55 import javax.crypto.spec.SecretKeySpec;
56 import javax.net.ssl.SSLEngine;
57 import javax.net.ssl.SSLEngineResult;
58 import javax.net.ssl.SSLException;
59 import javax.net.ssl.SSLHandshakeException;
60 import javax.net.ssl.SSLParameters;
61 import javax.net.ssl.SSLPeerUnverifiedException;
62 import javax.net.ssl.SSLSession;
63 import javax.net.ssl.SSLSessionBindingEvent;
64 import javax.net.ssl.SSLSessionBindingListener;
65 import javax.security.cert.X509Certificate;
66
67 import static io.netty.handler.ssl.OpenSsl.memoryAddress;
68 import static io.netty.handler.ssl.SslUtils.SSL_RECORD_HEADER_LENGTH;
69 import static io.netty.util.internal.EmptyArrays.EMPTY_STRINGS;
70 import static io.netty.util.internal.ObjectUtil.checkNotNull;
71 import static io.netty.util.internal.ObjectUtil.checkNotNullArrayParam;
72 import static io.netty.util.internal.ObjectUtil.checkNotNullWithIAE;
73 import static java.lang.Integer.MAX_VALUE;
74 import static java.lang.Math.min;
75 import static javax.net.ssl.SSLEngineResult.HandshakeStatus.FINISHED;
76 import static javax.net.ssl.SSLEngineResult.HandshakeStatus.NEED_TASK;
77 import static javax.net.ssl.SSLEngineResult.HandshakeStatus.NEED_UNWRAP;
78 import static javax.net.ssl.SSLEngineResult.HandshakeStatus.NEED_WRAP;
79 import static javax.net.ssl.SSLEngineResult.HandshakeStatus.NOT_HANDSHAKING;
80 import static javax.net.ssl.SSLEngineResult.Status.BUFFER_OVERFLOW;
81 import static javax.net.ssl.SSLEngineResult.Status.BUFFER_UNDERFLOW;
82 import static javax.net.ssl.SSLEngineResult.Status.CLOSED;
83 import static javax.net.ssl.SSLEngineResult.Status.OK;
84
85
86
87
88
89
90
91
92
93
94 public class ReferenceCountedOpenSslEngine extends SSLEngine implements ReferenceCounted, ApplicationProtocolAccessor {
95
96 private static final InternalLogger logger = InternalLoggerFactory.getInstance(ReferenceCountedOpenSslEngine.class);
97
98 private static final ResourceLeakDetector<ReferenceCountedOpenSslEngine> leakDetector =
99 ResourceLeakDetectorFactory.instance().newResourceLeakDetector(ReferenceCountedOpenSslEngine.class);
100 private static final int OPENSSL_OP_NO_PROTOCOL_INDEX_SSLV2 = 0;
101 private static final int OPENSSL_OP_NO_PROTOCOL_INDEX_SSLV3 = 1;
102 private static final int OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1 = 2;
103 private static final int OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_1 = 3;
104 private static final int OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_2 = 4;
105 private static final int OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_3 = 5;
106 private static final int[] OPENSSL_OP_NO_PROTOCOLS = {
107 SSL.SSL_OP_NO_SSLv2,
108 SSL.SSL_OP_NO_SSLv3,
109 SSL.SSL_OP_NO_TLSv1,
110 SSL.SSL_OP_NO_TLSv1_1,
111 SSL.SSL_OP_NO_TLSv1_2,
112 SSL.SSL_OP_NO_TLSv1_3
113 };
114
115
116
117
118 static final int MAX_PLAINTEXT_LENGTH = SSL.SSL_MAX_PLAINTEXT_LENGTH;
119
120
121
122 static final int MAX_RECORD_SIZE = SSL.SSL_MAX_RECORD_LENGTH;
123
124 private static final SSLEngineResult NEED_UNWRAP_OK = new SSLEngineResult(OK, NEED_UNWRAP, 0, 0);
125 private static final SSLEngineResult NEED_UNWRAP_CLOSED = new SSLEngineResult(CLOSED, NEED_UNWRAP, 0, 0);
126 private static final SSLEngineResult NEED_WRAP_OK = new SSLEngineResult(OK, NEED_WRAP, 0, 0);
127 private static final SSLEngineResult NEED_WRAP_CLOSED = new SSLEngineResult(CLOSED, NEED_WRAP, 0, 0);
128 private static final SSLEngineResult CLOSED_NOT_HANDSHAKING = new SSLEngineResult(CLOSED, NOT_HANDSHAKING, 0, 0);
129
130
131 private long ssl;
132 private long networkBIO;
133
134 private enum HandshakeState {
135
136
137
138 NOT_STARTED,
139
140
141
142 STARTED_IMPLICITLY,
143
144
145
146 STARTED_EXPLICITLY,
147
148
149
150 FINISHED
151 }
152
153 private HandshakeState handshakeState = HandshakeState.NOT_STARTED;
154 private boolean receivedShutdown;
155 private volatile boolean destroyed;
156 private volatile String applicationProtocol;
157 private volatile boolean needTask;
158 private boolean hasTLSv13Cipher;
159 private boolean sessionSet;
160
161
162 private final ResourceLeakTracker<ReferenceCountedOpenSslEngine> leak;
163 private final AbstractReferenceCounted refCnt = new AbstractReferenceCounted() {
164 @Override
165 public ReferenceCounted touch(Object hint) {
166 if (leak != null) {
167 leak.record(hint);
168 }
169
170 return ReferenceCountedOpenSslEngine.this;
171 }
172
173 @Override
174 protected void deallocate() {
175 shutdown();
176 if (leak != null) {
177 boolean closed = leak.close(ReferenceCountedOpenSslEngine.this);
178 assert closed;
179 }
180 parentContext.release();
181 }
182 };
183
184 private final Set<String> enabledProtocols = new LinkedHashSet<String>();
185
186 private volatile ClientAuth clientAuth = ClientAuth.NONE;
187
188 private String endpointIdentificationAlgorithm;
189
190 private Object algorithmConstraints;
191 private List<String> sniHostNames;
192
193
194
195 private volatile Collection<?> matchers;
196
197
198 private boolean isInboundDone;
199 private boolean outboundClosed;
200
201 final boolean jdkCompatibilityMode;
202 private final boolean clientMode;
203 final ByteBufAllocator alloc;
204 private final OpenSslEngineMap engineMap;
205 private final OpenSslApplicationProtocolNegotiator apn;
206 private final ReferenceCountedOpenSslContext parentContext;
207 private final OpenSslInternalSession session;
208 private final ByteBuffer[] singleSrcBuffer = new ByteBuffer[1];
209 private final ByteBuffer[] singleDstBuffer = new ByteBuffer[1];
210 private final boolean enableOcsp;
211 private int maxWrapOverhead;
212 private int maxWrapBufferSize;
213 private Throwable pendingException;
214
215
216
217
218
219
220
221
222
223
224
225
226
227 ReferenceCountedOpenSslEngine(ReferenceCountedOpenSslContext context, final ByteBufAllocator alloc, String peerHost,
228 int peerPort, boolean jdkCompatibilityMode, boolean leakDetection,
229 String endpointIdentificationAlgorithm) {
230 super(peerHost, peerPort);
231 OpenSsl.ensureAvailability();
232 engineMap = context.engineMap;
233 enableOcsp = context.enableOcsp;
234 this.jdkCompatibilityMode = jdkCompatibilityMode;
235 this.alloc = checkNotNull(alloc, "alloc");
236 apn = (OpenSslApplicationProtocolNegotiator) context.applicationProtocolNegotiator();
237 clientMode = context.isClient();
238 this.endpointIdentificationAlgorithm = endpointIdentificationAlgorithm;
239
240 if (PlatformDependent.javaVersion() >= 7) {
241 session = new ExtendedOpenSslSession(new DefaultOpenSslSession(context.sessionContext())) {
242 private String[] peerSupportedSignatureAlgorithms;
243 private List requestedServerNames;
244
245 @Override
246 public List getRequestedServerNames() {
247 if (clientMode) {
248 return Java8SslUtils.getSniHostNames(sniHostNames);
249 } else {
250 synchronized (ReferenceCountedOpenSslEngine.this) {
251 if (requestedServerNames == null) {
252 if (isDestroyed()) {
253 requestedServerNames = Collections.emptyList();
254 } else {
255 String name = SSL.getSniHostname(ssl);
256 if (name == null) {
257 requestedServerNames = Collections.emptyList();
258 } else {
259
260
261 requestedServerNames =
262 Java8SslUtils.getSniHostName(
263 SSL.getSniHostname(ssl).getBytes(CharsetUtil.UTF_8));
264 }
265 }
266 }
267 return requestedServerNames;
268 }
269 }
270 }
271
272 @Override
273 public String[] getPeerSupportedSignatureAlgorithms() {
274 synchronized (ReferenceCountedOpenSslEngine.this) {
275 if (peerSupportedSignatureAlgorithms == null) {
276 if (isDestroyed()) {
277 peerSupportedSignatureAlgorithms = EMPTY_STRINGS;
278 } else {
279 String[] algs = SSL.getSigAlgs(ssl);
280 if (algs == null) {
281 peerSupportedSignatureAlgorithms = EMPTY_STRINGS;
282 } else {
283 Set<String> algorithmList = new LinkedHashSet<String>(algs.length);
284 for (String alg: algs) {
285 String converted = SignatureAlgorithmConverter.toJavaName(alg);
286
287 if (converted != null) {
288 algorithmList.add(converted);
289 }
290 }
291 peerSupportedSignatureAlgorithms = algorithmList.toArray(EMPTY_STRINGS);
292 }
293 }
294 }
295 return peerSupportedSignatureAlgorithms.clone();
296 }
297 }
298
299 @Override
300 public List<byte[]> getStatusResponses() {
301 byte[] ocspResponse = null;
302 if (enableOcsp && clientMode) {
303 synchronized (ReferenceCountedOpenSslEngine.this) {
304 if (!isDestroyed()) {
305 ocspResponse = SSL.getOcspResponse(ssl);
306 }
307 }
308 }
309 return ocspResponse == null ?
310 Collections.<byte[]>emptyList() : Collections.singletonList(ocspResponse);
311 }
312 };
313 } else {
314 session = new DefaultOpenSslSession(context.sessionContext());
315 }
316
317 if (!context.sessionContext().useKeyManager()) {
318 session.setLocalCertificate(context.keyCertChain);
319 }
320
321 Lock readerLock = context.ctxLock.readLock();
322 readerLock.lock();
323 final long finalSsl;
324 try {
325 finalSsl = SSL.newSSL(context.ctx, !context.isClient());
326 } finally {
327 readerLock.unlock();
328 }
329 synchronized (this) {
330 ssl = finalSsl;
331 try {
332 networkBIO = SSL.bioNewByteBuffer(ssl, context.getBioNonApplicationBufferSize());
333
334
335
336 setClientAuth(clientMode ? ClientAuth.NONE : context.clientAuth);
337
338 assert context.protocols != null;
339 this.hasTLSv13Cipher = context.hasTLSv13Cipher;
340
341 setEnabledProtocols(context.protocols);
342
343
344
345 if (clientMode && SslUtils.isValidHostNameForSNI(peerHost)) {
346
347
348 if (PlatformDependent.javaVersion() >= 8) {
349 if (Java8SslUtils.isValidHostNameForSNI(peerHost)) {
350 SSL.setTlsExtHostName(ssl, peerHost);
351 sniHostNames = Collections.singletonList(peerHost);
352 }
353 } else {
354 SSL.setTlsExtHostName(ssl, peerHost);
355 sniHostNames = Collections.singletonList(peerHost);
356 }
357 }
358
359 if (enableOcsp) {
360 SSL.enableOcsp(ssl);
361 }
362
363 if (!jdkCompatibilityMode) {
364 SSL.setMode(ssl, SSL.getMode(ssl) | SSL.SSL_MODE_ENABLE_PARTIAL_WRITE);
365 }
366
367 if (isProtocolEnabled(SSL.getOptions(ssl), SSL.SSL_OP_NO_TLSv1_3, SslProtocols.TLS_v1_3)) {
368 final boolean enableTickets = clientMode ?
369 ReferenceCountedOpenSslContext.CLIENT_ENABLE_SESSION_TICKET_TLSV13 :
370 ReferenceCountedOpenSslContext.SERVER_ENABLE_SESSION_TICKET_TLSV13;
371 if (enableTickets) {
372
373
374
375
376
377
378
379 SSL.clearOptions(ssl, SSL.SSL_OP_NO_TICKET);
380 }
381 }
382
383 if (OpenSsl.isBoringSSL() && clientMode) {
384
385
386
387
388 SSL.setRenegotiateMode(ssl, SSL.SSL_RENEGOTIATE_ONCE);
389 }
390
391 calculateMaxWrapOverhead();
392
393
394 configureEndpointVerification(endpointIdentificationAlgorithm);
395 } catch (Throwable cause) {
396
397
398 shutdown();
399
400 PlatformDependent.throwException(cause);
401 }
402 }
403
404
405
406 parentContext = context;
407 parentContext.retain();
408
409
410
411 leak = leakDetection ? leakDetector.track(this) : null;
412 }
413
414 final synchronized String[] authMethods() {
415 if (isDestroyed()) {
416 return EMPTY_STRINGS;
417 }
418 return SSL.authenticationMethods(ssl);
419 }
420
421 final boolean setKeyMaterial(OpenSslKeyMaterial keyMaterial) throws Exception {
422 synchronized (this) {
423 if (isDestroyed()) {
424 return false;
425 }
426 SSL.setKeyMaterial(ssl, keyMaterial.certificateChainAddress(), keyMaterial.privateKeyAddress());
427 }
428 session.setLocalCertificate(keyMaterial.certificateChain());
429 return true;
430 }
431
432 final synchronized SecretKeySpec masterKey() {
433 if (isDestroyed()) {
434 return null;
435 }
436 return new SecretKeySpec(SSL.getMasterKey(ssl), "AES");
437 }
438
439 synchronized boolean isSessionReused() {
440 if (isDestroyed()) {
441 return false;
442 }
443 return SSL.isSessionReused(ssl);
444 }
445
446
447
448
449 @UnstableApi
450 public void setOcspResponse(byte[] response) {
451 if (!enableOcsp) {
452 throw new IllegalStateException("OCSP stapling is not enabled");
453 }
454
455 if (clientMode) {
456 throw new IllegalStateException("Not a server SSLEngine");
457 }
458
459 synchronized (this) {
460 if (!isDestroyed()) {
461 SSL.setOcspResponse(ssl, response);
462 }
463 }
464 }
465
466
467
468
469 @UnstableApi
470 public byte[] getOcspResponse() {
471 if (!enableOcsp) {
472 throw new IllegalStateException("OCSP stapling is not enabled");
473 }
474
475 if (!clientMode) {
476 throw new IllegalStateException("Not a client SSLEngine");
477 }
478
479 synchronized (this) {
480 if (isDestroyed()) {
481 return EmptyArrays.EMPTY_BYTES;
482 }
483 return SSL.getOcspResponse(ssl);
484 }
485 }
486
487 @Override
488 public final int refCnt() {
489 return refCnt.refCnt();
490 }
491
492 @Override
493 public final ReferenceCounted retain() {
494 refCnt.retain();
495 return this;
496 }
497
498 @Override
499 public final ReferenceCounted retain(int increment) {
500 refCnt.retain(increment);
501 return this;
502 }
503
504 @Override
505 public final ReferenceCounted touch() {
506 refCnt.touch();
507 return this;
508 }
509
510 @Override
511 public final ReferenceCounted touch(Object hint) {
512 refCnt.touch(hint);
513 return this;
514 }
515
516 @Override
517 public final boolean release() {
518 return refCnt.release();
519 }
520
521 @Override
522 public final boolean release(int decrement) {
523 return refCnt.release(decrement);
524 }
525
526
527
528 public String getApplicationProtocol() {
529 return applicationProtocol;
530 }
531
532
533
534 public String getHandshakeApplicationProtocol() {
535 return applicationProtocol;
536 }
537
538 @Override
539 public final synchronized SSLSession getHandshakeSession() {
540
541
542
543
544 switch(handshakeState) {
545 case NOT_STARTED:
546 case FINISHED:
547 return null;
548 default:
549 return session;
550 }
551 }
552
553
554
555
556
557
558 public final synchronized long sslPointer() {
559 return ssl;
560 }
561
562
563
564
565 public final synchronized void shutdown() {
566 if (!destroyed) {
567 destroyed = true;
568
569
570
571 if (engineMap != null) {
572 engineMap.remove(ssl);
573 }
574 SSL.freeSSL(ssl);
575 ssl = networkBIO = 0;
576
577 isInboundDone = outboundClosed = true;
578 }
579
580
581 SSL.clearError();
582 }
583
584
585
586
587
588
589 private int writePlaintextData(final ByteBuffer src, int len) {
590 final int pos = src.position();
591 final int limit = src.limit();
592 final int sslWrote;
593
594 if (src.isDirect()) {
595 sslWrote = SSL.writeToSSL(ssl, bufferAddress(src) + pos, len);
596 if (sslWrote > 0) {
597 src.position(pos + sslWrote);
598 }
599 } else {
600 ByteBuf buf = alloc.directBuffer(len);
601 try {
602 src.limit(pos + len);
603
604 buf.setBytes(0, src);
605 src.limit(limit);
606
607 sslWrote = SSL.writeToSSL(ssl, memoryAddress(buf), len);
608 if (sslWrote > 0) {
609 src.position(pos + sslWrote);
610 } else {
611 src.position(pos);
612 }
613 } finally {
614 buf.release();
615 }
616 }
617 return sslWrote;
618 }
619
620 synchronized void bioSetFd(int fd) {
621 if (!isDestroyed()) {
622 SSL.bioSetFd(this.ssl, fd);
623 }
624 }
625
626
627
628
629 private ByteBuf writeEncryptedData(final ByteBuffer src, int len) throws SSLException {
630 final int pos = src.position();
631 if (src.isDirect()) {
632 SSL.bioSetByteBuffer(networkBIO, bufferAddress(src) + pos, len, false);
633 } else {
634 final ByteBuf buf = alloc.directBuffer(len);
635 try {
636 final int limit = src.limit();
637 src.limit(pos + len);
638 buf.writeBytes(src);
639
640 src.position(pos);
641 src.limit(limit);
642
643 SSL.bioSetByteBuffer(networkBIO, memoryAddress(buf), len, false);
644 return buf;
645 } catch (Throwable cause) {
646 buf.release();
647 PlatformDependent.throwException(cause);
648 }
649 }
650 return null;
651 }
652
653
654
655
656 private int readPlaintextData(final ByteBuffer dst) throws SSLException {
657 final int sslRead;
658 final int pos = dst.position();
659 if (dst.isDirect()) {
660 sslRead = SSL.readFromSSL(ssl, bufferAddress(dst) + pos, dst.limit() - pos);
661 if (sslRead > 0) {
662 dst.position(pos + sslRead);
663 }
664 } else {
665 final int limit = dst.limit();
666 final int len = min(maxEncryptedPacketLength0(), limit - pos);
667 final ByteBuf buf = alloc.directBuffer(len);
668 try {
669 sslRead = SSL.readFromSSL(ssl, memoryAddress(buf), len);
670 if (sslRead > 0) {
671 dst.limit(pos + sslRead);
672 buf.getBytes(buf.readerIndex(), dst);
673 dst.limit(limit);
674 }
675 } finally {
676 buf.release();
677 }
678 }
679
680 return sslRead;
681 }
682
683
684
685
686 final synchronized int maxWrapOverhead() {
687 return maxWrapOverhead;
688 }
689
690
691
692
693 final synchronized int maxEncryptedPacketLength() {
694 return maxEncryptedPacketLength0();
695 }
696
697
698
699
700
701 final int maxEncryptedPacketLength0() {
702 return maxWrapOverhead + MAX_PLAINTEXT_LENGTH;
703 }
704
705
706
707
708
709
710
711
712
713 final int calculateMaxLengthForWrap(int plaintextLength, int numComponents) {
714 return (int) min(maxWrapBufferSize, plaintextLength + (long) maxWrapOverhead * numComponents);
715 }
716
717
718
719
720
721
722
723
724 final int calculateOutNetBufSize(int plaintextLength, int numComponents) {
725 return (int) min(MAX_VALUE, plaintextLength + (long) maxWrapOverhead * numComponents);
726 }
727
728 final synchronized int sslPending() {
729 return sslPending0();
730 }
731
732
733
734
735 private void calculateMaxWrapOverhead() {
736 maxWrapOverhead = SSL.getMaxWrapOverhead(ssl);
737
738
739
740
741 maxWrapBufferSize = jdkCompatibilityMode ? maxEncryptedPacketLength0() : maxEncryptedPacketLength0() << 4;
742 }
743
744 private int sslPending0() {
745
746
747
748
749 return handshakeState != HandshakeState.FINISHED ? 0 : SSL.sslPending(ssl);
750 }
751
752 private boolean isBytesAvailableEnoughForWrap(int bytesAvailable, int plaintextLength, int numComponents) {
753 return bytesAvailable - (long) maxWrapOverhead * numComponents >= plaintextLength;
754 }
755
756 @Override
757 public final SSLEngineResult wrap(
758 final ByteBuffer[] srcs, int offset, final int length, final ByteBuffer dst) throws SSLException {
759
760 checkNotNullWithIAE(srcs, "srcs");
761 checkNotNullWithIAE(dst, "dst");
762
763 if (offset >= srcs.length || offset + length > srcs.length) {
764 throw new IndexOutOfBoundsException(
765 "offset: " + offset + ", length: " + length +
766 " (expected: offset <= offset + length <= srcs.length (" + srcs.length + "))");
767 }
768
769 if (dst.isReadOnly()) {
770 throw new ReadOnlyBufferException();
771 }
772
773 synchronized (this) {
774 if (isOutboundDone()) {
775
776 return isInboundDone() || isDestroyed() ? CLOSED_NOT_HANDSHAKING : NEED_UNWRAP_CLOSED;
777 }
778
779 int bytesProduced = 0;
780 ByteBuf bioReadCopyBuf = null;
781 try {
782
783 if (dst.isDirect()) {
784 SSL.bioSetByteBuffer(networkBIO, bufferAddress(dst) + dst.position(), dst.remaining(),
785 true);
786 } else {
787 bioReadCopyBuf = alloc.directBuffer(dst.remaining());
788 SSL.bioSetByteBuffer(networkBIO, memoryAddress(bioReadCopyBuf), bioReadCopyBuf.writableBytes(),
789 true);
790 }
791
792 int bioLengthBefore = SSL.bioLengthByteBuffer(networkBIO);
793
794
795 if (outboundClosed) {
796
797
798
799
800
801 if (!isBytesAvailableEnoughForWrap(dst.remaining(), 2, 1)) {
802 return new SSLEngineResult(BUFFER_OVERFLOW, getHandshakeStatus(), 0, 0);
803 }
804
805
806
807 bytesProduced = SSL.bioFlushByteBuffer(networkBIO);
808 if (bytesProduced <= 0) {
809 return newResultMayFinishHandshake(NOT_HANDSHAKING, 0, 0);
810 }
811
812
813
814 if (!doSSLShutdown()) {
815 return newResultMayFinishHandshake(NOT_HANDSHAKING, 0, bytesProduced);
816 }
817 bytesProduced = bioLengthBefore - SSL.bioLengthByteBuffer(networkBIO);
818 return newResultMayFinishHandshake(NEED_WRAP, 0, bytesProduced);
819 }
820
821
822 SSLEngineResult.HandshakeStatus status = NOT_HANDSHAKING;
823 HandshakeState oldHandshakeState = handshakeState;
824
825
826 if (handshakeState != HandshakeState.FINISHED) {
827 if (handshakeState != HandshakeState.STARTED_EXPLICITLY) {
828
829 handshakeState = HandshakeState.STARTED_IMPLICITLY;
830 }
831
832
833 bytesProduced = SSL.bioFlushByteBuffer(networkBIO);
834
835 if (pendingException != null) {
836
837
838
839
840
841
842
843
844
845
846
847 if (bytesProduced > 0) {
848 return newResult(NEED_WRAP, 0, bytesProduced);
849 }
850
851
852
853 return newResult(handshakeException(), 0, 0);
854 }
855
856 status = handshake();
857
858
859
860 bytesProduced = bioLengthBefore - SSL.bioLengthByteBuffer(networkBIO);
861
862 if (status == NEED_TASK) {
863 return newResult(status, 0, bytesProduced);
864 }
865
866 if (bytesProduced > 0) {
867
868
869
870 return newResult(mayFinishHandshake(status != FINISHED ?
871 bytesProduced == bioLengthBefore ? NEED_WRAP :
872 getHandshakeStatus(SSL.bioLengthNonApplication(networkBIO)) : FINISHED),
873 0, bytesProduced);
874 }
875
876 if (status == NEED_UNWRAP) {
877
878 return isOutboundDone() ? NEED_UNWRAP_CLOSED : NEED_UNWRAP_OK;
879 }
880
881
882
883 if (outboundClosed) {
884 bytesProduced = SSL.bioFlushByteBuffer(networkBIO);
885 return newResultMayFinishHandshake(status, 0, bytesProduced);
886 }
887 }
888
889 final int endOffset = offset + length;
890 if (jdkCompatibilityMode ||
891
892
893
894 oldHandshakeState != HandshakeState.FINISHED) {
895 int srcsLen = 0;
896 for (int i = offset; i < endOffset; ++i) {
897 final ByteBuffer src = srcs[i];
898 if (src == null) {
899 throw new IllegalArgumentException("srcs[" + i + "] is null");
900 }
901 if (srcsLen == MAX_PLAINTEXT_LENGTH) {
902 continue;
903 }
904
905 srcsLen += src.remaining();
906 if (srcsLen > MAX_PLAINTEXT_LENGTH || srcsLen < 0) {
907
908
909
910 srcsLen = MAX_PLAINTEXT_LENGTH;
911 }
912 }
913
914
915
916 if (!isBytesAvailableEnoughForWrap(dst.remaining(), srcsLen, 1)) {
917 return new SSLEngineResult(BUFFER_OVERFLOW, getHandshakeStatus(), 0, 0);
918 }
919 }
920
921
922 int bytesConsumed = 0;
923 assert bytesProduced == 0;
924
925
926 bytesProduced = SSL.bioFlushByteBuffer(networkBIO);
927
928 if (bytesProduced > 0) {
929 return newResultMayFinishHandshake(status, bytesConsumed, bytesProduced);
930 }
931
932
933 if (pendingException != null) {
934 Throwable error = pendingException;
935 pendingException = null;
936 shutdown();
937
938
939 throw new SSLException(error);
940 }
941
942 for (; offset < endOffset; ++offset) {
943 final ByteBuffer src = srcs[offset];
944 final int remaining = src.remaining();
945 if (remaining == 0) {
946 continue;
947 }
948
949 final int bytesWritten;
950 if (jdkCompatibilityMode) {
951
952
953
954 bytesWritten = writePlaintextData(src, min(remaining, MAX_PLAINTEXT_LENGTH - bytesConsumed));
955 } else {
956
957
958
959 final int availableCapacityForWrap = dst.remaining() - bytesProduced - maxWrapOverhead;
960 if (availableCapacityForWrap <= 0) {
961 return new SSLEngineResult(BUFFER_OVERFLOW, getHandshakeStatus(), bytesConsumed,
962 bytesProduced);
963 }
964 bytesWritten = writePlaintextData(src, min(remaining, availableCapacityForWrap));
965 }
966
967
968
969
970
971
972 final int pendingNow = SSL.bioLengthByteBuffer(networkBIO);
973 bytesProduced += bioLengthBefore - pendingNow;
974 bioLengthBefore = pendingNow;
975
976 if (bytesWritten > 0) {
977 bytesConsumed += bytesWritten;
978
979 if (jdkCompatibilityMode || bytesProduced == dst.remaining()) {
980 return newResultMayFinishHandshake(status, bytesConsumed, bytesProduced);
981 }
982 } else {
983 int sslError = SSL.getError(ssl, bytesWritten);
984 if (sslError == SSL.SSL_ERROR_ZERO_RETURN) {
985
986 if (!receivedShutdown) {
987 closeAll();
988
989 bytesProduced += bioLengthBefore - SSL.bioLengthByteBuffer(networkBIO);
990
991
992
993
994 SSLEngineResult.HandshakeStatus hs = mayFinishHandshake(
995 status != FINISHED ? bytesProduced == dst.remaining() ? NEED_WRAP
996 : getHandshakeStatus(SSL.bioLengthNonApplication(networkBIO))
997 : FINISHED);
998 return newResult(hs, bytesConsumed, bytesProduced);
999 }
1000
1001 return newResult(NOT_HANDSHAKING, bytesConsumed, bytesProduced);
1002 } else if (sslError == SSL.SSL_ERROR_WANT_READ) {
1003
1004
1005
1006 return newResult(NEED_UNWRAP, bytesConsumed, bytesProduced);
1007 } else if (sslError == SSL.SSL_ERROR_WANT_WRITE) {
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020 if (bytesProduced > 0) {
1021
1022
1023 return newResult(NEED_WRAP, bytesConsumed, bytesProduced);
1024 }
1025 return newResult(BUFFER_OVERFLOW, status, bytesConsumed, bytesProduced);
1026 } else if (sslError == SSL.SSL_ERROR_WANT_X509_LOOKUP ||
1027 sslError == SSL.SSL_ERROR_WANT_CERTIFICATE_VERIFY ||
1028 sslError == SSL.SSL_ERROR_WANT_PRIVATE_KEY_OPERATION) {
1029
1030 return newResult(NEED_TASK, bytesConsumed, bytesProduced);
1031 } else {
1032
1033 throw shutdownWithError("SSL_write", sslError, SSL.getLastErrorNumber());
1034 }
1035 }
1036 }
1037 return newResultMayFinishHandshake(status, bytesConsumed, bytesProduced);
1038 } finally {
1039 SSL.bioClearByteBuffer(networkBIO);
1040 if (bioReadCopyBuf == null) {
1041 dst.position(dst.position() + bytesProduced);
1042 } else {
1043 assert bioReadCopyBuf.readableBytes() <= dst.remaining() : "The destination buffer " + dst +
1044 " didn't have enough remaining space to hold the encrypted content in " + bioReadCopyBuf;
1045 dst.put(bioReadCopyBuf.internalNioBuffer(bioReadCopyBuf.readerIndex(), bytesProduced));
1046 bioReadCopyBuf.release();
1047 }
1048 }
1049 }
1050 }
1051
1052 private SSLEngineResult newResult(SSLEngineResult.HandshakeStatus hs, int bytesConsumed, int bytesProduced) {
1053 return newResult(OK, hs, bytesConsumed, bytesProduced);
1054 }
1055
1056 private SSLEngineResult newResult(SSLEngineResult.Status status, SSLEngineResult.HandshakeStatus hs,
1057 int bytesConsumed, int bytesProduced) {
1058
1059
1060
1061 if (isOutboundDone()) {
1062 if (isInboundDone()) {
1063
1064 hs = NOT_HANDSHAKING;
1065
1066
1067 shutdown();
1068 }
1069 return new SSLEngineResult(CLOSED, hs, bytesConsumed, bytesProduced);
1070 }
1071 if (hs == NEED_TASK) {
1072
1073 needTask = true;
1074 }
1075 return new SSLEngineResult(status, hs, bytesConsumed, bytesProduced);
1076 }
1077
1078 private SSLEngineResult newResultMayFinishHandshake(SSLEngineResult.HandshakeStatus hs,
1079 int bytesConsumed, int bytesProduced) throws SSLException {
1080 return newResult(mayFinishHandshake(hs, bytesConsumed, bytesProduced), bytesConsumed, bytesProduced);
1081 }
1082
1083 private SSLEngineResult newResultMayFinishHandshake(SSLEngineResult.Status status,
1084 SSLEngineResult.HandshakeStatus hs,
1085 int bytesConsumed, int bytesProduced) throws SSLException {
1086 return newResult(status, mayFinishHandshake(hs, bytesConsumed, bytesProduced), bytesConsumed, bytesProduced);
1087 }
1088
1089
1090
1091
1092 private SSLException shutdownWithError(String operation, int sslError, int error) {
1093 if (logger.isDebugEnabled()) {
1094 String errorString = SSL.getErrorString(error);
1095 logger.debug("{} failed with {}: OpenSSL error: {} {}",
1096 operation, sslError, error, errorString);
1097 }
1098
1099
1100 shutdown();
1101
1102 SSLException exception = newSSLExceptionForError(error);
1103
1104 if (pendingException != null) {
1105 exception.initCause(pendingException);
1106 pendingException = null;
1107 }
1108 return exception;
1109 }
1110
1111 private SSLEngineResult handleUnwrapException(int bytesConsumed, int bytesProduced, SSLException e)
1112 throws SSLException {
1113 int lastError = SSL.getLastErrorNumber();
1114 if (lastError != 0) {
1115 return sslReadErrorResult(SSL.SSL_ERROR_SSL, lastError, bytesConsumed,
1116 bytesProduced);
1117 }
1118 throw e;
1119 }
1120
1121 public final SSLEngineResult unwrap(
1122 final ByteBuffer[] srcs, int srcsOffset, final int srcsLength,
1123 final ByteBuffer[] dsts, int dstsOffset, final int dstsLength) throws SSLException {
1124
1125
1126 checkNotNullWithIAE(srcs, "srcs");
1127 if (srcsOffset >= srcs.length
1128 || srcsOffset + srcsLength > srcs.length) {
1129 throw new IndexOutOfBoundsException(
1130 "offset: " + srcsOffset + ", length: " + srcsLength +
1131 " (expected: offset <= offset + length <= srcs.length (" + srcs.length + "))");
1132 }
1133 checkNotNullWithIAE(dsts, "dsts");
1134 if (dstsOffset >= dsts.length || dstsOffset + dstsLength > dsts.length) {
1135 throw new IndexOutOfBoundsException(
1136 "offset: " + dstsOffset + ", length: " + dstsLength +
1137 " (expected: offset <= offset + length <= dsts.length (" + dsts.length + "))");
1138 }
1139 long capacity = 0;
1140 final int dstsEndOffset = dstsOffset + dstsLength;
1141 for (int i = dstsOffset; i < dstsEndOffset; i ++) {
1142 ByteBuffer dst = checkNotNullArrayParam(dsts[i], i, "dsts");
1143 if (dst.isReadOnly()) {
1144 throw new ReadOnlyBufferException();
1145 }
1146 capacity += dst.remaining();
1147 }
1148
1149 final int srcsEndOffset = srcsOffset + srcsLength;
1150 long len = 0;
1151 for (int i = srcsOffset; i < srcsEndOffset; i++) {
1152 ByteBuffer src = checkNotNullArrayParam(srcs[i], i, "srcs");
1153 len += src.remaining();
1154 }
1155
1156 synchronized (this) {
1157 if (isInboundDone()) {
1158 return isOutboundDone() || isDestroyed() ? CLOSED_NOT_HANDSHAKING : NEED_WRAP_CLOSED;
1159 }
1160
1161 SSLEngineResult.HandshakeStatus status = NOT_HANDSHAKING;
1162 HandshakeState oldHandshakeState = handshakeState;
1163
1164 if (handshakeState != HandshakeState.FINISHED) {
1165 if (handshakeState != HandshakeState.STARTED_EXPLICITLY) {
1166
1167 handshakeState = HandshakeState.STARTED_IMPLICITLY;
1168 }
1169
1170 status = handshake();
1171
1172 if (status == NEED_TASK) {
1173 return newResult(status, 0, 0);
1174 }
1175
1176 if (status == NEED_WRAP) {
1177 return NEED_WRAP_OK;
1178 }
1179
1180 if (isInboundDone) {
1181 return NEED_WRAP_CLOSED;
1182 }
1183 }
1184
1185 int sslPending = sslPending0();
1186 int packetLength;
1187
1188
1189
1190
1191 if (jdkCompatibilityMode ||
1192
1193
1194
1195 oldHandshakeState != HandshakeState.FINISHED) {
1196 if (len < SSL_RECORD_HEADER_LENGTH) {
1197 return newResultMayFinishHandshake(BUFFER_UNDERFLOW, status, 0, 0);
1198 }
1199
1200 packetLength = SslUtils.getEncryptedPacketLength(srcs, srcsOffset);
1201 if (packetLength == SslUtils.NOT_ENCRYPTED) {
1202 throw new NotSslRecordException("not an SSL/TLS record");
1203 }
1204
1205 final int packetLengthDataOnly = packetLength - SSL_RECORD_HEADER_LENGTH;
1206 if (packetLengthDataOnly > capacity) {
1207
1208
1209 if (packetLengthDataOnly > MAX_RECORD_SIZE) {
1210
1211
1212
1213
1214
1215 throw new SSLException("Illegal packet length: " + packetLengthDataOnly + " > " +
1216 session.getApplicationBufferSize());
1217 } else {
1218 session.tryExpandApplicationBufferSize(packetLengthDataOnly);
1219 }
1220 return newResultMayFinishHandshake(BUFFER_OVERFLOW, status, 0, 0);
1221 }
1222
1223 if (len < packetLength) {
1224
1225
1226 return newResultMayFinishHandshake(BUFFER_UNDERFLOW, status, 0, 0);
1227 }
1228 } else if (len == 0 && sslPending <= 0) {
1229 return newResultMayFinishHandshake(BUFFER_UNDERFLOW, status, 0, 0);
1230 } else if (capacity == 0) {
1231 return newResultMayFinishHandshake(BUFFER_OVERFLOW, status, 0, 0);
1232 } else {
1233 packetLength = (int) min(MAX_VALUE, len);
1234 }
1235
1236
1237 assert srcsOffset < srcsEndOffset;
1238
1239
1240 assert capacity > 0;
1241
1242
1243 int bytesProduced = 0;
1244 int bytesConsumed = 0;
1245 try {
1246 srcLoop:
1247 for (;;) {
1248 ByteBuffer src = srcs[srcsOffset];
1249 int remaining = src.remaining();
1250 final ByteBuf bioWriteCopyBuf;
1251 int pendingEncryptedBytes;
1252 if (remaining == 0) {
1253 if (sslPending <= 0) {
1254
1255
1256 if (++srcsOffset >= srcsEndOffset) {
1257 break;
1258 }
1259 continue;
1260 } else {
1261 bioWriteCopyBuf = null;
1262 pendingEncryptedBytes = SSL.bioLengthByteBuffer(networkBIO);
1263 }
1264 } else {
1265
1266
1267 pendingEncryptedBytes = min(packetLength, remaining);
1268 try {
1269 bioWriteCopyBuf = writeEncryptedData(src, pendingEncryptedBytes);
1270 } catch (SSLException e) {
1271
1272 return handleUnwrapException(bytesConsumed, bytesProduced, e);
1273 }
1274 }
1275 try {
1276 for (;;) {
1277 ByteBuffer dst = dsts[dstsOffset];
1278 if (!dst.hasRemaining()) {
1279
1280 if (++dstsOffset >= dstsEndOffset) {
1281 break srcLoop;
1282 }
1283 continue;
1284 }
1285
1286 int bytesRead;
1287 try {
1288 bytesRead = readPlaintextData(dst);
1289 } catch (SSLException e) {
1290
1291 return handleUnwrapException(bytesConsumed, bytesProduced, e);
1292 }
1293
1294
1295
1296 int localBytesConsumed = pendingEncryptedBytes - SSL.bioLengthByteBuffer(networkBIO);
1297 bytesConsumed += localBytesConsumed;
1298 packetLength -= localBytesConsumed;
1299 pendingEncryptedBytes -= localBytesConsumed;
1300 src.position(src.position() + localBytesConsumed);
1301
1302 if (bytesRead > 0) {
1303 bytesProduced += bytesRead;
1304
1305 if (!dst.hasRemaining()) {
1306 sslPending = sslPending0();
1307
1308 if (++dstsOffset >= dstsEndOffset) {
1309 return sslPending > 0 ?
1310 newResult(BUFFER_OVERFLOW, status, bytesConsumed, bytesProduced) :
1311 newResultMayFinishHandshake(isInboundDone() ? CLOSED : OK, status,
1312 bytesConsumed, bytesProduced);
1313 }
1314 } else if (packetLength == 0 || jdkCompatibilityMode) {
1315
1316
1317 break srcLoop;
1318 }
1319 } else {
1320 int sslError = SSL.getError(ssl, bytesRead);
1321 if (sslError == SSL.SSL_ERROR_WANT_READ || sslError == SSL.SSL_ERROR_WANT_WRITE) {
1322
1323
1324 break;
1325 } else if (sslError == SSL.SSL_ERROR_ZERO_RETURN) {
1326
1327 if (!receivedShutdown) {
1328 closeAll();
1329 }
1330 return newResultMayFinishHandshake(isInboundDone() ? CLOSED : OK, status,
1331 bytesConsumed, bytesProduced);
1332 } else if (sslError == SSL.SSL_ERROR_WANT_X509_LOOKUP ||
1333 sslError == SSL.SSL_ERROR_WANT_CERTIFICATE_VERIFY ||
1334 sslError == SSL.SSL_ERROR_WANT_PRIVATE_KEY_OPERATION) {
1335 return newResult(isInboundDone() ? CLOSED : OK,
1336 NEED_TASK, bytesConsumed, bytesProduced);
1337 } else {
1338 return sslReadErrorResult(sslError, SSL.getLastErrorNumber(), bytesConsumed,
1339 bytesProduced);
1340 }
1341 }
1342 }
1343
1344 if (++srcsOffset >= srcsEndOffset) {
1345 break;
1346 }
1347 } finally {
1348 if (bioWriteCopyBuf != null) {
1349 bioWriteCopyBuf.release();
1350 }
1351 }
1352 }
1353 } finally {
1354 SSL.bioClearByteBuffer(networkBIO);
1355 rejectRemoteInitiatedRenegotiation();
1356 }
1357
1358
1359 if (!receivedShutdown && (SSL.getShutdown(ssl) & SSL.SSL_RECEIVED_SHUTDOWN) == SSL.SSL_RECEIVED_SHUTDOWN) {
1360 closeAll();
1361 }
1362
1363 return newResultMayFinishHandshake(isInboundDone() ? CLOSED : OK, status, bytesConsumed, bytesProduced);
1364 }
1365 }
1366
1367 private boolean needWrapAgain(int stackError) {
1368
1369
1370
1371
1372 if (SSL.bioLengthNonApplication(networkBIO) > 0) {
1373
1374
1375 if (pendingException == null) {
1376 pendingException = newSSLExceptionForError(stackError);
1377 } else if (shouldAddSuppressed(pendingException, stackError)) {
1378 ThrowableUtil.addSuppressed(pendingException, newSSLExceptionForError(stackError));
1379 }
1380
1381
1382 SSL.clearError();
1383 return true;
1384 }
1385 return false;
1386 }
1387
1388 private SSLException newSSLExceptionForError(int stackError) {
1389 String message = SSL.getErrorString(stackError);
1390 return handshakeState == HandshakeState.FINISHED ?
1391 new OpenSslException(message, stackError) : new OpenSslHandshakeException(message, stackError);
1392 }
1393
1394 private static boolean shouldAddSuppressed(Throwable target, int errorCode) {
1395 for (Throwable suppressed: ThrowableUtil.getSuppressed(target)) {
1396 if (suppressed instanceof NativeSslException &&
1397 ((NativeSslException) suppressed).errorCode() == errorCode) {
1398
1399 return false;
1400 }
1401 }
1402 return true;
1403 }
1404
1405 private SSLEngineResult sslReadErrorResult(int error, int stackError, int bytesConsumed, int bytesProduced)
1406 throws SSLException {
1407 if (needWrapAgain(stackError)) {
1408
1409
1410 return new SSLEngineResult(OK, NEED_WRAP, bytesConsumed, bytesProduced);
1411 }
1412 throw shutdownWithError("SSL_read", error, stackError);
1413 }
1414
1415 private void closeAll() throws SSLException {
1416 receivedShutdown = true;
1417 closeOutbound();
1418 closeInbound();
1419 }
1420
1421 private void rejectRemoteInitiatedRenegotiation() throws SSLHandshakeException {
1422
1423
1424
1425 if (!isDestroyed() && (!clientMode && SSL.getHandshakeCount(ssl) > 1 ||
1426
1427 clientMode && SSL.getHandshakeCount(ssl) > 2) &&
1428
1429
1430 !SslProtocols.TLS_v1_3.equals(session.getProtocol()) && handshakeState == HandshakeState.FINISHED) {
1431
1432
1433 shutdown();
1434 throw new SSLHandshakeException("remote-initiated renegotiation not allowed");
1435 }
1436 }
1437
1438 public final SSLEngineResult unwrap(final ByteBuffer[] srcs, final ByteBuffer[] dsts) throws SSLException {
1439 return unwrap(srcs, 0, srcs.length, dsts, 0, dsts.length);
1440 }
1441
1442 private ByteBuffer[] singleSrcBuffer(ByteBuffer src) {
1443 singleSrcBuffer[0] = src;
1444 return singleSrcBuffer;
1445 }
1446
1447 private void resetSingleSrcBuffer() {
1448 singleSrcBuffer[0] = null;
1449 }
1450
1451 private ByteBuffer[] singleDstBuffer(ByteBuffer src) {
1452 singleDstBuffer[0] = src;
1453 return singleDstBuffer;
1454 }
1455
1456 private void resetSingleDstBuffer() {
1457 singleDstBuffer[0] = null;
1458 }
1459
1460 @Override
1461 public final synchronized SSLEngineResult unwrap(
1462 final ByteBuffer src, final ByteBuffer[] dsts, final int offset, final int length) throws SSLException {
1463 try {
1464 return unwrap(singleSrcBuffer(src), 0, 1, dsts, offset, length);
1465 } finally {
1466 resetSingleSrcBuffer();
1467 }
1468 }
1469
1470 @Override
1471 public final synchronized SSLEngineResult wrap(ByteBuffer src, ByteBuffer dst) throws SSLException {
1472 try {
1473 return wrap(singleSrcBuffer(src), dst);
1474 } finally {
1475 resetSingleSrcBuffer();
1476 }
1477 }
1478
1479 @Override
1480 public final synchronized SSLEngineResult unwrap(ByteBuffer src, ByteBuffer dst) throws SSLException {
1481 try {
1482 return unwrap(singleSrcBuffer(src), singleDstBuffer(dst));
1483 } finally {
1484 resetSingleSrcBuffer();
1485 resetSingleDstBuffer();
1486 }
1487 }
1488
1489 @Override
1490 public final synchronized SSLEngineResult unwrap(ByteBuffer src, ByteBuffer[] dsts) throws SSLException {
1491 try {
1492 return unwrap(singleSrcBuffer(src), dsts);
1493 } finally {
1494 resetSingleSrcBuffer();
1495 }
1496 }
1497
1498 private class TaskDecorator<R extends Runnable> implements Runnable {
1499 protected final R task;
1500 TaskDecorator(R task) {
1501 this.task = task;
1502 }
1503
1504 @Override
1505 public void run() {
1506 runAndResetNeedTask(task);
1507 }
1508 }
1509
1510 private final class AsyncTaskDecorator extends TaskDecorator<AsyncTask> implements AsyncRunnable {
1511 AsyncTaskDecorator(AsyncTask task) {
1512 super(task);
1513 }
1514
1515 @Override
1516 public void run(final Runnable runnable) {
1517 if (isDestroyed()) {
1518
1519 return;
1520 }
1521 task.runAsync(new TaskDecorator<Runnable>(runnable));
1522 }
1523 }
1524
1525 private void runAndResetNeedTask(Runnable task) {
1526
1527
1528 synchronized (ReferenceCountedOpenSslEngine.this) {
1529 try {
1530 if (isDestroyed()) {
1531
1532 return;
1533 }
1534 task.run();
1535 if (handshakeState != HandshakeState.FINISHED && !isDestroyed()) {
1536
1537
1538
1539 if (SSL.doHandshake(ssl) <= 0) {
1540 SSL.clearError();
1541 }
1542 }
1543 } finally {
1544
1545 needTask = false;
1546 }
1547 }
1548 }
1549
1550 @Override
1551 public final synchronized Runnable getDelegatedTask() {
1552 if (isDestroyed()) {
1553 return null;
1554 }
1555 final Runnable task = SSL.getTask(ssl);
1556 if (task == null) {
1557 return null;
1558 }
1559 if (task instanceof AsyncTask) {
1560 return new AsyncTaskDecorator((AsyncTask) task);
1561 }
1562 return new TaskDecorator<Runnable>(task);
1563 }
1564
1565 @Override
1566 public final synchronized void closeInbound() throws SSLException {
1567 if (isInboundDone) {
1568 return;
1569 }
1570
1571 isInboundDone = true;
1572
1573 if (isOutboundDone()) {
1574
1575
1576 shutdown();
1577 }
1578
1579 if (handshakeState != HandshakeState.NOT_STARTED && !receivedShutdown) {
1580 throw new SSLException(
1581 "Inbound closed before receiving peer's close_notify: possible truncation attack?");
1582 }
1583 }
1584
1585 @Override
1586 public final synchronized boolean isInboundDone() {
1587 return isInboundDone;
1588 }
1589
1590 @Override
1591 public final synchronized void closeOutbound() {
1592 if (outboundClosed) {
1593 return;
1594 }
1595
1596 outboundClosed = true;
1597
1598 if (handshakeState != HandshakeState.NOT_STARTED && !isDestroyed()) {
1599 int mode = SSL.getShutdown(ssl);
1600 if ((mode & SSL.SSL_SENT_SHUTDOWN) != SSL.SSL_SENT_SHUTDOWN) {
1601 doSSLShutdown();
1602 }
1603 } else {
1604
1605 shutdown();
1606 }
1607 }
1608
1609
1610
1611
1612
1613 private boolean doSSLShutdown() {
1614 if (SSL.isInInit(ssl) != 0) {
1615
1616
1617
1618
1619 return false;
1620 }
1621 int err = SSL.shutdownSSL(ssl);
1622 if (err < 0) {
1623 int sslErr = SSL.getError(ssl, err);
1624 if (sslErr == SSL.SSL_ERROR_SYSCALL || sslErr == SSL.SSL_ERROR_SSL) {
1625 if (logger.isDebugEnabled()) {
1626 int error = SSL.getLastErrorNumber();
1627 logger.debug("SSL_shutdown failed: OpenSSL error: {} {}", error, SSL.getErrorString(error));
1628 }
1629
1630 shutdown();
1631 return false;
1632 }
1633 SSL.clearError();
1634 }
1635 return true;
1636 }
1637
1638 @Override
1639 public final synchronized boolean isOutboundDone() {
1640
1641
1642 return outboundClosed && (networkBIO == 0 || SSL.bioLengthNonApplication(networkBIO) == 0);
1643 }
1644
1645 @Override
1646 public final String[] getSupportedCipherSuites() {
1647 return OpenSsl.AVAILABLE_CIPHER_SUITES.toArray(EMPTY_STRINGS);
1648 }
1649
1650 @Override
1651 public final String[] getEnabledCipherSuites() {
1652 final String[] extraCiphers;
1653 final String[] enabled;
1654 final boolean tls13Enabled;
1655 synchronized (this) {
1656 if (!isDestroyed()) {
1657 enabled = SSL.getCiphers(ssl);
1658 int opts = SSL.getOptions(ssl);
1659 if (isProtocolEnabled(opts, SSL.SSL_OP_NO_TLSv1_3, SslProtocols.TLS_v1_3)) {
1660 extraCiphers = OpenSsl.EXTRA_SUPPORTED_TLS_1_3_CIPHERS;
1661 tls13Enabled = true;
1662 } else {
1663 extraCiphers = EMPTY_STRINGS;
1664 tls13Enabled = false;
1665 }
1666 } else {
1667 return EMPTY_STRINGS;
1668 }
1669 }
1670 if (enabled == null) {
1671 return EMPTY_STRINGS;
1672 } else {
1673 Set<String> enabledSet = new LinkedHashSet<String>(enabled.length + extraCiphers.length);
1674 synchronized (this) {
1675 for (int i = 0; i < enabled.length; i++) {
1676 String mapped = toJavaCipherSuite(enabled[i]);
1677 final String cipher = mapped == null ? enabled[i] : mapped;
1678 if ((!tls13Enabled || !OpenSsl.isTlsv13Supported()) && SslUtils.isTLSv13Cipher(cipher)) {
1679 continue;
1680 }
1681 enabledSet.add(cipher);
1682 }
1683 Collections.addAll(enabledSet, extraCiphers);
1684 }
1685 return enabledSet.toArray(EMPTY_STRINGS);
1686 }
1687 }
1688
1689 @Override
1690 public final void setEnabledCipherSuites(String[] cipherSuites) {
1691 checkNotNull(cipherSuites, "cipherSuites");
1692
1693 final StringBuilder buf = new StringBuilder();
1694 final StringBuilder bufTLSv13 = new StringBuilder();
1695
1696 CipherSuiteConverter.convertToCipherStrings(Arrays.asList(cipherSuites), buf, bufTLSv13, OpenSsl.isBoringSSL());
1697 final String cipherSuiteSpec = buf.toString();
1698 final String cipherSuiteSpecTLSv13 = bufTLSv13.toString();
1699
1700 if (!OpenSsl.isTlsv13Supported() && !cipherSuiteSpecTLSv13.isEmpty()) {
1701 throw new IllegalArgumentException("TLSv1.3 is not supported by this java version.");
1702 }
1703 synchronized (this) {
1704 hasTLSv13Cipher = !cipherSuiteSpecTLSv13.isEmpty();
1705 if (!isDestroyed()) {
1706 try {
1707
1708 SSL.setCipherSuites(ssl, cipherSuiteSpec, false);
1709 if (OpenSsl.isTlsv13Supported()) {
1710
1711 SSL.setCipherSuites(ssl, OpenSsl.checkTls13Ciphers(logger, cipherSuiteSpecTLSv13), true);
1712 }
1713
1714
1715
1716 Set<String> protocols = new HashSet<String>(enabledProtocols);
1717
1718
1719
1720 if (cipherSuiteSpec.isEmpty()) {
1721 protocols.remove(SslProtocols.TLS_v1);
1722 protocols.remove(SslProtocols.TLS_v1_1);
1723 protocols.remove(SslProtocols.TLS_v1_2);
1724 protocols.remove(SslProtocols.SSL_v3);
1725 protocols.remove(SslProtocols.SSL_v2);
1726 protocols.remove(SslProtocols.SSL_v2_HELLO);
1727 }
1728
1729 if (cipherSuiteSpecTLSv13.isEmpty()) {
1730 protocols.remove(SslProtocols.TLS_v1_3);
1731 }
1732
1733
1734 setEnabledProtocols0(protocols.toArray(EMPTY_STRINGS), !hasTLSv13Cipher);
1735 } catch (Exception e) {
1736 throw new IllegalStateException("failed to enable cipher suites: " + cipherSuiteSpec, e);
1737 }
1738 } else {
1739 throw new IllegalStateException("failed to enable cipher suites: " + cipherSuiteSpec);
1740 }
1741 }
1742 }
1743
1744 @Override
1745 public final String[] getSupportedProtocols() {
1746 return OpenSsl.SUPPORTED_PROTOCOLS_SET.toArray(EMPTY_STRINGS);
1747 }
1748
1749 @Override
1750 public final String[] getEnabledProtocols() {
1751 return enabledProtocols.toArray(EMPTY_STRINGS);
1752 }
1753
1754 private static boolean isProtocolEnabled(int opts, int disableMask, String protocolString) {
1755
1756
1757 return (opts & disableMask) == 0 && OpenSsl.SUPPORTED_PROTOCOLS_SET.contains(protocolString);
1758 }
1759
1760
1761
1762
1763
1764
1765
1766
1767
1768
1769 @Override
1770 public final void setEnabledProtocols(String[] protocols) {
1771 checkNotNullWithIAE(protocols, "protocols");
1772 synchronized (this) {
1773 enabledProtocols.clear();
1774
1775 enabledProtocols.add(SslProtocols.SSL_v2_HELLO);
1776
1777 Collections.addAll(enabledProtocols, protocols);
1778
1779 setEnabledProtocols0(protocols, !hasTLSv13Cipher);
1780 }
1781 }
1782
1783 private void setEnabledProtocols0(String[] protocols, boolean explicitDisableTLSv13) {
1784 assert Thread.holdsLock(this);
1785
1786 int minProtocolIndex = OPENSSL_OP_NO_PROTOCOLS.length;
1787 int maxProtocolIndex = 0;
1788 for (String p: protocols) {
1789 if (!OpenSsl.SUPPORTED_PROTOCOLS_SET.contains(p)) {
1790 throw new IllegalArgumentException("Protocol " + p + " is not supported.");
1791 }
1792 if (p.equals(SslProtocols.SSL_v2)) {
1793 if (minProtocolIndex > OPENSSL_OP_NO_PROTOCOL_INDEX_SSLV2) {
1794 minProtocolIndex = OPENSSL_OP_NO_PROTOCOL_INDEX_SSLV2;
1795 }
1796 if (maxProtocolIndex < OPENSSL_OP_NO_PROTOCOL_INDEX_SSLV2) {
1797 maxProtocolIndex = OPENSSL_OP_NO_PROTOCOL_INDEX_SSLV2;
1798 }
1799 } else if (p.equals(SslProtocols.SSL_v3)) {
1800 if (minProtocolIndex > OPENSSL_OP_NO_PROTOCOL_INDEX_SSLV3) {
1801 minProtocolIndex = OPENSSL_OP_NO_PROTOCOL_INDEX_SSLV3;
1802 }
1803 if (maxProtocolIndex < OPENSSL_OP_NO_PROTOCOL_INDEX_SSLV3) {
1804 maxProtocolIndex = OPENSSL_OP_NO_PROTOCOL_INDEX_SSLV3;
1805 }
1806 } else if (p.equals(SslProtocols.TLS_v1)) {
1807 if (minProtocolIndex > OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1) {
1808 minProtocolIndex = OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1;
1809 }
1810 if (maxProtocolIndex < OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1) {
1811 maxProtocolIndex = OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1;
1812 }
1813 } else if (p.equals(SslProtocols.TLS_v1_1)) {
1814 if (minProtocolIndex > OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_1) {
1815 minProtocolIndex = OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_1;
1816 }
1817 if (maxProtocolIndex < OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_1) {
1818 maxProtocolIndex = OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_1;
1819 }
1820 } else if (p.equals(SslProtocols.TLS_v1_2)) {
1821 if (minProtocolIndex > OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_2) {
1822 minProtocolIndex = OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_2;
1823 }
1824 if (maxProtocolIndex < OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_2) {
1825 maxProtocolIndex = OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_2;
1826 }
1827 } else if (!explicitDisableTLSv13 && p.equals(SslProtocols.TLS_v1_3)) {
1828 if (minProtocolIndex > OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_3) {
1829 minProtocolIndex = OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_3;
1830 }
1831 if (maxProtocolIndex < OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_3) {
1832 maxProtocolIndex = OPENSSL_OP_NO_PROTOCOL_INDEX_TLSv1_3;
1833 }
1834 }
1835 }
1836 if (!isDestroyed()) {
1837
1838 SSL.clearOptions(ssl, SSL.SSL_OP_NO_SSLv2 | SSL.SSL_OP_NO_SSLv3 | SSL.SSL_OP_NO_TLSv1 |
1839 SSL.SSL_OP_NO_TLSv1_1 | SSL.SSL_OP_NO_TLSv1_2 | SSL.SSL_OP_NO_TLSv1_3);
1840
1841 int opts = 0;
1842 for (int i = 0; i < minProtocolIndex; ++i) {
1843 opts |= OPENSSL_OP_NO_PROTOCOLS[i];
1844 }
1845 assert maxProtocolIndex != MAX_VALUE;
1846 for (int i = maxProtocolIndex + 1; i < OPENSSL_OP_NO_PROTOCOLS.length; ++i) {
1847 opts |= OPENSSL_OP_NO_PROTOCOLS[i];
1848 }
1849
1850
1851 SSL.setOptions(ssl, opts);
1852 } else {
1853 throw new IllegalStateException("failed to enable protocols: " + Arrays.asList(protocols));
1854 }
1855 }
1856
1857 @Override
1858 public final SSLSession getSession() {
1859 return session;
1860 }
1861
1862 @Override
1863 public final synchronized void beginHandshake() throws SSLException {
1864 switch (handshakeState) {
1865 case STARTED_IMPLICITLY:
1866 checkEngineClosed();
1867
1868
1869
1870
1871
1872
1873
1874 handshakeState = HandshakeState.STARTED_EXPLICITLY;
1875 calculateMaxWrapOverhead();
1876
1877 break;
1878 case STARTED_EXPLICITLY:
1879
1880 break;
1881 case FINISHED:
1882 throw new SSLException("renegotiation unsupported");
1883 case NOT_STARTED:
1884 handshakeState = HandshakeState.STARTED_EXPLICITLY;
1885 if (handshake() == NEED_TASK) {
1886
1887 needTask = true;
1888 }
1889 calculateMaxWrapOverhead();
1890 break;
1891 default:
1892 throw new Error();
1893 }
1894 }
1895
1896 private void checkEngineClosed() throws SSLException {
1897 if (isDestroyed()) {
1898 throw new SSLException("engine closed");
1899 }
1900 }
1901
1902 private static SSLEngineResult.HandshakeStatus pendingStatus(int pendingStatus) {
1903
1904 return pendingStatus > 0 ? NEED_WRAP : NEED_UNWRAP;
1905 }
1906
1907 private static boolean isEmpty(Object[] arr) {
1908 return arr == null || arr.length == 0;
1909 }
1910
1911 private static boolean isEmpty(byte[] cert) {
1912 return cert == null || cert.length == 0;
1913 }
1914
1915 private SSLEngineResult.HandshakeStatus handshakeException() throws SSLException {
1916 if (SSL.bioLengthNonApplication(networkBIO) > 0) {
1917
1918 return NEED_WRAP;
1919 }
1920
1921 Throwable exception = pendingException;
1922 assert exception != null;
1923 pendingException = null;
1924 shutdown();
1925 if (exception instanceof SSLHandshakeException) {
1926 throw (SSLHandshakeException) exception;
1927 }
1928 SSLHandshakeException e = new SSLHandshakeException("General OpenSslEngine problem");
1929 e.initCause(exception);
1930 throw e;
1931 }
1932
1933
1934
1935
1936
1937 final void initHandshakeException(Throwable cause) {
1938 if (pendingException == null) {
1939 pendingException = cause;
1940 } else {
1941 ThrowableUtil.addSuppressed(pendingException, cause);
1942 }
1943 }
1944
1945 private SSLEngineResult.HandshakeStatus handshake() throws SSLException {
1946 if (needTask) {
1947 return NEED_TASK;
1948 }
1949 if (handshakeState == HandshakeState.FINISHED) {
1950 return FINISHED;
1951 }
1952
1953 checkEngineClosed();
1954
1955 if (pendingException != null) {
1956
1957
1958 if (SSL.doHandshake(ssl) <= 0) {
1959
1960 SSL.clearError();
1961 }
1962 return handshakeException();
1963 }
1964
1965
1966 engineMap.add(this);
1967
1968 if (!sessionSet) {
1969 if (!parentContext.sessionContext().setSessionFromCache(ssl, session, getPeerHost(), getPeerPort())) {
1970
1971
1972 session.prepareHandshake();
1973 }
1974 sessionSet = true;
1975 }
1976
1977 int code = SSL.doHandshake(ssl);
1978 if (code <= 0) {
1979 int sslError = SSL.getError(ssl, code);
1980 if (sslError == SSL.SSL_ERROR_WANT_READ || sslError == SSL.SSL_ERROR_WANT_WRITE) {
1981 return pendingStatus(SSL.bioLengthNonApplication(networkBIO));
1982 }
1983
1984 if (sslError == SSL.SSL_ERROR_WANT_X509_LOOKUP ||
1985 sslError == SSL.SSL_ERROR_WANT_CERTIFICATE_VERIFY ||
1986 sslError == SSL.SSL_ERROR_WANT_PRIVATE_KEY_OPERATION) {
1987 return NEED_TASK;
1988 }
1989
1990 int errorNumber = SSL.getLastErrorNumber();
1991 if (needWrapAgain(errorNumber)) {
1992
1993
1994 return NEED_WRAP;
1995 }
1996
1997
1998 if (pendingException != null) {
1999 return handshakeException();
2000 }
2001
2002
2003 throw shutdownWithError("SSL_do_handshake", sslError, errorNumber);
2004 }
2005
2006 if (SSL.bioLengthNonApplication(networkBIO) > 0) {
2007 return NEED_WRAP;
2008 }
2009
2010 session.handshakeFinished(SSL.getSessionId(ssl), SSL.getCipherForSSL(ssl), SSL.getVersion(ssl),
2011 SSL.getPeerCertificate(ssl), SSL.getPeerCertChain(ssl),
2012 SSL.getTime(ssl) * 1000L, parentContext.sessionTimeout() * 1000L);
2013 selectApplicationProtocol();
2014 return FINISHED;
2015 }
2016
2017 private SSLEngineResult.HandshakeStatus mayFinishHandshake(
2018 SSLEngineResult.HandshakeStatus hs, int bytesConsumed, int bytesProduced) throws SSLException {
2019 return hs == NEED_UNWRAP && bytesProduced > 0 || hs == NEED_WRAP && bytesConsumed > 0 ?
2020 handshake() : mayFinishHandshake(hs != FINISHED ? getHandshakeStatus() : FINISHED);
2021 }
2022
2023 private SSLEngineResult.HandshakeStatus mayFinishHandshake(SSLEngineResult.HandshakeStatus status)
2024 throws SSLException {
2025 if (status == NOT_HANDSHAKING) {
2026 if (handshakeState != HandshakeState.FINISHED) {
2027
2028
2029 return handshake();
2030 }
2031 if (!isDestroyed() && SSL.bioLengthNonApplication(networkBIO) > 0) {
2032
2033 return NEED_WRAP;
2034 }
2035 }
2036 return status;
2037 }
2038
2039 @Override
2040 public final synchronized SSLEngineResult.HandshakeStatus getHandshakeStatus() {
2041
2042 if (needPendingStatus()) {
2043 if (needTask) {
2044
2045 return NEED_TASK;
2046 }
2047 return pendingStatus(SSL.bioLengthNonApplication(networkBIO));
2048 }
2049 return NOT_HANDSHAKING;
2050 }
2051
2052 private SSLEngineResult.HandshakeStatus getHandshakeStatus(int pending) {
2053
2054 if (needPendingStatus()) {
2055 if (needTask) {
2056
2057 return NEED_TASK;
2058 }
2059 return pendingStatus(pending);
2060 }
2061 return NOT_HANDSHAKING;
2062 }
2063
2064 private boolean needPendingStatus() {
2065 return handshakeState != HandshakeState.NOT_STARTED && !isDestroyed()
2066 && (handshakeState != HandshakeState.FINISHED || isInboundDone() || isOutboundDone());
2067 }
2068
2069
2070
2071
2072 private String toJavaCipherSuite(String openSslCipherSuite) {
2073 if (openSslCipherSuite == null) {
2074 return null;
2075 }
2076
2077 String version = SSL.getVersion(ssl);
2078 String prefix = toJavaCipherSuitePrefix(version);
2079 return CipherSuiteConverter.toJava(openSslCipherSuite, prefix);
2080 }
2081
2082
2083
2084
2085 private static String toJavaCipherSuitePrefix(String protocolVersion) {
2086 final char c;
2087 if (protocolVersion == null || protocolVersion.isEmpty()) {
2088 c = 0;
2089 } else {
2090 c = protocolVersion.charAt(0);
2091 }
2092
2093 switch (c) {
2094 case 'T':
2095 return "TLS";
2096 case 'S':
2097 return "SSL";
2098 default:
2099 return "UNKNOWN";
2100 }
2101 }
2102
2103 @Override
2104 public final void setUseClientMode(boolean clientMode) {
2105 if (clientMode != this.clientMode) {
2106 throw new UnsupportedOperationException();
2107 }
2108 }
2109
2110 @Override
2111 public final boolean getUseClientMode() {
2112 return clientMode;
2113 }
2114
2115 @Override
2116 public final void setNeedClientAuth(boolean b) {
2117 setClientAuth(b ? ClientAuth.REQUIRE : ClientAuth.NONE);
2118 }
2119
2120 @Override
2121 public final boolean getNeedClientAuth() {
2122 return clientAuth == ClientAuth.REQUIRE;
2123 }
2124
2125 @Override
2126 public final void setWantClientAuth(boolean b) {
2127 setClientAuth(b ? ClientAuth.OPTIONAL : ClientAuth.NONE);
2128 }
2129
2130 @Override
2131 public final boolean getWantClientAuth() {
2132 return clientAuth == ClientAuth.OPTIONAL;
2133 }
2134
2135
2136
2137
2138
2139 @UnstableApi
2140 public final synchronized void setVerify(int verifyMode, int depth) {
2141 if (!isDestroyed()) {
2142 SSL.setVerify(ssl, verifyMode, depth);
2143 }
2144 }
2145
2146 private void setClientAuth(ClientAuth mode) {
2147 if (clientMode) {
2148 return;
2149 }
2150 synchronized (this) {
2151 if (clientAuth == mode) {
2152
2153 return;
2154 }
2155 if (!isDestroyed()) {
2156 switch (mode) {
2157 case NONE:
2158 SSL.setVerify(ssl, SSL.SSL_CVERIFY_NONE, ReferenceCountedOpenSslContext.VERIFY_DEPTH);
2159 break;
2160 case REQUIRE:
2161 SSL.setVerify(ssl, SSL.SSL_CVERIFY_REQUIRED, ReferenceCountedOpenSslContext.VERIFY_DEPTH);
2162 break;
2163 case OPTIONAL:
2164 SSL.setVerify(ssl, SSL.SSL_CVERIFY_OPTIONAL, ReferenceCountedOpenSslContext.VERIFY_DEPTH);
2165 break;
2166 default:
2167 throw new Error(mode.toString());
2168 }
2169 }
2170 clientAuth = mode;
2171 }
2172 }
2173
2174 @Override
2175 public final void setEnableSessionCreation(boolean b) {
2176 if (b) {
2177 throw new UnsupportedOperationException();
2178 }
2179 }
2180
2181 @Override
2182 public final boolean getEnableSessionCreation() {
2183 return false;
2184 }
2185
2186 @SuppressJava6Requirement(reason = "Usage guarded by java version check")
2187 @Override
2188 public final synchronized SSLParameters getSSLParameters() {
2189 SSLParameters sslParameters = super.getSSLParameters();
2190
2191 int version = PlatformDependent.javaVersion();
2192 if (version >= 7) {
2193 Java7SslParametersUtils.setEndpointIdentificationAlgorithm(sslParameters, endpointIdentificationAlgorithm);
2194 Java7SslParametersUtils.setAlgorithmConstraints(sslParameters, algorithmConstraints);
2195 if (version >= 8) {
2196 if (sniHostNames != null) {
2197 Java8SslUtils.setSniHostNames(sslParameters, sniHostNames);
2198 }
2199 if (!isDestroyed()) {
2200 Java8SslUtils.setUseCipherSuitesOrder(
2201 sslParameters, (SSL.getOptions(ssl) & SSL.SSL_OP_CIPHER_SERVER_PREFERENCE) != 0);
2202 }
2203
2204 Java8SslUtils.setSNIMatchers(sslParameters, matchers);
2205 }
2206 }
2207 return sslParameters;
2208 }
2209
2210 @SuppressJava6Requirement(reason = "Usage guarded by java version check")
2211 @Override
2212 public final synchronized void setSSLParameters(SSLParameters sslParameters) {
2213 int version = PlatformDependent.javaVersion();
2214 if (version >= 7) {
2215 if (sslParameters.getAlgorithmConstraints() != null) {
2216 throw new IllegalArgumentException("AlgorithmConstraints are not supported.");
2217 }
2218
2219 boolean isDestroyed = isDestroyed();
2220 if (version >= 8) {
2221 if (!isDestroyed) {
2222 if (clientMode) {
2223 final List<String> sniHostNames = Java8SslUtils.getSniHostNames(sslParameters);
2224 for (String name: sniHostNames) {
2225 SSL.setTlsExtHostName(ssl, name);
2226 }
2227 this.sniHostNames = sniHostNames;
2228 }
2229 if (Java8SslUtils.getUseCipherSuitesOrder(sslParameters)) {
2230 SSL.setOptions(ssl, SSL.SSL_OP_CIPHER_SERVER_PREFERENCE);
2231 } else {
2232 SSL.clearOptions(ssl, SSL.SSL_OP_CIPHER_SERVER_PREFERENCE);
2233 }
2234 }
2235 matchers = sslParameters.getSNIMatchers();
2236 }
2237
2238 final String endpointIdentificationAlgorithm = sslParameters.getEndpointIdentificationAlgorithm();
2239 if (!isDestroyed) {
2240 configureEndpointVerification(endpointIdentificationAlgorithm);
2241 }
2242 this.endpointIdentificationAlgorithm = endpointIdentificationAlgorithm;
2243 algorithmConstraints = sslParameters.getAlgorithmConstraints();
2244 }
2245 super.setSSLParameters(sslParameters);
2246 }
2247
2248 private void configureEndpointVerification(String endpointIdentificationAlgorithm) {
2249
2250
2251 if (clientMode && isEndPointVerificationEnabled(endpointIdentificationAlgorithm)) {
2252 SSL.setVerify(ssl, SSL.SSL_CVERIFY_REQUIRED, -1);
2253 }
2254 }
2255
2256 private static boolean isEndPointVerificationEnabled(String endPointIdentificationAlgorithm) {
2257 return endPointIdentificationAlgorithm != null && !endPointIdentificationAlgorithm.isEmpty();
2258 }
2259
2260 private boolean isDestroyed() {
2261 return destroyed;
2262 }
2263
2264 final boolean checkSniHostnameMatch(byte[] hostname) {
2265 return Java8SslUtils.checkSniHostnameMatch(matchers, hostname);
2266 }
2267
2268 @Override
2269 public String getNegotiatedApplicationProtocol() {
2270 return applicationProtocol;
2271 }
2272
2273 private static long bufferAddress(ByteBuffer b) {
2274 assert b.isDirect();
2275 if (PlatformDependent.hasUnsafe()) {
2276 return PlatformDependent.directBufferAddress(b);
2277 }
2278 return Buffer.address(b);
2279 }
2280
2281
2282
2283
2284 private void selectApplicationProtocol() throws SSLException {
2285 ApplicationProtocolConfig.SelectedListenerFailureBehavior behavior = apn.selectedListenerFailureBehavior();
2286 List<String> protocols = apn.protocols();
2287 String applicationProtocol;
2288 switch (apn.protocol()) {
2289 case NONE:
2290 break;
2291
2292
2293 case ALPN:
2294 applicationProtocol = SSL.getAlpnSelected(ssl);
2295 if (applicationProtocol != null) {
2296 ReferenceCountedOpenSslEngine.this.applicationProtocol = selectApplicationProtocol(
2297 protocols, behavior, applicationProtocol);
2298 }
2299 break;
2300 case NPN:
2301 applicationProtocol = SSL.getNextProtoNegotiated(ssl);
2302 if (applicationProtocol != null) {
2303 ReferenceCountedOpenSslEngine.this.applicationProtocol = selectApplicationProtocol(
2304 protocols, behavior, applicationProtocol);
2305 }
2306 break;
2307 case NPN_AND_ALPN:
2308 applicationProtocol = SSL.getAlpnSelected(ssl);
2309 if (applicationProtocol == null) {
2310 applicationProtocol = SSL.getNextProtoNegotiated(ssl);
2311 }
2312 if (applicationProtocol != null) {
2313 ReferenceCountedOpenSslEngine.this.applicationProtocol = selectApplicationProtocol(
2314 protocols, behavior, applicationProtocol);
2315 }
2316 break;
2317 default:
2318 throw new Error();
2319 }
2320 }
2321
2322 private String selectApplicationProtocol(List<String> protocols,
2323 ApplicationProtocolConfig.SelectedListenerFailureBehavior behavior,
2324 String applicationProtocol) throws SSLException {
2325 if (behavior == ApplicationProtocolConfig.SelectedListenerFailureBehavior.ACCEPT) {
2326 return applicationProtocol;
2327 } else {
2328 int size = protocols.size();
2329 assert size > 0;
2330 if (protocols.contains(applicationProtocol)) {
2331 return applicationProtocol;
2332 } else {
2333 if (behavior == ApplicationProtocolConfig.SelectedListenerFailureBehavior.CHOOSE_MY_LAST_PROTOCOL) {
2334 return protocols.get(size - 1);
2335 } else {
2336 throw new SSLException("unknown protocol " + applicationProtocol);
2337 }
2338 }
2339 }
2340 }
2341
2342 private static final X509Certificate[] JAVAX_CERTS_NOT_SUPPORTED = new X509Certificate[0];
2343
2344 private final class DefaultOpenSslSession implements OpenSslInternalSession {
2345 private final OpenSslSessionContext sessionContext;
2346
2347
2348
2349 private X509Certificate[] x509PeerCerts;
2350 private Certificate[] peerCerts;
2351
2352 private boolean valid = true;
2353 private String protocol;
2354 private String cipher;
2355 private OpenSslSessionId id = OpenSslSessionId.NULL_ID;
2356 private long creationTime;
2357
2358
2359 private long lastAccessed = -1;
2360
2361 private volatile int applicationBufferSize = MAX_PLAINTEXT_LENGTH;
2362 private volatile Certificate[] localCertificateChain;
2363 private volatile Map<String, Object> keyValueStorage = new ConcurrentHashMap<String, Object>();
2364
2365 DefaultOpenSslSession(OpenSslSessionContext sessionContext) {
2366 this.sessionContext = sessionContext;
2367 }
2368
2369 private SSLSessionBindingEvent newSSLSessionBindingEvent(String name) {
2370 return new SSLSessionBindingEvent(session, name);
2371 }
2372
2373 @Override
2374 public void prepareHandshake() {
2375 keyValueStorage.clear();
2376 }
2377
2378 @Override
2379 public void setSessionDetails(
2380 long creationTime, long lastAccessedTime, OpenSslSessionId sessionId,
2381 Map<String, Object> keyValueStorage) {
2382 synchronized (ReferenceCountedOpenSslEngine.this) {
2383 if (this.id == OpenSslSessionId.NULL_ID) {
2384 this.id = sessionId;
2385 this.creationTime = creationTime;
2386 this.lastAccessed = lastAccessedTime;
2387
2388
2389
2390
2391 this.keyValueStorage = keyValueStorage;
2392 }
2393 }
2394 }
2395
2396 @Override
2397 public Map<String, Object> keyValueStorage() {
2398 return keyValueStorage;
2399 }
2400
2401 @Override
2402 public OpenSslSessionId sessionId() {
2403 synchronized (ReferenceCountedOpenSslEngine.this) {
2404 if (this.id == OpenSslSessionId.NULL_ID && !isDestroyed()) {
2405 byte[] sessionId = SSL.getSessionId(ssl);
2406 if (sessionId != null) {
2407 id = new OpenSslSessionId(sessionId);
2408 }
2409 }
2410
2411 return id;
2412 }
2413 }
2414
2415 @Override
2416 public void setLocalCertificate(Certificate[] localCertificate) {
2417 this.localCertificateChain = localCertificate;
2418 }
2419
2420 @Override
2421 public byte[] getId() {
2422 return sessionId().cloneBytes();
2423 }
2424
2425 @Override
2426 public OpenSslSessionContext getSessionContext() {
2427 return sessionContext;
2428 }
2429
2430 @Override
2431 public long getCreationTime() {
2432 synchronized (ReferenceCountedOpenSslEngine.this) {
2433 return creationTime;
2434 }
2435 }
2436
2437 @Override
2438 public void setLastAccessedTime(long time) {
2439 synchronized (ReferenceCountedOpenSslEngine.this) {
2440 this.lastAccessed = time;
2441 }
2442 }
2443
2444 @Override
2445 public long getLastAccessedTime() {
2446
2447 synchronized (ReferenceCountedOpenSslEngine.this) {
2448 return lastAccessed == -1 ? creationTime : lastAccessed;
2449 }
2450 }
2451
2452 @Override
2453 public void invalidate() {
2454 synchronized (ReferenceCountedOpenSslEngine.this) {
2455 valid = false;
2456 sessionContext.removeFromCache(id);
2457 }
2458 }
2459
2460 @Override
2461 public boolean isValid() {
2462 synchronized (ReferenceCountedOpenSslEngine.this) {
2463 return valid || sessionContext.isInCache(id);
2464 }
2465 }
2466
2467 @Override
2468 public void putValue(String name, Object value) {
2469 checkNotNull(name, "name");
2470 checkNotNull(value, "value");
2471
2472 final Object old = keyValueStorage.put(name, value);
2473 if (value instanceof SSLSessionBindingListener) {
2474
2475 ((SSLSessionBindingListener) value).valueBound(newSSLSessionBindingEvent(name));
2476 }
2477 notifyUnbound(old, name);
2478 }
2479
2480 @Override
2481 public Object getValue(String name) {
2482 checkNotNull(name, "name");
2483 return keyValueStorage.get(name);
2484 }
2485
2486 @Override
2487 public void removeValue(String name) {
2488 checkNotNull(name, "name");
2489 final Object old = keyValueStorage.remove(name);
2490 notifyUnbound(old, name);
2491 }
2492
2493 @Override
2494 public String[] getValueNames() {
2495 return keyValueStorage.keySet().toArray(EMPTY_STRINGS);
2496 }
2497
2498 private void notifyUnbound(Object value, String name) {
2499 if (value instanceof SSLSessionBindingListener) {
2500
2501 ((SSLSessionBindingListener) value).valueUnbound(newSSLSessionBindingEvent(name));
2502 }
2503 }
2504
2505
2506
2507
2508
2509 @Override
2510 public void handshakeFinished(byte[] id, String cipher, String protocol, byte[] peerCertificate,
2511 byte[][] peerCertificateChain, long creationTime, long timeout)
2512 throws SSLException {
2513 synchronized (ReferenceCountedOpenSslEngine.this) {
2514 if (!isDestroyed()) {
2515 if (this.id == OpenSslSessionId.NULL_ID) {
2516
2517
2518 this.id = id == null ? OpenSslSessionId.NULL_ID : new OpenSslSessionId(id);
2519
2520
2521 this.creationTime = lastAccessed = creationTime;
2522 }
2523 this.cipher = toJavaCipherSuite(cipher);
2524 this.protocol = protocol;
2525
2526 if (clientMode) {
2527 if (isEmpty(peerCertificateChain)) {
2528 peerCerts = EmptyArrays.EMPTY_CERTIFICATES;
2529 if (OpenSsl.JAVAX_CERTIFICATE_CREATION_SUPPORTED) {
2530 x509PeerCerts = EmptyArrays.EMPTY_JAVAX_X509_CERTIFICATES;
2531 } else {
2532 x509PeerCerts = JAVAX_CERTS_NOT_SUPPORTED;
2533 }
2534 } else {
2535 peerCerts = new Certificate[peerCertificateChain.length];
2536 if (OpenSsl.JAVAX_CERTIFICATE_CREATION_SUPPORTED) {
2537 x509PeerCerts = new X509Certificate[peerCertificateChain.length];
2538 } else {
2539 x509PeerCerts = JAVAX_CERTS_NOT_SUPPORTED;
2540 }
2541 initCerts(peerCertificateChain, 0);
2542 }
2543 } else {
2544
2545
2546
2547
2548
2549 if (isEmpty(peerCertificate)) {
2550 peerCerts = EmptyArrays.EMPTY_CERTIFICATES;
2551 x509PeerCerts = EmptyArrays.EMPTY_JAVAX_X509_CERTIFICATES;
2552 } else {
2553 if (isEmpty(peerCertificateChain)) {
2554 peerCerts = new Certificate[] {new LazyX509Certificate(peerCertificate)};
2555 if (OpenSsl.JAVAX_CERTIFICATE_CREATION_SUPPORTED) {
2556 x509PeerCerts = new X509Certificate[] {
2557 new LazyJavaxX509Certificate(peerCertificate)
2558 };
2559 } else {
2560 x509PeerCerts = JAVAX_CERTS_NOT_SUPPORTED;
2561 }
2562 } else {
2563 peerCerts = new Certificate[peerCertificateChain.length + 1];
2564 peerCerts[0] = new LazyX509Certificate(peerCertificate);
2565
2566 if (OpenSsl.JAVAX_CERTIFICATE_CREATION_SUPPORTED) {
2567 x509PeerCerts = new X509Certificate[peerCertificateChain.length + 1];
2568 x509PeerCerts[0] = new LazyJavaxX509Certificate(peerCertificate);
2569 } else {
2570 x509PeerCerts = JAVAX_CERTS_NOT_SUPPORTED;
2571 }
2572
2573 initCerts(peerCertificateChain, 1);
2574 }
2575 }
2576 }
2577
2578 calculateMaxWrapOverhead();
2579
2580 handshakeState = HandshakeState.FINISHED;
2581 } else {
2582 throw new SSLException("Already closed");
2583 }
2584 }
2585 }
2586
2587 private void initCerts(byte[][] chain, int startPos) {
2588 for (int i = 0; i < chain.length; i++) {
2589 int certPos = startPos + i;
2590 peerCerts[certPos] = new LazyX509Certificate(chain[i]);
2591 if (x509PeerCerts != JAVAX_CERTS_NOT_SUPPORTED) {
2592 x509PeerCerts[certPos] = new LazyJavaxX509Certificate(chain[i]);
2593 }
2594 }
2595 }
2596
2597 @Override
2598 public Certificate[] getPeerCertificates() throws SSLPeerUnverifiedException {
2599 synchronized (ReferenceCountedOpenSslEngine.this) {
2600 if (isEmpty(peerCerts)) {
2601 throw new SSLPeerUnverifiedException("peer not verified");
2602 }
2603 return peerCerts.clone();
2604 }
2605 }
2606
2607 @Override
2608 public boolean hasPeerCertificates() {
2609 synchronized (ReferenceCountedOpenSslEngine.this) {
2610 return !isEmpty(peerCerts);
2611 }
2612 }
2613
2614 @Override
2615 public Certificate[] getLocalCertificates() {
2616 Certificate[] localCerts = this.localCertificateChain;
2617 if (localCerts == null) {
2618 return null;
2619 }
2620 return localCerts.clone();
2621 }
2622
2623 @Override
2624 public X509Certificate[] getPeerCertificateChain() throws SSLPeerUnverifiedException {
2625 synchronized (ReferenceCountedOpenSslEngine.this) {
2626 if (x509PeerCerts == JAVAX_CERTS_NOT_SUPPORTED) {
2627
2628
2629 throw new UnsupportedOperationException();
2630 }
2631 if (isEmpty(x509PeerCerts)) {
2632 throw new SSLPeerUnverifiedException("peer not verified");
2633 }
2634 return x509PeerCerts.clone();
2635 }
2636 }
2637
2638 @Override
2639 public Principal getPeerPrincipal() throws SSLPeerUnverifiedException {
2640 Certificate[] peer = getPeerCertificates();
2641
2642
2643 return ((java.security.cert.X509Certificate) peer[0]).getSubjectX500Principal();
2644 }
2645
2646 @Override
2647 public Principal getLocalPrincipal() {
2648 Certificate[] local = this.localCertificateChain;
2649 if (local == null || local.length == 0) {
2650 return null;
2651 }
2652 return ((java.security.cert.X509Certificate) local[0]).getSubjectX500Principal();
2653 }
2654
2655 @Override
2656 public String getCipherSuite() {
2657 synchronized (ReferenceCountedOpenSslEngine.this) {
2658 if (cipher == null) {
2659 return SslUtils.INVALID_CIPHER;
2660 }
2661 return cipher;
2662 }
2663 }
2664
2665 @Override
2666 public String getProtocol() {
2667 String protocol = this.protocol;
2668 if (protocol == null) {
2669 synchronized (ReferenceCountedOpenSslEngine.this) {
2670 if (!isDestroyed()) {
2671 protocol = SSL.getVersion(ssl);
2672 } else {
2673 protocol = StringUtil.EMPTY_STRING;
2674 }
2675 }
2676 }
2677 return protocol;
2678 }
2679
2680 @Override
2681 public String getPeerHost() {
2682 return ReferenceCountedOpenSslEngine.this.getPeerHost();
2683 }
2684
2685 @Override
2686 public int getPeerPort() {
2687 return ReferenceCountedOpenSslEngine.this.getPeerPort();
2688 }
2689
2690 @Override
2691 public int getPacketBufferSize() {
2692 return SSL.SSL_MAX_ENCRYPTED_LENGTH;
2693 }
2694
2695 @Override
2696 public int getApplicationBufferSize() {
2697 return applicationBufferSize;
2698 }
2699
2700 @Override
2701 public void tryExpandApplicationBufferSize(int packetLengthDataOnly) {
2702 if (packetLengthDataOnly > MAX_PLAINTEXT_LENGTH && applicationBufferSize != MAX_RECORD_SIZE) {
2703 applicationBufferSize = MAX_RECORD_SIZE;
2704 }
2705 }
2706
2707 @Override
2708 public String toString() {
2709 return "DefaultOpenSslSession{" +
2710 "sessionContext=" + sessionContext +
2711 ", id=" + id +
2712 '}';
2713 }
2714
2715 @Override
2716 public int hashCode() {
2717 return sessionId().hashCode();
2718 }
2719
2720 @Override
2721 public boolean equals(Object o) {
2722 if (o == this) {
2723 return true;
2724 }
2725
2726 if (!(o instanceof OpenSslInternalSession)) {
2727 return false;
2728 }
2729 return sessionId().equals(((OpenSslInternalSession) o).sessionId());
2730 }
2731 }
2732
2733 private interface NativeSslException {
2734 int errorCode();
2735 }
2736
2737 private static final class OpenSslException extends SSLException implements NativeSslException {
2738 private final int errorCode;
2739
2740 OpenSslException(String reason, int errorCode) {
2741 super(reason);
2742 this.errorCode = errorCode;
2743 }
2744
2745 @Override
2746 public int errorCode() {
2747 return errorCode;
2748 }
2749 }
2750
2751 private static final class OpenSslHandshakeException extends SSLHandshakeException implements NativeSslException {
2752 private final int errorCode;
2753
2754 OpenSslHandshakeException(String reason, int errorCode) {
2755 super(reason);
2756 this.errorCode = errorCode;
2757 }
2758
2759 @Override
2760 public int errorCode() {
2761 return errorCode;
2762 }
2763 }
2764 }