pkcs7verifier.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. // Copyright (C) 2004-2021 Artifex Software, Inc.
  2. //
  3. // This file is part of MuPDF.
  4. //
  5. // MuPDF is free software: you can redistribute it and/or modify it under the
  6. // terms of the GNU Affero General Public License as published by the Free
  7. // Software Foundation, either version 3 of the License, or (at your option)
  8. // any later version.
  9. //
  10. // MuPDF is distributed in the hope that it will be useful, but WITHOUT ANY
  11. // WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
  12. // FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
  13. // details.
  14. //
  15. // You should have received a copy of the GNU Affero General Public License
  16. // along with MuPDF. If not, see <https://www.gnu.org/licenses/agpl-3.0.en.html>
  17. //
  18. // Alternative licensing terms are available from the licensor.
  19. // For commercial licensing, see <https://www.artifex.com/> or contact
  20. // Artifex Software, Inc., 39 Mesa Street, Suite 108A, San Francisco,
  21. // CA 94129, USA, for further information.
  22. /* PKCS7Verifier interface */
  23. static void java_pkcs7_drop_verifier(fz_context *ctx, pdf_pkcs7_verifier *verifier_)
  24. {
  25. java_pkcs7_verifier *verifier = (java_pkcs7_verifier *) verifier_;
  26. jboolean detach = JNI_FALSE;
  27. JNIEnv *env = NULL;
  28. env = jni_attach_thread(&detach);
  29. if (!env)
  30. fz_throw(ctx, FZ_ERROR_GENERIC, "cannot attach to JVM in java_pkcs7_check_digest");
  31. (*env)->DeleteGlobalRef(env, verifier->jverifier);
  32. fz_free(ctx, verifier);
  33. jni_detach_thread(detach);
  34. }
  35. static pdf_signature_error java_pkcs7_check_certificate(fz_context *ctx, pdf_pkcs7_verifier *verifier, unsigned char *signature, size_t len)
  36. {
  37. java_pkcs7_verifier *pkcs7_verifier = (java_pkcs7_verifier *) verifier;
  38. jobject jverifier = pkcs7_verifier->jverifier;
  39. jint result = PDF_SIGNATURE_ERROR_UNKNOWN;
  40. jboolean detach = JNI_FALSE;
  41. JNIEnv *env = NULL;
  42. jobject jsignature = NULL;
  43. env = jni_attach_thread(&detach);
  44. if (env == NULL)
  45. fz_throw(ctx, FZ_ERROR_GENERIC, "cannot attach to JVM in java_pkcs7_check_digest");
  46. fz_try(ctx)
  47. jsignature = to_byteArray(ctx, env, signature, (int)len);
  48. fz_catch(ctx)
  49. fz_rethrow_and_detach_thread(ctx, detach);
  50. result = (*env)->CallIntMethod(env, jverifier, mid_PKCS7Verifier_checkCertificate, jsignature);
  51. if ((*env)->ExceptionCheck(env))
  52. fz_throw_java_and_detach_thread(ctx, env, detach);
  53. jni_detach_thread(detach);
  54. return result;
  55. }
  56. static pdf_signature_error java_pkcs7_check_digest(fz_context *ctx, pdf_pkcs7_verifier *verifier, fz_stream *stm, unsigned char *signature, size_t len)
  57. {
  58. java_pkcs7_verifier *pkcs7_verifier = (java_pkcs7_verifier *) verifier;
  59. jobject jverifier = pkcs7_verifier->jverifier;
  60. jint result = PDF_SIGNATURE_ERROR_UNKNOWN;
  61. jboolean detach = JNI_FALSE;
  62. jobject jsignature = NULL;
  63. jobject jstm = NULL;
  64. JNIEnv *env = NULL;
  65. env = jni_attach_thread(&detach);
  66. if (env == NULL)
  67. fz_throw(ctx, FZ_ERROR_GENERIC, "cannot attach to JVM in java_pkcs7_check_digest");
  68. fz_try(ctx)
  69. {
  70. jsignature = to_byteArray(ctx, env, signature, (int)len);
  71. jstm = to_FitzInputStream(ctx, env, stm);
  72. }
  73. fz_catch(ctx)
  74. fz_rethrow_and_detach_thread(ctx, detach);
  75. result = (*env)->CallIntMethod(env, jverifier, mid_PKCS7Verifier_checkDigest, jstm, jsignature);
  76. if ((*env)->ExceptionCheck(env))
  77. fz_throw_java_and_detach_thread(ctx, env, detach);
  78. jni_detach_thread(detach);
  79. return result;
  80. }
  81. static pdf_pkcs7_verifier *java_pkcs7_new_verifier(fz_context *ctx, jobject jverifier)
  82. {
  83. java_pkcs7_verifier *verifier = fz_malloc_struct(ctx, java_pkcs7_verifier);
  84. verifier->base.drop = java_pkcs7_drop_verifier;
  85. verifier->base.check_digest = java_pkcs7_check_digest;
  86. verifier->base.check_certificate = java_pkcs7_check_certificate;
  87. verifier->jverifier = jverifier;
  88. return &verifier->base;
  89. }
  90. JNIEXPORT jlong JNICALL
  91. FUN(PKCS7Verifier_newNative)(JNIEnv *env, jobject self, jobject jverifier)
  92. {
  93. fz_context *ctx = get_context(env);
  94. pdf_pkcs7_verifier *verifier = NULL;
  95. if (!ctx) return 0;
  96. if (!jverifier) jni_throw_arg(env, "verifier must not be null");
  97. jverifier = (*env)->NewGlobalRef(env, jverifier);
  98. if (!jverifier) jni_throw_arg(env, "unable to get reference to verifier");
  99. fz_try(ctx)
  100. verifier = java_pkcs7_new_verifier(ctx, jverifier);
  101. fz_catch(ctx)
  102. {
  103. (*env)->DeleteGlobalRef(env, jverifier);
  104. jni_rethrow(env, ctx);
  105. }
  106. return jlong_cast(verifier);
  107. }
  108. JNIEXPORT void JNICALL
  109. FUN(PKCS7Verifier_finalize)(JNIEnv *env, jobject self)
  110. {
  111. fz_context *ctx = get_context(env);
  112. java_pkcs7_verifier *verifier = from_PKCS7Verifier_safe(env, self);
  113. if (!ctx || !verifier) return;
  114. (*env)->SetLongField(env, self, fid_PKCS7Verifier_pointer, 0);
  115. pdf_drop_verifier(ctx, &verifier->base);
  116. }