a421a46
diff --git openjdk/make/autoconf/jdk-options.m4 openjdk/make/autoconf/jdk-options.m4
a421a46
--- openjdk/make/autoconf/jdk-options.m4
a421a46
+++ openjdk/make/autoconf/jdk-options.m4
a421a46
@@ -264,9 +264,10 @@
a421a46
 #
a421a46
 AC_DEFUN_ONCE([JDKOPT_DETECT_INTREE_EC],
a421a46
 [
a421a46
+  AC_REQUIRE([LIB_SETUP_MISC_LIBS])
a421a46
   AC_MSG_CHECKING([if elliptic curve crypto implementation is present])
a421a46
 
a421a46
-  if test -d "${TOPDIR}/src/jdk.crypto.ec/share/native/libsunec/impl"; then
a421a46
+  if test "x${system_nss}" = "xyes" -o -d "${TOPDIR}/src/jdk.crypto.ec/share/native/libsunec/impl"; then
a421a46
     ENABLE_INTREE_EC=true
a421a46
     AC_MSG_RESULT([yes])
a421a46
   else
a421a46
diff --git openjdk/make/autoconf/libraries.m4 openjdk/make/autoconf/libraries.m4
a421a46
--- openjdk/make/autoconf/libraries.m4
a421a46
+++ openjdk/make/autoconf/libraries.m4
a421a46
@@ -175,6 +175,48 @@
a421a46
   AC_SUBST(LIBDL)
a421a46
   LIBS="$save_LIBS"
a421a46
 
a421a46
+  ###############################################################################
a421a46
+  #
a421a46
+  # Check for the NSS libraries
a421a46
+  #
a421a46
+
a421a46
+  AC_MSG_CHECKING([whether to build the Sun EC provider against the system NSS libraries])
a421a46
+
a421a46
+  # default is bundled
a421a46
+  DEFAULT_SYSTEM_NSS=no
a421a46
+
a421a46
+  AC_ARG_ENABLE([system-nss], [AS_HELP_STRING([--enable-system-nss],
a421a46
+     [build the SunEC provider using the system NSS libraries @<:@disabled@:>@])],
a421a46
+  [
a421a46
+    case "${enableval}" in
a421a46
+      yes)
a421a46
+        system_nss=yes
a421a46
+        ;;
a421a46
+      *)
a421a46
+        system_nss=no
a421a46
+        ;;
a421a46
+    esac
a421a46
+  ],
a421a46
+  [
a421a46
+    system_nss=${DEFAULT_SYSTEM_NSS}
a421a46
+  ])
a421a46
+  AC_MSG_RESULT([$system_nss])
a421a46
+
a421a46
+  if test "x${system_nss}" = "xyes"; then
a421a46
+      PKG_CHECK_MODULES(NSS_SOFTTKN, nss-softokn >= 3.16.1, [NSS_SOFTOKN_FOUND=yes], [NSS_SOFTOKN_FOUND=no])
a421a46
+      PKG_CHECK_MODULES(NSS, nss >= 3.16.1, [NSS_FOUND=yes], [NSS_FOUND=no])
a421a46
+      if test "x${NSS_SOFTOKN_FOUND}" = "xyes" -a "x${NSS_FOUND}" = "xyes"; then
a421a46
+          NSS_LIBS="$NSS_SOFTOKN_LIBS $NSS_LIBS -lfreebl";
a421a46
+	  USE_EXTERNAL_NSS=true
a421a46
+      else
a421a46
+	  AC_MSG_ERROR([--enable-system-nss specified, but NSS not found.])
a421a46
+      fi
a421a46
+  else
a421a46
+      USE_EXTERNAL_NSS=false
a421a46
+  fi
a421a46
+  AC_SUBST(USE_EXTERNAL_NSS)
a421a46
+
a421a46
+
a421a46
   # Deprecated libraries, keep the flags for backwards compatibility
a421a46
   if test "x$OPENJDK_TARGET_OS" = "xwindows"; then
a421a46
     BASIC_DEPRECATED_ARG_WITH([dxsdk])
a421a46
diff --git openjdk/make/autoconf/spec.gmk.in openjdk/make/autoconf/spec.gmk.in
a421a46
--- openjdk/make/autoconf/spec.gmk.in
a421a46
+++ openjdk/make/autoconf/spec.gmk.in
a421a46
@@ -764,6 +764,10 @@
a421a46
 # Libraries
a421a46
 #
a421a46
 
