• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2022 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  * http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include "ns_config.h"
17 
18 #include <ctype.h>
19 #include <stdarg.h>
20 
21 #include "ld_log.h"
22 /*---------------------------- Defines -------------------------------------*/
23 #define MAX_LINE_SIZE         (1024)
24 #define INI_INVALID_KEY     ((char*)-1)
25 
26 #ifdef UNIT_TEST_STATIC
27     #define UT_STATIC
28 #else
29     #define UT_STATIC static
30 #endif
31 typedef enum _line_status_ {
32     LINE_UNPROCESSED,
33     LINE_ERROR,
34     LINE_EMPTY,
35     LINE_COMMENT,
36     LINE_SECTION,
37     LINE_VALUE
38 } line_status;
39 
40 #define MAX_KEY_LEN 256
41 static char g_key[MAX_KEY_LEN + 1] = {0};
42 
config_key_join(const char * join,bool start)43 static char *config_key_join(const char *join, bool start)
44 {
45     if (start) g_key[0] = 0;
46     size_t cnt = MAX_KEY_LEN - strlen(g_key);
47     return strncat(g_key, join, cnt);
48 }
49 
default_error_callback(const char * format,...)50 static int default_error_callback(const char *format, ...)
51 {
52     int ret = 0;
53     va_list argptr;
54     va_start(argptr, format);
55     /* do not print
56     ret = vfprintf(stderr, format, argptr);
57     */
58     va_end(argptr);
59     return ret;
60 }
61 
62 static int (*config_error_callback)(const char *, ...) = default_error_callback;
63 
config_set_error_callback(int (* errback)(const char *,...))64 static void config_set_error_callback(int (*errback)(const char *, ...))
65 {
66     if (errback) {
67         config_error_callback = errback;
68     } else {
69         config_error_callback = default_error_callback;
70     }
71 }
72 
config_line(char * line,char * section,char * key,char * value)73 static line_status config_line(char *line, char *section, char *key, char *value)
74 {
75     size_t len;
76     char *split;
77 
78     if ((len = strcspn(line, "#;")) == 0) {
79         /* comment line */
80         return LINE_COMMENT;
81     }
82     line[len] = 0;
83     if ((len = strtrim(line)) == 0) {
84         /* empty line */
85         return LINE_EMPTY;
86     }
87     if (line[0] == '[' && line[len - 1] == ']') {
88         /* section name */
89         memcpy(section, line + 1, len - 2);
90         section[len - 2] = 0;
91         strtrim(section);
92         return LINE_SECTION;
93     }
94     if (split = strchr(line, '=')) {
95         /* key and value */
96         size_t klen, vlen;
97         klen = split - line;
98         vlen = len - klen - 1;
99         if (klen > 0) memcpy(key, line, klen);
100         if (vlen > 0) memcpy(value, split + 1, vlen);
101         key[klen] = 0;
102         value[vlen] = 0;
103         strtrim(key);
104         strtrim(value);
105         return LINE_VALUE;
106     }
107     return LINE_ERROR;
108 }
109 
110 #define SECTION_DEFAULT_SIZE   16
111 #define KV_DEFAULT_SIZE   64
112 
kvlist_alloc(size_t size)113 static kvlist * kvlist_alloc(size_t size)
114 {
115     kvlist *kvs;
116 
117     if (size < KV_DEFAULT_SIZE) size = KV_DEFAULT_SIZE;
118 
119     kvs = (kvlist *)__libc_calloc(1, sizeof *kvs);
120     if (kvs) {
121         kvs->key = (char **)__libc_calloc(size, sizeof *kvs->key);
122         kvs->val = (char **)__libc_calloc(size, sizeof *kvs->val);
123         if (kvs->key && kvs->val) {
124             kvs->size = size;
125         } else {
126             __libc_free(kvs->key);
127             __libc_free(kvs->val);
128             __libc_free(kvs);
129             kvs = NULL;
130         }
131     }
132     return kvs;
133 }
134 
kvlist_realloc(kvlist * kvs)135 UT_STATIC void kvlist_realloc(kvlist *kvs)
136 {
137     if (!kvs) return;
138     size_t size = 2 * kvs->size;
139     if (size) {
140         char **keys, **vals;
141         keys = (char **)__libc_realloc(kvs->key, size * (sizeof *kvs->key));
142         if (!keys) return;
143         kvs->key = keys;
144         vals = (char **)__libc_realloc(kvs->val, size * (sizeof *kvs->val));
145         if (!vals) return;
146         kvs->val = vals;
147         kvs->size = size;
148     }
149 
150     return;
151 }
152 
kvlist_free(kvlist * kvs)153 static void kvlist_free(kvlist *kvs)
154 {
155     size_t i;
156     if (!kvs) return;
157     for (i = 0; i < kvs->num; i++) {
158         __libc_free(kvs->key[i]);
159         __libc_free(kvs->val[i]);
160     }
161     __libc_free(kvs->key);
162     __libc_free(kvs->val);
163     __libc_free(kvs);
164 }
165 
sections_alloc(size_t size)166 static section_list *sections_alloc(size_t size)
167 {
168     section_list *sections;
169     if (size < SECTION_DEFAULT_SIZE) size = SECTION_DEFAULT_SIZE;
170 
171     sections = (section_list *)__libc_calloc(1, sizeof *sections);
172 
173     if (sections) {
174         sections->names = (char**)__libc_calloc(size, sizeof *sections->names);
175         sections->kvs = (kvlist**)__libc_calloc(size, sizeof *sections->kvs);
176         if (sections->names && sections->kvs) {
177             sections->size = size;
178         } else {
179             __libc_free(sections->names);
180             __libc_free(sections->kvs);
181             __libc_free(sections);
182             sections = NULL;
183         }
184     }
185     return sections;
186 }
187 
sections_realloc(section_list * sections)188 UT_STATIC void sections_realloc(section_list *sections)
189 {
190     if (!sections) return;
191     size_t size = 2 * sections->size;
192     if (size) {
193         char **names;
194         kvlist **kvs;
195         names = (char **)__libc_realloc(sections->names, size * (sizeof *sections->names));
196         if (!names) return;
197         sections->names = names;
198         kvs = (kvlist **)__libc_realloc(sections->kvs, size * (sizeof *sections->kvs));
199         if (!kvs) return;
200         sections->kvs = kvs;
201         sections->size = size;
202     }
203     return;
204 }
205 
sections_free(section_list * sections)206 static void sections_free(section_list *sections)
207 {
208     if (!sections) return;
209     for (size_t i = 0; i < sections->num; i++) {
210         __libc_free(sections->names[i]);
211         kvlist_free(sections->kvs[i]);
212     }
213     __libc_free(sections->names);
214     __libc_free(sections->kvs);
215     __libc_free(sections);
216 }
217 
kvlist_set(kvlist * kvs,const char * key,const char * val)218 static void kvlist_set(kvlist *kvs, const char *key, const char *val)
219 {
220     size_t i;
221     if (!kvs || !key || !val) return;
222 
223     for (i = 0; i < kvs->num; i++) {
224         if (!strcmp(kvs->key[i], key)) {
225             break;
226         }
227     }
228 
229     if (i < kvs->num) {
230         char * v = ld_strdup(val);
231         if (v) {
232             __libc_free(kvs->val[i]);
233             kvs->val[i] = v;
234         }
235         return;
236     }
237     if (kvs->num == kvs->size) {
238         kvlist_realloc(kvs);
239     }
240     if (kvs->num < kvs->size) {
241         kvs->key[kvs->num] = ld_strdup(key);
242         kvs->val[kvs->num] = ld_strdup(val);
243         if (kvs->key[kvs->num] && kvs->val[kvs->num]) {
244             kvs->num++;
245         } else {
246             __libc_free(kvs->key[kvs->num]);
247             __libc_free(kvs->val[kvs->num]);
248         }
249     }
250     return;
251 }
252 
sections_set(section_list * sections,const char * name,const char * key,const char * val)253 static void sections_set(section_list *sections, const char *name, const char *key, const char *val)
254 {
255     kvlist* kvs = NULL;
256     if (!sections || !name || !key || !val) return;
257 
258     for(size_t i = 0; i < sections->num; i++) {
259         if (!strcmp(sections->names[i], name)) {
260             kvs = sections->kvs[i];
261             break;
262         }
263     }
264     if (kvs) {
265         kvlist_set(kvs, key, val);
266         return;
267     }
268 
269     if (sections->num == sections->size) {
270        sections_realloc(sections);
271     }
272 
273     if (sections->num < sections->size) {
274         kvs = kvlist_alloc(0);
275         sections->names[sections->num] = ld_strdup(name);
276         sections->kvs[sections->num] = kvs;
277         if (sections->names[sections->num] && kvs) {
278             sections->num++;
279             kvlist_set(kvs, key, val);
280         } else {
281             __libc_free(sections->names[sections->num]);
282             kvlist_free(kvs);
283         }
284     }
285 }
286 
config_load(const char * filepath)287 static section_list *config_load(const char *filepath)
288 {
289     FILE *file;
290     char line[MAX_LINE_SIZE + 1];
291     char section[MAX_LINE_SIZE + 1];
292     char key[MAX_LINE_SIZE + 1];
293     char val[MAX_LINE_SIZE + 1];
294 
295     size_t  len;
296     int  lineno = 0;
297 
298     section_list *sections;
299 
300     if ((file = fopen(filepath, "r")) == NULL) {
301         config_error_callback("config: cannot open %s\n", filepath);
302         return NULL;
303     }
304 
305     sections = sections_alloc(0);
306     if (!sections) {
307         fclose(file);
308         return NULL;
309     }
310 
311     memset(line, 0, sizeof line);
312     memset(section, 0, sizeof section);
313     memset(key, 0, sizeof key);
314     memset(val, 0, sizeof val);
315 
316     while (fgets(line, sizeof line, file)) {
317         lineno++;
318         len = strlen(line);
319         if (len == 0) continue;
320 
321         if (line[len - 1] != '\n' && !feof(file)) {
322             config_error_callback(
323               "config: input line too long in %s (%d)\n", filepath, lineno);
324             sections_free(sections);
325             fclose(file);
326             return NULL;
327         }
328 
329         if (line[len - 1] == '\n') {
330             line[len - 1] = 0;
331             len--;
332         }
333 
334         switch (config_line(line, section, key, val)) {
335             case LINE_EMPTY:
336             case LINE_COMMENT:
337             case LINE_SECTION:
338                 break;
339             case LINE_VALUE:
340                 sections_set(sections, section, key, val);
341                 break;
342             case LINE_ERROR:
343                 config_error_callback(
344                     "config: syntax error in %s (%d):\n-> %s\n",
345                     filepath,
346                     lineno,
347                     line);
348                 break;
349             default:
350                 break;
351         }
352     }
353     fclose(file);
354     return sections;
355 }
356 
357 static ns_configor g_configor;
358 
359 /* const define */
360 #define CONFIG_DEFAULT_FILE "/etc/ld-musl-namespace-arm.ini"        /* default config file pathname */
361 #define SECTION_DIR_MAP "section.dir.map"   /* map of section and directory of app */
362 #define ATTR_NS_PREFIX "namespace"         /* prefix of namespace attribute */
363 #define ATTR_NS_ASAN "asan"                /* asan */
364 #define ATTR_NS_LIB_PATHS "lib.paths"         /* library search paths */
365 #define ATTR_NS_PERMITTED_PATHS "permitted.paths"         /* when separated, permitted dir paths of libs, including sub dirs */
366 #define ATTR_NS_INHERITS "inherits"          /* inherited namespace */
367 #define ATTR_NS_SEPARATED "separated"         /* if separated */
368 #define ATTR_ADDED_NSLIST "added.nslist"      /* all namespace names except default */
369 #define ATTR_NS_DEFAULT "default"           /* default namespace name */
370 #define ATTR_NS_ACQUIESCENCE "acquiescence"           /* acquiescence section name */
371 #define ATTR_NS_ALLOWED_LIBS "allowed.libs"      /* when separated, allowed library names */
372 #define ATTR_NS_INHERIT_SHARED_LIBS "shared.libs"      /* when inherited, shared library names */
373 #define SECTION_DIR_MAP_SYSTEM "system"      /* system path */
374 #define SECTION_DIR_MAP_ASAN_SYSTEM "asan_system"      /* asan system path */
375 
376 /* get key-value list of section */
config_get_kvs(const char * sname)377 static kvlist *config_get_kvs(const char *sname)
378 {
379     size_t  i;
380     for (i = 0; i < g_configor.sections->num; i++) {
381         if (!strcmp(g_configor.sections->names[i], sname)) {
382             return g_configor.sections->kvs[i];
383         }
384     }
385     return NULL;
386 }
387 
388 /* get value by acquiescence */
config_get_value_by_acquiescence(kvlist * acquiescence_kvs,const char * key)389 static char *config_get_value_by_acquiescence(kvlist *acquiescence_kvs, const char *key)
390 {
391     if (!acquiescence_kvs) {
392         return NULL;
393     }
394     size_t i;
395     for (i = 0; i < acquiescence_kvs->num; i++) {
396         if (!strcmp(acquiescence_kvs->key[i], key)) {
397             return acquiescence_kvs->val[i];
398         }
399     }
400     return NULL;
401 }
402 
403 /* get value by acquiescence lib path */
config_get_acquiescence_lib_path(kvlist * acquiescence_kvs)404 static char *config_get_acquiescence_lib_path(kvlist *acquiescence_kvs)
405 {
406     if (!acquiescence_kvs) {
407         return NULL;
408     }
409     config_key_join(ATTR_NS_PREFIX, true);
410     config_key_join(".", false);
411     config_key_join(ATTR_NS_DEFAULT, false);
412     config_key_join(".", false);
413     char *key = config_key_join(ATTR_NS_LIB_PATHS, false);
414     return config_get_value_by_acquiescence(acquiescence_kvs, key);
415 }
416 
417 /* get value by acquiescence asan lib path */
config_get_acquiescence_asan_lib_path(kvlist * acquiescence_kvs)418 static char *config_get_acquiescence_asan_lib_path(kvlist *acquiescence_kvs)
419 {
420     if (!acquiescence_kvs) {
421         return NULL;
422     }
423     config_key_join(ATTR_NS_PREFIX, true);
424     config_key_join(".", false);
425     config_key_join(ATTR_NS_DEFAULT, false);
426     config_key_join(".", false);
427     config_key_join(ATTR_NS_ASAN, false);
428     config_key_join(".", false);
429     char *key = config_key_join(ATTR_NS_LIB_PATHS, false);
430     return config_get_value_by_acquiescence(acquiescence_kvs, key);
431 }
432 
433 /* get value by key */
config_get_value(const char * key)434 static char *config_get_value(const char *key)
435 {
436     if (!g_configor.kvs) {
437         return NULL;
438     }
439     size_t i;
440     for (i = 0; i < g_configor.kvs->num; i++) {
441         if (!strcmp(g_configor.kvs->key[i], key)) return g_configor.kvs->val[i];
442     }
443     return NULL;
444 }
445 
446 /* get library search paths */
config_get_lib_paths(const char * ns_name)447 static char *config_get_lib_paths(const char *ns_name)
448 {
449     if (ns_name == NULL) {
450         return NULL;
451     }
452     config_key_join(ATTR_NS_PREFIX, true);
453     config_key_join(".", false);
454     config_key_join(ns_name, false);
455     config_key_join(".", false);
456     char *key = config_key_join(ATTR_NS_LIB_PATHS, false);
457     return config_get_value(key);
458 }
459 
460 /* get asan library search paths */
config_get_asan_lib_paths(const char * ns_name)461 static char *config_get_asan_lib_paths(const char *ns_name)
462 {
463     if (ns_name == NULL) {
464         return NULL;
465     }
466     config_key_join(ATTR_NS_PREFIX, true);
467     config_key_join(".", false);
468     config_key_join(ns_name, false);
469     config_key_join(".", false);
470     config_key_join(ATTR_NS_ASAN, false);
471     config_key_join(".", false);
472     char *key = config_key_join(ATTR_NS_LIB_PATHS, false);
473     return config_get_value(key);
474 }
475 
476 /* parse config, success 0, failure <0 */
config_parse(const char * file_path,const char * exe_path)477 static int config_parse(const char *file_path, const char *exe_path)
478 {
479     kvlist* dirkvs;
480     kvlist* acquiescence_kvs;
481     if (!exe_path) return -1;
482     g_configor.exe_path = ld_strdup(exe_path);
483     const char * fpath = CONFIG_DEFAULT_FILE;
484     if (file_path) fpath = file_path;
485     g_configor.file_path = ld_strdup(fpath);
486     g_configor.sections = config_load(fpath);
487 
488     if (!g_configor.sections) {
489         LD_LOGD("config_parse load ini config fail!");
490         return -2;
491     }
492     dirkvs = config_get_kvs(SECTION_DIR_MAP);
493     acquiescence_kvs = config_get_kvs(ATTR_NS_ACQUIESCENCE);
494     if (!dirkvs || !acquiescence_kvs) {
495         LD_LOGD("config_parse get dirkvs or acquiescence_kvs fail!");
496         return -3; /* no section directory map or acquiescence section found */
497     }
498     g_configor.config_sys_path = config_get_acquiescence_lib_path(acquiescence_kvs);
499     g_configor.config_asan_sys_path = config_get_acquiescence_asan_lib_path(acquiescence_kvs);
500     size_t i;
501     char * sname = NULL;
502     for (i = 0; i < dirkvs->num; i++) {
503        strlist * paths = strsplit(dirkvs->val[i], ":");
504        if (paths) {
505            size_t j;
506            for (j = 0; j < paths->num; j++) {
507                if (!strcmp(paths->strs[j], exe_path)) break;
508            }
509            if (j < paths->num) sname = dirkvs->key[i];
510         }
511         strlist_free(paths);
512         if (sname) break;
513     }
514     if (!sname) {
515         /* No matched section found, use the default section. */
516         sname = ATTR_NS_ACQUIESCENCE;
517         LD_LOGD("config_parse no section found!");
518     }
519     if (!(g_configor.kvs = config_get_kvs(sname))) {
520         LD_LOGD("config_parse no section key-value list found!");
521         return -5;/* no section key-value list found */
522     }
523 
524     char *default_lib_paths = config_get_lib_paths(ATTR_NS_DEFAULT);
525     if (default_lib_paths) {
526         g_configor.config_sys_path = default_lib_paths;
527     } else {
528         LD_LOGW("config_parse get default lib paths fail! Config namespace default lib paths,please!");
529     }
530     char *default_asan_lib_paths = config_get_asan_lib_paths(ATTR_NS_DEFAULT);
531     if (default_asan_lib_paths) {
532         g_configor.config_asan_sys_path = default_asan_lib_paths;
533     } else {
534         LD_LOGW("config_parse get default asan lib paths fail! Config namespace default asan lib paths,please!");
535     }
536     return 0;
537 }
538 
539 /* get namespace names except default */
config_get_namespaces()540 static strlist *config_get_namespaces()
541 {
542     char *key = config_key_join(ATTR_ADDED_NSLIST, true);
543     char *val = config_get_value(key);
544     return strsplit(val, ",");
545 }
546 
547 /* get permitted paths */
config_get_permitted_paths(const char * ns_name)548 static char *config_get_permitted_paths(const char *ns_name)
549 {
550     if (ns_name == NULL) {
551         return NULL;
552     }
553     config_key_join(ATTR_NS_PREFIX, true);
554     config_key_join(".", false);
555     config_key_join(ns_name, false);
556     config_key_join(".", false);
557     char *key = config_key_join(ATTR_NS_PERMITTED_PATHS, false);
558     return config_get_value(key);
559 }
560 
561 /* get asan permitted paths */
config_get_asan_permitted_paths(const char * ns_name)562 static char *config_get_asan_permitted_paths(const char *ns_name)
563 {
564     if (ns_name == NULL) {
565         return NULL;
566     }
567     config_key_join(ATTR_NS_PREFIX, true);
568     config_key_join(".", false);
569     config_key_join(ns_name, false);
570     config_key_join(".", false);
571     config_key_join(ATTR_NS_ASAN, false);
572     config_key_join(".", false);
573     char *key = config_key_join(ATTR_NS_PERMITTED_PATHS, false);
574     return config_get_value(key);
575 }
576 /* get inherited namespace names */
config_get_inherits(const char * ns_name)577 static strlist *config_get_inherits(const char *ns_name)
578 {
579     if (ns_name == NULL) {
580         return NULL;
581     }
582     config_key_join(ATTR_NS_PREFIX, true);
583     config_key_join(".", false);
584     config_key_join(ns_name, false);
585     config_key_join(".", false);
586     char *key = config_key_join(ATTR_NS_INHERITS, false);
587     char *val = config_get_value(key);
588     return strsplit(val, ",");
589 }
590 /* get separated */
config_get_separated(const char * ns_name)591 static bool config_get_separated(const char *ns_name)
592 {
593     if (ns_name == NULL) {
594         return false;
595     }
596     config_key_join(ATTR_NS_PREFIX, true);
597     config_key_join(".", false);
598     config_key_join(ns_name, false);
599     config_key_join(".", false);
600     char *key = config_key_join(ATTR_NS_SEPARATED, false);
601     char *val = config_get_value(key);
602     strlwc(val);
603     if (val && !strcmp("true", val)) return true;
604     return false;  /* default false */
605 }
606 
607 /* get allowed libs */
config_get_allowed_libs(const char * ns_name)608 static char *config_get_allowed_libs(const char *ns_name)
609 {
610     if (ns_name == NULL) {
611         return NULL;
612     }
613     config_key_join(ATTR_NS_PREFIX, true);
614     config_key_join(".", false);
615     config_key_join(ns_name, false);
616     config_key_join(".", false);
617     char *key = config_key_join(ATTR_NS_ALLOWED_LIBS, false);
618     return config_get_value(key);
619 }
620 /* get shared libs by inherited namespace */
config_get_inherit_shared_libs(const char * ns_name,const char * inherited_ns_name)621 static char *config_get_inherit_shared_libs(const char *ns_name, const char *inherited_ns_name)
622 {
623     if (ns_name == NULL || inherited_ns_name == NULL) {
624         return NULL;
625     }
626     config_key_join(ATTR_NS_PREFIX, true);
627     config_key_join(".", false);
628     config_key_join(ns_name, false);
629     config_key_join(".inherit.", false);
630     config_key_join(inherited_ns_name, false);
631     config_key_join(".", false);
632     char *key = config_key_join(ATTR_NS_INHERIT_SHARED_LIBS, false);
633     return config_get_value(key);
634 }
635 
636 /* The call time is after parse */
config_get_sys_paths(void)637 static char *config_get_sys_paths(void)
638 {
639     return g_configor.config_sys_path;
640 }
config_get_asan_sys_paths(void)641 static char *config_get_asan_sys_paths(void)
642 {
643     return g_configor.config_asan_sys_path;
644 }
configor_init()645 ns_configor *configor_init()
646 {
647     memset(&g_configor, 0, sizeof g_configor);
648     g_configor.set_error_callback = config_set_error_callback;
649     g_configor.parse = config_parse;
650     g_configor.get_namespaces = config_get_namespaces;
651     g_configor.get_lib_paths = config_get_lib_paths;
652     g_configor.get_asan_lib_paths = config_get_asan_lib_paths;
653     g_configor.get_permitted_paths = config_get_permitted_paths;
654     g_configor.get_asan_permitted_paths = config_get_asan_permitted_paths;
655     g_configor.get_separated = config_get_separated;
656     g_configor.get_inherits = config_get_inherits;
657     g_configor.get_allowed_libs = config_get_allowed_libs;
658     g_configor.get_inherit_shared_libs = config_get_inherit_shared_libs;
659     g_configor.get_sys_paths = config_get_sys_paths;
660     g_configor.get_asan_sys_paths = config_get_asan_sys_paths;
661     g_configor.config_sys_path = NULL; // init it in config_parse.
662     g_configor.config_asan_sys_path = NULL; // init it in config_parse.
663     return &g_configor;
664 }
665 
configor_free()666 void configor_free()
667 {
668     if (g_configor.sections) {
669         sections_free(g_configor.sections);
670         g_configor.sections = NULL;
671     }
672     if (g_configor.file_path) {
673         __libc_free(g_configor.file_path);
674         g_configor.file_path = NULL;
675     }
676     if (g_configor.exe_path) {
677         __libc_free(g_configor.exe_path);
678         g_configor.exe_path = NULL;
679     }
680 }