BMP 图像信息隐藏及检测

Posted qftm

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了BMP 图像信息隐藏及检测相关的知识,希望对你有一定的参考价值。

原理简介

针对文件结构的信息隐藏方法需详细掌握文件的格式,利用文件结构块之间的关系或根据块数据和块大小之间的关系来隐藏信息。

BMP(Bitmap-File)图形文件是 Windows 采用的常见图形文件格式,要利用 BMP 位图进行信息隐藏首先需要详细了解 BMP 文件的格式,BMP 图像文件结构比较单一而且固定,BMP  图像由文件头、信息头、调色板区和数据区四个部分组成,而 24 位真彩色图像中没有调色板信息。24 位真彩色 BMP 位图文件包括 3 部分。 第一部分是 BMP 文件头(14个字节)。前 2 个字节是“BM”,是用于识别 BMP文件的标志;第 3、4、5、6 字节存放的是位图文件的大小,以字节为单位;第7、8、9、10 字节是保留的,必须为 0;第 11、12、13、14 字节给出位图阵列相对于文件头的偏移,在 24 位真彩色图像中,这个值固定为 54;第二部分是位图信息头(40个字节)。第19,20,21,22字节表示的是图像文件的宽度,以像素为单位;第23,24,25,26 字节表示的是图像文件的高度,以像素为单位。从第 29个字节开始,第 29、30 字节描述的是像素的位数, 24 位真彩色位图。该位的值为 0x18; 第三部分是数据区。从第 55 个字节开始,每 3 个字节表示一个像素,这 3 个字节依次表示该像素的红、绿、蓝亮度分量值。

在不影响图像正常显示情况下,可使用以下四种方法在 24 位真彩色 BMP 图像中隐藏信息。

  • 在图像文件尾部添加任意长度的数据,秘密信息存放在文件尾部可以减少修改文件头的数据量,仅需修改文件头中文件长度的值即可。
  • 在调色板或者位图信息头和实际的图像数据之间隐藏数据,如果将秘密数据放在文件头与图像数据之间,则至少需要修改文件头中文件长度、数据起始偏移地址这两个域的值。
  • 修改文件头和信息头中的保留字段隐藏信息。
  • 在图像像素区利用图像宽度字节必须是 4 的倍数的特点,在补足位处隐藏数据。

操作环境

Windows 10 操作系统

MATLAB 2019a 版本软件

BMP 格式图片文件

010 编辑工具

技术过程

一、在实际的图像数据后隐藏信息

待隐藏的秘密信息文件名称为  hidden.txt,  Baboon.bmp  为载体图像,将载体和秘密信息放置在同一个目录下,在 Windows 的 MS-DOS 方式下执行命令 Copy baboon.bmp /b + hidden.txt /a baboon1.bmp  ,其中参数/b 指定以二进制格式复制、合并文件,参数/a 以 ASCII 格式复制、合并文件。执行该命令后,生成一个新的 baboon1.bmp 文件,使用图像浏览工具浏览该文件发现与原始载体图像几乎完全相同,信息隐藏在 baboon.bmp 文件的尾部。从 BMP 图像的结构中可知,图像的 3、4、5、6 四个字节存放整个 BMP 图像的长度。使用该方法隐藏信息时,未修改图像文件的文件长度字节,通过比较文件的实际长度和

文件中保存的文件长度,就可发现该图像是否隐藏秘密信息。

(1)制作隐藏信息的图片QftmModify1.bmp

创建Hidden.txt文件

技术图片

制作QftmModify1.bmp

copy  Qftm.bmp/b + hidden.txt/a QftmModify1.bmp

技术图片

(2)Matlab脚本检测文件是否存在隐藏信息

检测图片QftmModify1.bmp文件长度

clc;
clear;
fid=fopen(‘QftmModify1.bmp‘,‘r‘); %读入载体图像文件
[a,length]=fread(fid,inf,‘uint8‘);%length 是文件的实际长度
fclose(fid);
fid=fopen(‘QftmModify1.bmp‘,‘r‘);
status=fseek(fid,2,‘bof‘);
fileb=fread(fid,4,‘uint8‘);
filelength=fileb(1)*1+fileb(2)*256+fileb(3)*256^2+fileb(4)*256^3 %文件图像中保存的文件长度
diff=length-filelength;
%diff 表示隐藏的信息长度如果相同,表示图像没有隐藏任何信息。
fclose(fid);

运行脚本查看diff结果

 技术图片

 从结果可以看出来当没有对bmp的文件头文件长度进行修改时,隐藏的图片diff差值不等于0,得到图片存在信息隐藏。

二、文件头与图像数据之间隐藏信息

在数据区开始之前隐藏信息,隐藏的秘密信息从 hidden.txt 文件中读取(隐藏整个文件),此种方法修改图像数据的偏移量和图像数据的文件长度。

(1)Matlab脚本进行隐藏信息

