隐写术浅谈(二):LSB隐写与IDAT隐写

Misc 学习(番外篇) - 隐写分析:隐写术(2)

在本系列的其他文章中,我主要讲了讲自己对于隐写分析的一些浅薄理解,但是大都是针对于如何反隐写(做题嘛,不寒碜),基本上很少讲如何去隐写。上一篇我们讨论了直接附加和 IHDR,这一篇我们继续讨论如何去进行隐写(LSB 隐写和 IDAT 隐写)(非 CTF 向)。(此文并非教学,我只是在这里记下我的笔记、我的心得、我的体会,请辩证看待、理性思考,不要全都当成真理)

一. LSB 隐写(PNG、BMP)

PNG 文件中的图像像数一般是由 RGB 三原色(红绿蓝)组成(有的图片还包含A通道表示透明度),每一种颜色占用8位,取值范围为0x00至0xFF。LSB 隐写就是修改 RGB 颜色分量的最低二进制位(LSB),它修改了每个像数颜色的最低的1 bit,而人类的眼睛不会注意到这前后的变化,这样每个像素可以携带3比特的信息。

按照LSB隐写的原理LSB 隐写只能对 PNGBMP这种无损图片格式使用,像是 JPEG 格式的文件就并不能使用。

人工隐写

反正我不会用手隐写。

脚本隐写

脚本隐写无非分为三步:打开文件 --> 从一个像素点开始,按行或列改写其最低二进制位 ,这样的方式虽然容易被检测到,但是相比于其他的较为隐蔽,我们也可以根据最基础的开始逐步扩展。

对于 PNG 图片,我们可以使用如下 Python 脚本隐写一个文件到另一个 PNG 文件中。

from PIL import Image
import sys

def toasc(strr):
    return int(strr, 2)       
def plus(string): 
    return string.zfill(8)
def get_key(strr):
    #获取要隐藏的文件内容
    with open(strr,"rb")  as f:
        s = f.read()
        string=""
        for i in range(len(s)):
            string=string+""+plus(bin(s[i]).replace('0b',''))
    return string
def mod(x,y):
    return x%y

#str1为载体图片路径,str2为隐写文件,str3为加密图片保存的路径 
def encode(str1,str2,str3): 
    im = Image.open(str1) 
    width,height= im.size[0],im.size[1]
    print("width:"+str(width))
    print("height:"+str(height))
    count = 0
    key = get_key(str2) 
    keylen = len(key)
    for h in range(height):
        for w in range(width):
            pixel = im.getpixel((w,h))
            a=pixel[0]
            b=pixel[1]
            c=pixel[2]
            if count == keylen:
                break
            a= a-mod(a,2)+int(key[count])
            count+=1
            if count == keylen:
                im.putpixel((w,h),(a,b,c)) 
                break
            b =b-mod(b,2)+int(key[count])
            count+=1 
            if count == keylen:
                im.putpixel((w,h),(a,b,c)) 
                break
            c= c-mod(c,2)+int(key[count])
            count+=1
            if count == keylen:
                im.putpixel((w,h),(a,b,c))
                break
            if count % 3 == 0:
                im.putpixel((w,h),(a,b,c))
    im.save(str3)


if __name__ == '__main__':
    str1 = sys.argv[1]
    str2 = sys.argv[2]
    str3 = sys.argv[3]
    encode(str1,str2,str3)

来自于 https://blog.csdn.net/qq_26090065/article/details/82469266

反隐写

由于对方 LSB 隐写时很可能会发扬创新精神,进行个人定制,所以不推荐通过脚本进行反隐写,我们可以用 Stegsolve 去寻找这种 LSB 隐藏痕迹,通过下方的按钮观察每个通道的信息进行分析,捕捉异常点,抓住 LSB 隐写的蛛丝马迹(这玩意儿很难说,一般就是一看就感觉奇怪的n行或n列颜色块),进而利用 Stegsolve --> Analyse --> Data Extract 功能指定通道,点击 Preview 预览,Save Bin 进行提取。或者我们也可以使用 **zsteg **工具直接进行自动化的识别和提取。

其实这玩意儿意义并不大,主要用于双方不能传输文字,只能传输图片并且有第三方并不严格的审查下才算有点意义。(反正我是不会用的)

二. IDAT 隐写(PNG)

IDAT(图像数据块)
它存储实际的数据,在数据流中可包含多个 连续顺序图像数据块。它采用 LZ77 算法的派生算法进行压缩,可以用 zlib 解压缩。IDAT 块只有当上一个块充满(正常length最大65524)时,才会继续一个新的块。

根据 IDAT 块的定义与性质,我们有隐写方式如下:

人工隐写

由于 IDAT 块必须要与其他的 IDAT 块连续,我们需要找到原来的最后一个 IDAT 块,然后在它的 CRC 块之后加上我们自己的 IDAT 块,注意,我们加上的 IDAT 块必须完整(包含4字节的“数据块长度”(n),4字节的名称“IDAT”,n字节的数据,4字节的 CRC 块)。为了更好的隐写,我们最好将名称块+数据块的 CRC 计算出来,写到 CRC 块中,防止某些图像查看器报错。

虽然 IDAT 里存储的照片数据总是压缩过的,但是我们偷偷放进去的数据可以不压缩。

具体的照片就不贴了,隐写时有问题可以直接问我。

脚本隐写

脚本的话主要是分为两个部分:计算 CRC 和 IDAT 块写入。

# author : CHTXRT
# use : .\[脚本名] [要写入的图片] [要隐藏的数据] 
import zlib
import struct
import sys
import binascii

