kernel hexdump分析

浏览:
字体:
发布时间:2013-12-09 23:24:02
来源:

驱动调试中,很多时候是二进制的,这个时候hexdump就是个非常有用的工具了。
不要再自己去实现类似的功能,kernel代码里面就有:
参考: kernel/lib/hexdump.c
// 0Xxx转换成 ASCII ,实现简单巧妙

const char hex_asc[] = "0123456789abcdef";#define hex_asc_lo(x)	hex_asc[((x) & 0x0f)]#define hex_asc_hi(x)	hex_asc[((x) & 0xf0) >> 4]


void hex_dump_to_buffer(const void *buf, size_t len, int rowsize,			int groupsize, char *linebuf, size_t linebuflen,			bool ascii){	const u8 *ptr = buf;	u8 ch;	int j, lx = 0;	int ascii_column;        // 每行固定16或者32个字节,符合国际审美观	if (rowsize != 16 && rowsize != 32)		rowsize = 16;	if (!len)		goto nil;	if (len > rowsize)		/* limit to one line at a time */		len = rowsize;	if ((len % groupsize) != 0)	/* no mixed size output */		groupsize = 1;	switch (groupsize) {	......	default:                //  转换成ASCii 形式		for (j = 0; (j < len) && (lx + 3) <= linebuflen; j++) {			ch = ptr[j];			linebuf[lx++] = hex_asc_hi(ch);			linebuf[lx++] = hex_asc_lo(ch);			linebuf[lx++] = ' ';		}		if (j)			lx--;		ascii_column = 3 * rowsize + 2;		break;	}	if (!ascii)		goto nil;        // 加上对应的ASCII字符串,区别如下:        //  0009ab42: 40 41 42 43 44 45 46 47 48 49 4a 4b 4c 4d 4e 4f  @ABCDEFGHIJKLMNO        //  0009ab42: 40 41 42 43 44 45 46 47 48 49 4a 4b 4c 4d 4e 4f  	while (lx < (linebuflen - 1) && lx < (ascii_column - 1))		linebuf[lx++] = ' ';	for (j = 0; (j < len) && (lx + 2) < linebuflen; j++) {		ch = ptr[j];		linebuf[lx++] = (isascii(ch) && isprint(ch)) ? ch : '.';	}nil:        // 记得加上结束符号,有可能导致printk --> kernel panic 	linebuf[lx++] = '/0';}


void print_hex_dump(const char *level, const char *prefix_str, int prefix_type,		    int rowsize, int groupsize,		    const void *buf, size_t len, bool ascii){	const u8 *ptr = buf;	int i, linelen, remaining = len;        /* 每行数据类似下面           40 41 42 43 44 45 46 47 48 49 4a 4b 4c 4d 4e 4f  @ABCDEFGHIJKLMNO           |                                          |  |                 |           -----------------------------------------------                    16*3 or 32*3                       2      32+1         */	unsigned char linebuf[32 * 3 + 2 + 32 + 1];	if (rowsize != 16 && rowsize != 32)		rowsize = 16;	for (i = 0; i < len; i += rowsize) {		linelen = min(remaining, rowsize);		remaining -= rowsize;                //  linebuf 返回需要打印的字符串		hex_dump_to_buffer(ptr + i, linelen, rowsize, groupsize,				   linebuf, sizeof(linebuf), ascii);                                 // 是否使用前缀		switch (prefix_type) {                     // 加上绝对地址		case DUMP_PREFIX_ADDRESS:			printk("%s%s%p: %s/n",			       level, prefix_str, ptr + i, linebuf);			break;                     // 只是简单地加上偏移量而已		case DUMP_PREFIX_OFFSET:			printk("%s%s%.8x: %s/n", level, prefix_str, i, linebuf);			break;		default:			printk("%s%s%s/n", level, prefix_str, linebuf);			break;		}	}}


// 一般在kernel代码中,调用此接口,许多参数用默认的,不需要了解太多细节void print_hex_dump_bytes(const char *prefix_str, int prefix_type,			  const void *buf, size_t len){	print_hex_dump(KERN_DEBUG, prefix_str, prefix_type, 16, 1,		       buf, len, true);}


>更多相关文章
24小时热门资讯
24小时回复排行
资讯 | QQ | 安全 | 编程 | 数据库 | 系统 | 网络 | 考试 | 站长 | 关于东联 | 安全雇佣 | 搞笑视频大全 | 微信学院 | 视频课程 |
关于我们 | 联系我们 | 广告服务 | 免责申明 | 作品发布 | 网站地图 | 官方微博 | 技术培训
Copyright © 2007 - 2024 Vm888.Com. All Rights Reserved
粤公网安备 44060402001498号 粤ICP备19097316号 请遵循相关法律法规
');})();