a421a46
+USE_EXTERNAL_NSS:=@USE_EXTERNAL_NSS@
a421a46
+NSS_LIBS:=@NSS_LIBS@
a421a46
+NSS_CFLAGS:=@NSS_CFLAGS@
a421a46
+
a421a46
 USE_EXTERNAL_LCMS:=@USE_EXTERNAL_LCMS@
a421a46
 LCMS_CFLAGS:=@LCMS_CFLAGS@
a421a46
 LCMS_LIBS:=@LCMS_LIBS@
a421a46
diff --git openjdk/make/lib/Lib-jdk.crypto.ec.gmk openjdk/make/lib/Lib-jdk.crypto.ec.gmk
a421a46
--- openjdk/make/lib/Lib-jdk.crypto.ec.gmk
a421a46
+++ openjdk/make/lib/Lib-jdk.crypto.ec.gmk
a421a46
@@ -29,9 +29,15 @@
a421a46
 
a421a46
 ifeq ($(ENABLE_INTREE_EC), true)
a421a46
 
a421a46
-  LIBSUNEC_SRC := $(TOPDIR)/src/jdk.crypto.ec/share/native/libsunec
a421a46
   BUILD_LIBSUNEC_FLAGS := $(addprefix -I, $(SUNEC_SRC))
a421a46
 
a421a46
+  ifeq ($(USE_EXTERNAL_NSS), true)
a421a46
+    LIBSUNEC_SRC := $(TOPDIR)/src/jdk.crypto.ec/share/native/libsunec/ECC_JNI.cpp
a421a46
+    BUILD_LIBSUNEC_FLAGS += $(NSS_CFLAGS) -DSYSTEM_NSS -DNSS_ENABLE_ECC
a421a46
+  else
a421a46
+    LIBSUNEC_SRC := $(TOPDIR)/src/jdk.crypto.ec/share/native/libsunec
a421a46
+  endif
a421a46
+
a421a46
   #
a421a46
   # On sol-sparc...all libraries are compiled with -xregs=no%appl
a421a46
   # (set in CFLAGS_REQUIRED_sparc)
a421a46
@@ -60,7 +66,7 @@
a421a46
       LDFLAGS := $(LDFLAGS_JDKLIB) $(LDFLAGS_CXX_JDK), \
a421a46
       LDFLAGS_macosx := $(call SET_SHARED_LIBRARY_ORIGIN), \
a421a46
       LIBS := $(LIBCXX), \
a421a46
-      LIBS_linux := -lc, \
a421a46
+      LIBS_linux := -lc $(NSS_LIBS), \
a421a46
       LIBS_solaris := -lc, \
a421a46
       VERSIONINFO_RESOURCE := $(GLOBAL_VERSION_INFO_RESOURCE), \
a421a46
       RC_FLAGS := $(RC_FLAGS) \
a421a46
diff --git openjdk/make/mapfiles/libsunec/mapfile-vers openjdk/make/mapfiles/libsunec/mapfile-vers
a421a46
--- openjdk/make/mapfiles/libsunec/mapfile-vers
a421a46
+++ openjdk/make/mapfiles/libsunec/mapfile-vers
a421a46
@@ -31,6 +31,7 @@
a421a46
                 Java_sun_security_ec_ECDSASignature_signDigest;
a421a46
                 Java_sun_security_ec_ECDSASignature_verifySignedDigest;
a421a46
                 Java_sun_security_ec_ECDHKeyAgreement_deriveKey;
a421a46
+                Java_sun_security_ec_SunEC_initialize;
a421a46
         local:
a421a46
                 *;
a421a46
 };
a421a46
diff --git openjdk/src/java.base/unix/native/include/jni_md.h openjdk/src/java.base/unix/native/include/jni_md.h
a421a46
--- openjdk/src/java.base/unix/native/include/jni_md.h
a421a46
+++ openjdk/src/java.base/unix/native/include/jni_md.h
a421a46
@@ -36,6 +36,11 @@
a421a46
   #define JNIEXPORT
a421a46
   #define JNIIMPORT
a421a46
 #endif
a421a46
+#if (defined(__GNUC__)) || __has_attribute(unused)
a421a46
+  #define UNUSED(x) UNUSED_ ## x __attribute__((__unused__))
a421a46
+#else
a421a46
+  #define UNUSED(x) UNUSED_ ## x
a421a46
+#endif
a421a46
 
a421a46
 #define JNICALL
a421a46
 
