复制 dom 然后替换节点(xmlCopyDoc 和 xmlReplaceNode)时内存泄漏

Memory leak when copying dom and then replacing node (xmlCopyDoc and xmlReplaceNode)

提问人:JoseleMG 提问时间:8/8/2023 最后编辑:Paul FloydJoseleMG 更新时间:8/9/2023 访问量:44

问:

我正在构建一个程序,该程序使用 libxml2 v2.11 在两个 xml 文件之间进行合并验证。

总之,我正在创建一个目标 dom 的副本,然后将所有节点替换为替换 dom 的节点,当目标节点具有等于 true 的属性“overloadable”时。最后,我根据模式和 schematron 文件验证结果 dom。

目前,这段代码可以做到这一点(对不起,如果它很乱,我一直在调试它):

/**
 * replace_nodes:
 * @target_node: A node to be replaced
 * @replace_node: A node to copy
 * @Output:    0 on Success or error code.
 *
 * Replace target_node with a copy of replace_node. This function changes the dom tree.
 */
static xmlNodePtr replace_node(xmlNodePtr target_node, xmlNodePtr replace_node){
    xmlNodePtr new_target_node = xmlCopyNode(replace_node,1);
    xmlSetProp(new_target_node, (xmlChar *) "overloadable", (xmlChar *) "true");
    xmlReplaceNode (target_node, new_target_node);
    DEV_DEBUG_PRINT("Replaced node %s \n", replace_node->name);
    return new_target_node;
}

int validate_overloadables_merge(xmlDocPtr target_dom, xmlDocPtr replace_dom){
    struct xpath_exp {
        char target_xpath_exp[100];
        char replace_xpath_exp[100];
    };
    struct xpath_exp find_xpr_tb =
        {
            .target_xpath_exp = "/configuration",
            .replace_xpath_exp = "/configurations/replacement/*",
        };
    int ret = 0;

    // copy target dom
    xmlDocPtr copy_dom = xmlCopyDoc(target_dom, 1);
    xmlChar xpath_expr_replace[STR_LEN], xpath_expr_target[STR_LEN];
    xmlXPathObjectPtr xpath_obj_target, xpath_obj_replace;
    strcpy((char *)xpath_expr_target, find_xpr_tb.target_xpath_exp);
    strcpy((char *)xpath_expr_replace, find_xpr_tb.replace_xpath_exp);

    // Get all inputs from target dom with an xpath expr
    int ret_target = query_dom(xpath_expr_target, &xpath_obj_target, copy_dom);
    // Get all inputs from replacement dom with an xpath expr
    int ret_replace = query_dom(xpath_expr_replace, &xpath_obj_replace, replace_dom);

    // Error handling in case there is not matches
    if(ret_target == -EACCES  || ret_replace == -EACCES)
    {
        ERR_MSG("Error: Fail to query\n");
        if(ret_target != -EACCES)
        {
            xmlXPathFreeObject(xpath_obj_target);
        }
        xmlFreeDoc(copy_dom);
        return -EACCES;
    }
    else if (ret_target == -ESRCH || ret_replace == -ESRCH ||
            xpath_obj_target->nodesetval->nodeNr < 1 ||
            xpath_obj_replace->nodesetval->nodeNr < 1)
    {
        DEV_DEBUG_PRINT("DEBUG_INFO: No replaceomization nodes found\n");
        xmlXPathFreeObject(xpath_obj_target);
        xmlXPathFreeObject(xpath_obj_replace);
        xmlFreeDoc(copy_dom);
    }

    // Get first node to potentially replace it
    xmlNodePtr target_node =  xpath_obj_target->nodesetval->nodeTab[0];
    while(target_node)
    {
        // Get next sibling to potentially replace it. This is done here as we might free the target node
        xmlNodePtr nx_target_node = target_node->next;
        char * overloadable = (char *) xmlGetProp(target_node, (xmlChar *) "overloadable");
        //If overloadable, we replace it
        if(overloadable != NULL && 0==strcmp(overloadable,"true"))
        {
            for(int replace_idx=0; replace_idx < xpath_obj_replace->nodesetval->nodeNr; replace_idx++){
                xmlNodePtr replace_node = xpath_obj_replace->nodesetval->nodeTab[replace_idx];
                if(0==strcmp((char *)target_node->name,
                             (char *)replace_node->name))
                {
                    xmlNodePtr new_target_node = replace_node(target_node, replace_node);
                    //xmlFreeNode(target_node); 
                    xmlReconciliateNs(copy_dom, new_target_node);
                }
            }
        }
        if(overloadable)
        {
            xmlFree(overloadable);
        }        
        target_node = nx_target_node;
    }
    if( is_conf_sct && ret >= 0){
        DEV_DEBUG_PRINT("DEBUG_INFO: Validating copy sct\n");
        ret = xml_validate_sct(copy_dom, conf_sct_path);
    }
    if(is_conf_xsd && ret >= 0){
        DEV_DEBUG_PRINT("DEBUG_INFO: Validating copy xsd\n");
        ret = xml_validate_xsd(copy_dom, conf_xsd_path);
    }
    xmlXPathFreeObject(xpath_obj_target);
    xmlXPathFreeObject(xpath_obj_replace);
    xmlFreeDoc(copy_dom);
    return ret;
}

