1 diff -pu a/nss/lib/ssl/ssl3con.c b/nss/lib/ssl/ssl3con.c
2 --- a/nss/lib/ssl/ssl3con.c 2014-01-17 18:04:43.127747463 -0800
3 +++ b/nss/lib/ssl/ssl3con.c 2014-01-17 18:06:21.919386088 -0800
6 /* TODO(ekr): Implement HelloVerifyRequest on server side. OK for now. */
11 #include "cryptohi.h" /* for DSAU_ stuff */
13 #ifdef NSS_ENABLE_ZLIB
21 #define PK11_SETATTRS(x,id,v,l) (x)->type = (id); \
22 @@ -1842,6 +1846,63 @@ ssl3_BuildRecordPseudoHeader(unsigned ch
26 +typedef SECStatus (*PK11CryptFcn)(
27 + PK11SymKey *symKey, CK_MECHANISM_TYPE mechanism, SECItem *param,
28 + unsigned char *out, unsigned int *outLen, unsigned int maxLen,
29 + const unsigned char *in, unsigned int inLen);
31 +static PK11CryptFcn pk11_encrypt = NULL;
32 +static PK11CryptFcn pk11_decrypt = NULL;
34 +static PRCallOnceType resolvePK11CryptOnce;
37 +ssl3_ResolvePK11CryptFunctions(void)
40 + /* On Linux we use the system NSS libraries. Look up the PK11_Encrypt and
41 + * PK11_Decrypt functions at run time. */
42 + pk11_encrypt = (PK11CryptFcn)dlsym(RTLD_DEFAULT, "PK11_Encrypt");
43 + pk11_decrypt = (PK11CryptFcn)dlsym(RTLD_DEFAULT, "PK11_Decrypt");
46 + /* On other platforms we use our own copy of NSS. PK11_Encrypt and
47 + * PK11_Decrypt are known to be available. */
48 + pk11_encrypt = PK11_Encrypt;
49 + pk11_decrypt = PK11_Decrypt;
55 + * In NSS 3.15, PK11_Encrypt and PK11_Decrypt were added to provide access
56 + * to the AES GCM implementation in the NSS softoken. So the presence of
57 + * these two functions implies the NSS version supports AES GCM.
60 +ssl3_HasGCMSupport(void)
62 + (void)PR_CallOnce(&resolvePK11CryptOnce, ssl3_ResolvePK11CryptFunctions);
63 + return pk11_encrypt != NULL;
66 +/* On this socket, disable the GCM cipher suites */
68 +ssl3_DisableGCMSuites(sslSocket * ss)
72 + for (i = 0; i < PR_ARRAY_SIZE(cipher_suite_defs); i++) {
73 + const ssl3CipherSuiteDef *cipher_def = &cipher_suite_defs[i];
74 + if (cipher_def->bulk_cipher_alg == cipher_aes_128_gcm) {
75 + SECStatus rv = ssl3_CipherPrefSet(ss, cipher_def->cipher_suite,
77 + PORT_Assert(rv == SECSuccess); /* else is coding error */
84 ssl3_AESGCM(ssl3KeyMaterial *keys,
86 @@ -1893,10 +1960,10 @@ ssl3_AESGCM(ssl3KeyMaterial *keys,
87 gcmParams.ulTagBits = tagSize * 8;
90 - rv = PK11_Decrypt(keys->write_key, CKM_AES_GCM, ¶m, out, &uOutLen,
91 + rv = pk11_decrypt(keys->write_key, CKM_AES_GCM, ¶m, out, &uOutLen,
94 - rv = PK11_Encrypt(keys->write_key, CKM_AES_GCM, ¶m, out, &uOutLen,
95 + rv = pk11_encrypt(keys->write_key, CKM_AES_GCM, ¶m, out, &uOutLen,
98 *outlen += (int) uOutLen;
99 @@ -5103,6 +5170,10 @@ ssl3_SendClientHello(sslSocket *ss, PRBo
100 ssl3_DisableNonDTLSSuites(ss);
103 + if (!ssl3_HasGCMSupport()) {
104 + ssl3_DisableGCMSuites(ss);
107 /* how many suites are permitted by policy and user preference? */
108 num_suites = count_cipher_suites(ss, ss->ssl3.policy, PR_TRUE);
110 @@ -8080,6 +8151,10 @@ ssl3_HandleClientHello(sslSocket *ss, SS
111 ssl3_DisableNonDTLSSuites(ss);
114 + if (!ssl3_HasGCMSupport()) {
115 + ssl3_DisableGCMSuites(ss);
119 /* Look for a matching cipher suite. */
120 j = ssl3_config_match_init(ss);