设为首页 收藏本站
查看: 1360|回复: 0

[经验分享] linux驱动调试--段错误之oops信息分析

[复制链接]

尚未签到

发表于 2015-12-9 15:36:04 | 显示全部楼层 |阅读模式
发生段错误原因就是访问了不该访问的地址,例如访问了不存在的内存地址、访问了系统保护的内存地址、访问了只读的内存地址等。
下面根据Oops信息来分析一下段错误

first_drv.c



  • #include

  • #include
  • #include
  • #include
  • #include
  • #include
  • #include
  • #include
  • #include

  • static struct class *segment_class;
  • static struct device *segment_class_dev;
  • unsigned char *c = NULL;
  • int major;

  • static int segment_test_open(struct inode *inode, struct file *file)
  • {
  •     *c = 0x34;
  •     //printk("segment_test_open success!\n");
  •     return 0;
  • }

  • static struct file_operations segment_test_fops = {
  •     .owner = THIS_MODULE, /* 这是一个宏,推向编译模块时自动创建的__this_module变量 */
  •     .open = segment_test_open,
  • };

  • static int segment_drv_init(void)
  • {
  •     major = register_chrdev(0, "segment_test", &segment_test_fops); // 注册, 告诉内核
  •     segment_class = class_create(THIS_MODULE, "segment_test");
  •     segment_class_dev = device_create(segment_class, NULL, MKDEV(major, 0), NULL, "segment");

  •     c = (unsigned char *)0x48000000;

  •     printk("segment_drv_init success!\n");

  •     return 0;
  • }

  • static void segment_drv_exit(void)
  • {
  •     device_destroy(segment_class, MKDEV(major,0));
  •     class_destroy(segment_class);

  •     unregister_chrdev(major, "segment_test");
  • }

  • module_init(segment_drv_init);
  • module_exit(segment_drv_exit);


  • MODULE_LICENSE("GPL");

firstdrvtest.c


#include
#include
#include
#include

int main(int argc, char **argv)
{
    int fd;
    int val = 1;
    fd = open("/dev/segment", O_RDWR);
    if (fd first_drv.dis
打开first_drv.dis有下面这一段:

00000000 :
   0:   e1a0c00d        mov     ip, sp
   4:   e92dd800        push    {fp, ip, lr, pc}
   8:   e24cb004        sub     fp, ip, #4      ; 0x4
   c:   e59f3010        ldr     r3, [pc, #16]   ; 24
  10:   e3a00000        mov     r0, #0  ; 0x0
  14:   e5932000        ldr     r2, [r3]
  18:   e3a03034        mov     r3, #52 ; 0x34
  1c:   e5c23000        strb    r3, [r2]
  20:   e89da800        ldm     sp, {fp, sp, pc}
  24:   00000000        .word   0x00000000

这里代码的实际地址都要加上偏移地址 bf0d7000,发生错误的那句代码是:
1c:   e5c23000        strb    r3, [r2]
根据我们的C语言代码可以看出这里是把0x34赋给变量时产生错误,产生错误的原因是加载模块初始化时赋给的一个地址非法:
c = (unsigned char *)0x48000000;

我们这里的程序比较短,可以一眼看出来,如果代码很长,就可以根据发生错误的位置,大概确定代码的位置,
然后再去看代码和汇编,这里要求比较高的汇编阅读能力


2. 如果发生的错误函数是属于内核的

这个时候和发生在模块里类似,不过这里要反汇编整个内核:
# arm-none-linux-gnueabi-objdump -D vmlinux > vmlinux.dis
打开vmlinux.dis,然后直接查找地址bf0d7000,接下来像上面一样分析代码

关于上面的打印信息,还有栈那一段没有讲,放到下一篇博文说




运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.iyunv.com/thread-148866-1-1.html 上篇帖子: linux驱动的模块依赖问题 下篇帖子: RedHat Linux 安装oracle11g (转)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表