MD5文件校验

         最近一个项目,需要做OTA远程升级,MCU从服务器下载升级文件到本地,进行程序升级。本来如果使用TPC/IP协议传输数据(升级文件)的话,可以不进行文件校验,因为TCP/IP数据传输的时候,为了保证数据的正确性,TCP/IP会进行数据校验。也就是说,我们通过TCP/IP协议接受的数据包,是不需要再次进行校验的。但是这个项目,使用了移远的串口-GPRS模块。MCU通过串口发送AT指令给GPRS模块,GPRS模块从远程服务器下载升级文件,然后再通过串口传输到MCU。我们知道,串口在传输数据的过程中,可能会产生误码,虽然概率非常小,但是毕竟是关于升级的程序,不能出错。所以项目需要MCU对从串口接收到的文件再次进行校验。比较了常用的校验算法,决定使用MD5校验算法,对升级文件进行校验。MCU对升级文件进行MD5校验后,再跟收到的升级文件的MD5校验码进行比较。如果两个MD5校验码相同,则说明升级文件传输正确,否则表示升级文件传输错误,需要重新下载。百度了一下MD5校验,参考的算法都差不多,下面的这个MD5校验算法,经过我的测试,确认是可以使用的。

    补充一下:

            CRC校验一般用于通信数据的校验。

            MD5和SH1用于安全领域,比如文件校验、数字签名等。

测试时使用使用VS2013.

#include<stdio.h>
#include<stdlib.h>
#include<string.h>

#define F(x, y, z) (((x) & (y)) | ((~x) & (z)))
#define G(x, y, z) (((x) & (z)) | ((y) & (~z)))
#define H(x, y, z) ((x) ^ (y) ^ (z))
#define I(x, y, z) ((y) ^ ((x) | (~z)))

#define RL(x, y) (((x) << (y)) | ((x) >> (32 - (y)))) //x向左循环移y位

#define PP(x) (x<<24)|((x<<8)&0xff0000)|((x>>8)&0xff00)| (x>>24) //将x高低位互换,例如PP(aabbccdd)=ddccbbaa

#define FF(a, b, c, d, x, s, ac) a = b + (RL((a + F(b,c,d) + x + ac),s))
#define GG(a, b, c, d, x, s, ac) a = b + (RL((a + G(b,c,d) + x + ac),s))
#define HH(a, b, c, d, x, s, ac) a = b + (RL((a + H(b,c,d) + x + ac),s))
#define II(a, b, c, d, x, s, ac) a = b + (RL((a + I(b,c,d) + x + ac),s))

unsigned A, B, C, D, a, b, c, d, i, len, flen[2], x[16]; //i临时变量,len文件长,flen[2]为64位二进制表示的文件初始长度
char filen

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
文件分块进行MD5校验的步骤如下: 1. 将要校验文件进行分块,每块的大小自定,一般为几百KB或几MB。可以使用fread函数读取文件块到缓冲区中。 2. 对每个文件块进行MD5计算。可以使用openssl库中的MD5函数来计算MD5值。计算MD5值的函数原型如下: ``` #include <openssl/md5.h> int MD5(const unsigned char *d, size_t n, unsigned char *md); ``` 其中,d指向要计算MD5值的数据,n为数据的长度,md输出MD5值。 3. 将每个块的MD5值合并成一个整体的MD5值。可以使用MD5_Init、MD5_Update和MD5_Final函数来实现。其中,MD5_Init函数用于初始化MD5值,MD5_Update函数用于将数据加入到MD5值的计算中,MD5_Final函数用于输出最终的MD5值。这些函数的原型如下: ``` #include <openssl/md5.h> void MD5_Init(MD5_CTX *c); void MD5_Update(MD5_CTX *c, const void *data, size_t len); void MD5_Final(unsigned char *md, MD5_CTX *c); ``` 4. 将整体MD5值与预期MD5值进行比较,判断文件是否被篡改。 下面是一个示例代码,实现了大文件分块进行MD5校验的过程: ```c #include <stdio.h> #include <stdlib.h> #include <openssl/md5.h> int main(int argc, char *argv[]) { FILE *fp; unsigned char buffer[1024]; unsigned char md5sum[MD5_DIGEST_LENGTH]; MD5_CTX ctx; int bytes; long file_size, block_size, blocks; char *filename; if (argc != 3) { printf("Usage: %s <filename> <blocksize>\n", argv[0]); return 0; } filename = argv[1]; block_size = atol(argv[2]); fp = fopen(filename, "rb"); if (!fp) { printf("Can't open file %s\n", filename); return 0; } fseek(fp, 0L, SEEK_END); file_size = ftell(fp); fseek(fp, 0L, SEEK_SET); blocks = file_size / block_size + (file_size % block_size ? 1 : 0); MD5_Init(&ctx); for (int i = 0; i < blocks; i++) { bytes = fread(buffer, 1, block_size, fp); MD5_Update(&ctx, buffer, bytes); } MD5_Final(md5sum, &ctx); fclose(fp); printf("MD5(%s) = ", filename); for (int i = 0; i < MD5_DIGEST_LENGTH; i++) { printf("%02x", md5sum[i]); } printf("\n"); return 0; } ``` 在命令行中输入以下命令来编译程序: ``` gcc -o md5sum md5sum.c -lcrypto ``` 其中,-lcrypto选项指定链接openssl库。然后可以使用以下命令来计算文件MD5值: ``` ./md5sum <filename> <blocksize> ``` 其中,<filename>为要计算MD5值的文件路径,<blocksize>为每个文件块的大小,单位为字节。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值