mbedtls移植之CMAC算法

一、 mbedtls简介

MbedTLS是一个开源、可移植、易使用、可读性高的SSL库,实现了常所用的加解密算法、X.509证书操作以及TLS协议操作。MbedTLS各功能模块独立性高、耦合度低,可以通过配置宏定义进行功能裁剪,非常适合对空间和效率要求高的嵌入式系统。

二、CMAC算法简介

消息认证码Message Authentication Code(MAC)是一种完整性校验技术。对一个任意长度的消息,通过在发送者和接收者之间共享的密钥,基于MAC算法,可生成MAC值。接收者只需校验传递过的MAC值,即可检验消息的完整性是否有效,从而判断消息是否被篡改过。常用的MAC算法有CMAC、HMAC、GMAC,本文主要介绍CMAC算法。CMAC基于对称密钥的分组算法,例如基于AES算法的CMAC,成为AES-CMAC。AES-CMAC类似AES-CBC算法,总体流程如下图
在这里插入图片描述

详细算法说明请参考RFC4439或NIST.SP.800-38B。

三、实现

3.1 移植MbedTLS代码

移植自mbedTLS 2.16版本
需移植的文件如下:
在这里插入图片描述

修改config.h文件

#ifndef MBEDTLS_CONFIG_H
#define MBEDTLS_CONFIG_H

#define MBEDTLS_ERROR_C

#define MBEDTLS_AES_C
#define MBEDTLS_CMAC_C
#define MBEDTLS_CIPHER_C
#define MBEDTLS_CIPHER_MODE_CBC

#endif

3.2 引入头文件

引入相应的头文件

#include <stdio.h>
#include <time.h>
#include <string.h>
#include "../crypto/mbedtls/cipher.h"
#include "../crypto/mbedtls/cmac.h"
#include "../crypto/mbedtls/error.h"

3.3 计算CMAC

实例为从一个文件中读取信息并计算AES-CMAC,其中使用到的key设置为字符串“1234567890ABCDEF”,key的长度必须为16字节的unsigned char。需要计算的消息通过调用mbedtls_cipher_cmac_update不断输入,直到全部输入完成。

int get_cmac_from_file(){
    mbedtls_cipher_context_t cipher_context;
    unsigned char error[100];
    const unsigned char *key="1234567890ABCDEF";
    unsigned char output_cmac[16];
    const char *file_name = "D:\\tmp\\crypto\\rsa\\music.mp3";


    mbedtls_cipher_init(&cipher_context);
    const mbedtls_cipher_info_t *cipher_info = mbedtls_cipher_info_from_type(MBEDTLS_CIPHER_AES_128_ECB);
    if(cipher_info == NULL){
        printf("failed to get cipher info\n");
        return -1;
    }

    int result = mbedtls_cipher_setup(&cipher_context,cipher_info);
    if(result != 0){
        printf("failed to set cipher info:");
        mbedtls_strerror(result,error,sizeof(error));
        printf("%s\n",error);
        mbedtls_cipher_free(&cipher_context);
        return -2;
    }

    result = mbedtls_cipher_cmac_starts(&cipher_context,key,128);
    if(result != 0){
        printf("failed to set start:");
        mbedtls_strerror(result,error,sizeof(error));
        printf("%s\n",error);
        mbedtls_cipher_free(&cipher_context);
        return -3;
    }
    //读取文件
    FILE *input_file = fopen(file_name,"rb");
    unsigned char read_buffer[1024];
    size_t read_num;
    while (1){
        read_num = fread(read_buffer,sizeof(char),sizeof(read_buffer),input_file);
        result = mbedtls_cipher_cmac_update(&cipher_context,read_buffer,read_num);
        if(result != 0){
            printf("failed to set update:");
            mbedtls_strerror(result,error,sizeof(error));
            printf("%s\n",error);
            mbedtls_cipher_free(&cipher_context);
            fclose(input_file);
            return -3;
        }
        if(read_num < 1024){
            break;
        }
    }
    fclose(input_file);

    result = mbedtls_cipher_cmac_finish(&cipher_context,output_cmac);
    if(result != 0){
        printf("failed to set finish:");
        mbedtls_strerror(result,error,sizeof(error));
        printf("%s\n",error);
        mbedtls_cipher_free(&cipher_context);
        return -3;
    }
    printf("cmac:\n");
    for(int i=0;i<16;i++){
        printf("%02x",output_cmac[i]);
    }
    printf("\n");


    mbedtls_cipher_free(&cipher_context);
    return 0;

}

3.4 运行效果

在这里插入图片描述
使用openssl进行验证,计算出的mac值是一致的。
在这里插入图片描述

四、总结

AES-CMAC算法是最常用的CMAC算法,且通过mbedtls移植也很方便。若需要继续精简空间,可不需要error.c以及删除相关.c文件中未使用的函数,例如aes.c中的mbedtls_aes_decrypt。

  • 16
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值