244 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			244 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
/* Copyright (C) 2002 by  Red Hat, Incorporated. All rights reserved.
 | 
						|
 *
 | 
						|
 * Permission to use, copy, modify, and distribute this software
 | 
						|
 * is freely granted, provided that this notice is preserved.
 | 
						|
 */
 | 
						|
#ifndef _NO_WORDEXP
 | 
						|
 | 
						|
#include <sys/param.h>
 | 
						|
#include <sys/stat.h>
 | 
						|
 | 
						|
#include <ctype.h>
 | 
						|
#include <dirent.h>
 | 
						|
#include <errno.h>
 | 
						|
#include <glob.h>
 | 
						|
#include <pwd.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <sys/wait.h>
 | 
						|
#include <sys/queue.h>
 | 
						|
 | 
						|
#include <wordexp.h>
 | 
						|
#include "wordexp2.h"
 | 
						|
 | 
						|
#define MAXLINELEN 500
 | 
						|
 | 
						|
/* Note: This implementation of wordexp requires a version of bash
 | 
						|
   that supports the --wordexp and --protected arguments to be present
 | 
						|
   on the system.  It does not support the WRDE_UNDEF flag. */
 | 
						|
int
 | 
						|
wordexp(const char *__restrict words, wordexp_t *__restrict pwordexp, int flags)
 | 
						|
{
 | 
						|
  FILE *f = NULL;
 | 
						|
  FILE *f_err = NULL;
 | 
						|
  char tmp[MAXLINELEN];
 | 
						|
  int i = 0;
 | 
						|
  int offs = 0;
 | 
						|
  char *iter;
 | 
						|
  pid_t pid;
 | 
						|
  int num_words = 0;
 | 
						|
  int num_bytes = 0;
 | 
						|
  int fd[2];
 | 
						|
  int fd_err[2];
 | 
						|
  int err = WRDE_NOSPACE;
 | 
						|
  ext_wordv_t *wordv = NULL;
 | 
						|
  char *eword;
 | 
						|
  struct ewords_entry *entry;
 | 
						|
 | 
						|
  if (pwordexp == NULL)
 | 
						|
    {
 | 
						|
      return WRDE_NOSPACE;
 | 
						|
    }
 | 
						|
 | 
						|
  if (flags & WRDE_REUSE)
 | 
						|
    wordfree(pwordexp);
 | 
						|
 | 
						|
  if ((flags & WRDE_APPEND) == 0)
 | 
						|
    {
 | 
						|
      pwordexp->we_wordc = 0;
 | 
						|
      pwordexp->we_wordv = NULL;
 | 
						|
    }
 | 
						|
 | 
						|
  if (flags & WRDE_DOOFFS)
 | 
						|
    {
 | 
						|
      offs = pwordexp->we_offs;
 | 
						|
 | 
						|
      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)
 | 
						|
        return err;
 | 
						|
      if (!pwordexp->we_wordv)
 | 
						|
        SLIST_INIT(&wordv->list);
 | 
						|
      pwordexp->we_wordv = wordv->we_wordv;
 | 
						|
 | 
						|
      for (i = 0; i < offs; i++)
 | 
						|
        pwordexp->we_wordv[i] = NULL;
 | 
						|
    }
 | 
						|
 | 
						|
  if (pipe(fd))
 | 
						|
    return err;
 | 
						|
  if (pipe(fd_err))
 | 
						|
    {
 | 
						|
      close(fd[0]);
 | 
						|
      close(fd[1]);
 | 
						|
      return err;
 | 
						|
    }
 | 
						|
  pid = fork();
 | 
						|
 | 
						|
  if (pid == -1)
 | 
						|
    {
 | 
						|
      /* In "parent" process, but fork failed */
 | 
						|
      close(fd_err[0]);
 | 
						|
      close(fd_err[1]);
 | 
						|
      close(fd[0]);
 | 
						|
      close(fd[1]);
 | 
						|
      return err;
 | 
						|
    }
 | 
						|
  else if (pid > 0)
 | 
						|
    {
 | 
						|
      /* In parent process. */
 | 
						|
 | 
						|
      /* Close write end of parent's pipe. */
 | 
						|
      close(fd[1]);
 | 
						|
      close(fd_err[1]);
 | 
						|
 | 
						|
      /* f_err is the standard error from the shell command. */
 | 
						|
      if (!(f_err = fdopen(fd_err[0], "r")))
 | 
						|
        goto cleanup;
 | 
						|
 | 
						|
      /* Check for errors. */
 | 
						|
      if (fgets(tmp, MAXLINELEN, f_err))
 | 
						|
        {
 | 
						|
          if (strstr(tmp, "EOF"))
 | 
						|
            err = WRDE_SYNTAX;
 | 
						|
          else if (strstr(tmp, "`\n'") || strstr(tmp, "`|'")
 | 
						|
                   || strstr(tmp, "`&'") || strstr(tmp, "`;'")
 | 
						|
                   || strstr(tmp, "`<'") || strstr(tmp, "`>'")
 | 
						|
                   || strstr(tmp, "`('") || strstr(tmp, "`)'")
 | 
						|
                   || strstr(tmp, "`{'") || strstr(tmp, "`}'"))
 | 
						|
            err = WRDE_BADCHAR;
 | 
						|
          else if (strstr(tmp, "command substitution"))
 | 
						|
            err = WRDE_CMDSUB;
 | 
						|
          else
 | 
						|
            err = WRDE_SYNTAX;
 | 
						|
 | 
						|
          if (flags & WRDE_SHOWERR)
 | 
						|
            {
 | 
						|
              fprintf(stderr, tmp);
 | 
						|
              while(fgets(tmp, MAXLINELEN, f_err))
 | 
						|
                fprintf(stderr, tmp);
 | 
						|
            }
 | 
						|
 | 
						|
          goto cleanup;
 | 
						|
        }
 | 
						|
 | 
						|
      /* f is the standard output from the shell command. */
 | 
						|
      if (!(f = fdopen(fd[0], "r")))
 | 
						|
        goto cleanup;
 | 
						|
 | 
						|
      /* Get number of words expanded by shell. */
 | 
						|
      if (!fgets(tmp, MAXLINELEN, f))
 | 
						|
        goto cleanup;
 | 
						|
 | 
						|
      if((iter = strchr(tmp, '\n')))
 | 
						|
          *iter = '\0';
 | 
						|
 | 
						|
      num_words = atoi(tmp);
 | 
						|
 | 
						|
      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 + num_words) * sizeof(char *));
 | 
						|
      if (!wordv)
 | 
						|
        return err;
 | 
						|
      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. */
 | 
						|
      if (!fgets(tmp, MAXLINELEN, f))
 | 
						|
        goto cleanup;
 | 
						|
 | 
						|
      if((iter = strchr(tmp, '\n')))
 | 
						|
          *iter = '\0';
 | 
						|
 | 
						|
      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. */
 | 
						|
      if (!fread(entry->ewords, 1, num_bytes + num_words, f))
 | 
						|
        goto cleanup;
 | 
						|
      entry->ewords[num_bytes + num_words] = 0;
 | 
						|
 | 
						|
      /* Store each entry in pwordexp's we_wordv vector. */
 | 
						|
      eword = entry->ewords;
 | 
						|
      for(i = 0; i < num_words; i++, eword = iter)
 | 
						|
        {
 | 
						|
          if (!eword)
 | 
						|
            break;
 | 
						|
          pwordexp->we_wordv[offs + pwordexp->we_wordc + i] = eword;
 | 
						|
          if ((iter = strchr(eword, '\n')))
 | 
						|
            *iter++ = '\0';
 | 
						|
        }
 | 
						|
 | 
						|
      pwordexp->we_wordv[offs + pwordexp->we_wordc + i] = NULL;
 | 
						|
      pwordexp->we_wordc += num_words;
 | 
						|
      if (i == num_words)
 | 
						|
        err = WRDE_SUCCESS;
 | 
						|
 | 
						|
