Ubuntu 下 nginx-1.24.0 源码分析 - ngx_strerror_init()函数

server/2025/2/3 16:35:18/

目录

ngx_strerror_init()函数声明

ngx_int_t 类型声明定义 

intptr_t 类型 

ngx_strerror_init()函数实现

NGX_HAVE_STRERRORDESC_NP


ngx_strerror_init()函数声明

在 nginx.c 的开头引入了:

#include <ngx_core.h>

在 ngx_core.h 中引入了

#include <ngx_errno.h>

在 ngx_errno.h 这个文件中声明了 ngx_strerror_init()函数:

ngx_int_t ngx_strerror_init(void);

ngx_int_t 类型声明定义 

在ngx_config.h 中:

typedef intptr_t        ngx_int_t;

ngx_int_t 本质上是 intptr_t 类型 


intptr_t 类型 

intptr_t 是一种整数类型,它保证可以容纳指针的值,帮助我们安全地在指针和整数之间进行转换

在不同的系统和编译器中,指针的大小可能不同(比如 32 位系统和 64 位系统)

intptr_t 确保在这些系统上都能正确工作。

在我当前的Ubuntu环境下也可以通过引入

#include <unistd.h>

来使用 intptr_t 类型

在 ngx_linux_config.h 中:

#include <unistd.h>

ngx_strerror_init()函数实现

ngx_errno.c 中:

#if (NGX_HAVE_STRERRORDESC_NP)/** The strerrordesc_np() function, introduced in glibc 2.32, is* async-signal-safe.  This makes it possible to use it directly,* without copying error messages.*/u_char *
ngx_strerror(ngx_err_t err, u_char *errstr, size_t size)
{size_t       len;const char  *msg;msg = strerrordesc_np(err);if (msg == NULL) {msg = (char *) ngx_unknown_error.data;len = ngx_unknown_error.len;} else {len = ngx_strlen(msg);}size = ngx_min(size, len);return ngx_cpymem(errstr, msg, size);
}ngx_int_t
ngx_strerror_init(void)
{return NGX_OK;
}#else/** The strerror() messages are copied because:** 1) strerror() and strerror_r() functions are not Async-Signal-Safe,*    therefore, they cannot be used in signal handlers;** 2) a direct sys_errlist[] array may be used instead of these functions,*    but Linux linker warns about its usage:** warning: `sys_errlist' is deprecated; use `strerror' or `strerror_r' instead* warning: `sys_nerr' is deprecated; use `strerror' or `strerror_r' instead**    causing false bug reports.*/static ngx_str_t  *ngx_sys_errlist;
static ngx_err_t   ngx_first_error;
static ngx_err_t   ngx_last_error;u_char *
ngx_strerror(ngx_err_t err, u_char *errstr, size_t size)
{ngx_str_t  *msg;if (err >= ngx_first_error && err < ngx_last_error) {msg = &ngx_sys_errlist[err - ngx_first_error];} else {msg = &ngx_unknown_error;}size = ngx_min(size, msg->len);return ngx_cpymem(errstr, msg->data, size);
}ngx_int_t
ngx_strerror_init(void)
{char       *msg;u_char     *p;size_t      len;ngx_err_t   err;#if (NGX_SYS_NERR)ngx_first_error = 0;ngx_last_error = NGX_SYS_NERR;#elif (EPERM > 1000 && EPERM < 0x7fffffff - 1000)/** If number of errors is not known, and EPERM error code has large* but reasonable value, guess possible error codes based on the error* messages returned by strerror(), starting from EPERM.  Notably,* this covers GNU/Hurd, where errors start at 0x40000001.*/for (err = EPERM; err > EPERM - 1000; err--) {ngx_set_errno(0);msg = strerror(err);if (errno == EINVAL|| msg == NULL|| strncmp(msg, "Unknown error", 13) == 0){continue;}ngx_first_error = err;}for (err = EPERM; err < EPERM + 1000; err++) {ngx_set_errno(0);msg = strerror(err);if (errno == EINVAL|| msg == NULL|| strncmp(msg, "Unknown error", 13) == 0){continue;}ngx_last_error = err + 1;}#else/** If number of errors is not known, guess it based on the error* messages returned by strerror().*/ngx_first_error = 0;for (err = 0; err < 1000; err++) {ngx_set_errno(0);msg = strerror(err);if (errno == EINVAL|| msg == NULL|| strncmp(msg, "Unknown error", 13) == 0){continue;}ngx_last_error = err + 1;}#endif/** ngx_strerror() is not ready to work at this stage, therefore,* malloc() is used and possible errors are logged using strerror().*/len = (ngx_last_error - ngx_first_error) * sizeof(ngx_str_t);ngx_sys_errlist = malloc(len);if (ngx_sys_errlist == NULL) {goto failed;}for (err = ngx_first_error; err < ngx_last_error; err++) {msg = strerror(err);if (msg == NULL) {ngx_sys_errlist[err - ngx_first_error] = ngx_unknown_error;continue;}len = ngx_strlen(msg);p = malloc(len);if (p == NULL) {goto failed;}ngx_memcpy(p, msg, len);ngx_sys_errlist[err - ngx_first_error].len = len;ngx_sys_errlist[err - ngx_first_error].data = p;}return NGX_OK;failed:err = errno;ngx_log_stderr(0, "malloc(%uz) failed (%d: %s)", len, err, strerror(err));return NGX_ERROR;
}#endif

