summaryrefslogtreecommitdiff
path: root/ld/ld_options.c
diff options
context:
space:
mode:
Diffstat (limited to 'ld/ld_options.c')
-rw-r--r--ld/ld_options.c507
1 files changed, 507 insertions, 0 deletions
diff --git a/ld/ld_options.c b/ld/ld_options.c
new file mode 100644
index 000000000000..8a2cd1cbce9d
--- /dev/null
+++ b/ld/ld_options.c
@@ -0,0 +1,507 @@
+/*-
+ * Copyright (c) 2010-2013 Kai Wang
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+#include "ld.h"
+#include "ld_file.h"
+#include "ld_path.h"
+#include "ld_script.h"
+#include "ld_symbols.h"
+#include "ld_options.h"
+#include "ld_output.h"
+
+ELFTC_VCSID("$Id: ld_options.c 2926 2013-03-17 22:53:54Z kaiwang27 $");
+
+/*
+ * Support routines for parsing command line options.
+ */
+
+static const char *ld_short_opts =
+ "b:c:de:Ef:Fgh:iI:l:L:m:MnNo:O::qrR:sStT:xXyY:u:vV()";
+
+static struct ld_option ld_opts[] = {
+ {"aarchive", KEY_STATIC, ONE_DASH, NO_ARG},
+ {"adefault", KEY_DYNAMIC, ONE_DASH, NO_ARG},
+ {"ashared", KEY_DYNAMIC, ONE_DASH, NO_ARG},
+ {"accept-unknown-input-arch", KEY_ACCEPT_UNKNOWN, ANY_DASH, NO_ARG},
+ {"allow-multiple-definition", KEY_Z_MULDEFS, ANY_DASH, NO_ARG},
+ {"allow-shlib-undefined", KEY_ALLOW_SHLIB_UNDEF, ANY_DASH, NO_ARG},
+ {"assert", KEY_ASSERT, ANY_DASH, NO_ARG},
+ {"as-needed", KEY_AS_NEEDED, ANY_DASH, NO_ARG},
+ {"auxiliary", 'f', ANY_DASH, REQ_ARG},
+ {"build-id", KEY_BUILD_ID, ANY_DASH, OPT_ARG},
+ {"call_shared", KEY_DYNAMIC, ONE_DASH, NO_ARG},
+ {"check-sections", KEY_CHECK_SECTIONS, ANY_DASH, NO_ARG},
+ {"cref", KEY_CREF, ANY_DASH, NO_ARG},
+ {"defsym", KEY_DEFSYM, ANY_DASH, REQ_ARG},
+ {"demangle", KEY_DEMANGLE, ANY_DASH, OPT_ARG},
+ {"dc", 'd', ONE_DASH, NO_ARG},
+ {"dp", 'd', ONE_DASH, NO_ARG},
+ {"disable-new-dtags", KEY_DISABLE_NEW_DTAGS, ANY_DASH, NO_ARG},
+ {"discard-all", 'x', ANY_DASH, NO_ARG},
+ {"discard-locals", 'X', ANY_DASH, NO_ARG},
+ {"dn", KEY_STATIC, ONE_DASH, NO_ARG},
+ {"dy", KEY_DYNAMIC, ONE_DASH, NO_ARG},
+ {"dynamic-linker", 'I', ANY_DASH, REQ_ARG},
+ {"end-group", ')', ANY_DASH, NO_ARG},
+ {"entry", 'e', ANY_DASH, REQ_ARG},
+ {"error-unresolved-symbols", KEY_ERR_UNRESOLVE_SYM, ANY_DASH, NO_ARG},
+ {"export-dynamic", 'E', ANY_DASH, NO_ARG},
+ {"eh-frame-hdr", KEY_EH_FRAME_HDR, ANY_DASH, NO_ARG},
+ {"emit-relocs", 'q', ANY_DASH, NO_ARG},
+ {"emulation", 'm', ANY_DASH, REQ_ARG},
+ {"enable-new-dtags", KEY_ENABLE_NEW_DTAGS, ANY_DASH, NO_ARG},
+ {"fatal-warnings", KEY_FATAL_WARNINGS, ANY_DASH, NO_ARG},
+ {"filter", 'F', ANY_DASH, NO_ARG},
+ {"fini", KEY_FINI, ANY_DASH, NO_ARG},
+ {"format", 'b', ANY_DASH, REQ_ARG},
+ {"gc-sections", KEY_GC_SECTIONS, ANY_DASH, NO_ARG},
+ {"hash-style", KEY_HASH_STYLE, ANY_DASH, REQ_ARG},
+ {"help", KEY_HELP, ANY_DASH, NO_ARG},
+ {"init", KEY_INIT, ANY_DASH, REQ_ARG},
+ {"just-symbols", 'R', ANY_DASH, REQ_ARG},
+ {"library", 'l', ANY_DASH, REQ_ARG},
+ {"library-path", 'L', ANY_DASH, REQ_ARG},
+ {"mri-script", 'c', ANY_DASH, REQ_ARG},
+ {"nmagic", 'n', ANY_DASH, NO_ARG},
+ {"nostdlib", KEY_NO_STDLIB, ONE_DASH, NO_ARG},
+ {"no-accept-unknown-input-arch", KEY_NO_UNKNOWN, ANY_DASH, NO_ARG},
+ {"no-allow-shlib-undefined", KEY_NO_SHLIB_UNDEF, ANY_DASH, NO_ARG},
+ {"no-as-needed", KEY_NO_AS_NEEDED, ANY_DASH, NO_ARG},
+ {"no-check-sections", KEY_NO_CHECK_SECTIONS, ANY_DASH, NO_ARG},
+ {"no-define-common", KEY_NO_DEFINE_COMMON, ANY_DASH, NO_ARG},
+ {"no-demangle", KEY_NO_DEMANGLE, ANY_DASH, OPT_ARG},
+ {"no-gc-sections", KEY_NO_GC_SECTIONS, ANY_DASH, NO_ARG},
+ {"no-keep-memorg", KEY_NO_KEEP_MEMORY, ANY_DASH, NO_ARG},
+ {"no-omagic", KEY_NO_OMAGIC, ANY_DASH, NO_ARG},
+ {"no-print-gc-sections", KEY_NO_PRINT_GC_SECTIONS, ANY_DASH, NO_ARG},
+ {"no-undefined", KEY_Z_DEFS, ANY_DASH, NO_ARG},
+ {"no-undefined-version", KEY_NO_UNDEF_VERSION, ANY_DASH, NO_ARG},
+ {"no-whole-archive", KEY_NO_WHOLE_ARCHIVE, ANY_DASH, NO_ARG},
+ {"no-warn-mismatch", KEY_NO_WARN_MISMATCH, ANY_DASH, NO_ARG},
+ {"non_shared", KEY_STATIC, ONE_DASH, NO_ARG},
+ {"oformat", KEY_OFORMAT, TWO_DASH, REQ_ARG},
+ {"omagic", 'N', TWO_DASH, NO_ARG},
+ {"output", 'o', TWO_DASH, REQ_ARG},
+ {"pic-executable", KEY_PIE, ANY_DASH, NO_ARG},
+ {"pie", KEY_PIE, ONE_DASH, NO_ARG},
+ {"print-gc-sections", KEY_PRINT_GC_SECTIONS, ANY_DASH, NO_ARG},
+ {"print-map", 'M', ANY_DASH, NO_ARG},
+ {"qmagic", KEY_QMAGIC, ANY_DASH, NO_ARG},
+ {"relax", KEY_RELAX, ANY_DASH, NO_ARG},
+ {"relocatable", 'r', ANY_DASH, NO_ARG},
+ {"retain-symbols-file", KEY_RETAIN_SYM_FILE, ANY_DASH, REQ_ARG},
+ {"rpath", KEY_RPATH, ANY_DASH, REQ_ARG},
+ {"rpath-link", KEY_RPATH_LINK, ANY_DASH, REQ_ARG},
+ {"runpath", KEY_RUNPATH, ANY_DASH, REQ_ARG},
+ {"script", 'T', ANY_DASH, REQ_ARG},
+ {"section-start", KEY_SECTION_START, ANY_DASH, REQ_ARG},
+ {"shared", KEY_SHARED, ANY_DASH, NO_ARG},
+ {"soname", 'h', ONE_DASH, REQ_ARG},
+ {"sort-common", KEY_SORT_COMMON, ANY_DASH, NO_ARG},
+ {"split-by-file", KEY_SPLIT_BY_FILE, ANY_DASH, REQ_ARG},
+ {"split-by-reloc", KEY_SPLIT_BY_RELOC, ANY_DASH, REQ_ARG},
+ {"start-group", '(', ANY_DASH, NO_ARG},
+ {"stats", KEY_STATS, ANY_DASH, NO_ARG},
+ {"static", KEY_STATIC, ONE_DASH, NO_ARG},
+ {"strip-all", 's', ANY_DASH, NO_ARG},
+ {"strip-debug", 'S', ANY_DASH, NO_ARG},
+ {"trace", 't', ANY_DASH, NO_ARG},
+ {"trace_symbol", 'y', ANY_DASH, NO_ARG},
+ {"traditional-format", KEY_TRADITIONAL_FORMAT, ANY_DASH, NO_ARG},
+ {"undefined", 'u', ANY_DASH, REQ_ARG},
+ {"unique", KEY_UNIQUE, ANY_DASH, OPT_ARG},
+ {"unresolved-symbols", KEY_UNRESOLVED_SYMBOLS, ANY_DASH, REQ_ARG},
+ {"verbose" , 'v', ANY_DASH, NO_ARG},
+ {"version", KEY_VERSION, ANY_DASH, NO_ARG},
+ {"version-script", KEY_VERSION_SCRIPT, ANY_DASH, REQ_ARG},
+ {"warn-common", KEY_WARN_COMMON, ANY_DASH, NO_ARG},
+ {"warn-constructors", KEY_WARN_CONSTRUCTORS, ANY_DASH, NO_ARG},
+ {"warn-multiple-gp", KEY_WARN_MULTIPLE_GP, ANY_DASH, NO_ARG},
+ {"warn-once", KEY_WARN_ONCE, ANY_DASH, NO_ARG},
+ {"warn-section-align", KEY_WARN_SECTION_ALIGN, ANY_DASH, NO_ARG},
+ {"warn-shared-textrel", KEY_WARN_SHARED_TEXTREL, ANY_DASH, NO_ARG},
+ {"warn-unresolved-symbols", KEY_WARN_UNRESOLVE_SYM, ANY_DASH, NO_ARG},
+ {"whole_archive", KEY_WHOLE_ARCHIVE, ANY_DASH, NO_ARG},
+ {"wrap", KEY_WRAP, ANY_DASH, REQ_ARG},
+ {"EB", KEY_EB, ONE_DASH, NO_ARG},
+ {"EL", KEY_EL, ONE_DASH, NO_ARG},
+ {"Map", KEY_MAP, ONE_DASH, REQ_ARG},
+ {"Qy", KEY_QY, ONE_DASH, NO_ARG},
+ {"Tbss", KEY_TBSS, ONE_DASH, REQ_ARG},
+ {"Tdata", KEY_TDATA, ONE_DASH, REQ_ARG},
+ {"Ttext", KEY_TTEXT, ONE_DASH, REQ_ARG},
+ {"Ur", KEY_UR, ONE_DASH, NO_ARG},
+ {NULL, 0, 0, 0},
+};
+
+static struct ld_option ld_opts_B[] = {
+ {"shareable", KEY_SHARED, ONE_DASH, NO_ARG},
+ {"static", KEY_STATIC, ONE_DASH, NO_ARG},
+ {"dynamic", KEY_DYNAMIC, ONE_DASH, NO_ARG},
+ {"group", KEY_GROUP, ONE_DASH, NO_ARG},
+ {"symbolic", KEY_SYMBOLIC, ONE_DASH, NO_ARG},
+ {"symbolic_functions", KEY_SYMBOLIC_FUNC, ONE_DASH, NO_ARG},
+};
+
+static struct ld_option ld_opts_z[] = {
+ {"nodefaultlib", KEY_Z_NO_DEFAULT_LIB, ONE_DASH, NO_ARG},
+ {"allextract", KEY_WHOLE_ARCHIVE, ONE_DASH, NO_ARG},
+ {"defaultextract", KEY_Z_DEFAULT_EXTRACT, ONE_DASH, NO_ARG},
+ {"weakextract", KEY_Z_WEAK_EXTRACT, ONE_DASH, NO_ARG},
+ {"muldefs", KEY_Z_MULDEFS, ONE_DASH, NO_ARG},
+ {"defs", KEY_Z_DEFS, ONE_DASH, NO_ARG},
+ {"execstack", KEY_Z_EXEC_STACK, ONE_DASH, NO_ARG},
+ {"nodefs", KEY_Z_NO_DEFS, ONE_DASH, NO_ARG},
+ {"origin", KEY_Z_ORIGIN, ONE_DASH, NO_ARG},
+ {"now", KEY_Z_NOW, ONE_DASH, NO_ARG},
+ {"nodelete", KEY_Z_NO_DELETE, ONE_DASH, NO_ARG},
+ {"initfirst", KEY_Z_INIT_FIRST, ONE_DASH, NO_ARG},
+ {"lazyload", KEY_Z_LAZYLOAD, ONE_DASH, NO_ARG},
+ {"noexecstack", KEY_Z_NO_EXEC_STACK, ONE_DASH, NO_ARG},
+ {"nodlopen", KEY_Z_NO_DLOPEN, ONE_DASH, NO_ARG},
+ {"nolazyload", KEY_Z_NO_LAZYLOAD, ONE_DASH, NO_ARG},
+ {"ignore", KEY_Z_IGNORE, ONE_DASH, NO_ARG},
+ {"record", KEY_Z_RECORD, ONE_DASH, NO_ARG},
+ {"systemlibrary", KEY_Z_SYSTEM_LIBRARY, ONE_DASH, NO_ARG},
+};
+
+static void _copy_optarg(struct ld *ld, char **dst, char *src);
+static void _process_options(struct ld *ld, int key, char *arg);
+static int _parse_long_options(struct ld *, struct ld_option *, int,
+ int, char **, char *, enum ld_dash);
+static void _print_version(struct ld *ld);
+
+void
+ld_options_parse(struct ld* ld, int argc, char **argv)
+{
+ enum ld_dash d;
+ char *p, *p0, *oli;
+ int ac, ac0;
+
+ ac = 1;
+
+ while (ac < argc) {
+ p = argv[ac];
+ if (*p != '-' || p[1] == '\0') {
+ _process_options(ld, KEY_FILE, p);
+ ac++;
+ continue;
+ }
+
+ if (*++p == '-') {
+ if (p[1] == '\0') {
+ /* Option --. Ignore the rest of options. */
+ return;
+ }
+ p++;
+ d = TWO_DASH;
+ } else {
+ d = ONE_DASH;
+ if (*p == 'B' || *p == 'z') {
+ ac0 = ac;
+ if (*(p0 = p + 1) == '\0')
+ p0 = argv[++ac0];
+ ac = _parse_long_options(ld,
+ *p == 'B' ? ld_opts_B : ld_opts_z,
+ ac0, argc, argv, p0, d);
+ if (ac > 0)
+ continue;
+ ld_fatal(ld, "unrecognized options -%c: %s",
+ *p, p0);
+ }
+ }
+
+ ac0 = _parse_long_options(ld, ld_opts, ac, argc, argv, p, d);
+ if (ac0 > 0) {
+ ac = ac0;
+ continue;
+ }
+
+ if (d == TWO_DASH)
+ ld_fatal(ld, "unrecognized option %s", p);
+
+ /*
+ * Search short options.
+ */
+ while (*p != '\0') {
+ if ((oli = strchr(ld_short_opts, *p)) == NULL)
+ ld_fatal(ld, "unrecognized option -%c", *p);
+ if (*++oli != ':') {
+ _process_options(ld, *p++, NULL);
+ continue;
+ }
+ if (p[1] != '\0')
+ _process_options(ld, *p, &p[1]);
+ else if (oli[1] != ':') {
+ if (++ac >= argc)
+ ld_fatal(ld, "require arg for"
+ " option -%c", *p);
+ _process_options(ld, *p, argv[ac]);
+ }
+ break;
+ }
+
+ ac++;
+ }
+}
+
+static int
+_parse_long_options(struct ld *ld, struct ld_option *opts, int ac,
+ int argc, char **argv, char *opt, enum ld_dash dash)
+{
+ char *equal;
+ size_t av_len;
+ int i, match;
+
+ if ((equal = strchr(opt, '=')) != NULL) {
+ av_len = equal - opt;
+ equal++;
+ if (*equal == '\0')
+ ld_fatal(ld, "no argument after =");
+ } else
+ av_len = strlen(opt);
+
+ match = 0;
+ for (i = 0; opts[i].lo_long != NULL; i++) {
+ if (opts[i].lo_dash != ANY_DASH && opts[i].lo_dash != dash)
+ continue;
+ if (strlen(opts[i].lo_long) == av_len &&
+ !strncmp(opt, opts[i].lo_long, av_len)) {
+ match = 1;
+ break;
+ }
+ }
+ if (!match)
+ return (-1);
+
+ switch (opts[i].lo_arg) {
+ case NO_ARG:
+ if (equal != NULL) {
+ ld_fatal(ld, "option %s does not accept argument",
+ opts[i].lo_long);
+ }
+ _process_options(ld, opts[i].lo_key, NULL);
+ break;
+ case REQ_ARG:
+ if (equal != NULL)
+ _process_options(ld, opts[i].lo_key, equal);
+ else {
+ if (++ac >= argc)
+ ld_fatal(ld, "require arg for option %s",
+ opts[i].lo_long);
+ _process_options(ld, opts[i].lo_key, argv[ac]);
+ }
+ break;
+ case OPT_ARG:
+ _process_options(ld, opts[i].lo_key, equal);
+ break;
+ default:
+ assert(0);
+ break;
+ }
+
+ return (++ac);
+}
+
+static void
+_process_options(struct ld *ld, int key, char *arg)
+{
+ struct ld_state *ls;
+
+ assert(ld != NULL);
+ ls = &ld->ld_state;
+
+ switch (key) {
+ case 'b':
+ ls->ls_itgt = elftc_bfd_find_target(arg);
+ if (ls->ls_itgt == NULL)
+ ld_fatal(ld, "invalid BFD target `%s'", arg);
+ break;
+ case 'd':
+ ld->ld_common_alloc = 1;
+ break;
+ case 'e':
+ _copy_optarg(ld, &ld->ld_entry, arg);
+ break;
+ case 'h':
+ _copy_optarg(ld, &ld->ld_soname, arg);
+ break;
+ case 'I':
+ _copy_optarg(ld, &ld->ld_interp, arg);
+ break;
+ case 'l':
+ ld_path_search_library(ld, arg);
+ break;
+ case 'L':
+ ld_path_add(ld, arg, LPT_L);
+ break;
+ case 'M':
+ ld->ld_print_linkmap = 1;
+ break;
+ case 'o':
+ _copy_optarg(ld, &ld->ld_output_file, arg);
+ break;
+ case 'q':
+ ld->ld_emit_reloc = 1;
+ break;
+ case 'r':
+ ld->ld_reloc = 1;
+ break;
+ case 'T':
+ ld_script_parse(arg);
+ break;
+ case 'u':
+ ld_symbols_add_extern(ld, arg);
+ break;
+ case 'v':
+ case 'V':
+ _print_version(ld);
+ break;
+ case '(':
+ ls->ls_group_level++;
+ if (ls->ls_group_level > LD_MAX_NESTED_GROUP)
+ ld_fatal(ld, "too many nested archive groups");
+ break;
+ case ')':
+ ls->ls_group_level--;
+ break;
+ case KEY_AS_NEEDED:
+ ls->ls_as_needed = 1;
+ break;
+ case KEY_DYNAMIC:
+ ls->ls_static = 0;
+ break;
+ case KEY_EH_FRAME_HDR:
+ ld->ld_ehframe_hdr = 1;
+ break;
+ case KEY_GC_SECTIONS:
+ ld->ld_gc = 1;
+ break;
+ case KEY_NO_AS_NEEDED:
+ ls->ls_as_needed = 0;
+ break;
+ case KEY_NO_DEFINE_COMMON:
+ ld->ld_common_no_alloc = 1;
+ break;
+ case KEY_NO_GC_SECTIONS:
+ ld->ld_gc = 0;
+ break;
+ case KEY_NO_PRINT_GC_SECTIONS:
+ ld->ld_gc_print = 0;
+ break;
+ case KEY_NO_WHOLE_ARCHIVE:
+ ls->ls_whole_archive = 0;
+ break;
+ case KEY_OFORMAT:
+ ld_output_format(ld, arg, arg, arg);
+ break;
+ case KEY_PIE:
+ ld->ld_exec = 0;
+ ld->ld_pie = 1;
+ ld->ld_dynamic_link = 1;
+ break;
+ case KEY_PRINT_GC_SECTIONS:
+ ld->ld_gc_print = 1;
+ break;
+ case KEY_RPATH:
+ ld_path_add_multiple(ld, arg, LPT_RPATH);
+ break;
+ case KEY_RPATH_LINK:
+ ld_path_add_multiple(ld, arg, LPT_RPATH_LINK);
+ break;
+ case KEY_SHARED:
+ ld->ld_exec = 0;
+ ld->ld_dso = 1;
+ ld->ld_dynamic_link = 1;
+ break;
+ case KEY_STATIC:
+ ls->ls_static = 1;
+ break;
+ case KEY_WHOLE_ARCHIVE:
+ ls->ls_whole_archive = 1;
+ break;
+ case KEY_FILE:
+ ld_file_add(ld, arg, LFT_UNKNOWN);
+ break;
+ case KEY_VERSION_SCRIPT:
+ ld_script_parse(arg);
+ break;
+ case KEY_Z_EXEC_STACK:
+ ld->ld_gen_gnustack = 1;
+ ld->ld_stack_exec_set = 1;
+ ld->ld_stack_exec = 1;
+ break;
+ case KEY_Z_NO_EXEC_STACK:
+ ld->ld_gen_gnustack = 1;
+ ld->ld_stack_exec_set = 1;
+ ld->ld_stack_exec = 0;
+ break;
+ default:
+ break;
+ }
+}
+
+static void
+_print_version(struct ld *ld)
+{
+
+ (void) printf("%s (%s)\n", ELFTC_GETPROGNAME(), elftc_version());
+ ld->ld_print_version = 1;
+}
+
+static void
+_copy_optarg(struct ld *ld, char **dst, char *src)
+{
+
+ if (*dst != NULL)
+ free(*dst);
+ if ((*dst = strdup(src)) == NULL)
+ ld_fatal_std(ld, "strdup");
+}
+
+struct ld_wildcard *
+ld_wildcard_alloc(struct ld *ld)
+{
+ struct ld_wildcard *lw;
+
+ if ((lw = calloc(1, sizeof(*lw))) == NULL)
+ ld_fatal_std(ld, "calloc");
+
+ return (lw);
+}
+
+void
+ld_wildcard_free(void *ptr)
+{
+ struct ld_wildcard *lw;
+
+ lw = ptr;
+ if (lw == NULL)
+ return;
+
+ free(lw->lw_name);
+ free(lw);
+}