/* * Copyright (C) 2019 Sean Parkinson, wolfSSL Inc. * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ #include "wolfssl_common.h" #include "wolfssl_hasher.h" #include "wolfssl_util.h" #include typedef struct private_wolfssl_hasher_t private_wolfssl_hasher_t; /** * Private data of wolfssl_hasher_t */ struct private_wolfssl_hasher_t { /** * Public interface */ wolfssl_hasher_t public; /** * The hasher to use */ wc_HashAlg hasher; /** * The hash algorithm */ enum wc_HashType type; }; METHOD(hasher_t, get_hash_size, size_t, private_wolfssl_hasher_t *this) { return wc_HashGetDigestSize(this->type); } METHOD(hasher_t, reset, bool, private_wolfssl_hasher_t *this) { return wc_HashInit(&this->hasher, this->type) == 0; } METHOD(hasher_t, get_hash, bool, private_wolfssl_hasher_t *this, chunk_t chunk, uint8_t *hash) { int ret; ret = wc_HashUpdate(&this->hasher, this->type, chunk.ptr, chunk.len); if (ret == 0 && hash) { ret = wc_HashFinal(&this->hasher, this->type, hash); if (ret == 0) { return reset(this); } } return ret == 0; } METHOD(hasher_t, allocate_hash, bool, private_wolfssl_hasher_t *this, chunk_t chunk, chunk_t *hash) { if (hash) { *hash = chunk_alloc(get_hash_size(this)); return get_hash(this, chunk, hash->ptr); } return get_hash(this, chunk, NULL); } METHOD(hasher_t, destroy, void, private_wolfssl_hasher_t *this) { wc_HashFree(&this->hasher, this->type); free(this); } /* * Described in header */ wolfssl_hasher_t *wolfssl_hasher_create(hash_algorithm_t algo) { private_wolfssl_hasher_t *this; enum wc_HashType type; if (!wolfssl_hash2type(algo, &type)) { return NULL; } INIT(this, .public = { .hasher = { .get_hash = _get_hash, .allocate_hash = _allocate_hash, .get_hash_size = _get_hash_size, .reset = _reset, .destroy = _destroy, }, }, .type = type, ); /* initialization */ if (!reset(this)) { destroy(this); return NULL; } return &this->public; }