• 聊聊、Nginx 初始化错误信息


      这篇文章我们继续学习 main 方法,我们先来看看 ngx_debug_init() 这个方法。  

      从方法名我们也知道,debug初始化。我们先看看方法位置在哪。我们来断点在这个方法上面。 

      Function “ngx_debug_init” not defined。 我们去源码里面查找,grep “ngx_debug_init” -r *。 

      通过搜索结果大家应该知道,这个函数主要用于跨操作系统的东西。在 Linux 系统中应该是 src/os/unix/ngx_linux_config.h:#define ngx_debug_init()。 仅仅是一个宏定义,内容为空。大家可以去src/os/unix/ngx_linux_config.h 中看看。也就是说对于 Linux 平台,这个函数没有意义。所以 gdb 中运行 Nginx,直接就跳到了ngx_strerror_init() 方法。 那我们就来看看 ngx_strerror_init() 方法的作用。我们还是从启动 Nginx 开始,gdb nginx,b main,r,然后输入 s 命令。 

      ok,ngx_strerror_init () at src/os/unix/ngx_errno.c:58。去58行看看。 

      这就是 ngx_strerror_init 方法体了。前面的变量我们就不看了,直接看 len = NGX_SYS_NERR * sizeof(ngx_str_t); 

      ngx_str_t 长度为 8,我的是 32 位系统。如果是 64 位系统长度就是 16 了。那么 NGX_SYS_NERR 是多少呢? 

      objs/ngx_auto_config.h:#define NGX_SYS_NERR  132。 也就是 NGX_SYS_NERR=132。是一个宏定义,132。 那么 len = NGX_SYS_NERR * sizeof(ngx_str_t); len 就是 132*8=1056 了。 接着看!这行计算,根本没在 CPU 上计算,而是在编译的时候已经算完了。ngx_sys_errlist = malloc(len);是申请一块堆上的内存。类似于 Java 中的 new。在 gdb 中输入命令 n ,next 意思,我们看看下一行代码。 

      我们看看 len 的长度是多少?输入 p len。 

      没错,1056。再看看 ngx_sys_errlist。static ngx_str_t  *ngx_sys_errlist;一个结构指针。输入 p ngx_sys_errlist。 

      (ngx_str_t *) 0x0 说明什么呢?也就是在没有运行 ngx_sys_errlist = malloc(len);之前,堆栈上面它的地址字节都为 0。解释下 0x0地址。操作系统和 CPU,一起做了一个设定,地址是 0x0 的那个地址,在应用程序的进程地址空间,是不可访问的。主要是为了程序员写指针初始化的时候用的。如果你要是现在用代码操作这个地址,0x0,就会引发常规的非法地址访问。操作系统就把你的进程直接杀掉了。 我们继续 next。 

    再打印 ngx_sys_errlist 看看。 

      已经分配了地址。接下来 if (ngx_sys_errlist == NULL) { goto failed; } 如分配失败,为 NULL,则去到 failed。大家自己看。 再接下来就是一个循环了。

      for (err = 0; err < NGX_SYS_NERR; err++) {        

      msg = strerror(err);        

      len = ngx_strlen(msg);

        p = malloc(len);
        if (p == NULL) {
            goto failed;
        }
        ngx_memcpy(p, msg, len);
        ngx_sys_errlist[err].len = len;
        ngx_sys_errlist[err].data = p;

    }

      先看看 msg = strerror(err); 这个方法。一看就不是 Nginx 自己的方法,应该是底层 C 的方法。在 Linux 控制台输入 man strerror 打印看看。 

      于是我们就知道 strerror() 函数返回一个指针,指向一个“字符串”,就是一个地址,里面有一些字符,以 0x0 结尾。继续 next。 

      当 err=0 的时候,错误信息为 Success。 继续 next 

      当 err=1 的时候,msg 为 Operation not permitted。大家可以自己试试,看看不同的 err 都是什么错误信息。 大家有没有发现,msg 信息最前面为什么会有奇怪的地址呢? $15 = 0x235e94 “Success”。其实这是个动态链接库,有兴趣的同学可以查一下动态链接库的地址空间。 接下来继续 ngx_memcpy(p, msg, len);然后做一个字符串复制。 ngx_sys_errlist[err].len = len; ngx_sys_errlist[err].data = p; 这个大家一看就知道,向 ngx_sys_errlist 中赋值。 到这里,我们可以看出来了,ngx_sys_errlist 就是一个结构数组。一共是 132 个。用来存放所有的系统基本错误信息。 好,下节课我们讲 ngx_get_options()。

  • 相关阅读:
    2015PPTV校招笔试详解
    2015华数校招笔试详解
    2015我的校招之路
    2015美团校招笔试试题
    关于联想超极本装入Ubuntu系统无法开启无线硬件开关的解决
    RabbitMQ
    office 2016 for mac 下拉框 无法展开问题
    mac 下 tomcat7的安装
    mac 10.11.6,Xcode8下,ruby2.3安装,Cocoapods安装~
    学习的学习的方法
  • 原文地址:https://www.cnblogs.com/xums/p/6741097.html
Copyright © 2020-2023  润新知