mirror of https://github.com/ARMmbed/mbed-os.git
				
				
				
			Merge pull request #4157 from adustm/STM_sha1_F439ZI
NUCLEO_F439ZI/mbedtls: add SHA1 hw_accelerationpull/4639/head
						commit
						0a2a48c957
					
				| 
						 | 
				
			
			@ -21,6 +21,6 @@
 | 
			
		|||
#define MBEDTLS_DEVICE_H
 | 
			
		||||
 | 
			
		||||
#define MBEDTLS_AES_ALT
 | 
			
		||||
 | 
			
		||||
#define MBEDTLS_SHA1_ALT
 | 
			
		||||
 | 
			
		||||
#endif /* MBEDTLS_DEVICE_H */
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -0,0 +1,187 @@
 | 
			
		|||
/*
 | 
			
		||||
 *  sha1_alt.c for SHA1 HASH
 | 
			
		||||
 *******************************************************************************
 | 
			
		||||
 * Copyright (c) 2017, STMicroelectronics
 | 
			
		||||
 *  SPDX-License-Identifier: Apache-2.0
 | 
			
		||||
 *
 | 
			
		||||
 *  Licensed under the Apache License, Version 2.0 (the "License"); you may
 | 
			
		||||
 *  not use this file except in compliance with the License.
 | 
			
		||||
 *  You may obtain a copy of the License at
 | 
			
		||||
 *
 | 
			
		||||
 *  http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 *
 | 
			
		||||
 *  Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
 *  distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
 | 
			
		||||
 *  WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
 *  See the License for the specific language governing permissions and
 | 
			
		||||
 *  limitations under the License.
 | 
			
		||||
 *
 | 
			
		||||
 */
 | 
			
		||||
#include "mbedtls/sha1.h"
 | 
			
		||||
#if defined(MBEDTLS_SHA1_ALT)
 | 
			
		||||
#include "mbedtls/platform.h"
 | 
			
		||||
 | 
			
		||||
/* Implementation that should never be optimized out by the compiler */
 | 
			
		||||