clc;
clear;
wm=randsrc(1,300, [0 1]); %  产生随机水印
fid=fopen(‘Qftm.bmp‘,‘r‘); %读入载体图像文件
[a,length]=fread(fid,inf,‘uint8‘);
fclose(fid);
msgfid=fopen(‘hidden.txt‘,‘r‘);%打开秘密文件
[msg,count]=fread(msgfid);
fclose(msgfid);
wa=a;
j=1;
wa(11)=54+count;
wa(3)=wa(3)+count;
for i=55:54+count
    wa(i)=uint8(msg(j,1));%隐藏密码信息
    j=j+1;
end
for i=55:length
    wa(i+count)=a(i);
end
figure;
wa=uint8(wa);
fid=fopen(‘watermarked.bmp‘, ‘wb‘);
fwrite(fid,wa);
fclose(fid);
I=imread(‘Qftm.bmp‘);
J=imread(‘watermarked.bmp‘);
subplot(1,2,1)
imshow(I)
title(‘未修改图像‘);
subplot(1,2,2)
imshow(J)
title(‘修改后图像‘)

运行结果查看

技术图片

  分析《watermarked.bmp》16进制清晰查看隐藏效果

技术图片

三、BMP 图像文件隐藏信息的检测

在BMP图像中隐藏信息的时候一般都是通过修改文件的偏移量和图像文件中图像的长度来隐藏信息,但在BMP图像文件中,file_length=biwidth*biBytecount*biHeight+bfoffBits,其中  biwidth,biheight表示

图像文件的宽度和高度,bfoffBits表示文件头到实际位图图像数据之间的偏移量。

(1)Matlab脚本对Bmp进行信息隐藏多层Check

clc;
clear;
wm=randsrc(1,300, [0 1]); %  产生随机水印信息
fid=fopen(‘watermarked.bmp‘,‘r‘); %读入载体图像文件
[a,length]=fread(fid,inf,‘uint8‘);
status=fseek(fid,2,‘bof‘);
fileb=fread(fid,4,‘uint8‘);
filelength=fileb(1)*1+fileb(2)*256+fileb(3)*256^2+fileb(4)*256^3 %文件图像的理论长度
status=fseek(fid,10,‘bof‘);
b=fread(fid,4,‘uint8‘);
bfoffbitsmodify=b(1)*1+b(2)*256+b(3)*256^2+b(4)*256^3 %读取偏移量
status=fseek(fid,18,‘bof‘);
b=fread(fid,4,‘uint8‘);
biwidth=b(1)*1+b(2)*256+b(3)*256^2+b(4)*256^3
status=fseek(fid,22,‘bof‘);
b=fread(fid,4,‘uint8‘);
biHeight=b(1)*1+b(2)*256+b(3)*256^2+b(4)*256^3;
bfoffbits=54;%偏移量
biBytecount=3;%24 位真彩色图像为 3
filetruelength=biwidth*biBytecount*biHeight+bfoffbits %图片实际真实长度(固定偏移54)
filelengthbfoffbits=biwidth*biBytecount*biHeight+bfoffbitsmodify %读取偏移计算文件长度
fclose(fid);
diff1=length-filelength;        %未修改图片长度时Check1
diff2=filelength-filetruelength %修改了图片长度时Check2
diff3=filelengthbfoffbits-filetruelength %修改了图片偏移时Check3
diff=diff1+diff2+diff3

对watermarked.bmp(文件长度和文件偏移都被修改了)进行测试,根据diff的值进行判定该图片是否存在信息隐藏,运行结果查看:

技术图片

从结果diff != 0可以看出来该图片存在信息隐藏。

四、在图像文件头和信息头的保留字段中隐藏信息

BMP 图像文件中有很多从不使用的保留字节,如 7、8、9、10 字节是保留的,必须为 0,可在第 7、8、9、10 字节隐藏秘密信息。

(1)Matlab脚本对Bmp的保留字段隐藏信息

clc;
clear;
fid=fopen(‘Qftm.bmp‘,‘r‘); %读入载体图像文件
[a,length]=fread(fid,inf,‘uint8‘);
fclose(fid);
wa=a;
% 在BMP的7、8、9、10保留字中隐藏秘密信息Qftm,ASCII值为0x51 0x66 0x74 0x6d
wa(7)=81;
wa(8)=102;
wa(9)=116;
wa(10)=109;
figure;
wa=uint8(wa);
fid=fopen(‘watermarkedReserve.bmp‘, ‘wb‘);
fwrite(fid,wa);
fclose(fid);
I=imread(‘Qftm.bmp‘);
J=imread(‘watermarkedReserve.bmp‘);
subplot(1,2,1)
imshow(I)
title(‘未修改图像‘);
subplot(1,2,2)
imshow(J)
title(‘修改后图像‘)

技术图片

  分析《watermarkedReserve.bmp》16进制清晰查看隐藏效果

技术图片

 

 

 

 

 

以上是关于BMP 图像信息隐藏及检测的主要内容,如果未能解决你的问题,请参考以下文章

Bmp图像的数据格式及读取

用标准c读取某bmp文件的长宽及象素等信息需要哪些函数?

使用 C 的 24 位 bmp 中的边缘检测

隐藏在图片下的恶意程序Stegoloader,是否避开了你的杀软?

在 JPG 图像中隐藏消息

C++ bmp图像添加水印及旋转水印