From fc29ad10fd1e927a2b1584862faa69e7799fbe31 Mon Sep 17 00:00:00 2001 From: John Polstra Date: Tue, 6 Feb 1996 22:49:11 +0000 Subject: [PATCH] Fixed pkg_create to eliminate the limits on the size of the PLIST. Formerly, there were limits on both the number of files (hard-coded into the program) and the number of characters (because of the ARG_MAX limit in exec(2)). In this new version, the filenames are passed to tar through a pipe, using tar's "-T" option, rather than on the command line. --- usr.sbin/pkg_install/create/perform.c | 72 +++++++++++++++------------ 1 file changed, 41 insertions(+), 31 deletions(-) diff --git a/usr.sbin/pkg_install/create/perform.c b/usr.sbin/pkg_install/create/perform.c index b1a67ec82162..178b3a4307f8 100644 --- a/usr.sbin/pkg_install/create/perform.c +++ b/usr.sbin/pkg_install/create/perform.c @@ -1,5 +1,5 @@ #ifndef lint -static const char *rcsid = "$Id: perform.c,v 1.27.4.1 1995/10/14 19:11:22 jkh Exp $"; +static const char *rcsid = "$Id: perform.c,v 1.28 1995/10/25 15:38:01 jkh Exp $"; #endif /* @@ -25,8 +25,10 @@ static const char *rcsid = "$Id: perform.c,v 1.27.4.1 1995/10/14 19:11:22 jkh Ex #include "lib.h" #include "create.h" +#include #include #include +#include static void sanity_check(void); static void make_dist(char *, char *, char *, Package *); @@ -177,21 +179,17 @@ pkg_perform(char **pkgs) return TRUE; /* Success */ } -/* - * There doesn't seem to be a sysconf() datum for telling you how *many* - * arguments you can have, only how long the maximum list can be. - * Shamelessly guess at what seems to be a conservative value. - */ -#define MAX_NUM_ARGS 16384 - static void make_dist(char *home, char *pkg, char *suffix, Package *plist) { char tball[FILENAME_MAX]; PackingList p; int ret, max, len; - char *args[MAX_NUM_ARGS]; + char *args[50]; /* Much more than enough. */ int nargs = 0; + int pipefds[2]; + FILE *totar; + pid_t pid; args[nargs++] = "tar"; /* argv[0] */ @@ -211,44 +209,56 @@ make_dist(char *home, char *pkg, char *suffix, Package *plist) args[nargs++] = "-X"; args[nargs++] = ExcludeFrom; } + args[nargs++] = "-T"; /* Take filenames from file instead of args. */ + args[nargs++] = "-"; /* Use stdin for the file. */ + args[nargs] = NULL; if (Verbose) printf("Creating gzip'd tar ball in '%s'\n", tball); - args[nargs++] = CONTENTS_FNAME; - args[nargs++] = COMMENT_FNAME; - args[nargs++] = DESC_FNAME; + /* Set up a pipe for passing the filenames, and fork off a tar process. */ + if (pipe(pipefds) == -1) + barf("Cannot create pipe: %s", strerror(errno)); + if ((pid = fork()) == -1) + barf("Cannot fork process for tar: %s", strerror(errno)); + if (pid == 0) { /* The child */ + dup2(pipefds[0], 0); + close(pipefds[1]); + execv("/usr/bin/tar", args); + barf("Failed to execute tar command: %s", strerror(errno)); + } + + /* Meanwhile, back in the parent process ... */ + close(pipefds[0]); + if ((totar = fdopen(pipefds[1], "w")) == NULL) + barf("fdopen failed: %s", strerror(errno)); + + fprintf(totar, "%s\n", CONTENTS_FNAME); + fprintf(totar, "%s\n", COMMENT_FNAME); + fprintf(totar, "%s\n", DESC_FNAME); if (Install) - args[nargs++] = INSTALL_FNAME; + fprintf(totar, "%s\n", INSTALL_FNAME); if (DeInstall) - args[nargs++] = DEINSTALL_FNAME; + fprintf(totar, "%s\n", DEINSTALL_FNAME); if (Require) - args[nargs++] = REQUIRE_FNAME; + fprintf(totar, "%s\n", REQUIRE_FNAME); if (Display) - args[nargs++] = DISPLAY_FNAME; + fprintf(totar, "%s\n", DISPLAY_FNAME); if (Mtree) - args[nargs++] = MTREE_FNAME; + fprintf(totar, "%s\n", MTREE_FNAME); for (p = plist->head; p; p = p->next) { - if (nargs == (MAX_NUM_ARGS - 1)) - barf("More than %d arguments given in plist! :-(", MAX_NUM_ARGS); if (p->type == PLIST_FILE) - args[nargs++] = p->name; - else if (p->type == PLIST_CWD || p->type == PLIST_SRC) { - args[nargs++] = "-C"; - args[nargs++] = p->name; - } + fprintf(totar, "%s\n", p->name); + else if (p->type == PLIST_CWD || p->type == PLIST_SRC) + fprintf(totar, "-C\n%s\n", p->name); else if (p->type == PLIST_IGNORE) p = p->next; } - args[nargs] = NULL; - if (fork() == 0) { - execv("/usr/bin/tar", args); - barf("Failed to execute tar command!"); - } - else - wait(&ret); + + fclose(totar); + wait(&ret); /* assume either signal or bad exit is enough for us */ if (ret) barf("tar command failed with code %d", ret);