|
| 1 | +#include "ncrypto.h" |
| 2 | + |
| 3 | +namespace ncrypto { |
| 4 | + |
| 5 | +// ============================================================================ |
| 6 | +// Engine |
| 7 | + |
| 8 | +#ifndef OPENSSL_NO_ENGINE |
| 9 | +EnginePointer::EnginePointer(ENGINE* engine_, bool finish_on_exit_) |
| 10 | + : engine(engine_), finish_on_exit(finish_on_exit_) {} |
| 11 | + |
| 12 | +EnginePointer::EnginePointer(EnginePointer&& other) noexcept |
| 13 | + : engine(other.engine), finish_on_exit(other.finish_on_exit) { |
| 14 | + other.release(); |
| 15 | +} |
| 16 | + |
| 17 | +EnginePointer::~EnginePointer() { reset(); } |
| 18 | + |
| 19 | +EnginePointer& EnginePointer::operator=(EnginePointer&& other) noexcept { |
| 20 | + if (this == &other) return *this; |
| 21 | + this->~EnginePointer(); |
| 22 | + return *new (this) EnginePointer(std::move(other)); |
| 23 | +} |
| 24 | + |
| 25 | +void EnginePointer::reset(ENGINE* engine_, bool finish_on_exit_) { |
| 26 | + if (engine != nullptr) { |
| 27 | + if (finish_on_exit) { |
| 28 | + // This also does the equivalent of ENGINE_free. |
| 29 | + ENGINE_finish(engine); |
| 30 | + } else { |
| 31 | + ENGINE_free(engine); |
| 32 | + } |
| 33 | + } |
| 34 | + engine = engine_; |
| 35 | + finish_on_exit = finish_on_exit_; |
| 36 | +} |
| 37 | + |
| 38 | +ENGINE* EnginePointer::release() { |
| 39 | + ENGINE* ret = engine; |
| 40 | + engine = nullptr; |
| 41 | + finish_on_exit = false; |
| 42 | + return ret; |
| 43 | +} |
| 44 | + |
| 45 | +EnginePointer EnginePointer::getEngineByName(const std::string_view name, |
| 46 | + CryptoErrorList* errors) { |
| 47 | + MarkPopErrorOnReturn mark_pop_error_on_return(errors); |
| 48 | + EnginePointer engine(ENGINE_by_id(name.data())); |
| 49 | + if (!engine) { |
| 50 | + // Engine not found, try loading dynamically. |
| 51 | + engine = EnginePointer(ENGINE_by_id("dynamic")); |
| 52 | + if (engine) { |
| 53 | + if (!ENGINE_ctrl_cmd_string(engine.get(), "SO_PATH", name.data(), 0) || |
| 54 | + !ENGINE_ctrl_cmd_string(engine.get(), "LOAD", nullptr, 0)) { |
| 55 | + engine.reset(); |
| 56 | + } |
| 57 | + } |
| 58 | + } |
| 59 | + return engine; |
| 60 | +} |
| 61 | + |
| 62 | +bool EnginePointer::setAsDefault(uint32_t flags, CryptoErrorList* errors) { |
| 63 | + if (engine == nullptr) return false; |
| 64 | + ClearErrorOnReturn clear_error_on_return(errors); |
| 65 | + return ENGINE_set_default(engine, flags) != 0; |
| 66 | +} |
| 67 | + |
| 68 | +bool EnginePointer::init(bool finish_on_exit) { |
| 69 | + if (engine == nullptr) return false; |
| 70 | + if (finish_on_exit) setFinishOnExit(); |
| 71 | + return ENGINE_init(engine) == 1; |
| 72 | +} |
| 73 | + |
| 74 | +EVPKeyPointer EnginePointer::loadPrivateKey(const std::string_view key_name) { |
| 75 | + if (engine == nullptr) return EVPKeyPointer(); |
| 76 | + return EVPKeyPointer( |
| 77 | + ENGINE_load_private_key(engine, key_name.data(), nullptr, nullptr)); |
| 78 | +} |
| 79 | + |
| 80 | +void EnginePointer::initEnginesOnce() { |
| 81 | + static bool initialized = false; |
| 82 | + if (!initialized) { |
| 83 | + ENGINE_load_builtin_engines(); |
| 84 | + ENGINE_register_all_complete(); |
| 85 | + initialized = true; |
| 86 | + } |
| 87 | +} |
| 88 | + |
| 89 | +#endif // OPENSSL_NO_ENGINE |
| 90 | + |
| 91 | +} // namespace ncrypto |
0 commit comments