mirror of
git://git.savannah.gnu.org/libtool.git
synced 2024-11-21 01:40:57 +08:00
ec703fac9e
* AUTHORS, HACKING, Makefile.am, bootstrap.conf, build-aux/edit-readme-alpha, build-aux/no-bogus-m4-defines, cfg.mk, configure.ac, libltdl/configure.ac, libltdl/libltdl/lt__alloc.h, libltdl/libltdl/lt__argz_.h, libltdl/libltdl/lt__dirent.h, libltdl/libltdl/lt__glibc.h, libltdl/libltdl/lt__private.h, libltdl/libltdl/lt__strl.h, libltdl/libltdl/lt_dlloader.h, libltdl/libltdl/lt_error.h, libltdl/libltdl/lt_system.h, libltdl/libltdl/slist.h, libltdl/loaders/dld_link.c, libltdl/loaders/dlopen.c, libltdl/loaders/dyld.c, libltdl/loaders/load_add_on.c, libltdl/loaders/loadlibrary.c, libltdl/loaders/preopen.c, libltdl/loaders/shl_load.c, libltdl/lt__alloc.c, libltdl/lt__argz.c, libltdl/lt__dirent.c, libltdl/lt__strl.c, libltdl/lt_dlloader.c, libltdl/lt_error.c, libltdl/ltdl.c, libltdl/ltdl.h, libltdl/ltdl.mk, libltdl/slist.c, tests/am-subdir.at, tests/archive-in-archive.at, tests/bindir.at, tests/bug_62343.at, tests/cdemo.at, tests/cmdline_wrap.at, tests/configure-funcs.at, tests/configure-iface.at, tests/convenience.at, tests/ctor.at, tests/cwrapper.at, tests/darwin.at, tests/demo.at, tests/depdemo.at, tests/deplib-in-subdir.at, tests/deplibs-ident.at, tests/deplibs-mingw.at, tests/destdir.at, tests/dlloader-api.at, tests/dumpbin-symbols.at, tests/duplicate_conv.at, tests/duplicate_deps.at, tests/duplicate_members.at, tests/early-libtool.at, tests/exceptions.at, tests/execute-mode.at, tests/exeext.at, tests/export-def.at, tests/export.at, tests/f77demo.at, tests/fail.at, tests/fcdemo.at, tests/flags.at, tests/help.at, tests/indirect_deps.at, tests/infer-tag.at, tests/inherited_flags.at, tests/install.at, tests/lalib-syntax.at, tests/libtool.at, tests/libtoolize.at, tests/link-order.at, tests/link-order2.at, tests/loadlibrary.at, tests/localization.at, tests/lt_dladvise.at, tests/lt_dlexit.at, tests/lt_dlopen.at, tests/lt_dlopen_a.at, tests/lt_dlopenext.at, tests/ltdl-api.at, tests/ltdl-libdir.at, tests/mdemo.at, tests/need_lib_prefix.at, tests/no-executables.at, tests/nocase.at, tests/nonrecursive.at, tests/old-m4-iface.at, tests/pic_flag.at, tests/recursive.at, tests/resident.at, tests/runpath-in-lalib.at, tests/search-path.at, tests/shlibpath.at, tests/slist.at, tests/standalone.at, tests/static.at, tests/stresstest.at, tests/subproject.at, tests/sysroot.at, tests/tagdemo.at, tests/template.at, tests/testsuite.at, tests/versioning.at, tests/with-pic.at: Replace FSF address in each file's license block with a URL that points to licenses online hosted by GNU.
223 lines
5.6 KiB
C
223 lines
5.6 KiB
C
/* lt__argz.c -- argz implementation for non-glibc systems
|
|
|
|
Copyright (C) 2004, 2006-2008, 2011-2019, 2021-2024 Free Software
|
|
Foundation, Inc.
|
|
Written by Gary V. Vaughan, 2004
|
|
|
|
NOTE: The canonical source of this file is maintained with the
|
|
GNU Libtool package. Report bugs to bug-libtool@gnu.org.
|
|
|
|
GNU Libltdl is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2 of the License, or (at your option) any later version.
|
|
|
|
As a special exception to the GNU Lesser General Public License,
|
|
if you distribute this file as part of a program or library that
|
|
is built using GNU Libtool, you may include this file under the
|
|
same distribution terms that you use for the rest of that program.
|
|
|
|
GNU Libltdl is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with GNU Libltdl. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#if defined LTDL && defined LT_CONFIG_H
|
|
# include LT_CONFIG_H
|
|
#else
|
|
# include <config.h>
|
|
#endif
|
|
|
|
#include <lt__argz.h>
|
|
|
|
#include <assert.h>
|
|
#include <stdlib.h>
|
|
#include <sys/types.h>
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
|
|
#define EOS_CHAR '\0'
|
|
|
|
error_t
|
|
argz_append (char **pargz, size_t *pargz_len, const char *buf, size_t buf_len)
|
|
{
|
|
size_t argz_len;
|
|
char *argz;
|
|
|
|
assert (pargz);
|
|
assert (pargz_len);
|
|
assert ((*pargz && *pargz_len) || (!*pargz && !*pargz_len));
|
|
|
|
/* If nothing needs to be appended, no more work is required. */
|
|
if (buf_len == 0)
|
|
return 0;
|
|
|
|
/* Ensure there is enough room to append BUF_LEN. */
|
|
argz_len = *pargz_len + buf_len;
|
|
argz = (char *) realloc (*pargz, argz_len);
|
|
if (!argz)
|
|
return ENOMEM;
|
|
|
|
/* Copy characters from BUF after terminating '\0' in ARGZ. */
|
|
memcpy (argz + *pargz_len, buf, buf_len);
|
|
|
|
/* Assign new values. */
|
|
*pargz = argz;
|
|
*pargz_len = argz_len;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
error_t
|
|
argz_create_sep (const char *str, int delim, char **pargz, size_t *pargz_len)
|
|
{
|
|
size_t argz_len;
|
|
char *argz = 0;
|
|
|
|
assert (str);
|
|
assert (pargz);
|
|
assert (pargz_len);
|
|
|
|
/* Make a copy of STR, but replacing each occurrence of
|
|
DELIM with '\0'. */
|
|
argz_len = 1+ strlen (str);
|
|
if (argz_len)
|
|
{
|
|
const char *p;
|
|
char *q;
|
|
|
|
argz = (char *) malloc (argz_len);
|
|
if (!argz)
|
|
return ENOMEM;
|
|
|
|
for (p = str, q = argz; *p != EOS_CHAR; ++p)
|
|
{
|
|
if (*p == delim)
|
|
{
|
|
/* Ignore leading delimiters, and fold consecutive
|
|
delimiters in STR into a single '\0' in ARGZ. */
|
|
if ((q > argz) && (q[-1] != EOS_CHAR))
|
|
*q++ = EOS_CHAR;
|
|
else
|
|
--argz_len;
|
|
}
|
|
else
|
|
*q++ = *p;
|
|
}
|
|
/* Copy terminating EOS_CHAR. */
|
|
*q = *p;
|
|
}
|
|
|
|
/* If ARGZ_LEN has shrunk to nothing, release ARGZ's memory. */
|
|
if (!argz_len)
|
|
argz = (free (argz), (char *) 0);
|
|
|
|
/* Assign new values. */
|
|
*pargz = argz;
|
|
*pargz_len = argz_len;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
error_t
|
|
argz_insert (char **pargz, size_t *pargz_len, char *before, const char *entry)
|
|
{
|
|
assert (pargz);
|
|
assert (pargz_len);
|
|
assert (entry && *entry);
|
|
|
|
/* No BEFORE address indicates ENTRY should be inserted after the
|
|
current last element. */
|
|
if (!before)
|
|
return argz_append (pargz, pargz_len, entry, 1+ strlen (entry));
|
|
|
|
/* This probably indicates a programmer error, but to preserve
|
|
semantics, scan back to the start of an entry if BEFORE points
|
|
into the middle of it. */
|
|
while ((before > *pargz) && (before[-1] != EOS_CHAR))
|
|
--before;
|
|
|
|
{
|
|
size_t entry_len = 1+ strlen (entry);
|
|
size_t argz_len = *pargz_len + entry_len;
|
|
size_t offset = before - *pargz;
|
|
char *argz = (char *) realloc (*pargz, argz_len);
|
|
|
|
if (!argz)
|
|
return ENOMEM;
|
|
|
|
/* Make BEFORE point to the equivalent offset in ARGZ that it
|
|
used to have in *PARGZ in case realloc() moved the block. */
|
|
before = argz + offset;
|
|
|
|
/* Move the ARGZ entries starting at BEFORE up into the new
|
|
space at the end -- making room to copy ENTRY into the
|
|
resulting gap. */
|
|
memmove (before + entry_len, before, *pargz_len - offset);
|
|
memcpy (before, entry, entry_len);
|
|
|
|
/* Assign new values. */
|
|
*pargz = argz;
|
|
*pargz_len = argz_len;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
char *
|
|
argz_next (char *argz, size_t argz_len, const char *entry)
|
|
{
|
|
assert ((argz && argz_len) || (!argz && !argz_len));
|
|
|
|
if (entry)
|
|
{
|
|
/* Either ARGZ/ARGZ_LEN is empty, or ENTRY points into an address
|
|
within the ARGZ vector. */
|
|
assert ((!argz && !argz_len)
|
|
|| ((argz <= entry) && (entry < (argz + argz_len))));
|
|
|
|
/* Move to the char immediately after the terminating
|
|
'\0' of ENTRY. */
|
|
entry = 1+ strchr (entry, EOS_CHAR);
|
|
|
|
/* Return either the new ENTRY, or else NULL if ARGZ is
|
|
exhausted. */
|
|
return (entry >= argz + argz_len) ? 0 : (char *) entry;
|
|
}
|
|
else
|
|
{
|
|
/* This should probably be flagged as a programmer error,
|
|
since starting an argz_next loop with the iterator set
|
|
to ARGZ is safer. To preserve semantics, handle the NULL
|
|
case by returning the start of ARGZ (if any). */
|
|
if (argz_len > 0)
|
|
return argz;
|
|
else
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
argz_stringify (char *argz, size_t argz_len, int sep)
|
|
{
|
|
assert ((argz && argz_len) || (!argz && !argz_len));
|
|
|
|
if (sep)
|
|
{
|
|
--argz_len; /* don't stringify the terminating EOS */
|
|
while (--argz_len > 0)
|
|
{
|
|
if (argz[argz_len] == EOS_CHAR)
|
|
argz[argz_len] = sep;
|
|
}
|
|
}
|
|
}
|