cleanup:
 | 
						|
      if (f)
 | 
						|
        fclose(f);
 | 
						|
      else
 | 
						|
        close(fd[0]);
 | 
						|
      if (f_err)
 | 
						|
        fclose(f_err);
 | 
						|
      else
 | 
						|
        close(fd_err[0]);
 | 
						|
 | 
						|
      /* Wait for child to finish. */
 | 
						|
      waitpid (pid, NULL, 0);
 | 
						|
 | 
						|
      return err;
 | 
						|
    }
 | 
						|
  else
 | 
						|
    {
 | 
						|
      /* In child process. */
 | 
						|
 | 
						|
      /* Close read end of child's pipe. */
 | 
						|
      close(fd[0]);
 | 
						|
      close(fd_err[0]);
 | 
						|
 | 
						|
      /* Pipe standard output to parent process via fd. */
 | 
						|
      if (fd[1] != STDOUT_FILENO)
 | 
						|
        {
 | 
						|
          if (dup2(fd[1], STDOUT_FILENO) == -1)
 | 
						|
            _exit(EXIT_FAILURE);
 | 
						|
          /* fd[1] no longer required. */
 | 
						|
          close(fd[1]);
 | 
						|
        }
 | 
						|
 | 
						|
      /* Pipe standard error to parent process via fd_err. */
 | 
						|
      if (fd_err[1] != STDERR_FILENO)
 | 
						|
        {
 | 
						|
          if (dup2(fd_err[1], STDERR_FILENO) == -1)
 | 
						|
            _exit(EXIT_FAILURE);
 | 
						|
          /* fd_err[1] no longer required. */
 | 
						|
          close(fd_err[1]);
 | 
						|
        }
 | 
						|
 | 
						|
      if (flags & WRDE_NOCMD)
 | 
						|
        execl("/bin/bash", "bash", "--protected", "--wordexp", words, (char *)0);
 | 
						|
      else
 | 
						|
        execl("/bin/bash", "bash", "--wordexp", words, (char *)0);
 | 
						|
      _exit(EXIT_FAILURE);
 | 
						|
    }
 | 
						|
  return WRDE_SUCCESS;
 | 
						|
}
 | 
						|
#endif /* !_NO_WORDEXP  */
 |