a421a46
diff --git openjdk/src/jdk.crypto.ec/share/classes/sun/security/ec/SunEC.java openjdk/src/jdk.crypto.ec/share/classes/sun/security/ec/SunEC.java
a421a46
--- openjdk/src/jdk.crypto.ec/share/classes/sun/security/ec/SunEC.java
a421a46
+++ openjdk/src/jdk.crypto.ec/share/classes/sun/security/ec/SunEC.java
a421a46
@@ -61,6 +61,7 @@
a421a46
             AccessController.doPrivileged(new PrivilegedAction<Void>() {
a421a46
                 public Void run() {
a421a46
                     System.loadLibrary("sunec"); // check for native library
a421a46
+                    initialize();
a421a46
                     return null;
a421a46
                 }
a421a46
             });
a421a46
@@ -272,4 +273,9 @@
a421a46
         putService(new ProviderService(this, "KeyAgreement",
a421a46
             "ECDH", "sun.security.ec.ECDHKeyAgreement", null, ATTRS));
a421a46
     }
a421a46
+
a421a46
+    /**
a421a46
+     * Initialize the native code.
a421a46
+     */
a421a46
+    private static native void initialize();
a421a46
 }
a421a46
diff --git openjdk/src/jdk.crypto.ec/share/native/libsunec/ECC_JNI.cpp openjdk/src/jdk.crypto.ec/share/native/libsunec/ECC_JNI.cpp
a421a46
--- openjdk/src/jdk.crypto.ec/share/native/libsunec/ECC_JNI.cpp
a421a46
+++ openjdk/src/jdk.crypto.ec/share/native/libsunec/ECC_JNI.cpp
a421a46
@@ -25,7 +25,11 @@
a421a46
 
a421a46
 #include <jni.h>
a421a46
 #include "jni_util.h"
a421a46
+#ifdef SYSTEM_NSS
a421a46
+#include "ecc_impl.h"
a421a46
+#else
a421a46
 #include "impl/ecc_impl.h"
a421a46
+#endif
a421a46
 
a421a46
 #define ILLEGAL_STATE_EXCEPTION "java/lang/IllegalStateException"
a421a46
 #define INVALID_ALGORITHM_PARAMETER_EXCEPTION \
a421a46
@@ -33,6 +37,13 @@
a421a46
 #define INVALID_PARAMETER_EXCEPTION \
a421a46
         "java/security/InvalidParameterException"
a421a46
 #define KEY_EXCEPTION   "java/security/KeyException"
a421a46
+#define INTERNAL_ERROR "java/lang/InternalError"
a421a46
+
a421a46
+#ifdef SYSTEM_NSS
a421a46
+#define SYSTEM_UNUSED(x) UNUSED(x)
a421a46
+#else
a421a46
+#define SYSTEM_UNUSED(x) x
a421a46
+#endif
a421a46
 