filename = sys.argv[1]
with open(filename, 'rb+') as f:
    # 读取数据 计算 CRC 整理成 IDAT 块的形式
    data2 = bytearray(open(sys.argv[2],'rb').read())
    data2_len = len(data2)
    data2_crc = zlib.crc32(b'IDAT'+data2)
    data2 = data2_len.to_bytes(4,byteorder='big',signed=False) + b'IDAT' + data2 + data2_crc.to_bytes(4,byteorder='big',signed=False)
    #新 IDAT 块写入
    all_b = f.read()
    data = bytearray(all_b)
    lastidat = data.rfind(b'IDAT')
    data = data[0:lastidat + 8 + int.from_bytes(data[lastidat-4:lastidat],byteorder='big',signed=False)] + data2 + data[lastidat + 8 + int.from_bytes(data[lastidat-4:lastidat],byteorder='big',signed=False):len(data)]
    f.write(data)
    exit(0)

刚学 Python ,代码写的有点烂,好像还有点问题,我用编辑器肉眼看着写的挺好,但是 pngcheck 报错说 IEND 后面有块?但是用编辑器打开看 文件尾后面是没有问题的。而且下面的反隐写对我的代码无效,因为好像压根识别不出来我写进去的 IDAT 块。(好像莫名其妙提高了隐蔽性)

反隐写

IDAT 块只有当上一个块充满(正常length最大65524)时,才会继续一个新的块。程序读取图像的时候也会在第一个未满的块停止(查了下W3C标准,其实是PNG图片在压缩的时候会在最后一个块的标记位标明这是最后一个数据块)。所以如果某一块没有满但后面却还有 IDAT 块则说明后面的块是“假”的。

我们可以用 pngcheck -v [文件名] 去查看PNG文件数据块信息,然后利用 python zlib 解压多余IDAT块的内容,此时注意剔除长度数据块类型及末尾的CRC校验值(如果压缩过的话)。你可以先 binwalk 提取一下多的块,看看是不是 zlib。

import zlib
import binascii
IDAT = " ".decode('hex')	#双引号中填IDAT数据
result = binascii.hexlify(zlib.decompress(IDAT))
print(result)

如果碰到像我一样懒得压缩的,还是直接结合上期用肉眼观察法提取吧。

本期就先说到这里,针对于如何进行信息隐写,主要讲了讲隐写术中的直接附加和部分 PNG 图片隐写方法,写的不太好的地方还请包涵并提出您宝贵的建议,我们下期再见。

参考资料

[1] 从0开始学杂项 第三期:隐写分析(2) PNG 图片隐写 :https://blog.csdn.net/CHTXRT/article/details/128714931

以上内容仅供参考,水平不高,大佬见笑。

作者:CHTXRT

出处:https://blog.csdn.net/CHTXRT

本文使用「CC BY-ND 4.0」创作共享协议,转载请在文章明显位置注明作者及出处。

  • 6
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: libpng错误:idat:crc错误 这个错误通常是由于PNG图像文件的IDAT数据块中的CRC校验和不正确导致的。这可能是由于文件损坏或传输错误引起的。要解决此问题,您可以尝试重新下载或获取PNG文件,并确保它没有损坏。如果问题仍然存在,则可能需要使用其他工具或库来处理PNG文件。 ### 回答2: 这个错误是libpng库中的一个错误,它通常出现在使用libpng库时压缩IDAT数据块时发生的校验和错误。在读取PNG文件和处理PNG文件时,libpng库是一个非常常用的开源库。在使用libpng库时,我们可能会遇到各种各样的错误,其中“libpng error: idat: crc error”的错误就是其中之一。 crc error指的是校验和出错,即数据在转移过程中发生了损坏或错误。这个idat指的是PNG文件中的一种关键数据块类型,它存储了PNG文件中图像的像素数据。由于压缩过程中出错导致部分数据被损坏,因此就会出现idat数据块的crc错误。 解决这个错误需要查明是何种因素导致的校验和错误,可能是读取PNG文件本身时被损坏或在处理过程中进行了错误的修改操作。在处理PNG文件时,我们需要确保对PNG文件的读取和处理是正确的,以避免这种错误的发生。在处理PNG文件时,我们可以使用一些PNG图像处理工具或者代码库,如libpng库等,来进行处理操作。 另外,如果我们在处理PNG文件时出现了这个错误,我们需要检查一下PNG文件本身,以确保它没有损坏或者出现了其他的错误。在PNG文件传输或处理过程中,可能由于网络问题或错误的文件传输导致文件损坏,因此也需要检查文件本身是否正常。如果发现PNG文件本身存在问题,我们可以尝试重新下载或者重新生成一个更完整的PNG文件。 ### 回答3: libpng是一个免费的开源软件库,用于处理PNG格式的图像文件。当在使用libpng处理图像文件时出现“libpng error: idat: crc error”错误,这意味着文件中的IDAT(图像数据)块的CRC(循环冗余校验)值未能通过验证。 CRC校验是一种用于检测和处理数据传输错误的技术。在图像文件中,每个IDAT块包括图像的一部分数据,然后计算这些数据的CRC值。在解码文件时,libpng会计算CRC值并与文件中存储的值进行比较,以确保数据未被修改或破坏。如果CRC值与存储值不匹配,则会出现“libpng error: idat: crc error”的错误。 出现这种错误通常是因为在文件传输过程中发生了错误,例如文件损坏或传输中断。还可能是因为编码器或解码器中存在错误。解决这类问题的方法通常是确保文件完整性和正确性,并检查软件或环境中是否存在任何错误或问题。解决方案可能包括使用不同的软件或重新编码文件。 总之,“libpng error:idat:crc error”错误表明libpng无法验证文件的IDAT块的CRC值,因此需要检查文件完整性和环境中是否存在任何错误或问题以解决问题。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值