这里的 ngx_strerror_init 函数的定义有 2 个

具体使用的是哪一个

这要取决于

#if (NGX_HAVE_STRERRORDESC_NP)

 

NGX_HAVE_STRERRORDESC_NP

#if (NGX_HAVE_STRERRORDESC_NP) 是一个条件编译指令,

用于检查是否支持 strerrordesc_np 函数。

strerrordesc_np 是一个 GNU 扩展函数,

定义在 <string.h> 中。

它用于返回一个描述错误码的字符串,

strerror 不同的是,

strerrordesc_np 返回的描述不会根据当前的区域设置进行翻译

NGX_HAVE_STRERRORDESC_NP 用于判断当前系统是否支持 strerrordesc_np 函数。

如果支持,则在代码中会启用与该函数相关的功能,

例如直接调用 strerrordesc_np 来获取错误描述。

由于 strerrordesc_np 是 GNU 扩展,不是所有系统都支持。

Nginx 通过配置脚本检测系统是否支持该函数,

并在支持的情况下定义 NGX_HAVE_STRERRORDESC_NP

如果系统不支持 strerrordesc_np

Nginx 可能会使用其他方式(如 strerror)来获取错误描述。


ngx_errno.c 的开头引入了

#include <ngx_config.h>

ngx_config.h 中引入了

ngx_linux_config.h 这个头文件

在 ngx_linux_config.h 中引入了

#include <ngx_auto_config.h>

在 objs/ngx_auto_config.h  中:

#ifndef NGX_HAVE_STRERRORDESC_NP
#define NGX_HAVE_STRERRORDESC_NP  1
#endif

这里定义了 NGX_HAVE_STRERRORDESC_NP

 这个宏为 1


回到 ngx_errno.c 中 对于

#if (NGX_HAVE_STRERRORDESC_NP)

这个条件成立

于是使用的 ngx_strerror_init 函数的定义就是

ngx_int_t
ngx_strerror_init(void)
{return NGX_OK;
}

这里 NGX_OK 这个宏的定义在哪里呢?

在 ngx_errno.c 的开头引入了

#include <ngx_core.h>

打开 ngx_core.h 可以找到这样一行代码

#define  NGX_OK          0


http://www.ppmy.cn/server/164658.html

相关文章

Windsurf cursor vscode+cline 与Python快速开发指南

Windsurf简介 Windsurf是由Codeium推出的全球首个基于AI Flow范式的智能IDE&#xff0c;它通过强大的AI助手功能&#xff0c;显著提升开发效率。Windsurf集成了先进的代码补全、智能重构、代码生成等功能&#xff0c;特别适合Python开发者使用。 Python环境配置 1. Conda安装…

https数字签名手动验签

以bing.com 为例 1. CA 层级的基本概念 CA 层级是一种树状结构&#xff0c;由多个层级的 CA 组成。每个 CA 负责为其下一层级的实体&#xff08;如子 CA 或终端实体&#xff09;颁发证书。层级结构的顶端是 根 CA&#xff08;Root CA&#xff09;&#xff0c;它是整个 PKI 体…

K8S ReplicaSet 控制器

一、理论介绍 今天我们来实验 ReplicaSet 控制器&#xff08;也叫工作负载&#xff09;。官网描述如下&#xff1a; 1、是什么&#xff1f; ReplicaSet 副本集&#xff0c; 维护一组稳定的副本 Pod 集合。 2、为什么需要&#xff1f; 解决 pod 被删除了&#xff0c;不能自我恢…

Android 13 取色引擎详解

1、 应用如何获取壁纸颜色&#xff1f; 1.1、调用getWallpaperColors获取壁纸颜色 其实WallpaperManager从很早之前就提供了getWallpaperColors接口 /frameworks/base/core/java/android/app/WallpaperManager.java ...UnsupportedAppUsagepublic Nullable WallpaperColors …

PHP Composer:高效依赖管理工具详解

PHP Composer:高效依赖管理工具详解 引言 在PHP开发领域,依赖管理是项目构建过程中的重要环节。Composer的出现,极大地简化了PHP项目的依赖管理,使得开发者可以更加高效地构建和维护PHP应用程序。本文将深入探讨PHP Composer的使用方法、功能特点以及它在项目开发中的应用…

ubuntu 下使用deepseek

安装Ollama sudo snap install ollama 执行 ollama run deepseek-coder 然后进行等待。。。

【含文档+PPT+源码】基于微信小程序的校园快递平台

项目介绍 本课程演示的是一款基于微信小程序的校园快递平台&#xff0c;主要针对计算机相关专业的正在做毕设的学生与需要项目实战练习的 Java 学习者。 1.包含&#xff1a;项目源码、项目文档、数据库脚本、软件工具等所有资料 2.带你从零开始部署运行本套系统 3.该项目附带…

< OS 有关 > 阿里云:轻量应用服务器 的使用 :轻量化 阿里云 vpm 主机

原因&#xff1a; &#xff1c; OS 有关 &#xff1e; 阿里云&#xff1a;轻量应用服务器 的使用 &#xff1a;从新开始 配置 SSH 主机名 DNS Tailscale 更新OS安装包 最主要是 清除阿里云客户端这个性能杀手-CSDN博客 防止 I/O 祸害系统 操作&#xff1a; 查看进程&#x…