美文网首页区块链研习社
比特币源码研读2-程序入口函数解析(1)

比特币源码研读2-程序入口函数解析(1)

作者: Jacky_2c9f | 来源:发表于2017-09-12 22:45 被阅读0次

    在第一讲中,我们使用了which bitcoind和which bitcoin-cli两个命令来验证比特币客户端是否安装成功。那么我们就可以猜测会有个程序入口与bitcoind相关。

    果不其然,在src目录下有个bitcoind.cpp文件。

    从该文件中可以找到程序入口的main函数。

    int main(int argc, char* argv[])

    {

    SetupEnvironment();

    // Connect bitcoind signal handlers

    noui_connect();

    return (AppInit(argc, argv) ? EXIT_SUCCESS : EXIT_FAILURE);

    }

    这一小节主要讲解SetupEnvironment(), 该函数是在src/util.cpp文件中定义的函数。

    其作用有以下三个方面:

    1.内存分配区设置

    #ifdef HAVE_MALLOPT_ARENA_MAX

    // glibc-specific: On 32-bit systems set the number of arenas to 1.

    // By default, since glibc 2.10, the C library will create up to two heap

    // arenas per core. This is known to cause excessive virtual address space

    // usage in our usage. Work around it by setting the maximum number of

    // arenas to 1.

    if (sizeof(void*) == 4) {

    mallopt(M_ARENA_MAX, 1);

    }

    这里判断如果是32位系统(sizeof(void*) == 4),即只分配一个arena内存区,防止虚拟地址空间过度使用。自从glibc 2.10版本后,针对每个核心,C标准库都会默认创建两个堆内存区。这是公认会导致内存地址空间过度使用的问题。

    科普:arena:可以理解为一个较大且连续的内存分配区,需要手动来管理。

    2.本地化设置

    #endif

    // On most POSIX systems (e.g. Linux, but not BSD) the environment's locale

    // may be invalid, in which case the "C" locale is used as fallback.

    #if !defined(WIN32) && !defined(MAC_OSX) && !defined(__FreeBSD__) && !defined(__OpenBSD__)

    try {

    std::locale(""); // Raises a runtime error if current locale is invalid

    } catch (const std::runtime_error&) {

    setenv("LC_ALL", "C", 1);

    }

    在大部分POSIX系统(比如Linux,但非BSD),环境的locale系统区域设置可能会无效,假如当前的本地化设置无效,将抛出该运行时异常,同时在捕获后将该值设置为“C”。

    Tips:

     locale决定当前程序所使用的语言编码、日期格式、数字格式及其它与区域有关的设置,详情参考:http://blog.csdn.net/haiross/article/details/45074355

    C:缺省值:ANSI-C conventions,C语言标准规范、约定。比如函数命名规范等,由美国国家标准学会发布。

    POSIX:The Portable Operating System Interface。便携式操作系统接口(POSIX)是由IEEE计算机学会指定的用于维护操作系统之间的兼容性的一系列标准。

    关于函数setenv(const char *name, const char *value, int overwrite),作用是修改或增加一个环境变量,详情参考https://linux.die.net/man/3/setenv

    3.本地化文件路径设置

    #endif

    // The path locale is lazy initialized and to avoid deinitialization errors

    // in multithreading environments, it is set explicitly by the main thread.

    // A dummy locale is used to extract the internal default locale, used by

    // fs::path, which is then used to explicitly imbue the path.

    std::locale loc = fs::path::imbue(std::locale::classic());

    fs::path::imbue(loc);

    路径区域设置是采用懒初始化,即延迟初始化的方式,并且可以避免多线程环境下的析构过程错误,通过主线程来显式设置。一个dummy locale可以用来提取fs::path使用的内部默认locale(路径),然后就可以显式设置该路径。

    imbue:该函数将要设置的loc值与流和该流相关的流缓冲区(如果有的话)关联起来,作为新的系统区域设置对象。详情参考:http://www.cplusplus.com/reference/ios/ios/imbue/

    这里设置的是文件系统路径的本地化。

    下一小节将讲解比特币源码是如何处理信号连接的。

    区块链研习社源码研读班 Jacky

    相关文章

      网友评论

        本文标题:比特币源码研读2-程序入口函数解析(1)

        本文链接:https://www.haomeiwen.com/subject/mwuhsxtx.html