@@ -3516,8 +3516,17 @@ void CipherBase::InitIv(const char* cipher_type,
3516
3516
const int expected_iv_len = EVP_CIPHER_iv_length (cipher);
3517
3517
const int mode = EVP_CIPHER_mode (cipher);
3518
3518
const bool is_gcm_mode = (EVP_CIPH_GCM_MODE == mode);
3519
+ const bool has_iv = iv_len >= 0 ;
3519
3520
3520
- if (is_gcm_mode == false && iv_len != expected_iv_len) {
3521
+ // Throw if no IV was passed and the cipher requires an IV
3522
+ if (!has_iv && expected_iv_len != 0 ) {
3523
+ char msg[128 ];
3524
+ snprintf (msg, sizeof (msg), " Missing IV for cipher %s" , cipher_type);
3525
+ return env ()->ThrowError (msg);
3526
+ }
3527
+
3528
+ // Throw if an IV was passed which does not match the cipher's fixed IV length
3529
+ if (is_gcm_mode == false && has_iv && iv_len != expected_iv_len) {
3521
3530
return env ()->ThrowError (" Invalid IV length" );
3522
3531
}
3523
3532
@@ -3529,11 +3538,13 @@ void CipherBase::InitIv(const char* cipher_type,
3529
3538
const bool encrypt = (kind_ == kCipher );
3530
3539
EVP_CipherInit_ex (ctx_, cipher, nullptr , nullptr , nullptr , encrypt );
3531
3540
3532
- if (is_gcm_mode &&
3533
- !EVP_CIPHER_CTX_ctrl (ctx_, EVP_CTRL_GCM_SET_IVLEN, iv_len, nullptr )) {
3534
- EVP_CIPHER_CTX_free (ctx_);
3535
- ctx_ = nullptr ;
3536
- return env ()->ThrowError (" Invalid IV length" );
3541
+ if (is_gcm_mode) {
3542
+ CHECK (has_iv);
3543
+ if (!EVP_CIPHER_CTX_ctrl (ctx_, EVP_CTRL_GCM_SET_IVLEN, iv_len, nullptr )) {
3544
+ EVP_CIPHER_CTX_free (ctx_);
3545
+ ctx_ = nullptr ;
3546
+ return env ()->ThrowError (" Invalid IV length" );
3547
+ }
3537
3548
}
3538
3549
3539
3550
if (!EVP_CIPHER_CTX_set_key_length (ctx_, key_len)) {
@@ -3562,13 +3573,23 @@ void CipherBase::InitIv(const FunctionCallbackInfo<Value>& args) {
3562
3573
3563
3574
THROW_AND_RETURN_IF_NOT_STRING (args[0 ], " Cipher type" );
3564
3575
THROW_AND_RETURN_IF_NOT_BUFFER (args[1 ], " Key" );
3565
- THROW_AND_RETURN_IF_NOT_BUFFER (args[2 ], " IV" );
3576
+
3577
+ if (!args[2 ]->IsNull () && !Buffer::HasInstance (args[2 ])) {
3578
+ return env->ThrowTypeError (" IV must be a buffer" );
3579
+ }
3566
3580
3567
3581
const node::Utf8Value cipher_type (env->isolate (), args[0 ]);
3568
3582
ssize_t key_len = Buffer::Length (args[1 ]);
3569
3583
const char * key_buf = Buffer::Data (args[1 ]);
3570
- ssize_t iv_len = Buffer::Length (args[2 ]);
3571
- const char * iv_buf = Buffer::Data (args[2 ]);
3584
+ ssize_t iv_len;
3585
+ const char * iv_buf;
3586
+ if (args[2 ]->IsNull ()) {
3587
+ iv_buf = nullptr ;
3588
+ iv_len = -1 ;
3589
+ } else {
3590
+ iv_buf = Buffer::Data (args[2 ]);
3591
+ iv_len = Buffer::Length (args[2 ]);
3592
+ }
3572
3593
cipher->InitIv (*cipher_type, key_buf, key_len, iv_buf, iv_len);
3573
3594
}
3574
3595
0 commit comments