* libc/posix/wordfree.c (wordfree): The wrong words are freed

when WRDE_DOOFFS is in use. Restructure the code so that the memory
	needed to be freed is instead kept in an internal linked list...
	* libc/posix/wordexp2.h: ...as defined here...
	* libc/posix/wordexp.c (wordexp): ...and build this internal
	linked list here, avoiding wasteful strdup calls in the process.
This commit is contained in:
Corinna Vinschen 2012-10-09 12:05:51 +00:00
parent 277e7f0e2e
commit 509212aa98
4 changed files with 73 additions and 32 deletions

View File

@ -1,3 +1,12 @@
2012-10-09 Peter Rosin <peda@lysator.liu.se>
* libc/posix/wordfree.c (wordfree): The wrong words are freed
when WRDE_DOOFFS is in use. Restructure the code so that the memory
needed to be freed is instead kept in an internal linked list...
* libc/posix/wordexp2.h: ...as defined here...
* libc/posix/wordexp.c (wordexp): ...and build this internal
linked list here, avoiding wasteful strdup calls in the process.
2012-10-09 Peter Rosin <peda@lysator.liu.se> 2012-10-09 Peter Rosin <peda@lysator.liu.se>
* libc/posix/wordexp.c (wordexp): Return WRDE_NOSPACE on resource * libc/posix/wordexp.c (wordexp): Return WRDE_NOSPACE on resource

View File