a421a46
 extern "C" {
a421a46
 
a421a46
@@ -55,8 +66,13 @@
a421a46
 /*
a421a46
  * Deep free of the ECParams struct
a421a46
  */
a421a46
-void FreeECParams(ECParams *ecparams, jboolean freeStruct)
a421a46
+void FreeECParams(ECParams *ecparams, jboolean SYSTEM_UNUSED(freeStruct))
a421a46
 {
a421a46
+#ifdef SYSTEM_NSS
a421a46
+    // Needs to be freed using the matching method to the one
a421a46
+    // that allocated it. PR_TRUE means the memory is zeroed.
a421a46
+    PORT_FreeArena(ecparams->arena, PR_TRUE);
a421a46
+#else
a421a46
     // Use B_FALSE to free the SECItem->data element, but not the SECItem itself
a421a46
     // Use B_TRUE to free both
a421a46
 
a421a46
@@ -70,6 +86,7 @@
a421a46
     SECITEM_FreeItem(&ecparams->curveOID, B_FALSE);
a421a46
     if (freeStruct)
a421a46
         free(ecparams);
a421a46
+#endif
a421a46
 }
a421a46
 
a421a46
 jbyteArray getEncodedBytes(JNIEnv *env, SECItem *hSECItem)
a421a46
@@ -95,7 +112,7 @@
a421a46
  */
a421a46
 JNIEXPORT jobjectArray
a421a46
 JNICALL Java_sun_security_ec_ECKeyPairGenerator_generateECKeyPair
a421a46
-  (JNIEnv *env, jclass clazz, jint keySize, jbyteArray encodedParams, jbyteArray seed)
a421a46
+  (JNIEnv *env, jclass UNUSED(clazz), jint UNUSED(keySize), jbyteArray encodedParams, jbyteArray seed)
a421a46
 {
a421a46
     ECPrivateKey *privKey = NULL; // contains both public and private values
a421a46
     ECParams *ecparams = NULL;
a421a46
@@ -127,8 +144,17 @@
a421a46
     env->GetByteArrayRegion(seed, 0, jSeedLength, pSeedBuffer);
a421a46
 
a421a46
     // Generate the new keypair (using the supplied seed)
a421a46
+#ifdef SYSTEM_NSS
a421a46
+    if (RNG_RandomUpdate((unsigned char *) pSeedBuffer, jSeedLength)
a421a46
+        != SECSuccess) {
a421a46
+        ThrowException(env, KEY_EXCEPTION);
a421a46
+        goto cleanup;
a421a46
+    }
a421a46
+    if (EC_NewKey(ecparams, &privKey) != SECSuccess) {    
a421a46
+#else    
a421a46
     if (EC_NewKey(ecparams, &privKey, (unsigned char *) pSeedBuffer,
a421a46
         jSeedLength, 0) != SECSuccess) {
a421a46
+#endif
a421a46
         ThrowException(env, KEY_EXCEPTION);
a421a46
         goto cleanup;
a421a46
     }
a421a46
@@ -175,10 +201,15 @@
a421a46
         }
a421a46
         if (privKey) {
a421a46
             FreeECParams(&privKey->ecParams, false);
a421a46
+#ifndef SYSTEM_NSS
a421a46
+	    // The entire ECPrivateKey is allocated in the arena
a421a46
+	    // when using system NSS, so only the in-tree version
a421a46
+	    // needs to clear these manually.
a421a46
             SECITEM_FreeItem(&privKey->version, B_FALSE);
a421a46
             SECITEM_FreeItem(&privKey->privateValue, B_FALSE);
a421a46
             SECITEM_FreeItem(&privKey->publicValue, B_FALSE);
a421a46
             free(privKey);
a421a46
+#endif
a421a46
         }
a421a46
 
a421a46
         if (pSeedBuffer) {
a421a46
@@ -196,7 +227,7 @@
a421a46
  */
a421a46
 JNIEXPORT jbyteArray
a421a46
 JNICALL Java_sun_security_ec_ECDSASignature_signDigest
a421a46
-  (JNIEnv *env, jclass clazz, jbyteArray digest, jbyteArray privateKey, jbyteArray encodedParams, jbyteArray seed, jint timing)
a421a46
+  (JNIEnv *env, jclass UNUSED(clazz), jbyteArray digest, jbyteArray privateKey, jbyteArray encodedParams, jbyteArray seed, jint timing)
a421a46
 {
a421a46
     jbyte* pDigestBuffer = NULL;
a421a46
     jint jDigestLength = env->GetArrayLength(digest);
a421a46
@@ -255,8 +286,18 @@
a421a46
     env->GetByteArrayRegion(seed, 0, jSeedLength, pSeedBuffer);
a421a46
 
a421a46
     // Sign the digest (using the supplied seed)
a421a46
+#ifdef SYSTEM_NSS
a421a46
+    if (RNG_RandomUpdate((unsigned char *) pSeedBuffer, jSeedLength)
a421a46
+        != SECSuccess) {
a421a46
+        ThrowException(env, KEY_EXCEPTION);
a421a46
+        goto cleanup;
a421a46
+    }
a421a46
+    if (ECDSA_SignDigest(&privKey, &signature_item, &digest_item)
a421a46
+        != SECSuccess) {    
a421a46
+#else    
a421a46
     if (ECDSA_SignDigest(&privKey, &signature_item, &digest_item,
a421a46
         (unsigned char *) pSeedBuffer, jSeedLength, 0, timing) != SECSuccess) {
a421a46
+#endif
a421a46
         ThrowException(env, KEY_EXCEPTION);
a421a46
         goto cleanup;
a421a46
     }
a421a46
@@ -305,7 +346,7 @@
a421a46
  */
a421a46
 JNIEXPORT jboolean
a421a46
 JNICALL Java_sun_security_ec_ECDSASignature_verifySignedDigest
a421a46
-  (JNIEnv *env, jclass clazz, jbyteArray signedDigest, jbyteArray digest, jbyteArray publicKey, jbyteArray encodedParams)
a421a46
+  (JNIEnv *env, jclass UNUSED(clazz), jbyteArray signedDigest, jbyteArray digest, jbyteArray publicKey, jbyteArray encodedParams)
a421a46
 {
a421a46
     jboolean isValid = false;
a421a46
 
a421a46
@@ -362,9 +403,10 @@
a421a46
 
a421a46
 cleanup:
a421a46
     {
a421a46
-        if (params_item.data)
a421a46
+        if (params_item.data) {
a421a46
             env->ReleaseByteArrayElements(encodedParams,
a421a46
                 (jbyte *) params_item.data, JNI_ABORT);
a421a46
+	}
a421a46
 
a421a46
         if (pubKey.publicValue.data)
a421a46
             env->ReleaseByteArrayElements(publicKey,
a421a46
@@ -390,7 +432,7 @@
a421a46
  */
a421a46
 JNIEXPORT jbyteArray
a421a46
 JNICALL Java_sun_security_ec_ECDHKeyAgreement_deriveKey
a421a46
-  (JNIEnv *env, jclass clazz, jbyteArray privateKey, jbyteArray publicKey, jbyteArray encodedParams)
a421a46
+  (JNIEnv *env, jclass UNUSED(clazz), jbyteArray privateKey, jbyteArray publicKey, jbyteArray encodedParams)
a421a46
 {
a421a46
     jbyteArray jSecret = NULL;
a421a46
     ECParams *ecparams = NULL;
a421a46
@@ -466,9 +508,10 @@
a421a46
             env->ReleaseByteArrayElements(publicKey,
a421a46
                 (jbyte *) publicValue_item.data, JNI_ABORT);
a421a46
 
a421a46
-        if (params_item.data)
a421a46
+        if (params_item.data) {
a421a46
             env->ReleaseByteArrayElements(encodedParams,
a421a46
                 (jbyte *) params_item.data, JNI_ABORT);
a421a46
+	}
a421a46
 
a421a46
         if (ecparams)
a421a46
             FreeECParams(ecparams, true);
a421a46
@@ -477,4 +520,28 @@
a421a46
     return jSecret;
a421a46
 }
a421a46
 
a421a46
+JNIEXPORT void
a421a46
+JNICALL Java_sun_security_ec_SunEC_initialize
a421a46
+  (JNIEnv *env, jclass UNUSED(clazz))
a421a46
+{
a421a46
+#ifdef SYSTEM_NSS
a421a46
+    if (SECOID_Init() != SECSuccess) {
a421a46
+        ThrowException(env, INTERNAL_ERROR);
a421a46
+    }
a421a46
+    if (RNG_RNGInit() != SECSuccess) {
a421a46
+        ThrowException(env, INTERNAL_ERROR);
a421a46
+    }
a421a46
+#endif
a421a46
+}
a421a46
+
a421a46
+JNIEXPORT void
a421a46
+JNICALL JNI_OnUnload
a421a46
+  (JavaVM *vm, void *reserved)
a421a46
+{
a421a46
+#ifdef SYSTEM_NSS
a421a46
+    RNG_RNGShutdown();
a421a46
+    SECOID_Shutdown();
a421a46
+#endif
a421a46
+}
a421a46
+
a421a46
 } /* extern "C" */
a421a46
diff --git openjdk/src/jdk.crypto.ec/share/native/libsunec/ecc_impl.h openjdk/src/jdk.crypto.ec/share/native/libsunec/ecc_impl.h
a421a46
new file mode 100644
a421a46
--- /dev/null
a421a46
+++ openjdk/src/jdk.crypto.ec/share/native/libsunec/ecc_impl.h
a421a46
@@ -0,0 +1,298 @@
a421a46
+/*
a421a46
+ * Copyright (c) 2007, 2017, Oracle and/or its affiliates. All rights reserved.
a421a46
+ * Use is subject to license terms.
a421a46
+ *
a421a46
+ * This library is free software; you can redistribute it and/or
a421a46
+ * modify it under the terms of the GNU Lesser General Public
a421a46
+ * License as published by the Free Software Foundation; either
a421a46
+ * version 2.1 of the License, or (at your option) any later version.
a421a46
+ *
a421a46
+ * This library is distributed in the hope that it will be useful,
a421a46
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
a421a46
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
a421a46
+ * Lesser General Public License for more details.
a421a46
+ *
a421a46
+ * You should have received a copy of the GNU Lesser General Public License
a421a46
+ * along with this library; if not, write to the Free Software Foundation,
a421a46
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
a421a46
+ *
a421a46
+ * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
a421a46
+ * or visit www.oracle.com if you need additional information or have any
a421a46
+ * questions.
a421a46
+ */
a421a46
+
a421a46
+/* *********************************************************************
a421a46
+ *
a421a46
+ * The Original Code is the Netscape security libraries.
a421a46
+ *
a421a46
+ * The Initial Developer of the Original Code is
a421a46
+ * Netscape Communications Corporation.
a421a46
+ * Portions created by the Initial Developer are Copyright (C) 1994-2000
a421a46
+ * the Initial Developer. All Rights Reserved.
a421a46
+ *
a421a46
+ * Contributor(s):
a421a46
+ *   Dr Vipul Gupta <vipul.gupta@sun.com> and
a421a46
+ *   Douglas Stebila <douglas@stebila.ca>, Sun Microsystems Laboratories
a421a46
+ *
a421a46
+ * Last Modified Date from the Original Code: May 2017
a421a46
+ *********************************************************************** */
a421a46
+
a421a46
+#ifndef _ECC_IMPL_H
a421a46
+#define _ECC_IMPL_H
a421a46
+
a421a46
+#ifdef __cplusplus
a421a46
+extern "C" {
a421a46
+#endif
a421a46
+
a421a46
+#include <sys/types.h>
a421a46
+
a421a46
+#ifdef SYSTEM_NSS
a421a46
+#include <secitem.h>
a421a46
+#include <secerr.h>
a421a46
+#include <keythi.h>
a421a46
+#ifdef LEGACY_NSS
a421a46
+#include <softoken.h>
a421a46
+#else
a421a46
+#include <blapi.h>
a421a46
+#endif
a421a46
+#else
a421a46
+#include "ecl-exp.h"
a421a46
+#endif
a421a46
+
a421a46
+/*
a421a46
+ * Multi-platform definitions
a421a46
+ */
a421a46
+#ifdef __linux__
a421a46
+#define B_FALSE FALSE
a421a46
+#define B_TRUE TRUE
a421a46
+typedef unsigned char uint8_t;
a421a46
+typedef unsigned long ulong_t;
a421a46
+typedef enum { B_FALSE, B_TRUE } boolean_t;
a421a46
+#endif /* __linux__ */
a421a46
+
a421a46
+#ifdef _ALLBSD_SOURCE
a421a46
+#include <stdint.h>
a421a46
+#define B_FALSE FALSE
a421a46
+#define B_TRUE TRUE
a421a46
+typedef unsigned long ulong_t;
a421a46
+typedef enum boolean { B_FALSE, B_TRUE } boolean_t;
a421a46
+#endif /* _ALLBSD_SOURCE */
a421a46
+
a421a46
+#ifdef AIX
a421a46
+#define B_FALSE FALSE
a421a46
+#define B_TRUE TRUE
a421a46
+typedef unsigned char uint8_t;
a421a46
+typedef unsigned long ulong_t;
a421a46
+#endif /* AIX */
a421a46
+
a421a46
+#ifdef _WIN32
a421a46
+typedef unsigned char uint8_t;
a421a46
+typedef unsigned long ulong_t;
a421a46
+typedef enum boolean { B_FALSE, B_TRUE } boolean_t;
a421a46
+#define strdup _strdup          /* Replace POSIX name with ISO C++ name */
a421a46
+#endif /* _WIN32 */
a421a46
+
a421a46
+#ifndef _KERNEL
a421a46
+#include <stdlib.h>
a421a46
+#endif  /* _KERNEL */
a421a46
+
a421a46
+#define EC_MAX_DIGEST_LEN 1024  /* max digest that can be signed */
a421a46
+#define EC_MAX_POINT_LEN 145    /* max len of DER encoded Q */
a421a46
+#define EC_MAX_VALUE_LEN 72     /* max len of ANSI X9.62 private value d */
a421a46
+#define EC_MAX_SIG_LEN 144      /* max signature len for supported curves */
a421a46
+#define EC_MIN_KEY_LEN  112     /* min key length in bits */
a421a46
+#define EC_MAX_KEY_LEN  571     /* max key length in bits */
a421a46
+#define EC_MAX_OID_LEN 10       /* max length of OID buffer */
a421a46
+
a421a46
+/*
a421a46
+ * Various structures and definitions from NSS are here.
a421a46
+ */
a421a46
+
a421a46
+#ifndef SYSTEM_NSS
a421a46
+#ifdef _KERNEL
a421a46
+#define PORT_ArenaAlloc(a, n, f)        kmem_alloc((n), (f))
a421a46
+#define PORT_ArenaZAlloc(a, n, f)       kmem_zalloc((n), (f))
a421a46
+#define PORT_ArenaGrow(a, b, c, d)      NULL
a421a46
+#define PORT_ZAlloc(n, f)               kmem_zalloc((n), (f))
a421a46
+#define PORT_Alloc(n, f)                kmem_alloc((n), (f))
a421a46
+#else
a421a46
+#define PORT_ArenaAlloc(a, n, f)        malloc((n))
a421a46
+#define PORT_ArenaZAlloc(a, n, f)       calloc(1, (n))
a421a46
+#define PORT_ArenaGrow(a, b, c, d)      NULL
a421a46
+#define PORT_ZAlloc(n, f)               calloc(1, (n))
a421a46
+#define PORT_Alloc(n, f)                malloc((n))
a421a46
+#endif
a421a46
+
a421a46
+#define PORT_NewArena(b)                (char *)12345
a421a46
+#define PORT_ArenaMark(a)               NULL
a421a46
+#define PORT_ArenaUnmark(a, b)
a421a46
+#define PORT_ArenaRelease(a, m)
a421a46
+#define PORT_FreeArena(a, b)
a421a46
+#define PORT_Strlen(s)                  strlen((s))
a421a46
+#define PORT_SetError(e)
a421a46
+
a421a46
+#define PRBool                          boolean_t
a421a46
+#define PR_TRUE                         B_TRUE
a421a46
+#define PR_FALSE                        B_FALSE
a421a46
+
a421a46
+#ifdef _KERNEL
a421a46
+#define PORT_Assert                     ASSERT
a421a46
+#define PORT_Memcpy(t, f, l)            bcopy((f), (t), (l))
a421a46
+#else
a421a46
+#define PORT_Assert                     assert
a421a46
+#define PORT_Memcpy(t, f, l)            memcpy((t), (f), (l))
a421a46
+#endif
a421a46
+
a421a46
+#endif
a421a46
+
a421a46
+#define CHECK_OK(func) if (func == NULL) goto cleanup
a421a46
+#define CHECK_SEC_OK(func) if (SECSuccess != (rv = func)) goto cleanup
a421a46
+
a421a46
+#ifndef SYSTEM_NSS
a421a46
+typedef enum {
a421a46
+        siBuffer = 0,
a421a46
+        siClearDataBuffer = 1,
a421a46
+        siCipherDataBuffer = 2,
a421a46
+        siDERCertBuffer = 3,
a421a46
+        siEncodedCertBuffer = 4,
a421a46
+        siDERNameBuffer = 5,
a421a46
+        siEncodedNameBuffer = 6,
a421a46
+        siAsciiNameString = 7,
a421a46
+        siAsciiString = 8,
a421a46
+        siDEROID = 9,
a421a46
+        siUnsignedInteger = 10,
a421a46
+        siUTCTime = 11,
a421a46
+        siGeneralizedTime = 12
a421a46
+} SECItemType;
a421a46
+
a421a46
+typedef struct SECItemStr SECItem;
a421a46
+
a421a46
+struct SECItemStr {
a421a46
+        SECItemType type;
a421a46
+        unsigned char *data;
a421a46
+        unsigned int len;
a421a46
+};
a421a46
+
a421a46
+typedef SECItem SECKEYECParams;
a421a46
+
a421a46
+typedef enum { ec_params_explicit,
a421a46
+               ec_params_named
a421a46
+} ECParamsType;
a421a46
+
a421a46
+typedef enum { ec_field_GFp = 1,
a421a46
+               ec_field_GF2m
a421a46
+} ECFieldType;
a421a46
+
a421a46
+struct ECFieldIDStr {
a421a46
+    int         size;   /* field size in bits */
a421a46
+    ECFieldType type;
a421a46
+    union {
a421a46
+        SECItem  prime; /* prime p for (GFp) */
a421a46
+        SECItem  poly;  /* irreducible binary polynomial for (GF2m) */
a421a46
+    } u;
a421a46
+    int         k1;     /* first coefficient of pentanomial or
a421a46
+                         * the only coefficient of trinomial
a421a46
+                         */
a421a46
+    int         k2;     /* two remaining coefficients of pentanomial */
a421a46
+    int         k3;
a421a46
+};
a421a46
+typedef struct ECFieldIDStr ECFieldID;
a421a46
+
a421a46
+struct ECCurveStr {
a421a46
+        SECItem a;      /* contains octet stream encoding of
a421a46
+                         * field element (X9.62 section 4.3.3)
a421a46
+                         */
a421a46
+        SECItem b;
a421a46
+        SECItem seed;
a421a46
+};
a421a46
+typedef struct ECCurveStr ECCurve;
a421a46
+
a421a46
+typedef void PRArenaPool;
a421a46
+
a421a46
+struct ECParamsStr {
a421a46
+    PRArenaPool * arena;
a421a46
+    ECParamsType  type;
a421a46
+    ECFieldID     fieldID;
a421a46
+    ECCurve       curve;
a421a46
+    SECItem       base;
a421a46
+    SECItem       order;
a421a46
+    int           cofactor;
a421a46
+    SECItem       DEREncoding;
a421a46
+    ECCurveName   name;
a421a46
+    SECItem       curveOID;
a421a46
+};
a421a46
+typedef struct ECParamsStr ECParams;
a421a46
+
a421a46
+struct ECPublicKeyStr {
a421a46
+    ECParams ecParams;
a421a46
+    SECItem publicValue;   /* elliptic curve point encoded as
a421a46
+                            * octet stream.
a421a46
+                            */
a421a46
+};
a421a46
+typedef struct ECPublicKeyStr ECPublicKey;
a421a46
+
a421a46
+struct ECPrivateKeyStr {
a421a46
+    ECParams ecParams;
a421a46
+    SECItem publicValue;   /* encoded ec point */
a421a46
+    SECItem privateValue;  /* private big integer */
a421a46
+    SECItem version;       /* As per SEC 1, Appendix C, Section C.4 */
a421a46
+};
a421a46
+typedef struct ECPrivateKeyStr ECPrivateKey;
a421a46
+
a421a46
+typedef enum _SECStatus {
a421a46
+        SECBufferTooSmall = -3,
a421a46
+        SECWouldBlock = -2,
a421a46
+        SECFailure = -1,
a421a46
+        SECSuccess = 0
a421a46
+} SECStatus;
a421a46
+#endif
a421a46
+
a421a46
+#ifdef _KERNEL
a421a46
+#define RNG_GenerateGlobalRandomBytes(p,l) ecc_knzero_random_generator((p), (l))
a421a46
+#else
a421a46
+/*
a421a46
+ This function is no longer required because the random bytes are now
a421a46
+ supplied by the caller. Force a failure.
a421a46
+*/
a421a46
+#ifndef SYSTEM_NSS
a421a46
+#define RNG_GenerateGlobalRandomBytes(p,l) SECFailure
a421a46
+#endif
a421a46
+#endif
a421a46
+#define CHECK_MPI_OK(func) if (MP_OKAY > (err = func)) goto cleanup
a421a46
+#define MP_TO_SEC_ERROR(err)
a421a46
+
a421a46
+#define SECITEM_TO_MPINT(it, mp)                                        \
a421a46
+        CHECK_MPI_OK(mp_read_unsigned_octets((mp), (it).data, (it).len))
a421a46
+
a421a46
+extern int ecc_knzero_random_generator(uint8_t *, size_t);
a421a46
+extern ulong_t soft_nzero_random_generator(uint8_t *, ulong_t);
a421a46
+
a421a46
+#ifdef SYSTEM_NSS
a421a46
+#define EC_DecodeParams(a,b,c) EC_DecodeParams(a,b)
a421a46
+#define ECDSA_VerifyDigest(a,b,c,d) ECDSA_VerifyDigest(a,b,c)
a421a46
+#define ECDH_Derive(a,b,c,d,e,f) ECDH_Derive(a,b,c,d,e)
a421a46
+#else
a421a46
+extern SECStatus EC_DecodeParams(const SECItem *, ECParams **, int);
a421a46
+
a421a46
+extern SECItem * SECITEM_AllocItem(PRArenaPool *, SECItem *, unsigned int, int);
a421a46
+extern SECStatus SECITEM_CopyItem(PRArenaPool *, SECItem *, const SECItem *,
a421a46
+    int);
a421a46
+extern void SECITEM_FreeItem(SECItem *, boolean_t);
a421a46
+
a421a46
+/* This function has been modified to accept an array of random bytes */
a421a46
+extern SECStatus EC_NewKey(ECParams *ecParams, ECPrivateKey **privKey,
a421a46
+    const unsigned char* random, int randomlen, int);
a421a46
+/* This function has been modified to accept an array of random bytes */
a421a46
+extern SECStatus ECDSA_SignDigest(ECPrivateKey *, SECItem *, const SECItem *,
a421a46
+    const unsigned char* random, int randomlen, int, int timing);
a421a46
+extern SECStatus ECDSA_VerifyDigest(ECPublicKey *, const SECItem *,
a421a46
+    const SECItem *, int);
a421a46
+extern SECStatus ECDH_Derive(SECItem *, ECParams *, SECItem *, boolean_t,
a421a46
+    SECItem *, int);
a421a46
+#endif
a421a46
+
a421a46
+#ifdef  __cplusplus
a421a46
+}
a421a46
+#endif
a421a46
+
a421a46
+#endif /* _ECC_IMPL_H */