当我通过 valgrind 来识别内存泄漏时,问题就来了:

jmgs@esswlab3:~/Source/config$ valgrind  --leak-check=full ./config inputs/config.xml -c inputs/custom.xml -x schemas/config.xsd  -s schematrons/config.sct 
==96255== Memcheck, a memory error detector
==96255== Copyright (C) 2002-2017, and GNU GPL'd, by Julian Seward et al.
==96255== Using Valgrind-3.18.1 and LibVEX; rerun with -h for copyright info
==96255== Command: ./config inputs/config.xml -c inputs/custom.xml -x schemas/config.xsd  -s schematrons/config.sct 
==96255==
Document inputs/config.xml loaded
Document complient with schema
Document complient with schematron
Replaced node input_1
Replaced node input_4
Document complient with schematron
Document complient with schema
==96255==
==96255== HEAP SUMMARY:
==96255==     in use at exit: 3,343 bytes in 52 blocks
==96255==   total heap usage: 54,905 allocs, 54,853 frees, 5,176,922 bytes allocated
==96255==
==96255== 3,343 (240 direct, 3,103 indirect) bytes in 2 blocks are definitely lost in loss record 8 of 8
==96255==    at 0x4848899: malloc (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
==96255==    by 0x49B550F: xmlStaticCopyNode (tree.c:4311)
==96255==    by 0x49B569E: xmlStaticCopyNode (tree.c:4422)
==96255==    by 0x49B5BB3: xmlStaticCopyNodeList (tree.c:4495)
==96255==    by 0x49B60A0: xmlCopyDoc (tree.c:4716)
==96255==    by 0x10DFE8: validate_overloadables_merge (in /home/jmgs/Source/config)
==96255==    by 0x10D0DC: config_load_cust_to_shm (in  /home/jmgs/Source/config-manager)
==96255==    by 0x10FBFE: method_load_customization (in /home/jmgs/Source/config)
==96255==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96255==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96255==    by 0x111683: config_register_dbus (in  /home/jmgs/Source/config)
==96255==    by 0x10CD2B: main (in  /home/jmgs/Source/config)
==96255==
==96255== LEAK SUMMARY:
==96255==    definitely lost: 240 bytes in 2 blocks
==96255==    indirectly lost: 3,103 bytes in 50 blocks
==96255==      possibly lost: 0 bytes in 0 blocks
==96255==    still reachable: 0 bytes in 0 blocks
==96255==         suppressed: 0 bytes in 0 blocks
==96255==
==96255== For lists of detected and suppressed errors, rerun with: -s
==96255== ERROR SUMMARY: 1 errors from 1 contexts (suppressed: 0 from 0

我的猜测是内存泄漏来自更换target_node时没有释放它。但是,当我这样做时,我收到以下 valgrind 错误消息:

==96577== Memcheck, a memory error detector
==96577== Copyright (C) 2002-2017, and GNU GPL'd, by Julian Seward et al.
==96577== Using Valgrind-3.18.1 and LibVEX; rerun with -h for copyright info
==96577== Command: ./config inputs/config.xml -c inputs/custom.xml -x schemas/config.xsd  -s schematrons/config.sct
==96577==
Document inputs/config.xml loaded
Document complient with schema
Document complient with schematron
Document inputs/custom.xml loaded
Replaced node input_1
==96577== Invalid read of size 8
==96577==    at 0x10E229: validate_overloadables_merge (in /home/jmgs/Source/config)
==96577==    by 0x10D0DC: config_load_cust_to_shm (in /home/jmgs/Source/config)
==96577==    by 0x10FC0E: method_load_customization (in /home/jmgs/Source/config)
==96577==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x111693: config_register_dbus (in /home/jmgs/Source/config)
==96577==    by 0x10CD2B: main (in /home/jmgs/Source/config)
==96577==  Address 0x54811c0 is 16 bytes inside a block of size 120 free'd
==96577==    at 0x484B27F: free (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
==96577==    by 0x10E298: validate_overloadables_merge (in /home/jmgs/Source/config)
==96577==    by 0x10D0DC: config_load_cust_to_shm (in /home/jmgs/Source/config)
==96577==    by 0x10FC0E: method_load_customization (in /home/jmgs/Source/config)
==96577==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x111693: config_register_dbus (in /home/jmgs/Source/config)
==96577==    by 0x10CD2B: main (in /home/jmgs/Source/config)
==96577==  Block was alloc'd at
==96577==    at 0x4848899: malloc (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
==96577==    by 0x49B550F: xmlStaticCopyNode (tree.c:4311)
==96577==    by 0x49B569E: xmlStaticCopyNode (tree.c:4422)
==96577==    by 0x49B5BB3: xmlStaticCopyNodeList (tree.c:4495)
==96577==    by 0x49B60A0: xmlCopyDoc (tree.c:4716)
==96577==    by 0x10DFE8: validate_overloadables_merge (in /home/jmgs/Source/config)
==96577==    by 0x10D0DC: config_load_cust_to_shm (in /home/jmgs/Source/config)
==96577==    by 0x10FC0E: method_load_customization (in /home/jmgs/Source/config)
==96577==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x111693: config_register_dbus (in /home/jmgs/Source/config)
==96577==    by 0x10CD2B: main (in /home/jmgs/Source/config)
==96577==
==96577== Invalid read of size 1
==96577==    at 0x484FBD4: strcmp (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
==96577==    by 0x10E239: validate_overloadables_merge (in /home/jmgs/Source/config)
==96577==    by 0x10D0DC: config_load_cust_to_shm (in /home/jmgs/Source/config)
==96577==    by 0x10FC0E: method_load_customization (in /home/jmgs/Source/config)
==96577==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x111693: config_register_dbus (in /home/jmgs/Source/config)
==96577==    by 0x10CD2B: main (in /home/jmgs/Source/config)
==96577==  Address 0x5481270 is 0 bytes inside a block of size 7 free'd
==96577==    at 0x484B27F: free (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
==96577==    by 0x49B1512: xmlFreeNode (tree.c:3892)
==96577==    by 0x49B1512: xmlFreeNode (tree.c:3838)
==96577==    by 0x10E298: validate_overloadables_merge (in /home/jmgs/Source/config)
==96577==    by 0x10D0DC: config_load_cust_to_shm (in /home/jmgs/Source/config)
==96577==    by 0x10FC0E: method_load_customization (in /home/jmgs/Source/config)
==96577==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x111693: config_register_dbus (in /home/jmgs/Source/config)
==96577==    by 0x10CD2B: main (in /home/jmgs/Source/config)
==96577==  Block was alloc'd at
==96577==    at 0x4848899: malloc (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
==96577==    by 0x49CA8B5: xmlStrndup (xmlstring.c:49)
==96577==    by 0x49B58B4: xmlStaticCopyNode (tree.c:4331)
==96577==    by 0x49B569E: xmlStaticCopyNode (tree.c:4422)
==96577==    by 0x49B5BB3: xmlStaticCopyNodeList (tree.c:4495)
==96577==    by 0x49B60A0: xmlCopyDoc (tree.c:4716)
==96577==    by 0x10DFE8: validate_overloadables_merge (in /home/jmgs/Source/config)
==96577==    by 0x10D0DC: config_load_cust_to_shm (in /home/jmgs/Source/config)
==96577==    by 0x10FC0E: method_load_customization (in /home/jmgs/Source/config)
==96577==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x111693: config_register_dbus (in /home/jmgs/Source/config)
==96577==
==96577== Invalid read of size 1
==96577==    at 0x484FBE8: strcmp (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
==96577==    by 0x10E239: validate_overloadables_merge (in /home/jmgs/Source/config)
==96577==    by 0x10D0DC: config_load_cust_to_shm (in /home/jmgs/Source/config)
==96577==    by 0x10FC0E: method_load_customization (in /home/jmgs/Source/config)
==96577==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x111693: config_register_dbus (in /home/jmgs/Source/config)
==96577==    by 0x10CD2B: main (in /home/jmgs/Source/config)
==96577==  Address 0x5481271 is 1 bytes inside a block of size 7 free'd
==96577==    at 0x484B27F: free (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
==96577==    by 0x49B1512: xmlFreeNode (tree.c:3892)
==96577==    by 0x49B1512: xmlFreeNode (tree.c:3838)
==96577==    by 0x10E298: validate_overloadables_merge (in /home/jmgs/Source/config)
==96577==    by 0x10D0DC: config_load_cust_to_shm (in /home/jmgs/Source/config)
==96577==    by 0x10FC0E: method_load_customization (in /home/jmgs/Source/config)
==96577==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x111693: config_register_dbus (in /home/jmgs/Source/config)
==96577==    by 0x10CD2B: main (in /home/jmgs/Source/config)
==96577==  Block was alloc'd at
==96577==    at 0x4848899: malloc (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
==96577==    by 0x49CA8B5: xmlStrndup (xmlstring.c:49)
==96577==    by 0x49B58B4: xmlStaticCopyNode (tree.c:4331)
==96577==    by 0x49B569E: xmlStaticCopyNode (tree.c:4422)
==96577==    by 0x49B5BB3: xmlStaticCopyNodeList (tree.c:4495)
==96577==    by 0x49B60A0: xmlCopyDoc (tree.c:4716)
==96577==    by 0x10DFE8: validate_overloadables_merge (in /home/jmgs/Source/config)
==96577==    by 0x10D0DC: config_load_cust_to_shm (in /home/jmgs/Source/config)
==96577==    by 0x10FC0E: method_load_customization (in /home/jmgs/Source/config)
==96577==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x111693: config_register_dbus (in /home/jmgs/Source/config)
==96577==
Replaced node input_4
Document complient with schematron
Document complient with schema
==96577== Invalid read of size 4
==96577==    at 0x4A4A2FA: xmlXPathFreeNodeSet (xpath.c:4115)
==96577==    by 0x4A4C541: xmlXPathFreeObject (xpath.c:5467)
==96577==    by 0x10E1A6: validate_overloadables_merge (in /home/jmgs/Source/config)
==96577==    by 0x10D0DC: config_load_cust_to_shm (in /home/jmgs/Source/config)
==96577==    by 0x10FC0E: method_load_customization (in /home/jmgs/Source/config)
==96577==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x111693: config_register_dbus (in /home/jmgs/Source/config)
==96577==    by 0x10CD2B: main (in /home/jmgs/Source/config)
==96577==  Address 0x54811b8 is 8 bytes inside a block of size 120 free'd
==96577==    at 0x484B27F: free (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
==96577==    by 0x10E298: validate_overloadables_merge (in /home/jmgs/Source/config)
==96577==    by 0x10D0DC: config_load_cust_to_shm (in /home/jmgs/Source/config)
==96577==    by 0x10FC0E: method_load_customization (in /home/jmgs/Source/config)
==96577==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x111693: config_register_dbus (in /home/jmgs/Source/config)
==96577==    by 0x10CD2B: main (in /home/jmgs/Source/config)
==96577==  Block was alloc'd at
==96577==    at 0x4848899: malloc (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
==96577==    by 0x49B550F: xmlStaticCopyNode (tree.c:4311)
==96577==    by 0x49B569E: xmlStaticCopyNode (tree.c:4422)
==96577==    by 0x49B5BB3: xmlStaticCopyNodeList (tree.c:4495)
==96577==    by 0x49B60A0: xmlCopyDoc (tree.c:4716)
==96577==    by 0x10DFE8: validate_overloadables_merge (in /home/jmgs/Source/config)
==96577==    by 0x10D0DC: config_load_cust_to_shm (in /home/jmgs/Source/config)
==96577==    by 0x10FC0E: method_load_customization (in /home/jmgs/Source/config)
==96577==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
==96577==    by 0x111693: config_register_dbus (in /home/jmgs/Source/config)
==96577==    by 0x10CD2B: main (in /home/jmgs/Source/config)
==96577==
Document complient with schematron
Document complient with schema
==96577==
==96577== HEAP SUMMARY:
==96577==     in use at exit: 0 bytes in 0 blocks
==96577==   total heap usage: 54,910 allocs, 54,910 frees, 5,177,146 bytes allocated
==96577==
==96577== All heap blocks were freed -- no leaks are possible
==96577==
==96577== For lists of detected and suppressed errors, rerun with: -s
==96577== ERROR SUMMARY: 9 errors from 4 contexts (suppressed: 0 from 0)

提前致谢。

c valgrind libxml2

评论

0赞 Bodo 8/8/2023
你的代码相当大,尽管它不完整。我建议创建一个最小的可重复示例
0赞 JoseleMG 8/8/2023
是的,你是对的博多。我提供的信息不足以找出问题所在。但是,我在代码中发现了错误,因此我将发布解决方案和一条帮助我找到它的建议。

答:

0赞 JoseleMG 8/8/2023 #1

我发现了触发 valgrind 的 Invalid Read 消息的错误。我正在访问(在 if 语句中)target_node,一旦已经释放.一个有效的解决方案是添加一个 break 语句:strcmp((char *)target_node->name

            for(int replace_idx=0; replace_idx < xpath_obj_replace->nodesetval->nodeNr; replace_idx++)
            {
                xmlNodePtr replace_node = xpath_obj_replace->nodesetval->nodeTab[replace_idx];
                if(0==strcmp((char *)target_node->name,
                             (char *)replace_node->name))
                {
                    xmlNodePtr new_target_node = replace_node(target_node, replace_node);
                    xmlFreeNode(target_node); 
                    xmlReconciliateNs(copy_dom, new_target_node);
                    break;
                }
            }

对于类似情况,作为一条建议(我希望有),不要忘记使用 debug 标志 -g 进行编译。这使 valgrind 能够生成更详细的报告。 例如,就我而言

  ┆ ==100275== Invalid read of size 8
  ┆ ==100275==    at 0x10EC7F: validate_overloadables_merge (config-parsing.c:465)
  ┆ ==100275==    by 0x10CDCF: config_load_cust_to_shm (config.c:111)
  ┆ ==100275==    by 0x111685: method_load_customization (config-sdbus.c:46)
  ┆ ==100275==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
  ┆ ==100275==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
  ┆ ==100275==    by 0x114CE5: config_register_dbus (config-sdbus.c:807)
  ┆ ==100275==    by 0x10D741: main (config.c:328)
  ┆ ==100275==  Address 0x5cf2460 is 16 bytes inside a block of size 120 free'd
  ┆ ==100275==    at 0x484B27F: free (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
  ┆ ==100275==    by 0x10ECB9: validate_overloadables_merge (config-parsing.c:470)
  ┆ ==100275==    by 0x10CDCF: config_load_cust_to_shm (config.c:111)
  ┆ ==100275==    by 0x111685: method_load_customization (config-sdbus.c:46)
  ┆ ==100275==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
  ┆ ==100275==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
  ┆ ==100275==    by 0x114CE5: config_register_dbus (config-sdbus.c:807)
  ┆ ==100275==    by 0x10D741: main (config.c:328)
  ┆ ==100275==  Block was alloc'd at
  ┆ ==100275==    at 0x4848899: malloc (in /usr/libexec/valgrind/vgpreload_memcheck-amd64-linux.so)
  ┆ ==100275==    by 0x49B550F: xmlStaticCopyNode (tree.c:4311)
  ┆ ==100275==    by 0x49B569E: xmlStaticCopyNode (tree.c:4422)
  ┆ ==100275==    by 0x49B5BB3: xmlStaticCopyNodeList (tree.c:4495)
  ┆ ==100275==    by 0x49B60A0: xmlCopyDoc (tree.c:4716)
  ┆ ==100275==    by 0x10E9C5: validate_overloadables_merge (config-parsing.c:423)
  ┆ ==100275==    by 0x10CDCF: config_load_cust_to_shm (config.c:111)
  ┆ ==100275==    by 0x111685: method_load_customization (config-sdbus.c:46)
  ┆ ==100275==    by 0x48B327A: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
  ┆ ==100275==    by 0x48C176C: ??? (in /usr/lib/x86_64-linux-gnu/libsystemd.so.0.32.0)
  ┆ ==100275==    by 0x114CE5: config_register_dbus (config-sdbus.c:807)
  ┆ ==100275==    by 0x10D741: main (config.c:328)

我还推荐其他阅读材料:如何使用 valgrind 查找内存泄漏?