@ -18,8 +18,10 @@
#include <string.h> #include <string.h>
#include <unistd.h> #include <unistd.h>
#include <sys/wait.h> #include <sys/wait.h>
#include <sys/queue.h>
#include <wordexp.h> #include <wordexp.h>
#include "wordexp2.h"
#define MAXLINELEN 500 #define MAXLINELEN 500
@ -41,9 +43,9 @@ wordexp(const char *words, wordexp_t *pwordexp, int flags)
int fd[2]; int fd[2];
int fd_err[2]; int fd_err[2];
int err = WRDE_NOSPACE; int err = WRDE_NOSPACE;
char **wordv; ext_wordv_t *wordv = NULL;
char *ewords = NULL;
char *eword; char *eword;
struct ewords_entry *entry;
if (pwordexp == NULL) if (pwordexp == NULL)
{ {
@ -63,10 +65,14 @@ wordexp(const char *words, wordexp_t *pwordexp, int flags)
{ {
offs = pwordexp->we_offs; offs = pwordexp->we_offs;
wordv = (char **)realloc(pwordexp->we_wordv, (pwordexp->we_wordc + offs + 1) * sizeof(char *)); if (pwordexp->we_wordv)
wordv = WE_WORDV_TO_EXT_WORDV(pwordexp->we_wordv);
wordv = (ext_wordv_t *)realloc(wordv, sizeof(ext_wordv_t) + (offs + pwordexp->we_wordc) * sizeof(char *));
if (!wordv) if (!wordv)
return err; return err;
pwordexp->we_wordv = wordv; if (!pwordexp->we_wordv)
SLIST_INIT(&wordv->list);
pwordexp->we_wordv = wordv->we_wordv;
for (i = 0; i < offs; i++) for (i = 0; i < offs; i++)
pwordexp->we_wordv[i] = NULL; pwordexp->we_wordv[i] = NULL;
@ -142,11 +148,14 @@ wordexp(const char *words, wordexp_t *pwordexp, int flags)
num_words = atoi(tmp); num_words = atoi(tmp);
wordv = (char **)realloc(pwordexp->we_wordv, if (pwordexp->we_wordv)
(pwordexp->we_wordc + num_words + offs + 1) * sizeof(char *)); wordv = WE_WORDV_TO_EXT_WORDV(pwordexp->we_wordv);
wordv = (ext_wordv_t *)realloc(wordv, sizeof(ext_wordv_t) + (offs + pwordexp->we_wordc + num_words) * sizeof(char *));
if (!wordv) if (!wordv)
goto cleanup; return err;
pwordexp->we_wordv = wordv; if (!pwordexp->we_wordv)
SLIST_INIT(&wordv->list);
pwordexp->we_wordv = wordv->we_wordv;
/* Get number of bytes required for storage of all num_words words. */ /* Get number of bytes required for storage of all num_words words. */
if (!fgets(tmp, MAXLINELEN, f)) if (!fgets(tmp, MAXLINELEN, f))
@ -157,36 +166,32 @@ wordexp(const char *words, wordexp_t *pwordexp, int flags)
num_bytes = atoi(tmp); num_bytes = atoi(tmp);
if (!(entry = (struct ewords_entry *)malloc(sizeof(struct ewords_entry) + num_bytes + num_words)))
goto cleanup;
SLIST_INSERT_HEAD(&wordv->list, entry, next);
/* Get expansion from the shell output. */ /* Get expansion from the shell output. */
if (!(ewords = (char *)malloc(num_bytes + num_words + 1))) if (!fread(entry->ewords, 1, num_bytes + num_words, f))
goto cleanup; goto cleanup;
if (!fread(ewords, 1, num_bytes + num_words, f)) entry->ewords[num_bytes + num_words] = 0;
goto cleanup;
ewords[num_bytes + num_words] = 0;
/* Store each entry in pwordexp's we_wordv vector. */ /* Store each entry in pwordexp's we_wordv vector. */
eword = ewords; eword = entry->ewords;
for(i = 0; i < num_words; i++) for(i = 0; i < num_words; i++, eword = iter)
{ {
if (eword && (iter = strchr(eword, '\n'))) if (!eword)
*iter = '\0'; break;
pwordexp->we_wordv[offs + pwordexp->we_wordc + i] = eword;
if (!eword || if ((iter = strchr(eword, '\n')))
!(pwordexp->we_wordv[pwordexp->we_wordc + offs + i] = strdup(eword))) *iter++ = '\0';
{
pwordexp->we_wordv[pwordexp->we_wordc + offs + i] = NULL;
pwordexp->we_wordc += i;
goto cleanup;
}
eword = iter ? iter + 1 : iter;
} }
pwordexp->we_wordv[pwordexp->we_wordc + offs + i] = NULL; pwordexp->we_wordv[offs + pwordexp->we_wordc + i] = NULL;
pwordexp->we_wordc += num_words; pwordexp->we_wordc += num_words;
err = WRDE_SUCCESS; if (i == num_words)
err = WRDE_SUCCESS;
cleanup: cleanup:
free(ewords);
if (f) if (f)
fclose(f); fclose(f);
else else

View File

@ -0,0 +1,21 @@
/* Copyright (C) 2012 by Peter Rosin. All rights reserved.
*
* Permission to use, copy, modify, and distribute this software
* is freely granted, provided that this notice is preserved.
*/
#ifndef _WORDEXP2_H_
struct ewords_entry {
SLIST_ENTRY(ewords_entry) next;
char ewords[1];
};
typedef struct {
SLIST_HEAD(ewords_head, ewords_entry) list;
char *we_wordv[1];
} ext_wordv_t;
#define WE_WORDV_TO_EXT_WORDV(wordv) \
(ext_wordv_t *)((void *)(wordv) - offsetof(ext_wordv_t, we_wordv))
#endif /* !_WORDEXP2_H_ */

View File

@ -18,13 +18,15 @@
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
#include <unistd.h> #include <unistd.h>
#include <sys/queue.h>
#include <wordexp.h> #include <wordexp.h>
#include "wordexp2.h"
void void
wordfree(wordexp_t *pwordexp) wordfree(wordexp_t *pwordexp)
{ {
int i; ext_wordv_t *wordv;
if (pwordexp == NULL) if (pwordexp == NULL)
return; return;
@ -32,10 +34,14 @@ wordfree(wordexp_t *pwordexp)
if (pwordexp->we_wordv == NULL) if (pwordexp->we_wordv == NULL)
return; return;
for(i = 0; i < pwordexp->we_wordc; i++) wordv = WE_WORDV_TO_EXT_WORDV(pwordexp->we_wordv);
free(pwordexp->we_wordv[i]); while (!SLIST_EMPTY(&wordv->list)) {
struct ewords_entry *entry = SLIST_FIRST(&wordv->list);
SLIST_REMOVE_HEAD(&wordv->list, next);
free(entry);
}
free(pwordexp->we_wordv); free(wordv);
pwordexp->we_wordv = NULL; pwordexp->we_wordv = NULL;
} }