static void mbedtls_zeroize( void *v, size_t n ) {
 | 
			
		||||
    volatile unsigned char *p = (unsigned char*)v; while( n-- ) *p++ = 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int st_sha1_restore_hw_context(mbedtls_sha1_context *ctx)
 | 
			
		||||
{
 | 
			
		||||
    uint32_t i;
 | 
			
		||||
    uint32_t tickstart;
 | 
			
		||||
    /* allow multi-instance of HASH use: save context for HASH HW module CR */
 | 
			
		||||
    /* Check that there is no HASH activity on going */
 | 
			
		||||
    tickstart = HAL_GetTick();
 | 
			
		||||
    while ((HASH->SR & (HASH_FLAG_BUSY | HASH_FLAG_DMAS)) != 0) {
 | 
			
		||||
        if ((HAL_GetTick() - tickstart) > ST_SHA1_TIMEOUT) {
 | 
			
		||||
            return 0; // timeout: HASH processor is busy
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    HASH->STR = ctx->ctx_save_str;
 | 
			
		||||
    HASH->CR = (ctx->ctx_save_cr | HASH_CR_INIT);
 | 
			
		||||
    for (i=0;i<38;i++) {
 | 
			
		||||
        HASH->CSR[i] = ctx->ctx_save_csr[i];
 | 
			
		||||
    }
 | 
			
		||||
    return 1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int st_sha1_save_hw_context(mbedtls_sha1_context *ctx)
 | 
			
		||||
{
 | 
			
		||||
    uint32_t i;
 | 
			
		||||
    uint32_t tickstart;
 | 
			
		||||
    /* Check that there is no HASH activity on going */
 | 
			
		||||
    tickstart = HAL_GetTick();
 | 
			
		||||
    while ((HASH->SR & (HASH_FLAG_BUSY | HASH_FLAG_DMAS)) != 0) {
 | 
			
		||||
        if ((HAL_GetTick() - tickstart) > ST_SHA1_TIMEOUT) {
 | 
			
		||||
            return 0; // timeout: HASH processor is busy
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    /* allow multi-instance of HASH use: restore context for HASH HW module CR */
 | 
			
		||||
    ctx->ctx_save_cr = HASH->CR;
 | 
			
		||||
    ctx->ctx_save_str = HASH->STR;
 | 
			
		||||
    for (i=0;i<38;i++) {
 | 
			
		||||
        ctx->ctx_save_csr[i] = HASH->CSR[i];
 | 
			
		||||
    }
 | 
			
		||||
    return 1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mbedtls_sha1_init( mbedtls_sha1_context *ctx )
 | 
			
		||||
{
 | 
			
		||||
    mbedtls_zeroize( ctx, sizeof( mbedtls_sha1_context ) );
 | 
			
		||||
 | 
			
		||||
    /* Enable HASH clock */
 | 
			
		||||
    __HAL_RCC_HASH_CLK_ENABLE();
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mbedtls_sha1_free( mbedtls_sha1_context *ctx )
 | 
			
		||||
{
 | 
			
		||||
    if( ctx == NULL )
 | 
			
		||||
        return;
 | 
			
		||||
    mbedtls_zeroize( ctx, sizeof( mbedtls_sha1_context ) );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mbedtls_sha1_clone( mbedtls_sha1_context *dst,
 | 
			
		||||
                         const mbedtls_sha1_context *src )
 | 
			
		||||
{
 | 
			
		||||
    *dst = *src;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mbedtls_sha1_starts( mbedtls_sha1_context *ctx )
 | 
			
		||||
{
 | 
			
		||||
    /* Deinitializes the HASH peripheral */
 | 
			
		||||
    if (HAL_HASH_DeInit(&ctx->hhash_sha1) == HAL_ERROR) {
 | 
			
		||||
        // error found to be returned
 | 
			
		||||
        return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /* HASH Configuration */
 | 
			
		||||
    ctx->hhash_sha1.Init.DataType = HASH_DATATYPE_8B;
 | 
			
		||||
    if (HAL_HASH_Init(&ctx->hhash_sha1) == HAL_ERROR) {
 | 
			
		||||
        // error found to be returned
 | 
			
		||||
        return;
 | 
			
		||||
    }
 | 
			
		||||
    if (st_sha1_save_hw_context(ctx) != 1) {
 | 
			
		||||
        return; // return HASH_BUSY timeout Error here
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mbedtls_sha1_process( mbedtls_sha1_context *ctx, const unsigned char data[ST_SHA1_BLOCK_SIZE] )
 | 
			
		||||
{
 | 
			
		||||
    if (st_sha1_restore_hw_context(ctx) != 1) {
 | 
			
		||||
        return; // Return HASH_BUSY timout error here
 | 
			
		||||
    }
 | 
			
		||||
    if (HAL_HASH_SHA1_Accumulate(&ctx->hhash_sha1, (uint8_t *) data, ST_SHA1_BLOCK_SIZE) != 0) {
 | 
			
		||||
            return; // Return error code
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (st_sha1_save_hw_context(ctx) != 1) {
 | 
			
		||||
        return; // return HASH_BUSY timeout Error here
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mbedtls_sha1_update( mbedtls_sha1_context *ctx, const unsigned char *input, size_t ilen )
 | 
			
		||||
{
 | 
			
		||||
    size_t currentlen = ilen;
 | 
			
		||||
    if (st_sha1_restore_hw_context(ctx) != 1) {
 | 
			
		||||
        return; // Return HASH_BUSY timout error here
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // store mechanism to accumulate ST_SHA1_BLOCK_SIZE bytes (512 bits) in the HW
 | 
			
		||||
    if (currentlen == 0){ // only change HW status is size if 0
 | 
			
		||||
        if(ctx->hhash_sha1.Phase == HAL_HASH_PHASE_READY) {
 | 
			
		||||
          /* Select the SHA1 mode and reset the HASH processor core, so that the HASH will be ready to compute
 | 
			
		||||
             the message digest of a new message */
 | 
			
		||||
          HASH->CR |= HASH_ALGOSELECTION_SHA1 | HASH_CR_INIT;
 | 
			
		||||
        }
 | 
			
		||||
        ctx->hhash_sha1.Phase = HAL_HASH_PHASE_PROCESS;
 | 
			
		||||
    } else if (currentlen < (ST_SHA1_BLOCK_SIZE - ctx->sbuf_len)) {
 | 
			
		||||
        // only buffurize
 | 
			
		||||
        memcpy(ctx->sbuf + ctx->sbuf_len, input, currentlen);
 | 
			
		||||
        ctx->sbuf_len += currentlen;
 | 
			
		||||
    } else {
 | 
			
		||||
        // fill buffer and process it
 | 
			
		||||
        memcpy(ctx->sbuf + ctx->sbuf_len, input, (ST_SHA1_BLOCK_SIZE - ctx->sbuf_len));
 | 
			
		||||
        currentlen -= (ST_SHA1_BLOCK_SIZE - ctx->sbuf_len);
 | 
			
		||||
        mbedtls_sha1_process(ctx, ctx->sbuf);
 | 
			
		||||
        // Process every input as long as it is %64 bytes, ie 512 bits
 | 
			
		||||
        size_t iter = currentlen / ST_SHA1_BLOCK_SIZE;
 | 
			
		||||
        if (HAL_HASH_SHA1_Accumulate(&ctx->hhash_sha1, (uint8_t *)(input + ST_SHA1_BLOCK_SIZE - ctx->sbuf_len), (iter * ST_SHA1_BLOCK_SIZE)) != 0) {
 | 
			
		||||
            return; // Return error code here
 | 
			
		||||
        }
 | 
			
		||||
        // sbuf is completely accumulated, now copy up to 63 remaining bytes
 | 
			
		||||
        ctx->sbuf_len = currentlen % ST_SHA1_BLOCK_SIZE;
 | 
			
		||||
        if (ctx->sbuf_len !=0) {
 | 
			
		||||
            memcpy(ctx->sbuf, input + ilen - ctx->sbuf_len, ctx->sbuf_len);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    if (st_sha1_save_hw_context(ctx) != 1) {
 | 
			
		||||
        return; // return HASH_BUSY timeout Error here
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void mbedtls_sha1_finish( mbedtls_sha1_context *ctx, unsigned char output[20] )
 | 
			
		||||
{
 | 
			
		||||
    if (st_sha1_restore_hw_context(ctx) != 1) {
 | 
			
		||||
        return; // Return HASH_BUSY timout error here
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (ctx->sbuf_len > 0) {
 | 
			
		||||
        if (HAL_HASH_SHA1_Accumulate(&ctx->hhash_sha1, ctx->sbuf, ctx->sbuf_len) != 0) {
 | 
			
		||||
            return; // Return error code here
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
    mbedtls_zeroize(ctx->sbuf, ST_SHA1_BLOCK_SIZE);
 | 
			
		||||
    ctx->sbuf_len = 0;
 | 
			
		||||
    __HAL_HASH_START_DIGEST();
 | 
			
		||||
 | 
			
		||||
    if (HAL_HASH_SHA1_Finish(&ctx->hhash_sha1, output, 10) != 0){
 | 
			
		||||
        return; // error code to be returned
 | 
			
		||||
    }
 | 
			
		||||
    if (st_sha1_save_hw_context(ctx) != 1) {
 | 
			
		||||
        return; // return HASH_BUSY timeout Error here
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#endif /*MBEDTLS_SHA1_ALT*/
 | 
			
		||||
| 
						 | 
				
			
			@ -0,0 +1,111 @@
 | 
			
		|||
/*
 | 
			
		||||
 * \file sha1_alt.h
 | 
			
		||||
 *
 | 
			
		||||
 * \brief SHA1 hw acceleration (hash function)
 | 
			
		||||
 *
 | 
			
		||||
 *  Copyright (C) 2017, STMicroelectronics
 | 
			
		||||
 *  SPDX-License-Identifier: Apache-2.0
 | 
			
		||||
 *
 | 
			
		||||
 *  Licensed under the Apache License, Version 2.0 (the "License"); you may
 | 
			
		||||
 *  not use this file except in compliance with the License.
 | 
			
		||||
 *  You may obtain a copy of the License at
 | 
			
		||||
 *
 | 
			
		||||
 *  http://www.apache.org/licenses/LICENSE-2.0
 | 
			
		||||
 *
 | 
			
		||||
 *  Unless required by applicable law or agreed to in writing, software
 | 
			
		||||
 *  distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
 | 
			
		||||
 *  WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
			
		||||
 *  See the License for the specific language governing permissions and
 | 
			
		||||
 *  limitations under the License.
 | 
			
		||||
 *
 | 
			
		||||
 */
 | 
			
		||||
#ifndef MBEDTLS_SHA1_ALT_H
 | 
			
		||||
#define MBEDTLS_SHA1_ALT_H
 | 
			
		||||
 | 
			
		||||
#if defined (MBEDTLS_SHA1_ALT)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#include "cmsis.h"
 | 
			
		||||
#include <string.h>
 | 
			
		||||
 | 
			
		||||
#ifdef __cplusplus
 | 
			
		||||
extern "C" {
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#define ST_SHA1_BLOCK_SIZE   ((size_t) 64) // HW handles 512 bits, ie 64 bytes
 | 
			
		||||
#define ST_SHA1_TIMEOUT      ((uint32_t) 3)
 | 
			
		||||
/**
 | 
			
		||||
 * \brief    SHA-1 context structure
 | 
			
		||||
 * \note     HAL_HASH_SHA1_Accumulate will accumulate 512 bits packets, unless it is the last call to the  function
 | 
			
		||||
 *           A ST_SHA1_BLOCK_SIZE bytes buffer is used to save values and handle the processing
 | 
			
		||||
 *           multiples of ST_SHA1_BLOCK_SIZE bytes
 | 
			
		||||
 *           If SHA1_finish is called and sbuf_len>0, the remaining bytes are accumulated prior to the call to HAL_HASH_SHA1_Finish
 | 
			
		||||
 */
 | 
			
		||||
typedef struct
 | 
			
		||||
{
 | 
			
		||||
    unsigned char sbuf[ST_SHA1_BLOCK_SIZE]; /*!< MBEDTLS_SHA1_BLOCK_SIZE buffer to store values so that algorithm is caled once the buffer is filled */
 | 
			
		||||
    unsigned char sbuf_len; /*!< number of bytes remaining in sbuf to be processed */
 | 
			
		||||
    HASH_HandleTypeDef hhash_sha1; /*!< ST HAL HASH struct */
 | 
			
		||||
    uint32_t ctx_save_cr;
 | 
			
		||||
    uint32_t ctx_save_str;
 | 
			
		||||
    uint32_t ctx_save_csr[38];
 | 
			
		||||
}
 | 
			
		||||
mbedtls_sha1_context;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * \brief          Initialize SHA-1 context
 | 
			
		||||
 *
 | 
			
		||||
 * \param ctx      SHA-1 context to be initialized
 | 
			
		||||
 */
 | 
			
		||||
void mbedtls_sha1_init( mbedtls_sha1_context *ctx );
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * \brief          Clear SHA-1 context
 | 
			
		||||
 *
 | 
			
		||||
 * \param ctx      SHA-1 context to be cleared
 | 
			
		||||
 */
 | 
			
		||||
void mbedtls_sha1_free( mbedtls_sha1_context *ctx );
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * \brief          Clone (the state of) a SHA-1 context
 | 
			
		||||
 *
 | 
			
		||||
 * \param dst      The destination context
 | 
			
		||||
 * \param src      The context to be cloned
 | 
			
		||||
 */
 | 
			
		||||
void mbedtls_sha1_clone( mbedtls_sha1_context *dst,
 | 
			
		||||
                         const mbedtls_sha1_context *src );
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * \brief          SHA-1 context setup
 | 
			
		||||
 *
 | 
			
		||||
 * \param ctx      context to be initialized
 | 
			
		||||
 */
 | 
			
		||||
void mbedtls_sha1_starts( mbedtls_sha1_context *ctx );
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * \brief          SHA-1 process buffer
 | 
			
		||||
 *
 | 
			
		||||
 * \param ctx      SHA-1 context
 | 
			
		||||
 * \param input    buffer holding the  data
 | 
			
		||||
 * \param ilen     length of the input data
 | 
			
		||||
 */
 | 
			
		||||
void mbedtls_sha1_update( mbedtls_sha1_context *ctx, const unsigned char *input, size_t ilen );
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * \brief          SHA-1 final digest
 | 
			
		||||
 *
 | 
			
		||||
 * \param ctx      SHA-1 context
 | 
			
		||||
 * \param output   SHA-1 checksum result
 | 
			
		||||
 */
 | 
			
		||||
void mbedtls_sha1_finish( mbedtls_sha1_context *ctx, unsigned char output[20] );
 | 
			
		||||
 | 
			
		||||
/* Internal use */
 | 
			
		||||
void mbedtls_sha1_process( mbedtls_sha1_context *ctx, const unsigned char data[ST_SHA1_BLOCK_SIZE] );
 | 
			
		||||
 | 
			
		||||
#ifdef __cplusplus
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#endif /* MBEDTLS_SHA1_ALT */
 | 
			
		||||
 | 
			
		||||
#endif /* sha1_alt.h */
 | 
			
		||||
		Loading…
	
		Reference in New Issue