diff options
Diffstat (limited to 'gnu/usr.bin/patch')
| -rw-r--r-- | gnu/usr.bin/patch/EXTERN.h | 24 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/INTERN.h | 22 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/Makefile | 6 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/backupfile.c | 402 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/backupfile.h | 46 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/common.h | 197 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/config.h | 81 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/getopt.c | 731 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/getopt.h | 129 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/getopt1.c | 176 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/inp.c | 372 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/inp.h | 21 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/patch.1 | 591 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/patch.c | 956 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/patchlevel.h | 1 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/pch.c | 1316 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/pch.h | 39 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/util.c | 433 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/util.h | 91 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/version.c | 28 | ||||
| -rw-r--r-- | gnu/usr.bin/patch/version.h | 12 | 
21 files changed, 5674 insertions, 0 deletions
| diff --git a/gnu/usr.bin/patch/EXTERN.h b/gnu/usr.bin/patch/EXTERN.h new file mode 100644 index 000000000000..61ae1c3673ab --- /dev/null +++ b/gnu/usr.bin/patch/EXTERN.h @@ -0,0 +1,24 @@ +/* $Header: /home/cvs/386BSD/src/gnu/usr.bin/patch/EXTERN.h,v 1.1.1.1 1993/06/19 14:21:52 paul Exp $ + * + * $Log: EXTERN.h,v $ + * Revision 1.1.1.1  1993/06/19  14:21:52  paul + * b-maked patch-2.10 + * + * Revision 2.0  86/09/17  15:35:37  lwall + * Baseline for netwide release. + *  + */ + +#ifdef EXT +#undef EXT +#endif +#define EXT extern + +#ifdef INIT +#undef INIT +#endif +#define INIT(x) + +#ifdef DOINIT +#undef DOINIT +#endif diff --git a/gnu/usr.bin/patch/INTERN.h b/gnu/usr.bin/patch/INTERN.h new file mode 100644 index 000000000000..79fecd9b1939 --- /dev/null +++ b/gnu/usr.bin/patch/INTERN.h @@ -0,0 +1,22 @@ +/* $Header: /home/cvs/386BSD/src/gnu/usr.bin/patch/INTERN.h,v 1.1.1.1 1993/06/19 14:21:52 paul Exp $ + * + * $Log: INTERN.h,v $ + * Revision 1.1.1.1  1993/06/19  14:21:52  paul + * b-maked patch-2.10 + * + * Revision 2.0  86/09/17  15:35:58  lwall + * Baseline for netwide release. + *  + */ + +#ifdef EXT +#undef EXT +#endif +#define EXT + +#ifdef INIT +#undef INIT +#endif +#define INIT(x) = x + +#define DOINIT diff --git a/gnu/usr.bin/patch/Makefile b/gnu/usr.bin/patch/Makefile new file mode 100644 index 000000000000..d614e89235da --- /dev/null +++ b/gnu/usr.bin/patch/Makefile @@ -0,0 +1,6 @@ +PROG= 	patch +SRCS = backupfile.c getopt.c getopt1.c inp.c patch.c pch.c util.c \ +	version.c   +CFLAGS += -DHAVE_CONFIG_H +MAN=	patch.1 +.include <bsd.prog.mk> diff --git a/gnu/usr.bin/patch/backupfile.c b/gnu/usr.bin/patch/backupfile.c new file mode 100644 index 000000000000..d7eb5874dd78 --- /dev/null +++ b/gnu/usr.bin/patch/backupfile.c @@ -0,0 +1,402 @@ +/* backupfile.c -- make Emacs style backup file names +   Copyright (C) 1990, 1991, 1992 Free Software Foundation, Inc. + +   This program is free software; you can redistribute it and/or modify +   it under the terms of the GNU General Public License as published by +   the Free Software Foundation; either version 2, or (at your option) +   any later version. + +   This program 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 General Public License for more details. + +   You should have received a copy of the GNU General Public License +   along with this program; if not, write to the Free Software +   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.  */ + +/* Written by David MacKenzie <djm@gnu.ai.mit.edu>. +   Some algorithms adapted from GNU Emacs. */ + +#include "config.h" +#include <stdio.h> +#include <ctype.h> +#include <sys/types.h> +#include "backupfile.h" +#ifdef STDC_HEADERS +#include <string.h> +#include <stdlib.h> +#else +char *malloc (); +#endif + +#if defined (HAVE_UNISTD_H) +#include <unistd.h> +#endif + +#if defined(DIRENT) || defined(_POSIX_VERSION) +#include <dirent.h> +#define NLENGTH(direct) (strlen((direct)->d_name)) +#else /* not (DIRENT or _POSIX_VERSION) */ +#define dirent direct +#define NLENGTH(direct) ((direct)->d_namlen) +#ifdef SYSNDIR +#include <sys/ndir.h> +#endif /* SYSNDIR */ +#ifdef SYSDIR +#include <sys/dir.h> +#endif /* SYSDIR */ +#ifdef NDIR +#include <ndir.h> +#endif /* NDIR */ +#endif /* DIRENT or _POSIX_VERSION */ + +#ifndef isascii +#define ISDIGIT(c) (isdigit ((unsigned char) (c))) +#else +#define ISDIGIT(c) (isascii (c) && isdigit (c)) +#endif + +#if defined (_POSIX_VERSION) +/* POSIX does not require that the d_ino field be present, and some +   systems do not provide it. */ +#define REAL_DIR_ENTRY(dp) 1 +#else +#define REAL_DIR_ENTRY(dp) ((dp)->d_ino != 0) +#endif + +/* Which type of backup file names are generated. */ +enum backup_type backup_type = none; + +/* The extension added to file names to produce a simple (as opposed +   to numbered) backup file name. */ +char *simple_backup_suffix = "~"; + +char *basename (); +char *dirname (); +static char *concat (); +char *find_backup_file_name (); +static char *make_version_name (); +static int max_backup_version (); +static int version_number (); + +/* Return NAME with any leading path stripped off.  */ + +char * +basename (name) +     char *name; +{ +  char *r = name, *p = name; + +  while (*p) +    if (*p++ == '/') +      r = p; +  return r; +} + +#ifndef NODIR +/* Return the name of the new backup file for file FILE, +   allocated with malloc.  Return 0 if out of memory. +   FILE must not end with a '/' unless it is the root directory. +   Do not call this function if backup_type == none. */ + +char * +find_backup_file_name (file) +     char *file; +{ +  char *dir; +  char *base_versions; +  int highest_backup; + +  if (backup_type == simple) +    { +      char *s = malloc (strlen (file) + strlen (simple_backup_suffix) + 1); +      strcpy (s, file); +      addext (s, simple_backup_suffix, '~'); +      return s; +    } +  base_versions = concat (basename (file), ".~"); +  if (base_versions == 0) +    return 0; +  dir = dirname (file); +  if (dir == 0) +    { +      free (base_versions); +      return 0; +    } +  highest_backup = max_backup_version (base_versions, dir); +  free (base_versions); +  free (dir); +  if (backup_type == numbered_existing && highest_backup == 0) +    return concat (file, simple_backup_suffix); +  return make_version_name (file, highest_backup + 1); +} + +/* Return the number of the highest-numbered backup file for file +   FILE in directory DIR.  If there are no numbered backups +   of FILE in DIR, or an error occurs reading DIR, return 0. +   FILE should already have ".~" appended to it. */ + +static int +max_backup_version (file, dir) +     char *file, *dir; +{ +  DIR *dirp; +  struct dirent *dp; +  int highest_version; +  int this_version; +  int file_name_length; +   +  dirp = opendir (dir); +  if (!dirp) +    return 0; +   +  highest_version = 0; +  file_name_length = strlen (file); + +  while ((dp = readdir (dirp)) != 0) +    { +      if (!REAL_DIR_ENTRY (dp) || NLENGTH (dp) <= file_name_length) +	continue; +       +      this_version = version_number (file, dp->d_name, file_name_length); +      if (this_version > highest_version) +	highest_version = this_version; +    } +  closedir (dirp); +  return highest_version; +} + +/* Return a string, allocated with malloc, containing +   "FILE.~VERSION~".  Return 0 if out of memory. */ + +static char * +make_version_name (file, version) +     char *file; +     int version; +{ +  char *backup_name; + +  backup_name = malloc (strlen (file) + 16); +  if (backup_name == 0) +    return 0; +  sprintf (backup_name, "%s.~%d~", file, version); +  return backup_name; +} + +/* If BACKUP is a numbered backup of BASE, return its version number; +   otherwise return 0.  BASE_LENGTH is the length of BASE. +   BASE should already have ".~" appended to it. */ + +static int +version_number (base, backup, base_length) +     char *base; +     char *backup; +     int base_length; +{ +  int version; +  char *p; +   +  version = 0; +  if (!strncmp (base, backup, base_length) && ISDIGIT (backup[base_length])) +    { +      for (p = &backup[base_length]; ISDIGIT (*p); ++p) +	version = version * 10 + *p - '0'; +      if (p[0] != '~' || p[1]) +	version = 0; +    } +  return version; +} + +/* Return the newly-allocated concatenation of STR1 and STR2. +   If out of memory, return 0. */ + +static char * +concat (str1, str2) +     char *str1, *str2; +{ +  char *newstr; +  char str1_length = strlen (str1); + +  newstr = malloc (str1_length + strlen (str2) + 1); +  if (newstr == 0) +    return 0; +  strcpy (newstr, str1); +  strcpy (newstr + str1_length, str2); +  return newstr; +} + +/* Return the leading directories part of PATH, +   allocated with malloc.  If out of memory, return 0. +   Assumes that trailing slashes have already been +   removed.  */ + +char * +dirname (path) +     char *path; +{ +  char *newpath; +  char *slash; +  int length;    /* Length of result, not including NUL. */ + +  slash = basename (path); +  if (slash == path) +	{ +	  /* File is in the current directory.  */ +	  path = "."; +	  length = 1; +	} +  else +	{ +	  /* Remove any trailing slashes from result. */ +	  while (*--slash == '/' && slash > path) +	    ; + +	  length = slash - path + 1; +	} +  newpath = malloc (length + 1); +  if (newpath == 0) +    return 0; +  strncpy (newpath, path, length); +  newpath[length] = 0; +  return newpath; +} + +/* If ARG is an unambiguous match for an element of the +   null-terminated array OPTLIST, return the index in OPTLIST +   of the matched element, else -1 if it does not match any element +   or -2 if it is ambiguous (is a prefix of more than one element). */ + +int +argmatch (arg, optlist) +     char *arg; +     char **optlist; +{ +  int i;			/* Temporary index in OPTLIST. */ +  int arglen;			/* Length of ARG. */ +  int matchind = -1;		/* Index of first nonexact match. */ +  int ambiguous = 0;		/* If nonzero, multiple nonexact match(es). */ +   +  arglen = strlen (arg); +   +  /* Test all elements for either exact match or abbreviated matches.  */ +  for (i = 0; optlist[i]; i++) +    { +      if (!strncmp (optlist[i], arg, arglen)) +	{ +	  if (strlen (optlist[i]) == arglen) +	    /* Exact match found.  */ +	    return i; +	  else if (matchind == -1) +	    /* First nonexact match found.  */ +	    matchind = i; +	  else +	    /* Second nonexact match found.  */ +	    ambiguous = 1; +	} +    } +  if (ambiguous) +    return -2; +  else +    return matchind; +} + +/* Error reporting for argmatch. +   KIND is a description of the type of entity that was being matched. +   VALUE is the invalid value that was given. +   PROBLEM is the return value from argmatch. */ + +void +invalid_arg (kind, value, problem) +     char *kind; +     char *value; +     int problem; +{ +  fprintf (stderr, "patch: "); +  if (problem == -1) +    fprintf (stderr, "invalid"); +  else				/* Assume -2. */ +    fprintf (stderr, "ambiguous"); +  fprintf (stderr, " %s `%s'\n", kind, value); +} + +static char *backup_args[] = +{ +  "never", "simple", "nil", "existing", "t", "numbered", 0 +}; + +static enum backup_type backup_types[] = +{ +  simple, simple, numbered_existing, numbered_existing, numbered, numbered +}; + +/* Return the type of backup indicated by VERSION. +   Unique abbreviations are accepted. */ + +enum backup_type +get_version (version) +     char *version; +{ +  int i; + +  if (version == 0 || *version == 0) +    return numbered_existing; +  i = argmatch (version, backup_args); +  if (i >= 0) +    return backup_types[i]; +  invalid_arg ("version control type", version, i); +  exit (1); +} +#endif /* NODIR */ + +/* Append to FILENAME the extension EXT, unless the result would be too long, +   in which case just append the character E.  */ + +void +addext (filename, ext, e) +     char *filename, *ext; +     int e; +{ +  char *s = basename (filename); +  int slen = strlen (s), extlen = strlen (ext); +  long slen_max = -1; + +#if HAVE_PATHCONF && defined (_PC_NAME_MAX) +#ifndef _POSIX_NAME_MAX +#define _POSIX_NAME_MAX 14 +#endif +  if (slen + extlen <= _POSIX_NAME_MAX) +    /* The file name is so short there's no need to call pathconf.  */ +    slen_max = _POSIX_NAME_MAX; +  else if (s == filename) +    slen_max = pathconf (".", _PC_NAME_MAX); +  else +    { +      char c = *s; +      *s = 0; +      slen_max = pathconf (filename, _PC_NAME_MAX); +      *s = c; +    } +#endif +  if (slen_max == -1) { +#ifdef HAVE_LONG_FILE_NAMES +    slen_max = 255; +#else +    slen_max = 14; +#endif +  } +  if (slen + extlen <= slen_max) +    strcpy (s + slen, ext); +  else +    { +      if (slen_max <= slen) { +	/* Try to preserve difference between .h .c etc.  */ +	if (slen == slen_max && s[slen - 2] == '.') +	  s[slen - 2] = s[slen - 1]; + +	slen = slen_max - 1; +      } +      s[slen] = e; +      s[slen + 1] = 0; +    } +} diff --git a/gnu/usr.bin/patch/backupfile.h b/gnu/usr.bin/patch/backupfile.h new file mode 100644 index 000000000000..dfd1fc4e138e --- /dev/null +++ b/gnu/usr.bin/patch/backupfile.h @@ -0,0 +1,46 @@ +/* backupfile.h -- declarations for making Emacs style backup file names +   Copyright (C) 1990, 1991, 1992 Free Software Foundation, Inc. + +   This program is free software; you can redistribute it and/or modify +   it under the terms of the GNU General Public License as published by +   the Free Software Foundation; either version 2, or (at your option) +   any later version. + +   This program 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 General Public License for more details. + +   You should have received a copy of the GNU General Public License +   along with this program; if not, write to the Free Software +   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.  */ + +/* When to make backup files. */ +enum backup_type +{ +  /* Never make backups. */ +  none, + +  /* Make simple backups of every file. */ +  simple, + +  /* Make numbered backups of files that already have numbered backups, +     and simple backups of the others. */ +  numbered_existing, + +  /* Make numbered backups of every file. */ +  numbered +}; + +extern enum backup_type backup_type; +extern char *simple_backup_suffix; + +#ifdef __STDC__ +char *find_backup_file_name (char *file); +enum backup_type get_version (char *version); +void addext (char *, char *, int); +#else +char *find_backup_file_name (); +enum backup_type get_version (); +void addext (); +#endif diff --git a/gnu/usr.bin/patch/common.h b/gnu/usr.bin/patch/common.h new file mode 100644 index 000000000000..175e97b2d176 --- /dev/null +++ b/gnu/usr.bin/patch/common.h @@ -0,0 +1,197 @@ +/* $Header: /home/cvs/386BSD/src/gnu/usr.bin/patch/common.h,v 1.2 1993/09/22 16:51:03 rich Exp $ + * + * $Log: common.h,v $ + * Revision 1.2  1993/09/22  16:51:03  rich + * Increaed the fixed static buffer sizes and maximum hunk size that + * otherwise causes the XFree86 1.3 patch set to fail. + * + * Revision 1.1.1.1  1993/06/19  14:21:52  paul + * b-maked patch-2.10 + * + * Revision 2.0.1.2  88/06/22  20:44:53  lwall + * patch12: sprintf was declared wrong + *  + * Revision 2.0.1.1  88/06/03  15:01:56  lwall + * patch10: support for shorter extensions. + *  + * Revision 2.0  86/09/17  15:36:39  lwall + * Baseline for netwide release. + *  + */ + +#define DEBUGGING + +#define VOIDUSED 7 +#include "config.h" + +/* shut lint up about the following when return value ignored */ + +#define Signal (void)signal +#define Unlink (void)unlink +#define Lseek (void)lseek +#define Fseek (void)fseek +#define Fstat (void)fstat +#define Pclose (void)pclose +#define Close (void)close +#define Fclose (void)fclose +#define Fflush (void)fflush +#define Sprintf (void)sprintf +#define Mktemp (void)mktemp +#define Strcpy (void)strcpy +#define Strcat (void)strcat + +/* NeXT declares malloc and realloc incompatibly from us in some of +   these files.  Temporarily redefine them to prevent errors.  */ +#define malloc system_malloc +#define realloc system_realloc +#include <stdio.h> +#include <assert.h> +#include <sys/types.h> +#include <sys/stat.h> +#include <ctype.h> +#include <signal.h> +#undef malloc +#undef realloc + +/* constants */ + +/* AIX predefines these.  */ +#ifdef TRUE +#undef TRUE +#endif +#ifdef FALSE +#undef FALSE +#endif +#define TRUE (1) +#define FALSE (0) + +#define MAXHUNKSIZE 200000		/* is this enough lines? */ +#define INITHUNKMAX 125			/* initial dynamic allocation size */ +#define MAXLINELEN 4096 +#define BUFFERSIZE 4096 + +#define SCCSPREFIX "s." +#define GET "get %s" +#define GET_LOCKED "get -e %s" +#define SCCSDIFF "get -p %s | diff - %s >/dev/null" + +#define RCSSUFFIX ",v" +#define CHECKOUT "co %s" +#define CHECKOUT_LOCKED "co -l %s" +#define RCSDIFF "rcsdiff %s > /dev/null" + +/* handy definitions */ + +#define Null(t) ((t)0) +#define Nullch Null(char *) +#define Nullfp Null(FILE *) +#define Nulline Null(LINENUM) + +#define Ctl(ch) ((ch) & 037) + +#define strNE(s1,s2) (strcmp(s1, s2)) +#define strEQ(s1,s2) (!strcmp(s1, s2)) +#define strnNE(s1,s2,l) (strncmp(s1, s2, l)) +#define strnEQ(s1,s2,l) (!strncmp(s1, s2, l)) + +/* typedefs */ + +typedef char bool; +typedef long LINENUM;			/* must be signed */ +typedef unsigned MEM;			/* what to feed malloc */ + +/* globals */ + +EXT int Argc;				/* guess */ +EXT char **Argv; +EXT int optind_last;			/* for restarting plan_b */ + +EXT struct stat filestat;		/* file statistics area */ +EXT int filemode INIT(0644); + +EXT char buf[MAXLINELEN];		/* general purpose buffer */ +EXT FILE *ofp INIT(Nullfp);		/* output file pointer */ +EXT FILE *rejfp INIT(Nullfp);		/* reject file pointer */ + +EXT int myuid;				/* cache getuid return value */ + +EXT bool using_plan_a INIT(TRUE);	/* try to keep everything in memory */ +EXT bool out_of_mem INIT(FALSE);	/* ran out of memory in plan a */ + +#define MAXFILEC 2 +EXT int filec INIT(0);			/* how many file arguments? */ +EXT char *filearg[MAXFILEC]; +EXT bool ok_to_create_file INIT(FALSE); +EXT char *bestguess INIT(Nullch);	/* guess at correct filename */ + +EXT char *outname INIT(Nullch); +EXT char rejname[128]; + +EXT char *origprae INIT(Nullch); + +EXT char *TMPOUTNAME; +EXT char *TMPINNAME; +EXT char *TMPREJNAME; +EXT char *TMPPATNAME; +EXT bool toutkeep INIT(FALSE); +EXT bool trejkeep INIT(FALSE); + +EXT LINENUM last_offset INIT(0); +#ifdef DEBUGGING +EXT int debug INIT(0); +#endif +EXT LINENUM maxfuzz INIT(2); +EXT bool force INIT(FALSE); +EXT bool batch INIT(FALSE); +EXT bool verbose INIT(TRUE); +EXT bool reverse INIT(FALSE); +EXT bool noreverse INIT(FALSE); +EXT bool skip_rest_of_patch INIT(FALSE); +EXT int strippath INIT(957); +EXT bool canonicalize INIT(FALSE); + +#define CONTEXT_DIFF 1 +#define NORMAL_DIFF 2 +#define ED_DIFF 3 +#define NEW_CONTEXT_DIFF 4 +#define UNI_DIFF 5 +EXT int diff_type INIT(0); + +EXT bool do_defines INIT(FALSE);	/* patch using ifdef, ifndef, etc. */ +EXT char if_defined[128];		/* #ifdef xyzzy */ +EXT char not_defined[128];		/* #ifndef xyzzy */ +EXT char else_defined[] INIT("#else\n");/* #else */ +EXT char end_defined[128];		/* #endif xyzzy */ + +EXT char *revision INIT(Nullch);	/* prerequisite revision, if any */ + +#include <errno.h> +#ifdef STDC_HEADERS +#include <stdlib.h> +#include <string.h> +#else +extern int errno; +FILE *popen(); +char *malloc(); +char *realloc(); +long atol(); +char *getenv(); +char *strcpy(); +char *strcat(); +#endif +char *mktemp(); +#ifdef HAVE_UNISTD_H +#include <unistd.h> +#else +long lseek(); +#endif +#if defined(_POSIX_VERSION) || defined(HAVE_FCNTL_H) +#include <fcntl.h> +#endif + +#if !defined(S_ISDIR) && defined(S_IFDIR) +#define	S_ISDIR(m) (((m) & S_IFMT) == S_IFDIR) +#endif +#if !defined(S_ISREG) && defined(S_IFREG) +#define	S_ISREG(m) (((m) & S_IFMT) == S_IFREG) +#endif diff --git a/gnu/usr.bin/patch/config.h b/gnu/usr.bin/patch/config.h new file mode 100644 index 000000000000..71623d7523c9 --- /dev/null +++ b/gnu/usr.bin/patch/config.h @@ -0,0 +1,81 @@ +/* config.h.  Generated automatically by configure.  */ +/* Portability variables.  -*- C -*- */ + +/* Define if the system does not support the `const' keyword.  */ +/* #undef const */ + +/* Define if the system supports file names longer than 14 characters.  */ +#define HAVE_LONG_FILE_NAMES  + +/* Define if the system has pathconf().  */ +/* #undef HAVE_PATHCONF */ + +/* Define if the system has strerror().  */ +#define HAVE_STRERROR 1 + +/* Define if the system has ANSI C header files and library functions.  */ +#define STDC_HEADERS + +/* Define if the system uses strchr instead of index +   and strrchr instead of rindex.  */ +#define HAVE_STRING_H 1 + +#if defined(STDC_HEADERS) || defined(HAVE_STRING_H) +#define index strchr +#define rindex strrchr +#endif + +/* Define if the system has unistd.h.  */ +#define HAVE_UNISTD_H 1 + +/* Define if the system has fcntl.h.  */ +#define HAVE_FCNTL_H 1 + +/* Define as either int or void -- the type that signal handlers return.  */ +#define RETSIGTYPE void + +#ifndef RETSIGTYPE +#define RETSIGTYPE void +#endif + +/*  Which directory library header to use.  */ +#define DIRENT 1			/* dirent.h */ +/* #undef SYSNDIR */			/* sys/ndir.h */ +/* #undef SYSDIR */			/* sys/dir.h */ +/* #undef NDIR */			/* ndir.h */ +/* #undef NODIR */			/* none -- don't make numbered backup files */ + +/* Define if the system lets you pass fewer arguments to a function +   than the function actually accepts (in the absence of a prototype). +   Defining it makes I/O calls slightly more efficient. +   You need not bother defining it unless your C preprocessor chokes on +   multi-line arguments to macros.  */ +/* #undef CANVARARG */ + +/* Define Reg* as either `register' or nothing, depending on whether +   the C compiler pays attention to this many register declarations. +   The intent is that you don't have to order your register declarations +   in the order of importance, so you can freely declare register variables +   in sub-blocks of code and as function parameters. +   Do not use Reg<n> more than once per routine. + +   These don't really matter a lot, since most modern C compilers ignore +   register declarations and often do a better job of allocating +   registers than people do.  */ + +#define Reg1 register +#define Reg2 register +#define Reg3 register +#define Reg4 register +#define Reg5 register +#define Reg6 register +#define Reg7 +#define Reg8 +#define Reg9 +#define Reg10 +#define Reg11 +#define Reg12 +#define Reg13 +#define Reg14 +#define Reg15 +#define Reg16 diff --git a/gnu/usr.bin/patch/getopt.c b/gnu/usr.bin/patch/getopt.c new file mode 100644 index 000000000000..a59a01339835 --- /dev/null +++ b/gnu/usr.bin/patch/getopt.c @@ -0,0 +1,731 @@ +/* Getopt for GNU. +   NOTE: getopt is now part of the C library, so if you don't know what +   "Keep this file name-space clean" means, talk to roland@gnu.ai.mit.edu +   before changing it! + +   Copyright (C) 1987, 88, 89, 90, 91, 92, 1993 +   	Free Software Foundation, Inc. + +   This program is free software; you can redistribute it and/or modify it +   under the terms of the GNU General Public License as published by the +   Free Software Foundation; either version 2, or (at your option) any +   later version. + +   This program 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 General Public License for more details. + +   You should have received a copy of the GNU General Public License +   along with this program; if not, write to the Free Software +   Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */ + +/* NOTE!!!  AIX requires this to be the first thing in the file. +   Do not put ANYTHING before it!  */ +#if !defined (__GNUC__) && defined (_AIX) + #pragma alloca +#endif + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#ifdef __GNUC__ +#define alloca __builtin_alloca +#else /* not __GNUC__ */ +#if defined (HAVE_ALLOCA_H) || (defined(sparc) && (defined(sun) || (!defined(USG) && !defined(SVR4) && !defined(__svr4__)))) +#include <alloca.h> +#else +#ifndef _AIX +char *alloca (); +#endif +#endif /* alloca.h */ +#endif /* not __GNUC__ */ + +#if !__STDC__ && !defined(const) && IN_GCC +#define const +#endif + +/* This tells Alpha OSF/1 not to define a getopt prototype in <stdio.h>.  */ +#ifndef _NO_PROTO +#define _NO_PROTO +#endif + +#include <stdio.h> + +/* Comment out all this code if we are using the GNU C Library, and are not +   actually compiling the library itself.  This code is part of the GNU C +   Library, but also included in many other GNU distributions.  Compiling +   and linking in this code is a waste when using the GNU C library +   (especially if it is a shared library).  Rather than having every GNU +   program understand `configure --with-gnu-libc' and omit the object files, +   it is simpler to just do this in the source for each such file.  */ + +#if defined (_LIBC) || !defined (__GNU_LIBRARY__) + + +/* This needs to come after some library #include +   to get __GNU_LIBRARY__ defined.  */ +#ifdef	__GNU_LIBRARY__ +#undef	alloca +/* Don't include stdlib.h for non-GNU C libraries because some of them +   contain conflicting prototypes for getopt.  */ +#include <stdlib.h> +#else	/* Not GNU C library.  */ +#define	__alloca	alloca +#endif	/* GNU C library.  */ + +/* If GETOPT_COMPAT is defined, `+' as well as `--' can introduce a +   long-named option.  Because this is not POSIX.2 compliant, it is +   being phased out.  */ +/* #define GETOPT_COMPAT */ + +/* This version of `getopt' appears to the caller like standard Unix `getopt' +   but it behaves differently for the user, since it allows the user +   to intersperse the options with the other arguments. + +   As `getopt' works, it permutes the elements of ARGV so that, +   when it is done, all the options precede everything else.  Thus +   all application programs are extended to handle flexible argument order. + +   Setting the environment variable POSIXLY_CORRECT disables permutation. +   Then the behavior is completely standard. + +   GNU application programs can use a third alternative mode in which +   they can distinguish the relative order of options and other arguments.  */ + +#include "getopt.h" + +/* For communication from `getopt' to the caller. +   When `getopt' finds an option that takes an argument, +   the argument value is returned here. +   Also, when `ordering' is RETURN_IN_ORDER, +   each non-option ARGV-element is returned here.  */ + +char *optarg = 0; + +/* Index in ARGV of the next element to be scanned. +   This is used for communication to and from the caller +   and for communication between successive calls to `getopt'. + +   On entry to `getopt', zero means this is the first call; initialize. + +   When `getopt' returns EOF, this is the index of the first of the +   non-option elements that the caller should itself scan. + +   Otherwise, `optind' communicates from one call to the next +   how much of ARGV has been scanned so far.  */ + +/* XXX 1003.2 says this must be 1 before any call.  */ +int optind = 0; + +/* The next char to be scanned in the option-element +   in which the last option character we returned was found. +   This allows us to pick up the scan where we left off. + +   If this is zero, or a null string, it means resume the scan +   by advancing to the next ARGV-element.  */ + +static char *nextchar; + +/* Callers store zero here to inhibit the error message +   for unrecognized options.  */ + +int opterr = 1; + +/* Set to an option character which was unrecognized. +   This must be initialized on some systems to avoid linking in the +   system's own getopt implementation.  */ + +int optopt = '?'; + +/* Describe how to deal with options that follow non-option ARGV-elements. + +   If the caller did not specify anything, +   the default is REQUIRE_ORDER if the environment variable +   POSIXLY_CORRECT is defined, PERMUTE otherwise. + +   REQUIRE_ORDER means don't recognize them as options; +   stop option processing when the first non-option is seen. +   This is what Unix does. +   This mode of operation is selected by either setting the environment +   variable POSIXLY_CORRECT, or using `+' as the first character +   of the list of option characters. + +   PERMUTE is the default.  We permute the contents of ARGV as we scan, +   so that eventually all the non-options are at the end.  This allows options +   to be given in any order, even with programs that were not written to +   expect this. + +   RETURN_IN_ORDER is an option available to programs that were written +   to expect options and other ARGV-elements in any order and that care about +   the ordering of the two.  We describe each non-option ARGV-element +   as if it were the argument of an option with character code 1. +   Using `-' as the first character of the list of option characters +   selects this mode of operation. + +   The special argument `--' forces an end of option-scanning regardless +   of the value of `ordering'.  In the case of RETURN_IN_ORDER, only +   `--' can cause `getopt' to return EOF with `optind' != ARGC.  */ + +static enum +{ +  REQUIRE_ORDER, PERMUTE, RETURN_IN_ORDER +} ordering; + +#ifdef	__GNU_LIBRARY__ +/* We want to avoid inclusion of string.h with non-GNU libraries +   because there are many ways it can cause trouble. +   On some systems, it contains special magic macros that don't work +   in GCC.  */ +#include <string.h> +#define	my_index	strchr +#define	my_bcopy(src, dst, n)	memcpy ((dst), (src), (n)) +#else + +/* Avoid depending on library functions or files +   whose names are inconsistent.  */ + +char *getenv (); + +static char * +my_index (str, chr) +     const char *str; +     int chr; +{ +  while (*str) +    { +      if (*str == chr) +	return (char *) str; +      str++; +    } +  return 0; +} + +static void +my_bcopy (from, to, size) +     const char *from; +     char *to; +     int size; +{ +  int i; +  for (i = 0; i < size; i++) +    to[i] = from[i]; +} +#endif				/* GNU C library.  */ + +/* Handle permutation of arguments.  */ + +/* Describe the part of ARGV that contains non-options that have +   been skipped.  `first_nonopt' is the index in ARGV of the first of them; +   `last_nonopt' is the index after the last of them.  */ + +static int first_nonopt; +static int last_nonopt; + +/* Exchange two adjacent subsequences of ARGV. +   One subsequence is elements [first_nonopt,last_nonopt) +   which contains all the non-options that have been skipped so far. +   The other is elements [last_nonopt,optind), which contains all +   the options processed since those non-options were skipped. + +   `first_nonopt' and `last_nonopt' are relocated so that they describe +   the new indices of the non-options in ARGV after they are moved.  */ + +static void +exchange (argv) +     char **argv; +{ +  int nonopts_size = (last_nonopt - first_nonopt) * sizeof (char *); +  char **temp = (char **) __alloca (nonopts_size); + +  /* Interchange the two blocks of data in ARGV.  */ + +  my_bcopy ((char *) &argv[first_nonopt], (char *) temp, nonopts_size); +  my_bcopy ((char *) &argv[last_nonopt], (char *) &argv[first_nonopt], +	    (optind - last_nonopt) * sizeof (char *)); +  my_bcopy ((char *) temp, +	    (char *) &argv[first_nonopt + optind - last_nonopt], +	    nonopts_size); + +  /* Update records for the slots the non-options now occupy.  */ + +  first_nonopt += (optind - last_nonopt); +  last_nonopt = optind; +} + +/* Scan elements of ARGV (whose length is ARGC) for option characters +   given in OPTSTRING. + +   If an element of ARGV starts with '-', and is not exactly "-" or "--", +   then it is an option element.  The characters of this element +   (aside from the initial '-') are option characters.  If `getopt' +   is called repeatedly, it returns successively each of the option characters +   from each of the option elements. + +   If `getopt' finds another option character, it returns that character, +   updating `optind' and `nextchar' so that the next call to `getopt' can +   resume the scan with the following option character or ARGV-element. + +   If there are no more option characters, `getopt' returns `EOF'. +   Then `optind' is the index in ARGV of the first ARGV-element +   that is not an option.  (The ARGV-elements have been permuted +   so that those that are not options now come last.) + +   OPTSTRING is a string containing the legitimate option characters. +   If an option character is seen that is not listed in OPTSTRING, +   return '?' after printing an error message.  If you set `opterr' to +   zero, the error message is suppressed but we still return '?'. + +   If a char in OPTSTRING is followed by a colon, that means it wants an arg, +   so the following text in the same ARGV-element, or the text of the following +   ARGV-element, is returned in `optarg'.  Two colons mean an option that +   wants an optional arg; if there is text in the current ARGV-element, +   it is returned in `optarg', otherwise `optarg' is set to zero. + +   If OPTSTRING starts with `-' or `+', it requests different methods of +   handling the non-option ARGV-elements. +   See the comments about RETURN_IN_ORDER and REQUIRE_ORDER, above. + +   Long-named options begin with `--' instead of `-'. +   Their names may be abbreviated as long as the abbreviation is unique +   or is an exact match for some defined option.  If they have an +   argument, it follows the option name in the same ARGV-element, separated +   from the option name by a `=', or else the in next ARGV-element. +   When `getopt' finds a long-named option, it returns 0 if that option's +   `flag' field is nonzero, the value of the option's `val' field +   if the `flag' field is zero. + +   The elements of ARGV aren't really const, because we permute them. +   But we pretend they're const in the prototype to be compatible +   with other systems. + +   LONGOPTS is a vector of `struct option' terminated by an +   element containing a name which is zero. + +   LONGIND returns the index in LONGOPT of the long-named option found. +   It is only valid when a long-named option has been found by the most +   recent call. + +   If LONG_ONLY is nonzero, '-' as well as '--' can introduce +   long-named options.  */ + +int +_getopt_internal (argc, argv, optstring, longopts, longind, long_only) +     int argc; +     char *const *argv; +     const char *optstring; +     const struct option *longopts; +     int *longind; +     int long_only; +{ +  int option_index; + +  optarg = 0; + +  /* Initialize the internal data when the first call is made. +     Start processing options with ARGV-element 1 (since ARGV-element 0 +     is the program name); the sequence of previously skipped +     non-option ARGV-elements is empty.  */ + +  if (optind == 0) +    { +      first_nonopt = last_nonopt = optind = 1; + +      nextchar = NULL; + +      /* Determine how to handle the ordering of options and nonoptions.  */ + +      if (optstring[0] == '-') +	{ +	  ordering = RETURN_IN_ORDER; +	  ++optstring; +	} +      else if (optstring[0] == '+') +	{ +	  ordering = REQUIRE_ORDER; +	  ++optstring; +	} +      else if (getenv ("POSIXLY_CORRECT") != NULL) +	ordering = REQUIRE_ORDER; +      else +	ordering = PERMUTE; +    } + +  if (nextchar == NULL || *nextchar == '\0') +    { +      if (ordering == PERMUTE) +	{ +	  /* If we have just processed some options following some non-options, +	     exchange them so that the options come first.  */ + +	  if (first_nonopt != last_nonopt && last_nonopt != optind) +	    exchange ((char **) argv); +	  else if (last_nonopt != optind) +	    first_nonopt = optind; + +	  /* Now skip any additional non-options +	     and extend the range of non-options previously skipped.  */ + +	  while (optind < argc +		 && (argv[optind][0] != '-' || argv[optind][1] == '\0') +#ifdef GETOPT_COMPAT +		 && (longopts == NULL +		     || argv[optind][0] != '+' || argv[optind][1] == '\0') +#endif				/* GETOPT_COMPAT */ +		 ) +	    optind++; +	  last_nonopt = optind; +	} + +      /* Special ARGV-element `--' means premature end of options. +	 Skip it like a null option, +	 then exchange with previous non-options as if it were an option, +	 then skip everything else like a non-option.  */ + +      if (optind != argc && !strcmp (argv[optind], "--")) +	{ +	  optind++; + +	  if (first_nonopt != last_nonopt && last_nonopt != optind) +	    exchange ((char **) argv); +	  else if (first_nonopt == last_nonopt) +	    first_nonopt = optind; +	  last_nonopt = argc; + +	  optind = argc; +	} + +      /* If we have done all the ARGV-elements, stop the scan +	 and back over any non-options that we skipped and permuted.  */ + +      if (optind == argc) +	{ +	  /* Set the next-arg-index to point at the non-options +	     that we previously skipped, so the caller will digest them.  */ +	  if (first_nonopt != last_nonopt) +	    optind = first_nonopt; +	  return EOF; +	} + +      /* If we have come to a non-option and did not permute it, +	 either stop the scan or describe it to the caller and pass it by.  */ + +      if ((argv[optind][0] != '-' || argv[optind][1] == '\0') +#ifdef GETOPT_COMPAT +	  && (longopts == NULL +	      || argv[optind][0] != '+' || argv[optind][1] == '\0') +#endif				/* GETOPT_COMPAT */ +	  ) +	{ +	  if (ordering == REQUIRE_ORDER) +	    return EOF; +	  optarg = argv[optind++]; +	  return 1; +	} + +      /* We have found another option-ARGV-element. +	 Start decoding its characters.  */ + +      nextchar = (argv[optind] + 1 +		  + (longopts != NULL && argv[optind][1] == '-')); +    } + +  if (longopts != NULL +      && ((argv[optind][0] == '-' +	   && (argv[optind][1] == '-' || long_only)) +#ifdef GETOPT_COMPAT +	  || argv[optind][0] == '+' +#endif				/* GETOPT_COMPAT */ +	  )) +    { +      const struct option *p; +      char *s = nextchar; +      int exact = 0; +      int ambig = 0; +      const struct option *pfound = NULL; +      int indfound; + +      while (*s && *s != '=') +	s++; + +      /* Test all options for either exact match or abbreviated matches.  */ +      for (p = longopts, option_index = 0; p->name; +	   p++, option_index++) +	if (!strncmp (p->name, nextchar, s - nextchar)) +	  { +	    if (s - nextchar == strlen (p->name)) +	      { +		/* Exact match found.  */ +		pfound = p; +		indfound = option_index; +		exact = 1; +		break; +	      } +	    else if (pfound == NULL) +	      { +		/* First nonexact match found.  */ +		pfound = p; +		indfound = option_index; +	      } +	    else +	      /* Second nonexact match found.  */ +	      ambig = 1; +	  } + +      if (ambig && !exact) +	{ +	  if (opterr) +	    fprintf (stderr, "%s: option `%s' is ambiguous\n", +		     argv[0], argv[optind]); +	  nextchar += strlen (nextchar); +	  optind++; +	  return '?'; +	} + +      if (pfound != NULL) +	{ +	  option_index = indfound; +	  optind++; +	  if (*s) +	    { +	      /* Don't test has_arg with >, because some C compilers don't +		 allow it to be used on enums.  */ +	      if (pfound->has_arg) +		optarg = s + 1; +	      else +		{ +		  if (opterr) +		    { +		      if (argv[optind - 1][1] == '-') +			/* --option */ +			fprintf (stderr, +				 "%s: option `--%s' doesn't allow an argument\n", +				 argv[0], pfound->name); +		      else +			/* +option or -option */ +			fprintf (stderr, +			     "%s: option `%c%s' doesn't allow an argument\n", +			     argv[0], argv[optind - 1][0], pfound->name); +		    } +		  nextchar += strlen (nextchar); +		  return '?'; +		} +	    } +	  else if (pfound->has_arg == 1) +	    { +	      if (optind < argc) +		optarg = argv[optind++]; +	      else +		{ +		  if (opterr) +		    fprintf (stderr, "%s: option `%s' requires an argument\n", +			     argv[0], argv[optind - 1]); +		  nextchar += strlen (nextchar); +		  return optstring[0] == ':' ? ':' : '?'; +		} +	    } +	  nextchar += strlen (nextchar); +	  if (longind != NULL) +	    *longind = option_index; +	  if (pfound->flag) +	    { +	      *(pfound->flag) = pfound->val; +	      return 0; +	    } +	  return pfound->val; +	} +      /* Can't find it as a long option.  If this is not getopt_long_only, +	 or the option starts with '--' or is not a valid short +	 option, then it's an error. +	 Otherwise interpret it as a short option.  */ +      if (!long_only || argv[optind][1] == '-' +#ifdef GETOPT_COMPAT +	  || argv[optind][0] == '+' +#endif				/* GETOPT_COMPAT */ +	  || my_index (optstring, *nextchar) == NULL) +	{ +	  if (opterr) +	    { +	      if (argv[optind][1] == '-') +		/* --option */ +		fprintf (stderr, "%s: unrecognized option `--%s'\n", +			 argv[0], nextchar); +	      else +		/* +option or -option */ +		fprintf (stderr, "%s: unrecognized option `%c%s'\n", +			 argv[0], argv[optind][0], nextchar); +	    } +	  nextchar = (char *) ""; +	  optind++; +	  return '?'; +	} +    } + +  /* Look at and handle the next option-character.  */ + +  { +    char c = *nextchar++; +    char *temp = my_index (optstring, c); + +    /* Increment `optind' when we start to process its last character.  */ +    if (*nextchar == '\0') +      ++optind; + +    if (temp == NULL || c == ':') +      { +	if (opterr) +	  { +#if 0 +	    if (c < 040 || c >= 0177) +	      fprintf (stderr, "%s: unrecognized option, character code 0%o\n", +		       argv[0], c); +	    else +	      fprintf (stderr, "%s: unrecognized option `-%c'\n", argv[0], c); +#else +	    /* 1003.2 specifies the format of this message.  */ +	    fprintf (stderr, "%s: illegal option -- %c\n", argv[0], c); +#endif +	  } +	optopt = c; +	return '?'; +      } +    if (temp[1] == ':') +      { +	if (temp[2] == ':') +	  { +	    /* This is an option that accepts an argument optionally.  */ +	    if (*nextchar != '\0') +	      { +		optarg = nextchar; +		optind++; +	      } +	    else +	      optarg = 0; +	    nextchar = NULL; +	  } +	else +	  { +	    /* This is an option that requires an argument.  */ +	    if (*nextchar != '\0') +	      { +		optarg = nextchar; +		/* If we end this ARGV-element by taking the rest as an arg, +		   we must advance to the next element now.  */ +		optind++; +	      } +	    else if (optind == argc) +	      { +		if (opterr) +		  { +#if 0 +		    fprintf (stderr, "%s: option `-%c' requires an argument\n", +			     argv[0], c); +#else +		    /* 1003.2 specifies the format of this message.  */ +		    fprintf (stderr, "%s: option requires an argument -- %c\n", +			     argv[0], c); +#endif +		  } +		optopt = c; +		if (optstring[0] == ':') +		  c = ':'; +		else +		  c = '?'; +	      } +	    else +	      /* We already incremented `optind' once; +		 increment it again when taking next ARGV-elt as argument.  */ +	      optarg = argv[optind++]; +	    nextchar = NULL; +	  } +      } +    return c; +  } +} + +int +getopt (argc, argv, optstring) +     int argc; +     char *const *argv; +     const char *optstring; +{ +  return _getopt_internal (argc, argv, optstring, +			   (const struct option *) 0, +			   (int *) 0, +			   0); +} + +#endif	/* _LIBC or not __GNU_LIBRARY__.  */ + +#ifdef TEST + +/* Compile with -DTEST to make an executable for use in testing +   the above definition of `getopt'.  */ + +int +main (argc, argv) +     int argc; +     char **argv; +{ +  int c; +  int digit_optind = 0; + +  while (1) +    { +      int this_option_optind = optind ? optind : 1; + +      c = getopt (argc, argv, "abc:d:0123456789"); +      if (c == EOF) +	break; + +      switch (c) +	{ +	case '0': +	case '1': +	case '2': +	case '3': +	case '4': +	case '5': +	case '6': +	case '7': +	case '8': +	case '9': +	  if (digit_optind != 0 && digit_optind != this_option_optind) +	    printf ("digits occur in two different argv-elements.\n"); +	  digit_optind = this_option_optind; +	  printf ("option %c\n", c); +	  break; + +	case 'a': +	  printf ("option a\n"); +	  break; + +	case 'b': +	  printf ("option b\n"); +	  break; + +	case 'c': +	  printf ("option c with value `%s'\n", optarg); +	  break; + +	case '?': +	  break; + +	default: +	  printf ("?? getopt returned character code 0%o ??\n", c); +	} +    } + +  if (optind < argc) +    { +      printf ("non-option ARGV-elements: "); +      while (optind < argc) +	printf ("%s ", argv[optind++]); +      printf ("\n"); +    } + +  exit (0); +} + +#endif /* TEST */ diff --git a/gnu/usr.bin/patch/getopt.h b/gnu/usr.bin/patch/getopt.h new file mode 100644 index 000000000000..45541f5ac0f9 --- /dev/null +++ b/gnu/usr.bin/patch/getopt.h @@ -0,0 +1,129 @@ +/* Declarations for getopt. +   Copyright (C) 1989, 1990, 1991, 1992, 1993 Free Software Foundation, Inc. + +   This program is free software; you can redistribute it and/or modify it +   under the terms of the GNU General Public License as published by the +   Free Software Foundation; either version 2, or (at your option) any +   later version. + +   This program 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 General Public License for more details. + +   You should have received a copy of the GNU General Public License +   along with this program; if not, write to the Free Software +   Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */ + +#ifndef _GETOPT_H +#define _GETOPT_H 1 + +#ifdef	__cplusplus +extern "C" { +#endif + +/* For communication from `getopt' to the caller. +   When `getopt' finds an option that takes an argument, +   the argument value is returned here. +   Also, when `ordering' is RETURN_IN_ORDER, +   each non-option ARGV-element is returned here.  */ + +extern char *optarg; + +/* Index in ARGV of the next element to be scanned. +   This is used for communication to and from the caller +   and for communication between successive calls to `getopt'. + +   On entry to `getopt', zero means this is the first call; initialize. + +   When `getopt' returns EOF, this is the index of the first of the +   non-option elements that the caller should itself scan. + +   Otherwise, `optind' communicates from one call to the next +   how much of ARGV has been scanned so far.  */ + +extern int optind; + +/* Callers store zero here to inhibit the error message `getopt' prints +   for unrecognized options.  */ + +extern int opterr; + +/* Set to an option character which was unrecognized.  */ + +extern int optopt; + +/* Describe the long-named options requested by the application. +   The LONG_OPTIONS argument to getopt_long or getopt_long_only is a vector +   of `struct option' terminated by an element containing a name which is +   zero. + +   The field `has_arg' is: +   no_argument		(or 0) if the option does not take an argument, +   required_argument	(or 1) if the option requires an argument, +   optional_argument 	(or 2) if the option takes an optional argument. + +   If the field `flag' is not NULL, it points to a variable that is set +   to the value given in the field `val' when the option is found, but +   left unchanged if the option is not found. + +   To have a long-named option do something other than set an `int' to +   a compiled-in constant, such as set a value from `optarg', set the +   option's `flag' field to zero and its `val' field to a nonzero +   value (the equivalent single-letter option character, if there is +   one).  For long options that have a zero `flag' field, `getopt' +   returns the contents of the `val' field.  */ + +struct option +{ +#if	__STDC__ +  const char *name; +#else +  char *name; +#endif +  /* has_arg can't be an enum because some compilers complain about +     type mismatches in all the code that assumes it is an int.  */ +  int has_arg; +  int *flag; +  int val; +}; + +/* Names for the values of the `has_arg' field of `struct option'.  */ + +#define	no_argument		0 +#define required_argument	1 +#define optional_argument	2 + +#if __STDC__ +#if defined(__GNU_LIBRARY__) +/* Many other libraries have conflicting prototypes for getopt, with +   differences in the consts, in stdlib.h.  To avoid compilation +   errors, only prototype getopt for the GNU C library.  */ +extern int getopt (int argc, char *const *argv, const char *shortopts); +#else /* not __GNU_LIBRARY__ */ +extern int getopt (); +#endif /* not __GNU_LIBRARY__ */ +extern int getopt_long (int argc, char *const *argv, const char *shortopts, +		        const struct option *longopts, int *longind); +extern int getopt_long_only (int argc, char *const *argv, +			     const char *shortopts, +		             const struct option *longopts, int *longind); + +/* Internal only.  Users should not call this directly.  */ +extern int _getopt_internal (int argc, char *const *argv, +			     const char *shortopts, +		             const struct option *longopts, int *longind, +			     int long_only); +#else /* not __STDC__ */ +extern int getopt (); +extern int getopt_long (); +extern int getopt_long_only (); + +extern int _getopt_internal (); +#endif /* not __STDC__ */ + +#ifdef	__cplusplus +} +#endif + +#endif /* _GETOPT_H */ diff --git a/gnu/usr.bin/patch/getopt1.c b/gnu/usr.bin/patch/getopt1.c new file mode 100644 index 000000000000..a32615ce36e7 --- /dev/null +++ b/gnu/usr.bin/patch/getopt1.c @@ -0,0 +1,176 @@ +/* getopt_long and getopt_long_only entry points for GNU getopt. +   Copyright (C) 1987, 88, 89, 90, 91, 92, 1993 +	Free Software Foundation, Inc. + +   This program is free software; you can redistribute it and/or modify it +   under the terms of the GNU General Public License as published by the +   Free Software Foundation; either version 2, or (at your option) any +   later version. + +   This program 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 General Public License for more details. + +   You should have received a copy of the GNU General Public License +   along with this program; if not, write to the Free Software +   Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include "getopt.h" + +#if !__STDC__ && !defined(const) && IN_GCC +#define const +#endif + +#include <stdio.h> + +/* Comment out all this code if we are using the GNU C Library, and are not +   actually compiling the library itself.  This code is part of the GNU C +   Library, but also included in many other GNU distributions.  Compiling +   and linking in this code is a waste when using the GNU C library +   (especially if it is a shared library).  Rather than having every GNU +   program understand `configure --with-gnu-libc' and omit the object files, +   it is simpler to just do this in the source for each such file.  */ + +#if defined (_LIBC) || !defined (__GNU_LIBRARY__) + + +/* This needs to come after some library #include +   to get __GNU_LIBRARY__ defined.  */ +#ifdef __GNU_LIBRARY__ +#include <stdlib.h> +#else +char *getenv (); +#endif + +#ifndef	NULL +#define NULL 0 +#endif + +int +getopt_long (argc, argv, options, long_options, opt_index) +     int argc; +     char *const *argv; +     const char *options; +     const struct option *long_options; +     int *opt_index; +{ +  return _getopt_internal (argc, argv, options, long_options, opt_index, 0); +} + +/* Like getopt_long, but '-' as well as '--' can indicate a long option. +   If an option that starts with '-' (not '--') doesn't match a long option, +   but does match a short option, it is parsed as a short option +   instead.  */ + +int +getopt_long_only (argc, argv, options, long_options, opt_index) +     int argc; +     char *const *argv; +     const char *options; +     const struct option *long_options; +     int *opt_index; +{ +  return _getopt_internal (argc, argv, options, long_options, opt_index, 1); +} + + +#endif	/* _LIBC or not __GNU_LIBRARY__.  */ + +#ifdef TEST + +#include <stdio.h> + +int +main (argc, argv) +     int argc; +     char **argv; +{ +  int c; +  int digit_optind = 0; + +  while (1) +    { +      int this_option_optind = optind ? optind : 1; +      int option_index = 0; +      static struct option long_options[] = +      { +	{"add", 1, 0, 0}, +	{"append", 0, 0, 0}, +	{"delete", 1, 0, 0}, +	{"verbose", 0, 0, 0}, +	{"create", 0, 0, 0}, +	{"file", 1, 0, 0}, +	{0, 0, 0, 0} +      }; + +      c = getopt_long (argc, argv, "abc:d:0123456789", +		       long_options, &option_index); +      if (c == EOF) +	break; + +      switch (c) +	{ +	case 0: +	  printf ("option %s", long_options[option_index].name); +	  if (optarg) +	    printf (" with arg %s", optarg); +	  printf ("\n"); +	  break; + +	case '0': +	case '1': +	case '2': +	case '3': +	case '4': +	case '5': +	case '6': +	case '7': +	case '8': +	case '9': +	  if (digit_optind != 0 && digit_optind != this_option_optind) +	    printf ("digits occur in two different argv-elements.\n"); +	  digit_optind = this_option_optind; +	  printf ("option %c\n", c); +	  break; + +	case 'a': +	  printf ("option a\n"); +	  break; + +	case 'b': +	  printf ("option b\n"); +	  break; + +	case 'c': +	  printf ("option c with value `%s'\n", optarg); +	  break; + +	case 'd': +	  printf ("option d with value `%s'\n", optarg); +	  break; + +	case '?': +	  break; + +	default: +	  printf ("?? getopt returned character code 0%o ??\n", c); +	} +    } + +  if (optind < argc) +    { +      printf ("non-option ARGV-elements: "); +      while (optind < argc) +	printf ("%s ", argv[optind++]); +      printf ("\n"); +    } + +  exit (0); +} + +#endif /* TEST */ diff --git a/gnu/usr.bin/patch/inp.c b/gnu/usr.bin/patch/inp.c new file mode 100644 index 000000000000..cf17d8f2bdbb --- /dev/null +++ b/gnu/usr.bin/patch/inp.c @@ -0,0 +1,372 @@ +/* $Header: /home/cvs/386BSD/src/gnu/usr.bin/patch/inp.c,v 1.1.1.1 1993/06/19 14:21:52 paul Exp $ + * + * $Log: inp.c,v $ + * Revision 1.1.1.1  1993/06/19  14:21:52  paul + * b-maked patch-2.10 + * + * Revision 2.0.1.1  88/06/03  15:06:13  lwall + * patch10: made a little smarter about sccs files + *  + * Revision 2.0  86/09/17  15:37:02  lwall + * Baseline for netwide release. + *  + */ + +#include "EXTERN.h" +#include "common.h" +#include "util.h" +#include "pch.h" +#include "INTERN.h" +#include "inp.h" + +/* Input-file-with-indexable-lines abstract type */ + +static long i_size;			/* size of the input file */ +static char *i_womp;			/* plan a buffer for entire file */ +static char **i_ptr;			/* pointers to lines in i_womp */ + +static int tifd = -1;			/* plan b virtual string array */ +static char *tibuf[2];			/* plan b buffers */ +static LINENUM tiline[2] = {-1, -1};	/* 1st line in each buffer */ +static LINENUM lines_per_buf;		/* how many lines per buffer */ +static int tireclen;			/* length of records in tmp file */ + +/* New patch--prepare to edit another file. */ + +void +re_input() +{ +    if (using_plan_a) { +	i_size = 0; +#ifndef lint +	if (i_ptr != Null(char**)) +	    free((char *)i_ptr); +#endif +	if (i_womp != Nullch) +	    free(i_womp); +	i_womp = Nullch; +	i_ptr = Null(char **); +    } +    else { +	using_plan_a = TRUE;		/* maybe the next one is smaller */ +	Close(tifd); +	tifd = -1; +	free(tibuf[0]); +	free(tibuf[1]); +	tibuf[0] = tibuf[1] = Nullch; +	tiline[0] = tiline[1] = -1; +	tireclen = 0; +    } +} + +/* Constuct the line index, somehow or other. */ + +void +scan_input(filename) +char *filename; +{ +    if (!plan_a(filename)) +	plan_b(filename); +    if (verbose) { +	say3("Patching file %s using Plan %s...\n", filename, +	  (using_plan_a ? "A" : "B") ); +    } +} + +/* Try keeping everything in memory. */ + +bool +plan_a(filename) +char *filename; +{ +    int ifd, statfailed; +    Reg1 char *s; +    Reg2 LINENUM iline; +    char lbuf[MAXLINELEN]; +    int output_elsewhere = strcmp(filename, outname); + +    statfailed = stat(filename, &filestat); +    if (statfailed && ok_to_create_file) { +	if (verbose) +	    say2("(Creating file %s...)\n",filename); +	makedirs(filename, TRUE); +	close(creat(filename, 0666)); +	statfailed = stat(filename, &filestat); +    } +    /* For nonexistent or read-only files, look for RCS or SCCS versions.  */ +    if (statfailed +	|| (! output_elsewhere +	    && (/* No one can write to it.  */ +		(filestat.st_mode & 0222) == 0 +		/* I can't write to it.  */ +		|| ((filestat.st_mode & 0022) == 0 +		    && filestat.st_uid != myuid)))) { +	struct stat cstat; +	char *cs = Nullch; +	char *filebase; +	int pathlen; + +	filebase = basename(filename); +	pathlen = filebase - filename; + +	/* Put any leading path into `s'. +	   Leave room in lbuf for the diff command.  */ +	s = lbuf + 20; +	strncpy(s, filename, pathlen); + +#define try(f,a1,a2) (Sprintf(s + pathlen, f, a1, a2), stat(s, &cstat) == 0) +	if ((   try("RCS/%s%s", filebase, RCSSUFFIX) +	     || try("RCS/%s"  , filebase,         0) +	     || try(    "%s%s", filebase, RCSSUFFIX)) +	    && +	    /* Check that RCS file is not working file. +	       Some hosts don't report file name length errors.  */ +	    (statfailed +	     || (  (filestat.st_dev ^ cstat.st_dev) +		 | (filestat.st_ino ^ cstat.st_ino)))) { +	    Sprintf(buf, output_elsewhere?CHECKOUT:CHECKOUT_LOCKED, filename); +	    Sprintf(lbuf, RCSDIFF, filename); +	    cs = "RCS"; +	} else if (   try("SCCS/%s%s", SCCSPREFIX, filebase) +		   || try(     "%s%s", SCCSPREFIX, filebase)) { +	    Sprintf(buf, output_elsewhere?GET:GET_LOCKED, s); +	    Sprintf(lbuf, SCCSDIFF, s, filename); +	    cs = "SCCS"; +	} else if (statfailed) +	    fatal2("can't find %s\n", filename); +	/* else we can't write to it but it's not under a version +	   control system, so just proceed.  */ +	if (cs) { +	    if (!statfailed) { +		if ((filestat.st_mode & 0222) != 0) +		    /* The owner can write to it.  */ +		    fatal3("file %s seems to be locked by somebody else under %s\n", +			   filename, cs); +		/* It might be checked out unlocked.  See if it's safe to +		   check out the default version locked.  */ +		if (verbose) +		    say3("Comparing file %s to default %s version...\n", +			 filename, cs); +		if (system(lbuf)) +		    fatal3("can't check out file %s: differs from default %s version\n", +			   filename, cs); +	    } +	    if (verbose) +		say3("Checking out file %s from %s...\n", filename, cs); +	    if (system(buf) || stat(filename, &filestat)) +		fatal3("can't check out file %s from %s\n", filename, cs); +	} +    } +    filemode = filestat.st_mode; +    if (!S_ISREG(filemode)) +	fatal2("%s is not a normal file--can't patch\n", filename); +    i_size = filestat.st_size; +    if (out_of_mem) { +	set_hunkmax();		/* make sure dynamic arrays are allocated */ +	out_of_mem = FALSE; +	return FALSE;			/* force plan b because plan a bombed */ +    } +#ifdef lint +    i_womp = Nullch; +#else +    i_womp = malloc((MEM)(i_size+2));	/* lint says this may alloc less than */ +					/* i_size, but that's okay, I think. */ +#endif +    if (i_womp == Nullch) +	return FALSE; +    if ((ifd = open(filename, 0)) < 0) +	pfatal2("can't open file %s", filename); +#ifndef lint +    if (read(ifd, i_womp, (int)i_size) != i_size) { +	Close(ifd);	/* probably means i_size > 15 or 16 bits worth */ +	free(i_womp);	/* at this point it doesn't matter if i_womp was */ +	return FALSE;	/*   undersized. */ +    } +#endif +    Close(ifd); +    if (i_size && i_womp[i_size-1] != '\n') +	i_womp[i_size++] = '\n'; +    i_womp[i_size] = '\0'; + +    /* count the lines in the buffer so we know how many pointers we need */ + +    iline = 0; +    for (s=i_womp; *s; s++) { +	if (*s == '\n') +	    iline++; +    } +#ifdef lint +    i_ptr = Null(char**); +#else +    i_ptr = (char **)malloc((MEM)((iline + 2) * sizeof(char *))); +#endif +    if (i_ptr == Null(char **)) {	/* shucks, it was a near thing */ +	free((char *)i_womp); +	return FALSE; +    } +     +    /* now scan the buffer and build pointer array */ + +    iline = 1; +    i_ptr[iline] = i_womp; +    for (s=i_womp; *s; s++) { +	if (*s == '\n') +	    i_ptr[++iline] = s+1;	/* these are NOT null terminated */ +    } +    input_lines = iline - 1; + +    /* now check for revision, if any */ + +    if (revision != Nullch) {  +	if (!rev_in_string(i_womp)) { +	    if (force) { +		if (verbose) +		    say2( +"Warning: this file doesn't appear to be the %s version--patching anyway.\n", +			revision); +	    } +	    else if (batch) { +		fatal2( +"this file doesn't appear to be the %s version--aborting.\n", revision); +	    } +	    else { +		ask2( +"This file doesn't appear to be the %s version--patch anyway? [n] ", +		    revision); +	    if (*buf != 'y') +		fatal1("aborted\n"); +	    } +	} +	else if (verbose) +	    say2("Good.  This file appears to be the %s version.\n", +		revision); +    } +    return TRUE;			/* plan a will work */ +} + +/* Keep (virtually) nothing in memory. */ + +void +plan_b(filename) +char *filename; +{ +    Reg3 FILE *ifp; +    Reg1 int i = 0; +    Reg2 int maxlen = 1; +    Reg4 bool found_revision = (revision == Nullch); + +    using_plan_a = FALSE; +    if ((ifp = fopen(filename, "r")) == Nullfp) +	pfatal2("can't open file %s", filename); +    if ((tifd = creat(TMPINNAME, 0666)) < 0) +	pfatal2("can't open file %s", TMPINNAME); +    while (fgets(buf, sizeof buf, ifp) != Nullch) { +	if (revision != Nullch && !found_revision && rev_in_string(buf)) +	    found_revision = TRUE; +	if ((i = strlen(buf)) > maxlen) +	    maxlen = i;			/* find longest line */ +    } +    if (revision != Nullch) { +	if (!found_revision) { +	    if (force) { +		if (verbose) +		    say2( +"Warning: this file doesn't appear to be the %s version--patching anyway.\n", +			revision); +	    } +	    else if (batch) { +		fatal2( +"this file doesn't appear to be the %s version--aborting.\n", revision); +	    } +	    else { +		ask2( +"This file doesn't appear to be the %s version--patch anyway? [n] ", +		    revision); +		if (*buf != 'y') +		    fatal1("aborted\n"); +	    } +	} +	else if (verbose) +	    say2("Good.  This file appears to be the %s version.\n", +		revision); +    } +    Fseek(ifp, 0L, 0);		/* rewind file */ +    lines_per_buf = BUFFERSIZE / maxlen; +    tireclen = maxlen; +    tibuf[0] = malloc((MEM)(BUFFERSIZE + 1)); +    tibuf[1] = malloc((MEM)(BUFFERSIZE + 1)); +    if (tibuf[1] == Nullch) +	fatal1("out of memory\n"); +    for (i=1; ; i++) { +	if (! (i % lines_per_buf))	/* new block */ +	    if (write(tifd, tibuf[0], BUFFERSIZE) < BUFFERSIZE) +		pfatal1("can't write temp file"); +	if (fgets(tibuf[0] + maxlen * (i%lines_per_buf), maxlen + 1, ifp) +	  == Nullch) { +	    input_lines = i - 1; +	    if (i % lines_per_buf) +		if (write(tifd, tibuf[0], BUFFERSIZE) < BUFFERSIZE) +		    pfatal1("can't write temp file"); +	    break; +	} +    } +    Fclose(ifp); +    Close(tifd); +    if ((tifd = open(TMPINNAME, 0)) < 0) { +	pfatal2("can't reopen file %s", TMPINNAME); +    } +} + +/* Fetch a line from the input file, \n terminated, not necessarily \0. */ + +char * +ifetch(line,whichbuf) +Reg1 LINENUM line; +int whichbuf;				/* ignored when file in memory */ +{ +    if (line < 1 || line > input_lines) +	return ""; +    if (using_plan_a) +	return i_ptr[line]; +    else { +	LINENUM offline = line % lines_per_buf; +	LINENUM baseline = line - offline; + +	if (tiline[0] == baseline) +	    whichbuf = 0; +	else if (tiline[1] == baseline) +	    whichbuf = 1; +	else { +	    tiline[whichbuf] = baseline; +#ifndef lint		/* complains of long accuracy */ +	    Lseek(tifd, (long)baseline / lines_per_buf * BUFFERSIZE, 0); +#endif +	    if (read(tifd, tibuf[whichbuf], BUFFERSIZE) < 0) +		pfatal2("error reading tmp file %s", TMPINNAME); +	} +	return tibuf[whichbuf] + (tireclen*offline); +    } +} + +/* True if the string argument contains the revision number we want. */ + +bool +rev_in_string(string) +char *string; +{ +    Reg1 char *s; +    Reg2 int patlen; + +    if (revision == Nullch) +	return TRUE; +    patlen = strlen(revision); +    if (strnEQ(string,revision,patlen) && isspace(string[patlen])) +	return TRUE; +    for (s = string; *s; s++) { +	if (isspace(*s) && strnEQ(s+1, revision, patlen) &&  +		isspace(s[patlen+1] )) { +	    return TRUE; +	} +    } +    return FALSE; +} diff --git a/gnu/usr.bin/patch/inp.h b/gnu/usr.bin/patch/inp.h new file mode 100644 index 000000000000..f43d256897b2 --- /dev/null +++ b/gnu/usr.bin/patch/inp.h @@ -0,0 +1,21 @@ +/* $Header: /home/cvs/386BSD/src/gnu/usr.bin/patch/inp.h,v 1.1.1.1 1993/06/19 14:21:52 paul Exp $ + * + * $Log: inp.h,v $ + * Revision 1.1.1.1  1993/06/19  14:21:52  paul + * b-maked patch-2.10 + * + * Revision 2.0  86/09/17  15:37:25  lwall + * Baseline for netwide release. + *  + */ + +EXT LINENUM input_lines INIT(0);	/* how long is input file in lines */ +EXT LINENUM last_frozen_line INIT(0);	/* how many input lines have been */ +					/* irretractibly output */ + +bool rev_in_string(); +void scan_input(); +bool plan_a();			/* returns false if insufficient memory */ +void plan_b(); +char *ifetch(); + diff --git a/gnu/usr.bin/patch/patch.1 b/gnu/usr.bin/patch/patch.1 new file mode 100644 index 000000000000..a3353c99f57f --- /dev/null +++ b/gnu/usr.bin/patch/patch.1 @@ -0,0 +1,591 @@ +.\" -*- nroff -*- +.rn '' }` +'\" $Header: /home/cvs/386BSD/src/gnu/usr.bin/patch/patch.1,v 1.3.2.1 1994/05/01 16:04:21 jkh Exp $ +'\"  +'\" $Log: patch.1,v $ +.\" Revision 1.3.2.1  1994/05/01  16:04:21  jkh +.\" Bring man page changes over from -current (correct various misspellings, +.\" typos, etc). +.\" +.\" Revision 1.4  1994/02/25  21:45:59  phk +.\" added the -C/-check again. +.\" +.\" Revision 1.3  1994/02/17  22:20:33  jkh +.\" Put this back - I was somehow under the erroneous impression that patch was in +.\" ports, until I saw the the commit messages, that is! :-)  All changed backed out. +.\" +.\" Revision 1.2  1994/02/17  22:16:02  jkh +.\" From Poul-Henning Kamp -  Implement a -C option to verify the integrity of +.\" a patch before actually applying it. +.\" +.\" Revision 1.1.1.1  1993/06/19  14:21:51  paul +.\" b-maked patch-2.10 +.\" +'\" Revision 2.0.1.2  88/06/22  20:47:18  lwall +'\" patch12: now avoids Bell System Logo +'\"  +'\" Revision 2.0.1.1  88/06/03  15:12:51  lwall +'\" patch10: -B switch was contributed. +'\"  +'\" Revision 2.0  86/09/17  15:39:09  lwall +'\" Baseline for netwide release. +'\"  +'\" Revision 1.4  86/08/01  19:23:22  lwall +'\" Documented -v, -p, -F. +'\" Added notes to patch senders. +'\"  +'\" Revision 1.3  85/03/26  15:11:06  lwall +'\" Frozen. +'\"  +'\" Revision 1.2.1.4  85/03/12  16:14:27  lwall +'\" Documented -p. +'\"  +'\" Revision 1.2.1.3  85/03/12  16:09:41  lwall +'\" Documented -D. +'\"  +'\" Revision 1.2.1.2  84/12/05  11:06:55  lwall +'\" Added -l switch, and noted bistability bug. +'\"  +'\" Revision 1.2.1.1  84/12/04  17:23:39  lwall +'\" Branch for sdcrdcf changes. +'\"  +'\" Revision 1.2  84/12/04  17:22:02  lwall +'\" Baseline version. +'\"  +.de Sh +.br +.ne 5 +.PP +\fB\\$1\fR +.PP +.. +.de Sp +.if t .sp .5v +.if n .sp +.. +'\" +'\"     Set up \*(-- to give an unbreakable dash; +'\"     string Tr holds user defined translation string. +'\"     Bell System Logo is used as a dummy character. +'\" +.ie n \{\ +.tr \(*W-\*(Tr +.ds -- \(*W- +.if (\n(.H=4u)&(1m=24u) .ds -- \(*W\h'-12u'\(*W\h'-12u'-\" diablo 10 pitch +.if (\n(.H=4u)&(1m=20u) .ds -- \(*W\h'-12u'\(*W\h'-8u'-\" diablo 12 pitch +.ds L" "" +.ds R" "" +.ds L' ' +.ds R' ' +'br \} +.el \{\ +.ds -- \(em\| +.tr \*(Tr +.ds L" `` +.ds R" '' +.ds L' ` +.ds R' ' +'br\} +.TH PATCH 1 LOCAL +.SH NAME +patch - apply a diff file to an original +.SH SYNOPSIS +.B patch +[options] [origfile [patchfile]] [+ [options] [origfile]]... +.sp +but usually just +.sp +.B patch +<patchfile +.SH DESCRIPTION +.I Patch +will take a patch file containing any of the four forms of difference +listing produced by the +.I diff +program and apply those differences to an original file, producing a patched +version. +By default, the patched version is put in place of the original, with +the original file backed up to the same name with the +extension \*(L".orig\*(R" (\*(L"~\*(R" on systems that do not +support long file names), or as specified by the +\fB\-b\fP (\fB\-\-suffix\fP), +\fB\-B\fP (\fB\-\-prefix\fP), +or +\fB\-V\fP (\fB\-\-version\-control\fP) +options. +The extension used for making backup files may also be specified in the +.B SIMPLE_BACKUP_SUFFIX +environment variable, which is overridden by the above options. +.PP +If the backup file already exists, +.B patch +creates a new backup file name by changing the first lowercase letter +in the last component of the file's name into uppercase.  If there are +no more lowercase letters in the name, it removes the first character +from the name.  It repeats this process until it comes up with a +backup file that does not already exist. +.PP +You may also specify where you want the output to go with a +\fB\-o\fP (\fB\-\-output\fP) +option; if that file already exists, it is backed up first. +.PP +If +.I patchfile +is omitted, or is a hyphen, the patch will be read from standard input. +.PP +Upon startup, patch will attempt to determine the type of the diff listing, +unless over-ruled by a +\fB\-c\fP (\fB\-\-context\fP), +\fB\-e\fP (\fB\-\-ed\fP), +\fB\-n\fP (\fB\-\-normal\fP), +or +\fB\-u\fP (\fB\-\-unified\fP) +option. +Context diffs (old-style, new-style, and unified) and +normal diffs are applied by the +.I patch +program itself, while +.I ed +diffs are simply fed to the +.I ed +editor via a pipe. +.PP +.I Patch +will try to skip any leading garbage, apply the diff, +and then skip any trailing garbage. +Thus you could feed an article or message containing a +diff listing to +.IR patch , +and it should work. +If the entire diff is indented by a consistent amount, +this will be taken into account. +.PP +With context diffs, and to a lesser extent with normal diffs, +.I patch +can detect when the line numbers mentioned in the patch are incorrect, +and will attempt to find the correct place to apply each hunk of the patch. +As a first guess, it takes the line number mentioned for the hunk, plus or +minus any offset used in applying the previous hunk. +If that is not the correct place, +.I patch +will scan both forwards and backwards for a set of lines matching the context +given in the hunk. +First +.I patch +looks for a place where all lines of the context match. +If no such place is found, and it's a context diff, and the maximum fuzz factor +is set to 1 or more, then another scan takes place ignoring the first and last +line of context. +If that fails, and the maximum fuzz factor is set to 2 or more, +the first two and last two lines of context are ignored, +and another scan is made. +(The default maximum fuzz factor is 2.) +If +.I patch +cannot find a place to install that hunk of the patch, it will put the +hunk out to a reject file, which normally is the name of the output file +plus \*(L".rej\*(R" (\*(L"#\*(R" on systems that do not support +long file names). +(Note that the rejected hunk will come out in context diff form whether the +input patch was a context diff or a normal diff. +If the input was a normal diff, many of the contexts will simply be null.) +The line numbers on the hunks in the reject file may be different than +in the patch file: they reflect the approximate location patch thinks the +failed hunks belong in the new file rather than the old one. +.PP +As each hunk is completed, you will be told whether the hunk succeeded or +failed, and which line (in the new file) +.I patch +thought the hunk should go on. +If this is different from the line number specified in the diff you will +be told the offset. +A single large offset MAY be an indication that a hunk was installed in the +wrong place. +You will also be told if a fuzz factor was used to make the match, in which +case you should also be slightly suspicious. +.PP +If no original file is specified on the command line, +.I patch +will try to figure out from the leading garbage what the name of the file +to edit is. +In the header of a context diff, the file name is found from lines beginning +with \*(L"***\*(R" or \*(L"---\*(R", with the shortest name of an existing +file winning. +Only context diffs have lines like that, but if there is an \*(L"Index:\*(R" +line in the leading garbage, +.I patch +will try to use the file name from that line. +The context diff header takes precedence over an Index line. +If no file name can be intuited from the leading garbage, you will be asked +for the name of the file to patch. +.PP +If the original file cannot be found or is read-only, but a suitable +SCCS or RCS file is handy, +.I patch +will attempt to get or check out the file. +.PP +Additionally, if the leading garbage contains a \*(L"Prereq: \*(R" line, +.I patch +will take the first word from the prerequisites line (normally a version +number) and check the input file to see if that word can be found. +If not, +.I patch +will ask for confirmation before proceeding. +.PP +The upshot of all this is that you should be able to say, while in a news +interface, the following: +.Sp +	| patch -d /usr/src/local/blurfl +.Sp +and patch a file in the blurfl directory directly from the article containing +the patch. +.PP +If the patch file contains more than one patch, +.I patch +will try to apply each of them as if they came from separate patch files. +This means, among other things, that it is assumed that the name of the file +to patch must be determined for each diff listing, +and that the garbage before each diff listing will +be examined for interesting things such as file names and revision level, as +mentioned previously. +You can give options (and another original file name) for the second and +subsequent patches by separating the corresponding argument lists +by a \*(L'+\*(R'. +(The argument list for a second or subsequent patch may not specify a new +patch file, however.) +.PP +.I Patch +recognizes the following options: +.TP 5 +.B "\-b suff, \-\-suffix=suff" +causes +.B suff +to be interpreted as the backup extension, to be +used in place of \*(L".orig\*(R" or \*(L"~\*(R". +.TP 5 +.B "\-B pref, \-\-prefix=pref" +causes +.B pref +to be interpreted as a prefix to the backup file +name. If this argument is specified, any argument from +.B \-b +will be ignored. +.TP 5 +.B "\-c, \-\-context" +forces +.I patch +to interpret the patch file as a context diff. +.TP 5 +.B "\-C, \-\-check" +see what would happen, but don't do it. +.TP 5 +.B "\-d dir, \-\-directory=dir" +causes +.I patch +to interpret +.B dir +as a directory, and cd to it before doing +anything else. +.TP 5 +.B "\-D sym, \-\-ifdef=sym" +causes +.I patch +to use the "#ifdef...#endif" construct to mark changes. +.B sym +will be used as the differentiating symbol. +.TP 5 +.B "\-e, \-\-ed" +forces +.I patch +to interpret the patch file as an +.I ed +script. +.TP 5 +.B "\-E, \-\-remove\-empty\-files" +causes +.I patch +to remove output files that are empty after the patches have been applied. +.TP 5 +.B "\-f, \-\-force" +forces +.I patch +to assume that the user knows exactly what he or she is doing, and to not +ask any questions.  It assumes the following: skip patches for which a +file to patch can't be found; patch files even though they have the +wrong version for the ``Prereq:'' line in the patch; and assume that +patches are not reversed even if they look like they are. +This option does not suppress commentary; use +.B \-s +for that. +.TP 5 +.B "\-t, \-\-batch" +similar to +.BR \-f , +in that it suppresses questions, but makes some different assumptions: +skip patches for which a file to patch can't be found (the same as \fB\-f\fP); +skip patches for which the file has the wrong version for the ``Prereq:'' line +in the patch; and assume that patches are reversed if they look like +they are. +.TP 5 +.B "\-F number, \-\-fuzz=number" +sets the maximum fuzz factor. +This option only applies to context diffs, and causes +.I patch +to ignore up to that many lines in looking for places to install a hunk. +Note that a larger fuzz factor increases the odds of a faulty patch. +The default fuzz factor is 2, and it may not be set to more than +the number of lines of context in the context diff, ordinarily 3. +.TP 5 +.B "\-l, \-\-ignore\-whitespace" +causes the pattern matching to be done loosely, in case the tabs and +spaces have been munged in your input file. +Any sequence of whitespace in the pattern line will match any sequence +in the input file. +Normal characters must still match exactly. +Each line of the context must still match a line in the input file. +.TP 5 +.B "\-n, \-\-normal" +forces +.I patch +to interpret the patch file as a normal diff. +.TP 5 +.B "\-N, \-\-forward" +causes +.I patch +to ignore patches that it thinks are reversed or already applied. +See also +.B \-R . +.TP 5 +.B "\-o file, \-\-output=file" +causes +.B file +to be interpreted as the output file name. +.TP 5 +.B "\-p[number], \-\-strip[=number]" +sets the pathname strip count, +which controls how pathnames found in the patch file are treated, in case +the you keep your files in a different directory than the person who sent +out the patch. +The strip count specifies how many slashes are to be stripped from +the front of the pathname. +(Any intervening directory names also go away.) +For example, supposing the file name in the patch file was +.sp +	/u/howard/src/blurfl/blurfl.c +.sp +setting +.B \-p +or +.B \-p0 +gives the entire pathname unmodified, +.B \-p1 +gives +.sp +	u/howard/src/blurfl/blurfl.c +.sp +without the leading slash, +.B \-p4 +gives +.sp +	blurfl/blurfl.c +.sp +and not specifying +.B \-p +at all just gives you "blurfl.c", unless all of the directories in the +leading path (u/howard/src/blurfl) exist and that path is relative, +in which case you get the entire pathname unmodified. +Whatever you end up with is looked for either in the current directory, +or the directory specified by the +.B \-d +option. +.TP 5 +.B "\-r file, \-\-reject\-file=file" +causes +.B file +to be interpreted as the reject file name. +.TP 5 +.B "\-R, \-\-reverse" +tells +.I patch +that this patch was created with the old and new files swapped. +(Yes, I'm afraid that does happen occasionally, human nature being what it +is.) +.I Patch +will attempt to swap each hunk around before applying it. +Rejects will come out in the swapped format. +The +.B \-R +option will not work with +.I ed +diff scripts because there is too little +information to reconstruct the reverse operation. +.Sp +If the first hunk of a patch fails, +.I patch +will reverse the hunk to see if it can be applied that way. +If it can, you will be asked if you want to have the +.B \-R +option set. +If it can't, the patch will continue to be applied normally. +(Note: this method cannot detect a reversed patch if it is a normal diff +and if the first command is an append (i.e. it should have been a delete) +since appends always succeed, due to the fact that a null context will match +anywhere. +Luckily, most patches add or change lines rather than delete them, so most +reversed normal diffs will begin with a delete, which will fail, triggering +the heuristic.) +.TP 5 +.B "\-s, \-\-silent, \-\-quiet" +makes +.I patch +do its work silently, unless an error occurs. +.TP 5 +.B "\-S, \-\-skip" +causes +.I patch +to ignore this patch from the patch file, but continue on looking +for the next patch in the file. +Thus +.sp +	patch -S + -S + <patchfile +.sp +will ignore the first and second of three patches. +.TP 5 +.B "\-u, \-\-unified" +forces +.I patch +to interpret the patch file as a unified context diff (a unidiff). +.TP 5 +.B "\-v, \-\-version" +causes +.I patch +to print out its revision header and patch level. +.TP 5 +.B "\-V method, \-\-version\-\-control=method" +causes +.B method +to be interpreted as a method for creating +backup file names.  The type of backups made can also be given in the +.B VERSION_CONTROL +environment variable, which is overridden by this option. +The +.B -B +option overrides this option, causing the prefix to always be used for +making backup file names. +The value of the +.B VERSION_CONTROL +environment variable and the argument to the +.B -V +option are like the GNU +Emacs `version-control' variable; they also recognize synonyms that +are more descriptive.  The valid values are (unique abbreviations are +accepted): +.RS +.TP +`t' or `numbered' +Always make numbered backups. +.TP +`nil' or `existing' +Make numbered backups of files that already +have them, simple backups of the others. +This is the default. +.TP +`never' or `simple' +Always make simple backups. +.RE +.TP 5 +.B "\-x number, \-\-debug=number" +sets internal debugging flags, and is of interest only to +.I patch +patchers. +.SH AUTHOR +Larry Wall <lwall@netlabs.com> +.br +with many other contributors. +.SH ENVIRONMENT +.TP +.B TMPDIR +Directory to put temporary files in; default is /tmp. +.TP +.B SIMPLE_BACKUP_SUFFIX +Extension to use for backup file names instead of \*(L".orig\*(R" or +\*(L"~\*(R". +.TP +.B VERSION_CONTROL +Selects when numbered backup files are made. +.SH FILES +$TMPDIR/patch* +.SH SEE ALSO +diff(1) +.SH NOTES FOR PATCH SENDERS +There are several things you should bear in mind if you are going to +be sending out patches. +First, you can save people a lot of grief by keeping a patchlevel.h file +which is patched to increment the patch level as the first diff in the +patch file you send out. +If you put a Prereq: line in with the patch, it won't let them apply +patches out of order without some warning. +Second, make sure you've specified the file names right, either in a +context diff header, or with an Index: line. +If you are patching something in a subdirectory, be sure to tell the patch +user to specify a  +.B \-p +option as needed. +Third, you can create a file by sending out a diff that compares a +null file to the file you want to create. +This will only work if the file you want to create doesn't exist already in +the target directory. +Fourth, take care not to send out reversed patches, since it makes people wonder +whether they already applied the patch. +Fifth, while you may be able to get away with putting 582 diff listings into +one file, it is probably wiser to group related patches into separate files in +case something goes haywire. +.SH DIAGNOSTICS +Too many to list here, but generally indicative that +.I patch +couldn't parse your patch file. +.PP +The message \*(L"Hmm...\*(R" indicates that there is unprocessed text in +the patch file and that +.I patch +is attempting to intuit whether there is a patch in that text and, if so, +what kind of patch it is. +.PP +.I Patch +will exit with a non-zero status if any reject files were created. +When applying a set of patches in a loop it behooves you to check this +exit status so you don't apply a later patch to a partially patched file. +.SH CAVEATS +.I Patch +cannot tell if the line numbers are off in an +.I ed +script, and can only detect +bad line numbers in a normal diff when it finds a \*(L"change\*(R" or +a \*(L"delete\*(R" command. +A context diff using fuzz factor 3 may have the same problem. +Until a suitable interactive interface is added, you should probably do +a context diff in these cases to see if the changes made sense. +Of course, compiling without errors is a pretty good indication that the patch +worked, but not always. +.PP +.I Patch +usually produces the correct results, even when it has to do a lot of +guessing. +However, the results are guaranteed to be correct only when the patch is +applied to exactly the same version of the file that the patch was +generated from. +.SH BUGS +Could be smarter about partial matches, excessively \&deviant offsets and +swapped code, but that would take an extra pass. +.PP +If code has been duplicated (for instance with #ifdef OLDCODE ... #else ... +#endif), +.I patch +is incapable of patching both versions, and, if it works at all, will likely +patch the wrong one, and tell you that it succeeded to boot. +.PP +If you apply a patch you've already applied, +.I patch +will think it is a reversed patch, and offer to un-apply the patch. +This could be construed as a feature. +.rn }` '' diff --git a/gnu/usr.bin/patch/patch.c b/gnu/usr.bin/patch/patch.c new file mode 100644 index 000000000000..c315645ccc62 --- /dev/null +++ b/gnu/usr.bin/patch/patch.c @@ -0,0 +1,956 @@ +char rcsid[] = +	"$Header: /home/cvs/386BSD/src/gnu/usr.bin/patch/patch.c,v 1.3 1994/02/17 22:20:34 jkh Exp $"; + +/* patch - a program to apply diffs to original files + * + * Copyright 1986, Larry Wall + * + * This program may be copied as long as you don't try to make any + * money off of it, or pretend that you wrote it. + * + * $Log: patch.c,v $ + * Revision 1.3  1994/02/17  22:20:34  jkh + * Put this back - I was somehow under the erroneous impression that patch was in + * ports, until I saw the the commit messages, that is! :-)  All changed backed out. + * + * Revision 1.2  1994/02/17  22:16:03  jkh + * From Poul-Henning Kamp -  Implement a -C option to verify the integrity of + * a patch before actually applying it. + * + * Revision 1.1.1.1  1993/06/19  14:21:52  paul + * b-maked patch-2.10 + * + * Revision 2.0.2.0  90/05/01  22:17:50  davison + * patch12u: unidiff support added + *  + * Revision 2.0.1.6  88/06/22  20:46:39  lwall + * patch12: rindex() wasn't declared + *  + * Revision 2.0.1.5  88/06/03  15:09:37  lwall + * patch10: exit code improved. + * patch10: better support for non-flexfilenames. + *  + * Revision 2.0.1.4  87/02/16  14:00:04  lwall + * Short replacement caused spurious "Out of sync" message. + *  + * Revision 2.0.1.3  87/01/30  22:45:50  lwall + * Improved diagnostic on sync error. + * Moved do_ed_script() to pch.c. + *  + * Revision 2.0.1.2  86/11/21  09:39:15  lwall + * Fuzz factor caused offset of installed lines. + *  + * Revision 2.0.1.1  86/10/29  13:10:22  lwall + * Backwards search could terminate prematurely. + *  + * Revision 2.0  86/09/17  15:37:32  lwall + * Baseline for netwide release. + *  + * Revision 1.5  86/08/01  20:53:24  lwall + * Changed some %d's to %ld's. + * Linted. + *  + * Revision 1.4  86/08/01  19:17:29  lwall + * Fixes for machines that can't vararg. + * Added fuzz factor. + * Generalized -p. + * General cleanup. + *  + * 85/08/15 van%ucbmonet@berkeley + * Changes for 4.3bsd diff -c. + * + * Revision 1.3  85/03/26  15:07:43  lwall + * Frozen. + *  + * Revision 1.2.1.9  85/03/12  17:03:35  lwall + * Changed pfp->_file to fileno(pfp). + *  + * Revision 1.2.1.8  85/03/12  16:30:43  lwall + * Check i_ptr and i_womp to make sure they aren't null before freeing. + * Also allow ed output to be suppressed. + *  + * Revision 1.2.1.7  85/03/12  15:56:13  lwall + * Added -p option from jromine@uci-750a. + *  + * Revision 1.2.1.6  85/03/12  12:12:51  lwall + * Now checks for normalness of file to patch. + *  + * Revision 1.2.1.5  85/03/12  11:52:12  lwall + * Added -D (#ifdef) option from joe@fluke. + *  + * Revision 1.2.1.4  84/12/06  11:14:15  lwall + * Made smarter about SCCS subdirectories. + *  + * Revision 1.2.1.3  84/12/05  11:18:43  lwall + * Added -l switch to do loose string comparison. + *  + * Revision 1.2.1.2  84/12/04  09:47:13  lwall + * Failed hunk count not reset on multiple patch file. + *  + * Revision 1.2.1.1  84/12/04  09:42:37  lwall + * Branch for sdcrdcf changes. + *  + * Revision 1.2  84/11/29  13:29:51  lwall + * Linted.  Identifiers uniqified.  Fixed i_ptr malloc() bug.  Fixed + * multiple calls to mktemp().  Will now work on machines that can only + * read 32767 chars.  Added -R option for diffs with new and old swapped. + * Various cosmetic changes. + *  + * Revision 1.1  84/11/09  17:03:58  lwall + * Initial revision + *  + */ + +#include "INTERN.h" +#include "common.h" +#include "EXTERN.h" +#include "version.h" +#include "util.h" +#include "pch.h" +#include "inp.h" +#include "backupfile.h" +#include "getopt.h" + +/* procedures */ + +void reinitialize_almost_everything(); +void get_some_switches(); +LINENUM locate_hunk(); +void abort_hunk(); +void apply_hunk(); +void init_output(); +void init_reject(); +void copy_till(); +void spew_output(); +void dump_line(); +bool patch_match(); +bool similar(); +void re_input(); +void my_exit(); + +/* TRUE if -E was specified on command line.  */ +static int remove_empty_files = FALSE; + +/* TRUE if -R was specified on command line.  */ +static int reverse_flag_specified = FALSE; + +/* Apply a set of diffs as appropriate. */ + +int +main(argc,argv) +int argc; +char **argv; +{ +    LINENUM where; +    LINENUM newwhere; +    LINENUM fuzz; +    LINENUM mymaxfuzz; +    int hunk = 0; +    int failed = 0; +    int failtotal = 0; +    bool rev_okayed = 0; +    int i; + +    setbuf(stderr, serrbuf); +    for (i = 0; i<MAXFILEC; i++) +	filearg[i] = Nullch; + +    myuid = getuid(); + +    /* Cons up the names of the temporary files.  */ +    { +      /* Directory for temporary files.  */ +      char *tmpdir; +      int tmpname_len; + +      tmpdir = getenv ("TMPDIR"); +      if (tmpdir == NULL) { +	tmpdir = "/tmp"; +      } +      tmpname_len = strlen (tmpdir) + 20; + +      TMPOUTNAME = (char *) malloc (tmpname_len); +      strcpy (TMPOUTNAME, tmpdir); +      strcat (TMPOUTNAME, "/patchoXXXXXX"); +      Mktemp(TMPOUTNAME); + +      TMPINNAME = (char *) malloc (tmpname_len); +      strcpy (TMPINNAME, tmpdir); +      strcat (TMPINNAME, "/patchiXXXXXX"); +      Mktemp(TMPINNAME); + +      TMPREJNAME = (char *) malloc (tmpname_len); +      strcpy (TMPREJNAME, tmpdir); +      strcat (TMPREJNAME, "/patchrXXXXXX"); +      Mktemp(TMPREJNAME); + +      TMPPATNAME = (char *) malloc (tmpname_len); +      strcpy (TMPPATNAME, tmpdir); +      strcat (TMPPATNAME, "/patchpXXXXXX"); +      Mktemp(TMPPATNAME); +    } + +    { +      char *v; + +      v = getenv ("SIMPLE_BACKUP_SUFFIX"); +      if (v) +	simple_backup_suffix = v; +      else +	simple_backup_suffix = ".orig"; +#ifndef NODIR +      v = getenv ("VERSION_CONTROL"); +      backup_type = get_version (v); /* OK to pass NULL. */ +#endif +    } + +    /* parse switches */ +    Argc = argc; +    Argv = argv; +    get_some_switches(); +     +    /* make sure we clean up /tmp in case of disaster */ +    set_signals(0); + +    for ( +	open_patch_file(filearg[1]); +	there_is_another_patch(); +	reinitialize_almost_everything() +    ) {					/* for each patch in patch file */ + +	if (outname == Nullch) +	    outname = savestr(filearg[0]); +     +	/* for ed script just up and do it and exit */ +	if (diff_type == ED_DIFF) { +	    do_ed_script(); +	    continue; +	} +     +	/* initialize the patched file */ +	if (!skip_rest_of_patch) +	    init_output(TMPOUTNAME); +     +	/* initialize reject file */ +	init_reject(TMPREJNAME); +     +	/* find out where all the lines are */ +	if (!skip_rest_of_patch) +	    scan_input(filearg[0]); +     +	/* from here on, open no standard i/o files, because malloc */ +	/* might misfire and we can't catch it easily */ +     +	/* apply each hunk of patch */ +	hunk = 0; +	failed = 0; +	rev_okayed = FALSE; +	out_of_mem = FALSE; +	while (another_hunk()) { +	    hunk++; +	    fuzz = Nulline; +	    mymaxfuzz = pch_context(); +	    if (maxfuzz < mymaxfuzz) +		mymaxfuzz = maxfuzz; +	    if (!skip_rest_of_patch) { +		do { +		    where = locate_hunk(fuzz); +		    if (hunk == 1 && where == Nulline && !(force|rev_okayed)) { +						/* dwim for reversed patch? */ +			if (!pch_swap()) { +			    if (fuzz == Nulline) +				say1( +"Not enough memory to try swapped hunk!  Assuming unswapped.\n"); +			    continue; +			} +			reverse = !reverse; +			where = locate_hunk(fuzz);  /* try again */ +			if (where == Nulline) {	    /* didn't find it swapped */ +			    if (!pch_swap())         /* put it back to normal */ +				fatal1("lost hunk on alloc error!\n"); +			    reverse = !reverse; +			} +			else if (noreverse) { +			    if (!pch_swap())         /* put it back to normal */ +				fatal1("lost hunk on alloc error!\n"); +			    reverse = !reverse; +			    say1( +"Ignoring previously applied (or reversed) patch.\n"); +			    skip_rest_of_patch = TRUE; +			} +			else if (batch) { +			    if (verbose) +				say3( +"%seversed (or previously applied) patch detected!  %s -R.", +				reverse ? "R" : "Unr", +				reverse ? "Assuming" : "Ignoring"); +			} +			else { +			    ask3( +"%seversed (or previously applied) patch detected!  %s -R? [y] ", +				reverse ? "R" : "Unr", +				reverse ? "Assume" : "Ignore"); +			    if (*buf == 'n') { +				ask1("Apply anyway? [n] "); +				if (*buf == 'y') +				    rev_okayed = TRUE; +				else +				    skip_rest_of_patch = TRUE; +				where = Nulline; +				reverse = !reverse; +				if (!pch_swap())  /* put it back to normal */ +				    fatal1("lost hunk on alloc error!\n"); +			    } +			} +		    } +		} while (!skip_rest_of_patch && where == Nulline && +		    ++fuzz <= mymaxfuzz); + +		if (skip_rest_of_patch) {		/* just got decided */ +		    Fclose(ofp); +		    ofp = Nullfp; +		} +	    } + +	    newwhere = pch_newfirst() + last_offset; +	    if (skip_rest_of_patch) { +		abort_hunk(); +		failed++; +		if (verbose) +		    say3("Hunk #%d ignored at %ld.\n", hunk, newwhere); +	    } +	    else if (where == Nulline) { +		abort_hunk(); +		failed++; +		if (verbose) +		    say3("Hunk #%d failed at %ld.\n", hunk, newwhere); +	    } +	    else { +		apply_hunk(where); +		if (verbose) { +		    say3("Hunk #%d succeeded at %ld", hunk, newwhere); +		    if (fuzz) +			say2(" with fuzz %ld", fuzz); +		    if (last_offset) +			say3(" (offset %ld line%s)", +			    last_offset, last_offset==1L?"":"s"); +		    say1(".\n"); +		} +	    } +	} + +	if (out_of_mem && using_plan_a) { +	    optind = optind_last; +	    say1("\n\nRan out of memory using Plan A--trying again...\n\n"); +	    if (ofp) +	        Fclose(ofp); +	    ofp = Nullfp; +	    if (rejfp) +	        Fclose(rejfp); +	    rejfp = Nullfp; +	    continue; +	} +     +	assert(hunk); +     +	/* finish spewing out the new file */ +	if (!skip_rest_of_patch) +	    spew_output(); +	 +	/* and put the output where desired */ +	ignore_signals(); +	if (!skip_rest_of_patch) { +	    struct stat statbuf; +	    char *realout = outname; + +	    if (move_file(TMPOUTNAME, outname) < 0) { +		toutkeep = TRUE; +		realout = TMPOUTNAME; +		chmod(TMPOUTNAME, filemode); +	    } +	    else +		chmod(outname, filemode); + +	    if (remove_empty_files && stat(realout, &statbuf) == 0 +		&& statbuf.st_size == 0) { +		if (verbose) +		    say2("Removing %s (empty after patching).\n", realout); +	        while (unlink(realout) >= 0) ; /* while is for Eunice.  */ +	    } +	} +	Fclose(rejfp); +	rejfp = Nullfp; +	if (failed) { +	    failtotal += failed; +	    if (!*rejname) { +		Strcpy(rejname, outname); +		addext(rejname, ".rej", '#'); +	    } +	    if (skip_rest_of_patch) { +		say4("%d out of %d hunks ignored--saving rejects to %s\n", +		    failed, hunk, rejname); +	    } +	    else { +		say4("%d out of %d hunks failed--saving rejects to %s\n", +		    failed, hunk, rejname); +	    } +	    if (move_file(TMPREJNAME, rejname) < 0) +		trejkeep = TRUE; +	} +	set_signals(1); +    } +    my_exit(failtotal); +} + +/* Prepare to find the next patch to do in the patch file. */ + +void +reinitialize_almost_everything() +{ +    re_patch(); +    re_input(); + +    input_lines = 0; +    last_frozen_line = 0; + +    filec = 0; +    if (filearg[0] != Nullch && !out_of_mem) { +	free(filearg[0]); +	filearg[0] = Nullch; +    } + +    if (outname != Nullch) { +	free(outname); +	outname = Nullch; +    } + +    last_offset = 0; + +    diff_type = 0; + +    if (revision != Nullch) { +	free(revision); +	revision = Nullch; +    } + +    reverse = reverse_flag_specified; +    skip_rest_of_patch = FALSE; + +    get_some_switches(); + +    if (filec >= 2) +	fatal1("you may not change to a different patch file\n"); +} + +static char *shortopts = "-b:B:cd:D:eEfF:lnNo:p::r:RsStuvV:x:"; +static struct option longopts[] = +{ +  {"suffix", 1, NULL, 'b'}, +  {"prefix", 1, NULL, 'B'}, +  {"context", 0, NULL, 'c'}, +  {"directory", 1, NULL, 'd'}, +  {"ifdef", 1, NULL, 'D'}, +  {"ed", 0, NULL, 'e'}, +  {"remove-empty-files", 0, NULL, 'E'}, +  {"force", 0, NULL, 'f'}, +  {"fuzz", 1, NULL, 'F'}, +  {"ignore-whitespace", 0, NULL, 'l'}, +  {"normal", 0, NULL, 'n'}, +  {"forward", 0, NULL, 'N'}, +  {"output", 1, NULL, 'o'}, +  {"strip", 2, NULL, 'p'}, +  {"reject-file", 1, NULL, 'r'}, +  {"reverse", 0, NULL, 'R'}, +  {"quiet", 0, NULL, 's'}, +  {"silent", 0, NULL, 's'}, +  {"skip", 0, NULL, 'S'}, +  {"batch", 0, NULL, 't'}, +  {"unified", 0, NULL, 'u'}, +  {"version", 0, NULL, 'v'}, +  {"version-control", 1, NULL, 'V'}, +  {"debug", 1, NULL, 'x'}, +  {0, 0, 0, 0} +}; + +/* Process switches and filenames up to next '+' or end of list. */ + +void +get_some_switches() +{ +    Reg1 int optc; + +    rejname[0] = '\0'; +    optind_last = optind; +    if (optind == Argc) +	return; +    while ((optc = getopt_long (Argc, Argv, shortopts, longopts, (int *) 0)) +	   != -1) { +	if (optc == 1) { +	    if (strEQ(optarg, "+")) +		return; +	    if (filec == MAXFILEC) +		fatal1("too many file arguments\n"); +	    filearg[filec++] = savestr(optarg); +	} +	else { +	    switch (optc) { +	    case 'b': +		simple_backup_suffix = savestr(optarg); +		break; +	    case 'B': +		origprae = savestr(optarg); +		break; +	    case 'c': +		diff_type = CONTEXT_DIFF; +		break; +	    case 'd': +		if (chdir(optarg) < 0) +		    pfatal2("can't cd to %s", optarg); +		break; +	    case 'D': +	    	do_defines = TRUE; +		if (!isalpha(*optarg) && '_' != *optarg) +		    fatal1("argument to -D is not an identifier\n"); +		Sprintf(if_defined, "#ifdef %s\n", optarg); +		Sprintf(not_defined, "#ifndef %s\n", optarg); +		Sprintf(end_defined, "#endif /* %s */\n", optarg); +		break; +	    case 'e': +		diff_type = ED_DIFF; +		break; +	    case 'E': +		remove_empty_files = TRUE; +		break; +	    case 'f': +		force = TRUE; +		break; +	    case 'F': +		maxfuzz = atoi(optarg); +		break; +	    case 'l': +		canonicalize = TRUE; +		break; +	    case 'n': +		diff_type = NORMAL_DIFF; +		break; +	    case 'N': +		noreverse = TRUE; +		break; +	    case 'o': +		outname = savestr(optarg); +		break; +	    case 'p': +		if (optarg) +		    strippath = atoi(optarg); +		else +		    strippath = 0; +		break; +	    case 'r': +		Strcpy(rejname, optarg); +		break; +	    case 'R': +		reverse = TRUE; +		reverse_flag_specified = TRUE; +		break; +	    case 's': +		verbose = FALSE; +		break; +	    case 'S': +		skip_rest_of_patch = TRUE; +		break; +	    case 't': +		batch = TRUE; +		break; +	    case 'u': +		diff_type = UNI_DIFF; +		break; +	    case 'v': +		version(); +		break; +	    case 'V': +#ifndef NODIR +		backup_type = get_version (optarg); +#endif +		break; +#ifdef DEBUGGING +	    case 'x': +		debug = atoi(optarg); +		break; +#endif +	    default: +		fprintf(stderr, "\ +Usage: %s [options] [origfile [patchfile]] [+ [options] [origfile]]...\n", +			Argv[0]); +		fprintf(stderr, "\ +Options:\n\ +       [-ceEflnNRsStuv] [-b backup-ext] [-B backup-prefix] [-d directory]\n\ +       [-D symbol] [-F max-fuzz] [-o out-file] [-p[strip-count]]\n\ +       [-r rej-name] [-V {numbered,existing,simple}] [--context]\n\ +       [--prefix=backup-prefix] [--suffix=backup-ext] [--ifdef=symbol]\n\ +       [--directory=directory] [--ed] [--fuzz=max-fuzz] [--force] [--batch]\n\ +       [--ignore-whitespace] [--forward] [--reverse] [--output=out-file]\n"); +		fprintf(stderr, "\ +       [--strip[=strip-count]] [--normal] [--reject-file=rej-name] [--skip]\n\ +       [--remove-empty-files] [--quiet] [--silent] [--unified] [--version]\n\ +       [--version-control={numbered,existing,simple}]\n"); +		my_exit(1); +	    } +	} +    } + +    /* Process any filename args given after "--".  */ +    for (; optind < Argc; ++optind) { +	if (filec == MAXFILEC) +	    fatal1("too many file arguments\n"); +	filearg[filec++] = savestr(Argv[optind]); +    } +} + +/* Attempt to find the right place to apply this hunk of patch. */ + +LINENUM +locate_hunk(fuzz) +LINENUM fuzz; +{ +    Reg1 LINENUM first_guess = pch_first() + last_offset; +    Reg2 LINENUM offset; +    LINENUM pat_lines = pch_ptrn_lines(); +    Reg3 LINENUM max_pos_offset = input_lines - first_guess +				- pat_lines + 1;  +    Reg4 LINENUM max_neg_offset = first_guess - last_frozen_line - 1 +				+ pch_context(); + +    if (!pat_lines)			/* null range matches always */ +	return first_guess; +    if (max_neg_offset >= first_guess)	/* do not try lines < 0 */ +	max_neg_offset = first_guess - 1; +    if (first_guess <= input_lines && patch_match(first_guess, Nulline, fuzz)) +	return first_guess; +    for (offset = 1; ; offset++) { +	Reg5 bool check_after = (offset <= max_pos_offset); +	Reg6 bool check_before = (offset <= max_neg_offset); + +	if (check_after && patch_match(first_guess, offset, fuzz)) { +#ifdef DEBUGGING +	    if (debug & 1) +		say3("Offset changing from %ld to %ld\n", last_offset, offset); +#endif +	    last_offset = offset; +	    return first_guess+offset; +	} +	else if (check_before && patch_match(first_guess, -offset, fuzz)) { +#ifdef DEBUGGING +	    if (debug & 1) +		say3("Offset changing from %ld to %ld\n", last_offset, -offset); +#endif +	    last_offset = -offset; +	    return first_guess-offset; +	} +	else if (!check_before && !check_after) +	    return Nulline; +    } +} + +/* We did not find the pattern, dump out the hunk so they can handle it. */ + +void +abort_hunk() +{ +    Reg1 LINENUM i; +    Reg2 LINENUM pat_end = pch_end(); +    /* add in last_offset to guess the same as the previous successful hunk */ +    LINENUM oldfirst = pch_first() + last_offset; +    LINENUM newfirst = pch_newfirst() + last_offset; +    LINENUM oldlast = oldfirst + pch_ptrn_lines() - 1; +    LINENUM newlast = newfirst + pch_repl_lines() - 1; +    char *stars = (diff_type >= NEW_CONTEXT_DIFF ? " ****" : ""); +    char *minuses = (diff_type >= NEW_CONTEXT_DIFF ? " ----" : " -----"); + +    fprintf(rejfp, "***************\n"); +    for (i=0; i<=pat_end; i++) { +	switch (pch_char(i)) { +	case '*': +	    if (oldlast < oldfirst) +		fprintf(rejfp, "*** 0%s\n", stars); +	    else if (oldlast == oldfirst) +		fprintf(rejfp, "*** %ld%s\n", oldfirst, stars); +	    else +		fprintf(rejfp, "*** %ld,%ld%s\n", oldfirst, oldlast, stars); +	    break; +	case '=': +	    if (newlast < newfirst) +		fprintf(rejfp, "--- 0%s\n", minuses); +	    else if (newlast == newfirst) +		fprintf(rejfp, "--- %ld%s\n", newfirst, minuses); +	    else +		fprintf(rejfp, "--- %ld,%ld%s\n", newfirst, newlast, minuses); +	    break; +	case '\n': +	    fprintf(rejfp, "%s", pfetch(i)); +	    break; +	case ' ': case '-': case '+': case '!': +	    fprintf(rejfp, "%c %s", pch_char(i), pfetch(i)); +	    break; +	default: +	    fatal1("fatal internal error in abort_hunk\n");  +	} +    } +} + +/* We found where to apply it (we hope), so do it. */ + +void +apply_hunk(where) +LINENUM where; +{ +    Reg1 LINENUM old = 1; +    Reg2 LINENUM lastline = pch_ptrn_lines(); +    Reg3 LINENUM new = lastline+1; +#define OUTSIDE 0 +#define IN_IFNDEF 1 +#define IN_IFDEF 2 +#define IN_ELSE 3 +    Reg4 int def_state = OUTSIDE; +    Reg5 bool R_do_defines = do_defines; +    Reg6 LINENUM pat_end = pch_end(); + +    where--; +    while (pch_char(new) == '=' || pch_char(new) == '\n') +	new++; +     +    while (old <= lastline) { +	if (pch_char(old) == '-') { +	    copy_till(where + old - 1); +	    if (R_do_defines) { +		if (def_state == OUTSIDE) { +		    fputs(not_defined, ofp); +		    def_state = IN_IFNDEF; +		} +		else if (def_state == IN_IFDEF) { +		    fputs(else_defined, ofp); +		    def_state = IN_ELSE; +		} +		fputs(pfetch(old), ofp); +	    } +	    last_frozen_line++; +	    old++; +	} +	else if (new > pat_end) { +	    break; +	} +	else if (pch_char(new) == '+') { +	    copy_till(where + old - 1); +	    if (R_do_defines) { +		if (def_state == IN_IFNDEF) { +		    fputs(else_defined, ofp); +		    def_state = IN_ELSE; +		} +		else if (def_state == OUTSIDE) { +		    fputs(if_defined, ofp); +		    def_state = IN_IFDEF; +		} +	    } +	    fputs(pfetch(new), ofp); +	    new++; +	} +	else if (pch_char(new) != pch_char(old)) { +	    say3("Out-of-sync patch, lines %ld,%ld--mangled text or line numbers, maybe?\n", +		pch_hunk_beg() + old, +		pch_hunk_beg() + new); +#ifdef DEBUGGING +	    say3("oldchar = '%c', newchar = '%c'\n", +		pch_char(old), pch_char(new)); +#endif +	    my_exit(1); +	} +	else if (pch_char(new) == '!') { +	    copy_till(where + old - 1); +	    if (R_do_defines) { +	       fputs(not_defined, ofp); +	       def_state = IN_IFNDEF; +	    } +	    while (pch_char(old) == '!') { +		if (R_do_defines) { +		    fputs(pfetch(old), ofp); +		} +		last_frozen_line++; +		old++; +	    } +	    if (R_do_defines) { +		fputs(else_defined, ofp); +		def_state = IN_ELSE; +	    } +	    while (pch_char(new) == '!') { +		fputs(pfetch(new), ofp); +		new++; +	    } +	} +	else { +	    assert(pch_char(new) == ' '); +	    old++; +	    new++; +	    if (R_do_defines && def_state != OUTSIDE) { +		fputs(end_defined, ofp); +		def_state = OUTSIDE; +	    } +	} +    } +    if (new <= pat_end && pch_char(new) == '+') { +	copy_till(where + old - 1); +	if (R_do_defines) { +	    if (def_state == OUTSIDE) { +	    	fputs(if_defined, ofp); +		def_state = IN_IFDEF; +	    } +	    else if (def_state == IN_IFNDEF) { +		fputs(else_defined, ofp); +		def_state = IN_ELSE; +	    } +	} +	while (new <= pat_end && pch_char(new) == '+') { +	    fputs(pfetch(new), ofp); +	    new++; +	} +    } +    if (R_do_defines && def_state != OUTSIDE) { +	fputs(end_defined, ofp); +    } +} + +/* Open the new file. */ + +void +init_output(name) +char *name; +{ +    ofp = fopen(name, "w"); +    if (ofp == Nullfp) +	pfatal2("can't create %s", name); +} + +/* Open a file to put hunks we can't locate. */ + +void +init_reject(name) +char *name; +{ +    rejfp = fopen(name, "w"); +    if (rejfp == Nullfp) +	pfatal2("can't create %s", name); +} + +/* Copy input file to output, up to wherever hunk is to be applied. */ + +void +copy_till(lastline) +Reg1 LINENUM lastline; +{ +    Reg2 LINENUM R_last_frozen_line = last_frozen_line; + +    if (R_last_frozen_line > lastline) +	fatal1("misordered hunks! output would be garbled\n"); +    while (R_last_frozen_line < lastline) { +	dump_line(++R_last_frozen_line); +    } +    last_frozen_line = R_last_frozen_line; +} + +/* Finish copying the input file to the output file. */ + +void +spew_output() +{ +#ifdef DEBUGGING +    if (debug & 256) +	say3("il=%ld lfl=%ld\n",input_lines,last_frozen_line); +#endif +    if (input_lines) +	copy_till(input_lines);		/* dump remainder of file */ +    Fclose(ofp); +    ofp = Nullfp; +} + +/* Copy one line from input to output. */ + +void +dump_line(line) +LINENUM line; +{ +    Reg1 char *s; +    Reg2 char R_newline = '\n'; + +    /* Note: string is not null terminated. */ +    for (s=ifetch(line, 0); putc(*s, ofp) != R_newline; s++) ; +} + +/* Does the patch pattern match at line base+offset? */ + +bool +patch_match(base, offset, fuzz) +LINENUM base; +LINENUM offset; +LINENUM fuzz; +{ +    Reg1 LINENUM pline = 1 + fuzz; +    Reg2 LINENUM iline; +    Reg3 LINENUM pat_lines = pch_ptrn_lines() - fuzz; + +    for (iline=base+offset+fuzz; pline <= pat_lines; pline++,iline++) { +	if (canonicalize) { +	    if (!similar(ifetch(iline, (offset >= 0)), +			 pfetch(pline), +			 pch_line_len(pline) )) +		return FALSE; +	} +	else if (strnNE(ifetch(iline, (offset >= 0)), +		   pfetch(pline), +		   pch_line_len(pline) )) +	    return FALSE; +    } +    return TRUE; +} + +/* Do two lines match with canonicalized white space? */ + +bool +similar(a,b,len) +Reg1 char *a; +Reg2 char *b; +Reg3 int len; +{ +    while (len) { +	if (isspace(*b)) {		/* whitespace (or \n) to match? */ +	    if (!isspace(*a))		/* no corresponding whitespace? */ +		return FALSE; +	    while (len && isspace(*b) && *b != '\n') +		b++,len--;		/* skip pattern whitespace */ +	    while (isspace(*a) && *a != '\n') +		a++;			/* skip target whitespace */ +	    if (*a == '\n' || *b == '\n') +		return (*a == *b);	/* should end in sync */ +	} +	else if (*a++ != *b++)		/* match non-whitespace chars */ +	    return FALSE; +	else +	    len--;			/* probably not necessary */ +    } +    return TRUE;			/* actually, this is not reached */ +					/* since there is always a \n */ +} + +/* Exit with cleanup. */ + +void +my_exit(status) +int status; +{ +    Unlink(TMPINNAME); +    if (!toutkeep) { +	Unlink(TMPOUTNAME); +    } +    if (!trejkeep) { +	Unlink(TMPREJNAME); +    } +    Unlink(TMPPATNAME); +    exit(status); +} diff --git a/gnu/usr.bin/patch/patchlevel.h b/gnu/usr.bin/patch/patchlevel.h new file mode 100644 index 000000000000..d5de3a9e24dc --- /dev/null +++ b/gnu/usr.bin/patch/patchlevel.h @@ -0,0 +1 @@ +#define PATCH_VERSION "2.1" diff --git a/gnu/usr.bin/patch/pch.c b/gnu/usr.bin/patch/pch.c new file mode 100644 index 000000000000..cb058f7266cf --- /dev/null +++ b/gnu/usr.bin/patch/pch.c @@ -0,0 +1,1316 @@ +/* $Header: /home/cvs/386BSD/src/gnu/usr.bin/patch/pch.c,v 1.3 1994/02/17 22:20:36 jkh Exp $ + * + * $Log: pch.c,v $ + * Revision 1.3  1994/02/17  22:20:36  jkh + * Put this back - I was somehow under the erroneous impression that patch was in + * ports, until I saw the the commit messages, that is! :-)  All changed backed out. + * + * Revision 1.2  1994/02/17  22:16:05  jkh + * From Poul-Henning Kamp -  Implement a -C option to verify the integrity of + * a patch before actually applying it. + * + * Revision 1.1.1.1  1993/06/19  14:21:52  paul + * b-maked patch-2.10 + * + * Revision 2.0.2.0  90/05/01  22:17:51  davison + * patch12u: unidiff support added + * + * Revision 2.0.1.7  88/06/03  15:13:28  lwall + * patch10: Can now find patches in shar scripts. + * patch10: Hunks that swapped and then swapped back could core dump. + *  + * Revision 2.0.1.6  87/06/04  16:18:13  lwall + * pch_swap didn't swap p_bfake and p_efake. + *  + * Revision 2.0.1.5  87/01/30  22:47:42  lwall + * Improved responses to mangled patches. + *  + * Revision 2.0.1.4  87/01/05  16:59:53  lwall + * New-style context diffs caused double call to free(). + *  + * Revision 2.0.1.3  86/11/14  10:08:33  lwall + * Fixed problem where a long pattern wouldn't grow the hunk. + * Also restored p_input_line when backtracking so error messages are right. + *  + * Revision 2.0.1.2  86/11/03  17:49:52  lwall + * New-style delete triggers spurious assertion error. + *  + * Revision 2.0.1.1  86/10/29  15:52:08  lwall + * Could falsely report new-style context diff. + *  + * Revision 2.0  86/09/17  15:39:37  lwall + * Baseline for netwide release. + *  + */ + +#include "EXTERN.h" +#include "common.h" +#include "util.h" +#include "INTERN.h" +#include "pch.h" + +/* Patch (diff listing) abstract type. */ + +static long p_filesize;			/* size of the patch file */ +static LINENUM p_first;			/* 1st line number */ +static LINENUM p_newfirst;		/* 1st line number of replacement */ +static LINENUM p_ptrn_lines;		/* # lines in pattern */ +static LINENUM p_repl_lines;		/* # lines in replacement text */ +static LINENUM p_end = -1;		/* last line in hunk */ +static LINENUM p_max;			/* max allowed value of p_end */ +static LINENUM p_context = 3;		/* # of context lines */ +static LINENUM p_input_line = 0;	/* current line # from patch file */ +static char **p_line = Null(char**);	/* the text of the hunk */ +static short *p_len = Null(short*);	/* length of each line */ +static char *p_Char = Nullch;		/* +, -, and ! */ +static int hunkmax = INITHUNKMAX;	/* size of above arrays to begin with */ +static int p_indent;			/* indent to patch */ +static LINENUM p_base;			/* where to intuit this time */ +static LINENUM p_bline;			/* line # of p_base */ +static LINENUM p_start;			/* where intuit found a patch */ +static LINENUM p_sline;			/* and the line number for it */ +static LINENUM p_hunk_beg;		/* line number of current hunk */ +static LINENUM p_efake = -1;		/* end of faked up lines--don't free */ +static LINENUM p_bfake = -1;		/* beg of faked up lines */ + +/* Prepare to look for the next patch in the patch file. */ + +void +re_patch() +{ +    p_first = Nulline; +    p_newfirst = Nulline; +    p_ptrn_lines = Nulline; +    p_repl_lines = Nulline; +    p_end = (LINENUM)-1; +    p_max = Nulline; +    p_indent = 0; +} + +/* Open the patch file at the beginning of time. */ + +void +open_patch_file(filename) +char *filename; +{ +    if (filename == Nullch || !*filename || strEQ(filename, "-")) { +	pfp = fopen(TMPPATNAME, "w"); +	if (pfp == Nullfp) +	    pfatal2("can't create %s", TMPPATNAME); +	while (fgets(buf, sizeof buf, stdin) != Nullch) +	    fputs(buf, pfp); +	Fclose(pfp); +	filename = TMPPATNAME; +    } +    pfp = fopen(filename, "r"); +    if (pfp == Nullfp) +	pfatal2("patch file %s not found", filename); +    Fstat(fileno(pfp), &filestat); +    p_filesize = filestat.st_size; +    next_intuit_at(0L,1L);			/* start at the beginning */ +    set_hunkmax(); +} + +/* Make sure our dynamically realloced tables are malloced to begin with. */ + +void +set_hunkmax() +{ +#ifndef lint +    if (p_line == Null(char**)) +	p_line = (char**) malloc((MEM)hunkmax * sizeof(char *)); +    if (p_len == Null(short*)) +	p_len  = (short*) malloc((MEM)hunkmax * sizeof(short)); +#endif +    if (p_Char == Nullch) +	p_Char = (char*)  malloc((MEM)hunkmax * sizeof(char)); +} + +/* Enlarge the arrays containing the current hunk of patch. */ + +void +grow_hunkmax() +{ +    hunkmax *= 2; +    /*  +     * Note that on most systems, only the p_line array ever gets fresh memory +     * since p_len can move into p_line's old space, and p_Char can move into +     * p_len's old space.  Not on PDP-11's however.  But it doesn't matter. +     */ +    assert(p_line != Null(char**) && p_len != Null(short*) && p_Char != Nullch); +#ifndef lint +    p_line = (char**) realloc((char*)p_line, (MEM)hunkmax * sizeof(char *)); +    p_len  = (short*) realloc((char*)p_len,  (MEM)hunkmax * sizeof(short)); +    p_Char = (char*)  realloc((char*)p_Char, (MEM)hunkmax * sizeof(char)); +#endif +    if (p_line != Null(char**) && p_len != Null(short*) && p_Char != Nullch) +	return; +    if (!using_plan_a) +	fatal1("out of memory\n"); +    out_of_mem = TRUE;		/* whatever is null will be allocated again */ +				/* from within plan_a(), of all places */ +} + +/* True if the remainder of the patch file contains a diff of some sort. */ + +bool +there_is_another_patch() +{ +    if (p_base != 0L && p_base >= p_filesize) { +	if (verbose) +	    say1("done\n"); +	return FALSE; +    } +    if (verbose) +	say1("Hmm..."); +    diff_type = intuit_diff_type(); +    if (!diff_type) { +	if (p_base != 0L) { +	    if (verbose) +		say1("  Ignoring the trailing garbage.\ndone\n"); +	} +	else +	    say1("  I can't seem to find a patch in there anywhere.\n"); +	return FALSE; +    } +    if (verbose) +	say3("  %sooks like %s to me...\n", +	    (p_base == 0L ? "L" : "The next patch l"), +	    diff_type == UNI_DIFF ? "a unified diff" : +	    diff_type == CONTEXT_DIFF ? "a context diff" : +	    diff_type == NEW_CONTEXT_DIFF ? "a new-style context diff" : +	    diff_type == NORMAL_DIFF ? "a normal diff" : +	    "an ed script" ); +    if (p_indent && verbose) +	say3("(Patch is indented %d space%s.)\n", p_indent, p_indent==1?"":"s"); +    skip_to(p_start,p_sline); +    while (filearg[0] == Nullch) { +	if (force || batch) { +	    say1("No file to patch.  Skipping...\n"); +	    filearg[0] = savestr(bestguess); +	    return TRUE; +	} +	ask1("File to patch: "); +	if (*buf != '\n') { +	    if (bestguess) +		free(bestguess); +	    bestguess = savestr(buf); +	    filearg[0] = fetchname(buf, 0, FALSE); +	} +	if (filearg[0] == Nullch) { +	    ask1("No file found--skip this patch? [n] "); +	    if (*buf != 'y') { +		continue; +	    } +	    if (verbose) +		say1("Skipping patch...\n"); +	    filearg[0] = fetchname(bestguess, 0, TRUE); +	    skip_rest_of_patch = TRUE; +	    return TRUE; +	} +    } +    return TRUE; +} + +/* Determine what kind of diff is in the remaining part of the patch file. */ + +int +intuit_diff_type() +{ +    Reg4 long this_line = 0; +    Reg5 long previous_line; +    Reg6 long first_command_line = -1; +    long fcl_line; +    Reg7 bool last_line_was_command = FALSE; +    Reg8 bool this_is_a_command = FALSE; +    Reg9 bool stars_last_line = FALSE; +    Reg10 bool stars_this_line = FALSE; +    Reg3 int indent; +    Reg1 char *s; +    Reg2 char *t; +    char *indtmp = Nullch; +    char *oldtmp = Nullch; +    char *newtmp = Nullch; +    char *indname = Nullch; +    char *oldname = Nullch; +    char *newname = Nullch; +    Reg11 int retval; +    bool no_filearg = (filearg[0] == Nullch); + +    ok_to_create_file = FALSE; +    Fseek(pfp, p_base, 0); +    p_input_line = p_bline - 1; +    for (;;) { +	previous_line = this_line; +	last_line_was_command = this_is_a_command; +	stars_last_line = stars_this_line; +	this_line = ftell(pfp); +	indent = 0; +	p_input_line++; +	if (fgets(buf, sizeof buf, pfp) == Nullch) { +	    if (first_command_line >= 0L) { +					/* nothing but deletes!? */ +		p_start = first_command_line; +		p_sline = fcl_line; +		retval = ED_DIFF; +		goto scan_exit; +	    } +	    else { +		p_start = this_line; +		p_sline = p_input_line; +		retval = 0; +		goto scan_exit; +	    } +	} +	for (s = buf; *s == ' ' || *s == '\t' || *s == 'X'; s++) { +	    if (*s == '\t') +		indent += 8 - (indent % 8); +	    else +		indent++; +	} +	for (t=s; isdigit(*t) || *t == ','; t++) ;  +	this_is_a_command = (isdigit(*s) && +	  (*t == 'd' || *t == 'c' || *t == 'a') ); +	if (first_command_line < 0L && this_is_a_command) {  +	    first_command_line = this_line; +	    fcl_line = p_input_line; +	    p_indent = indent;		/* assume this for now */ +	} +	if (!stars_last_line && strnEQ(s, "*** ", 4)) +	    oldtmp = savestr(s+4); +	else if (strnEQ(s, "--- ", 4)) +	    newtmp = savestr(s+4); +	else if (strnEQ(s, "+++ ", 4)) +	    oldtmp = savestr(s+4);	/* pretend it is the old name */ +	else if (strnEQ(s, "Index:", 6)) +	    indtmp = savestr(s+6); +	else if (strnEQ(s, "Prereq:", 7)) { +	    for (t=s+7; isspace(*t); t++) ; +	    revision = savestr(t); +	    for (t=revision; *t && !isspace(*t); t++) ; +	    *t = '\0'; +	    if (!*revision) { +		free(revision); +		revision = Nullch; +	    } +	} +	if ((!diff_type || diff_type == ED_DIFF) && +	  first_command_line >= 0L && +	  strEQ(s, ".\n") ) { +	    p_indent = indent; +	    p_start = first_command_line; +	    p_sline = fcl_line; +	    retval = ED_DIFF; +	    goto scan_exit; +	} +	if ((!diff_type || diff_type == UNI_DIFF) && strnEQ(s, "@@ -", 4)) { +	    if (!atol(s+3)) +		ok_to_create_file = TRUE; +	    p_indent = indent; +	    p_start = this_line; +	    p_sline = p_input_line; +	    retval = UNI_DIFF; +	    goto scan_exit; +	} +	stars_this_line = strnEQ(s, "********", 8); +	if ((!diff_type || diff_type == CONTEXT_DIFF) && stars_last_line && +		 strnEQ(s, "*** ", 4)) { +	    if (!atol(s+4)) +		ok_to_create_file = TRUE; +	    /* if this is a new context diff the character just before */ +	    /* the newline is a '*'. */ +	    while (*s != '\n') +		s++; +	    p_indent = indent; +	    p_start = previous_line; +	    p_sline = p_input_line - 1; +	    retval = (*(s-1) == '*' ? NEW_CONTEXT_DIFF : CONTEXT_DIFF); +	    goto scan_exit; +	} +	if ((!diff_type || diff_type == NORMAL_DIFF) &&  +	  last_line_was_command && +	  (strnEQ(s, "< ", 2) || strnEQ(s, "> ", 2)) ) { +	    p_start = previous_line; +	    p_sline = p_input_line - 1; +	    p_indent = indent; +	    retval = NORMAL_DIFF; +	    goto scan_exit; +	} +    } +  scan_exit: +    if (no_filearg) { +	if (indtmp != Nullch) +	    indname = fetchname(indtmp, strippath, ok_to_create_file); +	if (oldtmp != Nullch) +	    oldname = fetchname(oldtmp, strippath, ok_to_create_file); +	if (newtmp != Nullch) +	    newname = fetchname(newtmp, strippath, ok_to_create_file); +	if (oldname && newname) { +	    if (strlen(oldname) < strlen(newname)) +		filearg[0] = savestr(oldname); +	    else +		filearg[0] = savestr(newname); +	} +	else if (oldname) +	    filearg[0] = savestr(oldname); +	else if (newname) +	    filearg[0] = savestr(newname); +	else if (indname) +	    filearg[0] = savestr(indname); +    } +    if (bestguess) { +	free(bestguess); +	bestguess = Nullch; +    } +    if (filearg[0] != Nullch) +	bestguess = savestr(filearg[0]); +    else if (indtmp != Nullch) +	bestguess = fetchname(indtmp, strippath, TRUE); +    else { +	if (oldtmp != Nullch) +	    oldname = fetchname(oldtmp, strippath, TRUE); +	if (newtmp != Nullch) +	    newname = fetchname(newtmp, strippath, TRUE); +	if (oldname && newname) { +	    if (strlen(oldname) < strlen(newname)) +		bestguess = savestr(oldname); +	    else +		bestguess = savestr(newname); +	} +	else if (oldname) +	    bestguess = savestr(oldname); +	else if (newname) +	    bestguess = savestr(newname); +    } +    if (indtmp != Nullch) +	free(indtmp); +    if (oldtmp != Nullch) +	free(oldtmp); +    if (newtmp != Nullch) +	free(newtmp); +    if (indname != Nullch) +	free(indname); +    if (oldname != Nullch) +	free(oldname); +    if (newname != Nullch) +	free(newname); +    return retval; +} + +/* Remember where this patch ends so we know where to start up again. */ + +void +next_intuit_at(file_pos,file_line) +long file_pos; +long file_line; +{ +    p_base = file_pos; +    p_bline = file_line; +} + +/* Basically a verbose fseek() to the actual diff listing. */ + +void +skip_to(file_pos,file_line) +long file_pos; +long file_line; +{ +    char *ret; + +    assert(p_base <= file_pos); +    if (verbose && p_base < file_pos) { +	Fseek(pfp, p_base, 0); +	say1("The text leading up to this was:\n--------------------------\n"); +	while (ftell(pfp) < file_pos) { +	    ret = fgets(buf, sizeof buf, pfp); +	    assert(ret != Nullch); +	    say2("|%s", buf); +	} +	say1("--------------------------\n"); +    } +    else +	Fseek(pfp, file_pos, 0); +    p_input_line = file_line - 1; +} + +/* Make this a function for better debugging.  */ +static void +malformed () +{ +    fatal3("malformed patch at line %ld: %s", p_input_line, buf); +		/* about as informative as "Syntax error" in C */ +} + +/* True if there is more of the current diff listing to process. */ + +bool +another_hunk() +{ +    Reg1 char *s; +    Reg8 char *ret; +    Reg2 int context = 0; + +    while (p_end >= 0) { +	if (p_end == p_efake) +	    p_end = p_bfake;		/* don't free twice */ +	else +	    free(p_line[p_end]); +	p_end--; +    } +    assert(p_end == -1); +    p_efake = -1; + +    p_max = hunkmax;			/* gets reduced when --- found */ +    if (diff_type == CONTEXT_DIFF || diff_type == NEW_CONTEXT_DIFF) { +	long line_beginning = ftell(pfp); +					/* file pos of the current line */ +	LINENUM repl_beginning = 0;	/* index of --- line */ +	Reg4 LINENUM fillcnt = 0;	/* #lines of missing ptrn or repl */ +	Reg5 LINENUM fillsrc;		/* index of first line to copy */ +	Reg6 LINENUM filldst;		/* index of first missing line */ +	bool ptrn_spaces_eaten = FALSE;	/* ptrn was slightly misformed */ +	Reg9 bool repl_could_be_missing = TRUE; +					/* no + or ! lines in this hunk */ +	bool repl_missing = FALSE;	/* we are now backtracking */ +	long repl_backtrack_position = 0; +					/* file pos of first repl line */ +	LINENUM repl_patch_line;	/* input line number for same */ +	Reg7 LINENUM ptrn_copiable = 0; +					/* # of copiable lines in ptrn */ + +	ret = pgets(buf, sizeof buf, pfp); +	p_input_line++; +	if (ret == Nullch || strnNE(buf, "********", 8)) { +	    next_intuit_at(line_beginning,p_input_line); +	    return FALSE; +	} +	p_context = 100; +	p_hunk_beg = p_input_line + 1; +	while (p_end < p_max) { +	    line_beginning = ftell(pfp); +	    ret = pgets(buf, sizeof buf, pfp); +	    p_input_line++; +	    if (ret == Nullch) { +		if (p_max - p_end < 4) +		    Strcpy(buf, "  \n");  /* assume blank lines got chopped */ +		else { +		    if (repl_beginning && repl_could_be_missing) { +			repl_missing = TRUE; +			goto hunk_done; +		    } +		    fatal1("unexpected end of file in patch\n"); +		} +	    } +	    p_end++; +	    assert(p_end < hunkmax); +	    p_Char[p_end] = *buf; +#ifdef zilog +	    p_line[(short)p_end] = Nullch; +#else +	    p_line[p_end] = Nullch; +#endif +	    switch (*buf) { +	    case '*': +		if (strnEQ(buf, "********", 8)) { +		    if (repl_beginning && repl_could_be_missing) { +			repl_missing = TRUE; +			goto hunk_done; +		    } +		    else +			fatal2("unexpected end of hunk at line %ld\n", +			    p_input_line); +		} +		if (p_end != 0) { +		    if (repl_beginning && repl_could_be_missing) { +			repl_missing = TRUE; +			goto hunk_done; +		    } +		    fatal3("unexpected *** at line %ld: %s", p_input_line, buf); +		} +		context = 0; +		p_line[p_end] = savestr(buf); +		if (out_of_mem) { +		    p_end--; +		    return FALSE; +		} +		for (s=buf; *s && !isdigit(*s); s++) ; +		if (!*s) +		    malformed (); +		if (strnEQ(s,"0,0",3)) +		    strcpy(s,s+2); +		p_first = (LINENUM) atol(s); +		while (isdigit(*s)) s++; +		if (*s == ',') { +		    for (; *s && !isdigit(*s); s++) ; +		    if (!*s) +			malformed (); +		    p_ptrn_lines = ((LINENUM)atol(s)) - p_first + 1; +		} +		else if (p_first) +		    p_ptrn_lines = 1; +		else { +		    p_ptrn_lines = 0; +		    p_first = 1; +		} +		p_max = p_ptrn_lines + 6;	/* we need this much at least */ +		while (p_max >= hunkmax) +		    grow_hunkmax(); +		p_max = hunkmax; +		break; +	    case '-': +		if (buf[1] == '-') { +		    if (repl_beginning || +			(p_end != p_ptrn_lines + 1 + (p_Char[p_end-1] == '\n'))) +		    { +			if (p_end == 1) { +			    /* `old' lines were omitted - set up to fill */ +			    /* them in from 'new' context lines. */ +			    p_end = p_ptrn_lines + 1; +			    fillsrc = p_end + 1; +			    filldst = 1; +			    fillcnt = p_ptrn_lines; +			} +			else { +			    if (repl_beginning) { +				if (repl_could_be_missing){ +				    repl_missing = TRUE; +				    goto hunk_done; +				} +				fatal3( +"duplicate \"---\" at line %ld--check line numbers at line %ld\n", +				    p_input_line, p_hunk_beg + repl_beginning); +			    } +			    else { +				fatal4( +"%s \"---\" at line %ld--check line numbers at line %ld\n", +				    (p_end <= p_ptrn_lines +					? "Premature" +					: "Overdue" ), +				    p_input_line, p_hunk_beg); +			    } +			} +		    } +		    repl_beginning = p_end; +		    repl_backtrack_position = ftell(pfp); +		    repl_patch_line = p_input_line; +		    p_line[p_end] = savestr(buf); +		    if (out_of_mem) { +			p_end--; +			return FALSE; +		    } +		    p_Char[p_end] = '='; +		    for (s=buf; *s && !isdigit(*s); s++) ; +		    if (!*s) +			malformed (); +		    p_newfirst = (LINENUM) atol(s); +		    while (isdigit(*s)) s++; +		    if (*s == ',') { +			for (; *s && !isdigit(*s); s++) ; +			if (!*s) +			    malformed (); +			p_repl_lines = ((LINENUM)atol(s)) - p_newfirst + 1; +		    } +		    else if (p_newfirst) +			p_repl_lines = 1; +		    else { +			p_repl_lines = 0; +			p_newfirst = 1; +		    } +		    p_max = p_repl_lines + p_end; +		    if (p_max > MAXHUNKSIZE) +			fatal4("hunk too large (%ld lines) at line %ld: %s", +			      p_max, p_input_line, buf); +		    while (p_max >= hunkmax) +			grow_hunkmax(); +		    if (p_repl_lines != ptrn_copiable +		     && (p_context != 0 || p_repl_lines != 1)) +			repl_could_be_missing = FALSE; +		    break; +		} +		goto change_line; +	    case '+':  case '!': +		repl_could_be_missing = FALSE; +	      change_line: +		if (buf[1] == '\n' && canonicalize) +		    strcpy(buf+1," \n"); +		if (!isspace(buf[1]) && buf[1] != '>' && buf[1] != '<' && +		  repl_beginning && repl_could_be_missing) { +		    repl_missing = TRUE; +		    goto hunk_done; +		} +		if (context >= 0) { +		    if (context < p_context) +			p_context = context; +		    context = -1000; +		} +		p_line[p_end] = savestr(buf+2); +		if (out_of_mem) { +		    p_end--; +		    return FALSE; +		} +		break; +	    case '\t': case '\n':	/* assume the 2 spaces got eaten */ +		if (repl_beginning && repl_could_be_missing && +		  (!ptrn_spaces_eaten || diff_type == NEW_CONTEXT_DIFF) ) { +		    repl_missing = TRUE; +		    goto hunk_done; +		} +		p_line[p_end] = savestr(buf); +		if (out_of_mem) { +		    p_end--; +		    return FALSE; +		} +		if (p_end != p_ptrn_lines + 1) { +		    ptrn_spaces_eaten |= (repl_beginning != 0); +		    context++; +		    if (!repl_beginning) +			ptrn_copiable++; +		    p_Char[p_end] = ' '; +		} +		break; +	    case ' ': +		if (!isspace(buf[1]) && +		  repl_beginning && repl_could_be_missing) { +		    repl_missing = TRUE; +		    goto hunk_done; +		} +		context++; +		if (!repl_beginning) +		    ptrn_copiable++; +		p_line[p_end] = savestr(buf+2); +		if (out_of_mem) { +		    p_end--; +		    return FALSE; +		} +		break; +	    default: +		if (repl_beginning && repl_could_be_missing) { +		    repl_missing = TRUE; +		    goto hunk_done; +		} +		malformed (); +	    } +	    /* set up p_len for strncmp() so we don't have to */ +	    /* assume null termination */ +	    if (p_line[p_end]) +		p_len[p_end] = strlen(p_line[p_end]); +	    else +		p_len[p_end] = 0; +	} +	 +    hunk_done: +	if (p_end >=0 && !repl_beginning) +	    fatal2("no --- found in patch at line %ld\n", pch_hunk_beg()); + +	if (repl_missing) { +	     +	    /* reset state back to just after --- */ +	    p_input_line = repl_patch_line; +	    for (p_end--; p_end > repl_beginning; p_end--) +		free(p_line[p_end]); +	    Fseek(pfp, repl_backtrack_position, 0); +	     +	    /* redundant 'new' context lines were omitted - set */ +	    /* up to fill them in from the old file context */ +	    if (!p_context && p_repl_lines == 1) { +		p_repl_lines = 0; +		p_max--; +	    } +	    fillsrc = 1; +	    filldst = repl_beginning+1; +	    fillcnt = p_repl_lines; +	    p_end = p_max; +	} +	else if (!p_context && fillcnt == 1) { +	    /* the first hunk was a null hunk with no context */ +	    /* and we were expecting one line -- fix it up. */ +	    while (filldst < p_end) { +		p_line[filldst] = p_line[filldst+1]; +		p_Char[filldst] = p_Char[filldst+1]; +		p_len[filldst] = p_len[filldst+1]; +		filldst++; +	    } +#if 0 +	    repl_beginning--;		/* this doesn't need to be fixed */ +#endif +	    p_end--; +	    p_first++;			/* do append rather than insert */ +	    fillcnt = 0; +	    p_ptrn_lines = 0; +	} + +	if (diff_type == CONTEXT_DIFF && +	  (fillcnt || (p_first > 1 && ptrn_copiable > 2*p_context)) ) { +	    if (verbose) +		say4("%s\n%s\n%s\n", +"(Fascinating--this is really a new-style context diff but without", +"the telltale extra asterisks on the *** line that usually indicate", +"the new style...)"); +	    diff_type = NEW_CONTEXT_DIFF; +	} +	 +	/* if there were omitted context lines, fill them in now */ +	if (fillcnt) { +	    p_bfake = filldst;		/* remember where not to free() */ +	    p_efake = filldst + fillcnt - 1; +	    while (fillcnt-- > 0) { +		while (fillsrc <= p_end && p_Char[fillsrc] != ' ') +		    fillsrc++; +		if (fillsrc > p_end) +		    fatal2("replacement text or line numbers mangled in hunk at line %ld\n", +			p_hunk_beg); +		p_line[filldst] = p_line[fillsrc]; +		p_Char[filldst] = p_Char[fillsrc]; +		p_len[filldst] = p_len[fillsrc]; +		fillsrc++; filldst++; +	    } +	    while (fillsrc <= p_end && fillsrc != repl_beginning && +	      p_Char[fillsrc] != ' ') +		fillsrc++; +#ifdef DEBUGGING +	    if (debug & 64) +		printf("fillsrc %ld, filldst %ld, rb %ld, e+1 %ld\n", +		    fillsrc,filldst,repl_beginning,p_end+1); +#endif +	    assert(fillsrc==p_end+1 || fillsrc==repl_beginning); +	    assert(filldst==p_end+1 || filldst==repl_beginning); +	} +    } +    else if (diff_type == UNI_DIFF) { +	long line_beginning = ftell(pfp); +					/* file pos of the current line */ +	Reg4 LINENUM fillsrc;		/* index of old lines */ +	Reg5 LINENUM filldst;		/* index of new lines */ +	char ch; + +	ret = pgets(buf, sizeof buf, pfp); +	p_input_line++; +	if (ret == Nullch || strnNE(buf, "@@ -", 4)) { +	    next_intuit_at(line_beginning,p_input_line); +	    return FALSE; +	} +	s = buf+4; +	if (!*s) +	    malformed (); +	p_first = (LINENUM) atol(s); +	while (isdigit(*s)) s++; +	if (*s == ',') { +	    p_ptrn_lines = (LINENUM) atol(++s); +	    while (isdigit(*s)) s++; +	} else +	    p_ptrn_lines = 1; +	if (*s == ' ') s++; +	if (*s != '+' || !*++s) +	    malformed (); +	p_newfirst = (LINENUM) atol(s); +	while (isdigit(*s)) s++; +	if (*s == ',') { +	    p_repl_lines = (LINENUM) atol(++s); +	    while (isdigit(*s)) s++; +	} else +	    p_repl_lines = 1; +	if (*s == ' ') s++; +	if (*s != '@') +	    malformed (); +	if (!p_ptrn_lines) +	    p_first++;			/* do append rather than insert */ +	p_max = p_ptrn_lines + p_repl_lines + 1; +	while (p_max >= hunkmax) +	    grow_hunkmax(); +	fillsrc = 1; +	filldst = fillsrc + p_ptrn_lines; +	p_end = filldst + p_repl_lines; +	Sprintf(buf,"*** %ld,%ld ****\n",p_first,p_first + p_ptrn_lines - 1); +	p_line[0] = savestr(buf); +	if (out_of_mem) { +	    p_end = -1; +	    return FALSE; +	} +	p_Char[0] = '*'; +        Sprintf(buf,"--- %ld,%ld ----\n",p_newfirst,p_newfirst+p_repl_lines-1); +	p_line[filldst] = savestr(buf); +	if (out_of_mem) { +	    p_end = 0; +	    return FALSE; +	} +	p_Char[filldst++] = '='; +	p_context = 100; +	context = 0; +	p_hunk_beg = p_input_line + 1; +	while (fillsrc <= p_ptrn_lines || filldst <= p_end) { +	    line_beginning = ftell(pfp); +	    ret = pgets(buf, sizeof buf, pfp); +	    p_input_line++; +	    if (ret == Nullch) { +		if (p_max - filldst < 3) +		    Strcpy(buf, " \n");  /* assume blank lines got chopped */ +		else { +		    fatal1("unexpected end of file in patch\n"); +		} +	    } +	    if (*buf == '\t' || *buf == '\n') { +		ch = ' ';		/* assume the space got eaten */ +		s = savestr(buf); +	    } +	    else { +		ch = *buf; +		s = savestr(buf+1); +	    } +	    if (out_of_mem) { +		while (--filldst > p_ptrn_lines) +		    free(p_line[filldst]); +		p_end = fillsrc-1; +		return FALSE; +	    } +	    switch (ch) { +	    case '-': +		if (fillsrc > p_ptrn_lines) { +		    free(s); +		    p_end = filldst-1; +		    malformed (); +		} +		p_Char[fillsrc] = ch; +		p_line[fillsrc] = s; +		p_len[fillsrc++] = strlen(s); +		break; +	    case '=': +		ch = ' '; +		/* FALL THROUGH */ +	    case ' ': +		if (fillsrc > p_ptrn_lines) { +		    free(s); +		    while (--filldst > p_ptrn_lines) +			free(p_line[filldst]); +		    p_end = fillsrc-1; +		    malformed (); +		} +		context++; +		p_Char[fillsrc] = ch; +		p_line[fillsrc] = s; +		p_len[fillsrc++] = strlen(s); +		s = savestr(s); +		if (out_of_mem) { +		    while (--filldst > p_ptrn_lines) +			free(p_line[filldst]); +		    p_end = fillsrc-1; +		    return FALSE; +		} +		/* FALL THROUGH */ +	    case '+': +		if (filldst > p_end) { +		    free(s); +		    while (--filldst > p_ptrn_lines) +			free(p_line[filldst]); +		    p_end = fillsrc-1; +		    malformed (); +		} +		p_Char[filldst] = ch; +		p_line[filldst] = s; +		p_len[filldst++] = strlen(s); +		break; +	    default: +		p_end = filldst; +		malformed (); +	    } +	    if (ch != ' ' && context > 0) { +		if (context < p_context) +		    p_context = context; +		context = -1000; +	    } +	}/* while */ +    } +    else {				/* normal diff--fake it up */ +	char hunk_type; +	Reg3 int i; +	LINENUM min, max; +	long line_beginning = ftell(pfp); + +	p_context = 0; +	ret = pgets(buf, sizeof buf, pfp); +	p_input_line++; +	if (ret == Nullch || !isdigit(*buf)) { +	    next_intuit_at(line_beginning,p_input_line); +	    return FALSE; +	} +	p_first = (LINENUM)atol(buf); +	for (s=buf; isdigit(*s); s++) ; +	if (*s == ',') { +	    p_ptrn_lines = (LINENUM)atol(++s) - p_first + 1; +	    while (isdigit(*s)) s++; +	} +	else +	    p_ptrn_lines = (*s != 'a'); +	hunk_type = *s; +	if (hunk_type == 'a') +	    p_first++;			/* do append rather than insert */ +	min = (LINENUM)atol(++s); +	for (; isdigit(*s); s++) ; +	if (*s == ',') +	    max = (LINENUM)atol(++s); +	else +	    max = min; +	if (hunk_type == 'd') +	    min++; +	p_end = p_ptrn_lines + 1 + max - min + 1; +	if (p_end > MAXHUNKSIZE) +	    fatal4("hunk too large (%ld lines) at line %ld: %s", +		  p_end, p_input_line, buf); +	while (p_end >= hunkmax) +	    grow_hunkmax(); +	p_newfirst = min; +	p_repl_lines = max - min + 1; +	Sprintf(buf, "*** %ld,%ld\n", p_first, p_first + p_ptrn_lines - 1); +	p_line[0] = savestr(buf); +	if (out_of_mem) { +	    p_end = -1; +	    return FALSE; +	} +	p_Char[0] = '*'; +	for (i=1; i<=p_ptrn_lines; i++) { +	    ret = pgets(buf, sizeof buf, pfp); +	    p_input_line++; +	    if (ret == Nullch) +		fatal2("unexpected end of file in patch at line %ld\n", +		  p_input_line); +	    if (*buf != '<') +		fatal2("< expected at line %ld of patch\n", p_input_line); +	    p_line[i] = savestr(buf+2); +	    if (out_of_mem) { +		p_end = i-1; +		return FALSE; +	    } +	    p_len[i] = strlen(p_line[i]); +	    p_Char[i] = '-'; +	} +	if (hunk_type == 'c') { +	    ret = pgets(buf, sizeof buf, pfp); +	    p_input_line++; +	    if (ret == Nullch) +		fatal2("unexpected end of file in patch at line %ld\n", +		    p_input_line); +	    if (*buf != '-') +		fatal2("--- expected at line %ld of patch\n", p_input_line); +	} +	Sprintf(buf, "--- %ld,%ld\n", min, max); +	p_line[i] = savestr(buf); +	if (out_of_mem) { +	    p_end = i-1; +	    return FALSE; +	} +	p_Char[i] = '='; +	for (i++; i<=p_end; i++) { +	    ret = pgets(buf, sizeof buf, pfp); +	    p_input_line++; +	    if (ret == Nullch) +		fatal2("unexpected end of file in patch at line %ld\n", +		    p_input_line); +	    if (*buf != '>') +		fatal2("> expected at line %ld of patch\n", p_input_line); +	    p_line[i] = savestr(buf+2); +	    if (out_of_mem) { +		p_end = i-1; +		return FALSE; +	    } +	    p_len[i] = strlen(p_line[i]); +	    p_Char[i] = '+'; +	} +    } +    if (reverse)			/* backwards patch? */ +	if (!pch_swap()) +	    say1("Not enough memory to swap next hunk!\n"); +#ifdef DEBUGGING +    if (debug & 2) { +	int i; +	char special; + +	for (i=0; i <= p_end; i++) { +	    if (i == p_ptrn_lines) +		special = '^'; +	    else +		special = ' '; +	    fprintf(stderr, "%3d %c %c %s", i, p_Char[i], special, p_line[i]); +	    Fflush(stderr); +	} +    } +#endif +    if (p_end+1 < hunkmax)	/* paranoia reigns supreme... */ +	p_Char[p_end+1] = '^';  /* add a stopper for apply_hunk */ +    return TRUE; +} + +/* Input a line from the patch file, worrying about indentation. */ + +char * +pgets(bf,sz,fp) +char *bf; +int sz; +FILE *fp; +{ +    char *ret = fgets(bf, sz, fp); +    Reg1 char *s; +    Reg2 int indent = 0; + +    if (p_indent && ret != Nullch) { +	for (s=buf; +	  indent < p_indent && (*s == ' ' || *s == '\t' || *s == 'X'); s++) { +	    if (*s == '\t') +		indent += 8 - (indent % 7); +	    else +		indent++; +	} +	if (buf != s) +	    Strcpy(buf, s); +    } +    return ret; +} + +/* Reverse the old and new portions of the current hunk. */ + +bool +pch_swap() +{ +    char **tp_line;		/* the text of the hunk */ +    short *tp_len;		/* length of each line */ +    char *tp_char;		/* +, -, and ! */ +    Reg1 LINENUM i; +    Reg2 LINENUM n; +    bool blankline = FALSE; +    Reg3 char *s; + +    i = p_first; +    p_first = p_newfirst; +    p_newfirst = i; +     +    /* make a scratch copy */ + +    tp_line = p_line; +    tp_len = p_len; +    tp_char = p_Char; +    p_line = Null(char**);	/* force set_hunkmax to allocate again */ +    p_len = Null(short*); +    p_Char = Nullch; +    set_hunkmax(); +    if (p_line == Null(char**) || p_len == Null(short*) || p_Char == Nullch) { +#ifndef lint +	if (p_line == Null(char**)) +	    free((char*)p_line); +	p_line = tp_line; +	if (p_len == Null(short*)) +	    free((char*)p_len); +	p_len = tp_len; +#endif +	if (p_Char == Nullch) +	    free((char*)p_Char); +	p_Char = tp_char; +	return FALSE;		/* not enough memory to swap hunk! */ +    } + +    /* now turn the new into the old */ + +    i = p_ptrn_lines + 1; +    if (tp_char[i] == '\n') {		/* account for possible blank line */ +	blankline = TRUE; +	i++; +    } +    if (p_efake >= 0) {			/* fix non-freeable ptr range */ +	if (p_efake <= i) +	    n = p_end - i + 1; +	else +	    n = -i; +	p_efake += n; +	p_bfake += n; +    } +    for (n=0; i <= p_end; i++,n++) { +	p_line[n] = tp_line[i]; +	p_Char[n] = tp_char[i]; +	if (p_Char[n] == '+') +	    p_Char[n] = '-'; +	p_len[n] = tp_len[i]; +    } +    if (blankline) { +	i = p_ptrn_lines + 1; +	p_line[n] = tp_line[i]; +	p_Char[n] = tp_char[i]; +	p_len[n] = tp_len[i]; +	n++; +    } +    assert(p_Char[0] == '='); +    p_Char[0] = '*'; +    for (s=p_line[0]; *s; s++) +	if (*s == '-') +	    *s = '*'; + +    /* now turn the old into the new */ + +    assert(tp_char[0] == '*'); +    tp_char[0] = '='; +    for (s=tp_line[0]; *s; s++) +	if (*s == '*') +	    *s = '-'; +    for (i=0; n <= p_end; i++,n++) { +	p_line[n] = tp_line[i]; +	p_Char[n] = tp_char[i]; +	if (p_Char[n] == '-') +	    p_Char[n] = '+'; +	p_len[n] = tp_len[i]; +    } +    assert(i == p_ptrn_lines + 1); +    i = p_ptrn_lines; +    p_ptrn_lines = p_repl_lines; +    p_repl_lines = i; +#ifndef lint +    if (tp_line == Null(char**)) +	free((char*)tp_line); +    if (tp_len == Null(short*)) +	free((char*)tp_len); +#endif +    if (tp_char == Nullch) +	free((char*)tp_char); +    return TRUE; +} + +/* Return the specified line position in the old file of the old context. */ + +LINENUM +pch_first() +{ +    return p_first; +} + +/* Return the number of lines of old context. */ + +LINENUM +pch_ptrn_lines() +{ +    return p_ptrn_lines; +} + +/* Return the probable line position in the new file of the first line. */ + +LINENUM +pch_newfirst() +{ +    return p_newfirst; +} + +/* Return the number of lines in the replacement text including context. */ + +LINENUM +pch_repl_lines() +{ +    return p_repl_lines; +} + +/* Return the number of lines in the whole hunk. */ + +LINENUM +pch_end() +{ +    return p_end; +} + +/* Return the number of context lines before the first changed line. */ + +LINENUM +pch_context() +{ +    return p_context; +} + +/* Return the length of a particular patch line. */ + +short +pch_line_len(line) +LINENUM line; +{ +    return p_len[line]; +} + +/* Return the control character (+, -, *, !, etc) for a patch line. */ + +char +pch_char(line) +LINENUM line; +{ +    return p_Char[line]; +} + +/* Return a pointer to a particular patch line. */ + +char * +pfetch(line) +LINENUM line; +{ +    return p_line[line]; +} + +/* Return where in the patch file this hunk began, for error messages. */ + +LINENUM +pch_hunk_beg() +{ +    return p_hunk_beg; +} + +/* Apply an ed script by feeding ed itself. */ + +void +do_ed_script() +{ +    Reg1 char *t; +    Reg2 long beginning_of_this_line; +    Reg3 bool this_line_is_command = FALSE; +    Reg4 FILE *pipefp; + +    if (!skip_rest_of_patch) { +	Unlink(TMPOUTNAME); +	copy_file(filearg[0], TMPOUTNAME); +	if (verbose) +	    Sprintf(buf, "/bin/ed %s", TMPOUTNAME); +	else +	    Sprintf(buf, "/bin/ed - %s", TMPOUTNAME); +	pipefp = popen(buf, "w"); +    } +    for (;;) { +	beginning_of_this_line = ftell(pfp); +	if (pgets(buf, sizeof buf, pfp) == Nullch) { +	    next_intuit_at(beginning_of_this_line,p_input_line); +	    break; +	} +	p_input_line++; +	for (t=buf; isdigit(*t) || *t == ','; t++) ; +	this_line_is_command = (isdigit(*buf) && +	  (*t == 'd' || *t == 'c' || *t == 'a') ); +	if (this_line_is_command) { +	    if (!skip_rest_of_patch) +		fputs(buf, pipefp); +	    if (*t != 'd') { +		while (pgets(buf, sizeof buf, pfp) != Nullch) { +		    p_input_line++; +		    if (!skip_rest_of_patch) +			fputs(buf, pipefp); +		    if (strEQ(buf, ".\n")) +			break; +		} +	    } +	} +	else { +	    next_intuit_at(beginning_of_this_line,p_input_line); +	    break; +	} +    } +    if (skip_rest_of_patch) +	return; +    fprintf(pipefp, "w\n"); +    fprintf(pipefp, "q\n"); +    Fflush(pipefp); +    Pclose(pipefp); +    ignore_signals(); +    if (move_file(TMPOUTNAME, outname) < 0) { +	toutkeep = TRUE; +	chmod(TMPOUTNAME, filemode); +    } +    else +	chmod(outname, filemode); +    set_signals(1); +} diff --git a/gnu/usr.bin/patch/pch.h b/gnu/usr.bin/patch/pch.h new file mode 100644 index 000000000000..8d4d169edbad --- /dev/null +++ b/gnu/usr.bin/patch/pch.h @@ -0,0 +1,39 @@ +/* $Header: /home/cvs/386BSD/src/gnu/usr.bin/patch/pch.h,v 1.1.1.1 1993/06/19 14:21:52 paul Exp $ + * + * $Log: pch.h,v $ + * Revision 1.1.1.1  1993/06/19  14:21:52  paul + * b-maked patch-2.10 + * + * Revision 2.0.1.1  87/01/30  22:47:16  lwall + * Added do_ed_script(). + *  + * Revision 2.0  86/09/17  15:39:57  lwall + * Baseline for netwide release. + *  + */ + +EXT FILE *pfp INIT(Nullfp);		/* patch file pointer */ + +void re_patch(); +void open_patch_file(); +void set_hunkmax(); +void grow_hunkmax(); +bool there_is_another_patch(); +int intuit_diff_type(); +void next_intuit_at(); +void skip_to(); +bool another_hunk(); +bool pch_swap(); +char *pfetch(); +short pch_line_len(); +LINENUM pch_first(); +LINENUM pch_ptrn_lines(); +LINENUM pch_newfirst(); +LINENUM pch_repl_lines(); +LINENUM pch_end(); +LINENUM pch_context(); +LINENUM pch_hunk_beg(); +char pch_char(); +char *pfetch(); +char *pgets(); +void do_ed_script(); diff --git a/gnu/usr.bin/patch/util.c b/gnu/usr.bin/patch/util.c new file mode 100644 index 000000000000..ecb85fffd105 --- /dev/null +++ b/gnu/usr.bin/patch/util.c @@ -0,0 +1,433 @@ +#include "EXTERN.h" +#include "common.h" +#include "INTERN.h" +#include "util.h" +#include "backupfile.h" + +void my_exit(); + +#ifndef HAVE_STRERROR +static char * +private_strerror (errnum) +     int errnum; +{ +  extern char *sys_errlist[]; +  extern int sys_nerr; + +  if (errnum > 0 && errnum <= sys_nerr) +    return sys_errlist[errnum]; +  return "Unknown system error"; +} +#define strerror private_strerror +#endif /* !HAVE_STRERROR */ + +/* Rename a file, copying it if necessary. */ + +int +move_file(from,to) +char *from, *to; +{ +    char bakname[512]; +    Reg1 char *s; +    Reg2 int i; +    Reg3 int fromfd; + +    /* to stdout? */ + +    if (strEQ(to, "-")) { +#ifdef DEBUGGING +	if (debug & 4) +	    say2("Moving %s to stdout.\n", from); +#endif +	fromfd = open(from, 0); +	if (fromfd < 0) +	    pfatal2("internal error, can't reopen %s", from); +	while ((i=read(fromfd, buf, sizeof buf)) > 0) +	    if (write(1, buf, i) != 1) +		pfatal1("write failed"); +	Close(fromfd); +	return 0; +    } + +    if (origprae) { +	Strcpy(bakname, origprae); +	Strcat(bakname, to); +    } else { +#ifndef NODIR +	char *backupname = find_backup_file_name(to); +	if (backupname == (char *) 0) +	    fatal1("out of memory\n"); +	Strcpy(bakname, backupname); +	free(backupname); +#else /* NODIR */ +	Strcpy(bakname, to); +    	Strcat(bakname, simple_backup_suffix); +#endif /* NODIR */ +    } + +    if (stat(to, &filestat) == 0) {	/* output file exists */ +	dev_t to_device = filestat.st_dev; +	ino_t to_inode  = filestat.st_ino; +	char *simplename = bakname; +	 +	for (s=bakname; *s; s++) { +	    if (*s == '/') +		simplename = s+1; +	} +	/* Find a backup name that is not the same file. +	   Change the first lowercase char into uppercase; +	   if that isn't sufficient, chop off the first char and try again.  */ +	while (stat(bakname, &filestat) == 0 && +		to_device == filestat.st_dev && to_inode == filestat.st_ino) { +	    /* Skip initial non-lowercase chars.  */ +	    for (s=simplename; *s && !islower(*s); s++) ; +	    if (*s) +		*s = toupper(*s); +	    else +		Strcpy(simplename, simplename+1); +	} +	while (unlink(bakname) >= 0) ;	/* while() is for benefit of Eunice */ +#ifdef DEBUGGING +	if (debug & 4) +	    say3("Moving %s to %s.\n", to, bakname); +#endif +	if (rename(to, bakname) < 0) { +	    say4("Can't backup %s, output is in %s: %s\n", to, from, +		 strerror(errno)); +	    return -1; +	} +	while (unlink(to) >= 0) ; +    } +#ifdef DEBUGGING +    if (debug & 4) +	say3("Moving %s to %s.\n", from, to); +#endif +    if (rename(from, to) < 0) {		/* different file system? */ +	Reg4 int tofd; +	 +	tofd = creat(to, 0666); +	if (tofd < 0) { +	    say4("Can't create %s, output is in %s: %s\n", +	      to, from, strerror(errno)); +	    return -1; +	} +	fromfd = open(from, 0); +	if (fromfd < 0) +	    pfatal2("internal error, can't reopen %s", from); +	while ((i=read(fromfd, buf, sizeof buf)) > 0) +	    if (write(tofd, buf, i) != i) +		pfatal1("write failed"); +	Close(fromfd); +	Close(tofd); +    } +    Unlink(from); +    return 0; +} + +/* Copy a file. */ + +void +copy_file(from,to) +char *from, *to; +{ +    Reg3 int tofd; +    Reg2 int fromfd; +    Reg1 int i; +     +    tofd = creat(to, 0666); +    if (tofd < 0) +	pfatal2("can't create %s", to); +    fromfd = open(from, 0); +    if (fromfd < 0) +	pfatal2("internal error, can't reopen %s", from); +    while ((i=read(fromfd, buf, sizeof buf)) > 0) +	if (write(tofd, buf, i) != i) +	    pfatal2("write to %s failed", to); +    Close(fromfd); +    Close(tofd); +} + +/* Allocate a unique area for a string. */ + +char * +savestr(s) +Reg1 char *s; +{ +    Reg3 char *rv; +    Reg2 char *t; + +    if (!s) +	s = "Oops"; +    t = s; +    while (*t++); +    rv = malloc((MEM) (t - s)); +    if (rv == Nullch) { +	if (using_plan_a) +	    out_of_mem = TRUE; +	else +	    fatal1("out of memory\n"); +    } +    else { +	t = rv; +	while (*t++ = *s++); +    } +    return rv; +} + +#if defined(lint) && defined(CANVARARG) + +/*VARARGS ARGSUSED*/ +say(pat) char *pat; { ; } +/*VARARGS ARGSUSED*/ +fatal(pat) char *pat; { ; } +/*VARARGS ARGSUSED*/ +pfatal(pat) char *pat; { ; } +/*VARARGS ARGSUSED*/ +ask(pat) char *pat; { ; } + +#else + +/* Vanilla terminal output (buffered). */ + +void +say(pat,arg1,arg2,arg3) +char *pat; +long arg1,arg2,arg3; +{ +    fprintf(stderr, pat, arg1, arg2, arg3); +    Fflush(stderr); +} + +/* Terminal output, pun intended. */ + +void				/* very void */ +fatal(pat,arg1,arg2,arg3) +char *pat; +long arg1,arg2,arg3; +{ +    fprintf(stderr, "patch: **** "); +    fprintf(stderr, pat, arg1, arg2, arg3); +    my_exit(1); +} + +/* Say something from patch, something from the system, then silence . . . */ + +void				/* very void */ +pfatal(pat,arg1,arg2,arg3) +char *pat; +long arg1,arg2,arg3; +{ +    int errnum = errno; + +    fprintf(stderr, "patch: **** "); +    fprintf(stderr, pat, arg1, arg2, arg3); +    fprintf(stderr, ": %s\n", strerror(errnum)); +    my_exit(1); +} + +/* Get a response from the user, somehow or other. */ + +void +ask(pat,arg1,arg2,arg3) +char *pat; +long arg1,arg2,arg3; +{ +    int ttyfd; +    int r; +    bool tty2 = isatty(2); + +    Sprintf(buf, pat, arg1, arg2, arg3); +    Fflush(stderr); +    write(2, buf, strlen(buf)); +    if (tty2) {				/* might be redirected to a file */ +	r = read(2, buf, sizeof buf); +    } +    else if (isatty(1)) {		/* this may be new file output */ +	Fflush(stdout); +	write(1, buf, strlen(buf)); +	r = read(1, buf, sizeof buf); +    } +    else if ((ttyfd = open("/dev/tty", 2)) >= 0 && isatty(ttyfd)) { +					/* might be deleted or unwriteable */ +	write(ttyfd, buf, strlen(buf)); +	r = read(ttyfd, buf, sizeof buf); +	Close(ttyfd); +    } +    else if (isatty(0)) {		/* this is probably patch input */ +	Fflush(stdin); +	write(0, buf, strlen(buf)); +	r = read(0, buf, sizeof buf); +    } +    else {				/* no terminal at all--default it */ +	buf[0] = '\n'; +	r = 1; +    } +    if (r <= 0) +	buf[0] = 0; +    else +	buf[r] = '\0'; +    if (!tty2) +	say1(buf); +} +#endif /* lint */ + +/* How to handle certain events when not in a critical region. */ + +void +set_signals(reset) +int reset; +{ +#ifndef lint +    static RETSIGTYPE (*hupval)(),(*intval)(); + +    if (!reset) { +	hupval = signal(SIGHUP, SIG_IGN); +	if (hupval != SIG_IGN) +	    hupval = (RETSIGTYPE(*)())my_exit; +	intval = signal(SIGINT, SIG_IGN); +	if (intval != SIG_IGN) +	    intval = (RETSIGTYPE(*)())my_exit; +    } +    Signal(SIGHUP, hupval); +    Signal(SIGINT, intval); +#endif +} + +/* How to handle certain events when in a critical region. */ + +void +ignore_signals() +{ +#ifndef lint +    Signal(SIGHUP, SIG_IGN); +    Signal(SIGINT, SIG_IGN); +#endif +} + +/* Make sure we'll have the directories to create a file. +   If `striplast' is TRUE, ignore the last element of `filename'.  */ + +void +makedirs(filename,striplast) +Reg1 char *filename; +bool striplast; +{ +    char tmpbuf[256]; +    Reg2 char *s = tmpbuf; +    char *dirv[20];		/* Point to the NULs between elements.  */ +    Reg3 int i; +    Reg4 int dirvp = 0;		/* Number of finished entries in dirv. */ + +    /* Copy `filename' into `tmpbuf' with a NUL instead of a slash +       between the directories.  */ +    while (*filename) { +	if (*filename == '/') { +	    filename++; +	    dirv[dirvp++] = s; +	    *s++ = '\0'; +	} +	else { +	    *s++ = *filename++; +	} +    } +    *s = '\0'; +    dirv[dirvp] = s; +    if (striplast) +	dirvp--; +    if (dirvp < 0) +	return; + +    strcpy(buf, "mkdir"); +    s = buf; +    for (i=0; i<=dirvp; i++) { +	struct stat sbuf; + +	if (stat(tmpbuf, &sbuf) && errno == ENOENT) { +	    while (*s) s++; +	    *s++ = ' '; +	    strcpy(s, tmpbuf); +	} +	*dirv[i] = '/'; +    } +    if (s != buf) +	system(buf); +} + +/* Make filenames more reasonable. */ + +char * +fetchname(at,strip_leading,assume_exists) +char *at; +int strip_leading; +int assume_exists; +{ +    char *fullname; +    char *name; +    Reg1 char *t; +    char tmpbuf[200]; +    int sleading = strip_leading; + +    if (!at) +	return Nullch; +    while (isspace(*at)) +	at++; +#ifdef DEBUGGING +    if (debug & 128) +	say4("fetchname %s %d %d\n",at,strip_leading,assume_exists); +#endif +    if (strnEQ(at, "/dev/null", 9))	/* so files can be created by diffing */ +	return Nullch;			/*   against /dev/null. */ +    name = fullname = t = savestr(at); + +    /* Strip off up to `sleading' leading slashes and null terminate.  */ +    for (; *t && !isspace(*t); t++) +	if (*t == '/') +	    if (--sleading >= 0) +		name = t+1; +    *t = '\0'; + +    /* If no -p option was given (957 is the default value!), +       we were given a relative pathname, +       and the leading directories that we just stripped off all exist, +       put them back on.  */ +    if (strip_leading == 957 && name != fullname && *fullname != '/') { +	name[-1] = '\0'; +	if (stat(fullname, &filestat) == 0 && S_ISDIR (filestat.st_mode)) { +	    name[-1] = '/'; +	    name=fullname; +	} +    } + +    name = savestr(name); +    free(fullname); + +    if (stat(name, &filestat) && !assume_exists) { +	char *filebase = basename(name); +	int pathlen = filebase - name; + +	/* Put any leading path into `tmpbuf'.  */ +	strncpy(tmpbuf, name, pathlen); + +#define try(f, a1, a2) (Sprintf(tmpbuf + pathlen, f, a1, a2), stat(tmpbuf, &filestat) == 0) +	if (   try("RCS/%s%s", filebase, RCSSUFFIX) +	    || try("RCS/%s"  , filebase,         0) +	    || try(    "%s%s", filebase, RCSSUFFIX) +	    || try("SCCS/%s%s", SCCSPREFIX, filebase) +	    || try(     "%s%s", SCCSPREFIX, filebase)) +	  return name; +	free(name); +	name = Nullch; +    } + +    return name; +} + +char * +xmalloc (size) +     unsigned size; +{ +  register char *p = (char *) malloc (size); +  if (!p) +    fatal("out of memory"); +  return p; +} diff --git a/gnu/usr.bin/patch/util.h b/gnu/usr.bin/patch/util.h new file mode 100644 index 000000000000..9c9949318f14 --- /dev/null +++ b/gnu/usr.bin/patch/util.h @@ -0,0 +1,91 @@ +/* $Header: /home/cvs/386BSD/src/gnu/usr.bin/patch/util.h,v 1.1.1.1 1993/06/19 14:21:52 paul Exp $ + * + * $Log: util.h,v $ + * Revision 1.1.1.1  1993/06/19  14:21:52  paul + * b-maked patch-2.10 + * + * Revision 2.0  86/09/17  15:40:06  lwall + * Baseline for netwide release. + *  + */ + +/* and for those machine that can't handle a variable argument list */ + +#ifdef CANVARARG + +#define say1 say +#define say2 say +#define say3 say +#define say4 say +#define ask1 ask +#define ask2 ask +#define ask3 ask +#define ask4 ask +#define fatal1 fatal +#define fatal2 fatal +#define fatal3 fatal +#define fatal4 fatal +#define pfatal1 pfatal +#define pfatal2 pfatal +#define pfatal3 pfatal +#define pfatal4 pfatal + +#else /* hope they allow multi-line macro actual arguments */ + +#ifdef lint + +#define say1(a) say(a, 0, 0, 0) +#define say2(a,b) say(a, (b)==(b), 0, 0) +#define say3(a,b,c) say(a, (b)==(b), (c)==(c), 0) +#define say4(a,b,c,d) say(a, (b)==(b), (c)==(c), (d)==(d)) +#define ask1(a) ask(a, 0, 0, 0) +#define ask2(a,b) ask(a, (b)==(b), 0, 0) +#define ask3(a,b,c) ask(a, (b)==(b), (c)==(c), 0) +#define ask4(a,b,c,d) ask(a, (b)==(b), (c)==(c), (d)==(d)) +#define fatal1(a) fatal(a, 0, 0, 0) +#define fatal2(a,b) fatal(a, (b)==(b), 0, 0) +#define fatal3(a,b,c) fatal(a, (b)==(b), (c)==(c), 0) +#define fatal4(a,b,c,d) fatal(a, (b)==(b), (c)==(c), (d)==(d)) +#define pfatal1(a) pfatal(a, 0, 0, 0) +#define pfatal2(a,b) pfatal(a, (b)==(b), 0, 0) +#define pfatal3(a,b,c) pfatal(a, (b)==(b), (c)==(c), 0) +#define pfatal4(a,b,c,d) pfatal(a, (b)==(b), (c)==(c), (d)==(d)) + +#else /* lint */ +    /* if this doesn't work, try defining CANVARARG above */ +#define say1(a) say(a, Nullch, Nullch, Nullch) +#define say2(a,b) say(a, b, Nullch, Nullch) +#define say3(a,b,c) say(a, b, c, Nullch) +#define say4 say +#define ask1(a) ask(a, Nullch, Nullch, Nullch) +#define ask2(a,b) ask(a, b, Nullch, Nullch) +#define ask3(a,b,c) ask(a, b, c, Nullch) +#define ask4 ask +#define fatal1(a) fatal(a, Nullch, Nullch, Nullch) +#define fatal2(a,b) fatal(a, b, Nullch, Nullch) +#define fatal3(a,b,c) fatal(a, b, c, Nullch) +#define fatal4 fatal +#define pfatal1(a) pfatal(a, Nullch, Nullch, Nullch) +#define pfatal2(a,b) pfatal(a, b, Nullch, Nullch) +#define pfatal3(a,b,c) pfatal(a, b, c, Nullch) +#define pfatal4 pfatal + +#endif /* lint */ + +/* if neither of the above work, join all multi-line macro calls. */ +#endif + +EXT char serrbuf[BUFSIZ];		/* buffer for stderr */ + +char *fetchname(); +int move_file(); +void copy_file(); +void say(); +void fatal(); +void pfatal(); +void ask(); +char *savestr(); +void set_signals(); +void ignore_signals(); +void makedirs(); +char *basename(); diff --git a/gnu/usr.bin/patch/version.c b/gnu/usr.bin/patch/version.c new file mode 100644 index 000000000000..76491e9405d7 --- /dev/null +++ b/gnu/usr.bin/patch/version.c @@ -0,0 +1,28 @@ +/* $Header: /home/cvs/386BSD/src/gnu/usr.bin/patch/version.c,v 1.1.1.1 1993/06/19 14:21:52 paul Exp $ + * + * $Log: version.c,v $ + * Revision 1.1.1.1  1993/06/19  14:21:52  paul + * b-maked patch-2.10 + * + * Revision 2.0  86/09/17  15:40:11  lwall + * Baseline for netwide release. + *  + */ + +#include "EXTERN.h" +#include "common.h" +#include "util.h" +#include "INTERN.h" +#include "patchlevel.h" +#include "version.h" + +void my_exit(); + +/* Print out the version number and die. */ + +void +version() +{ +    fprintf(stderr, "Patch version %s\n", PATCH_VERSION); +    my_exit(0); +} diff --git a/gnu/usr.bin/patch/version.h b/gnu/usr.bin/patch/version.h new file mode 100644 index 000000000000..36a42e37e7bc --- /dev/null +++ b/gnu/usr.bin/patch/version.h @@ -0,0 +1,12 @@ +/* $Header: /home/cvs/386BSD/src/gnu/usr.bin/patch/version.h,v 1.1.1.1 1993/06/19 14:21:52 paul Exp $ + * + * $Log: version.h,v $ + * Revision 1.1.1.1  1993/06/19  14:21:52  paul + * b-maked patch-2.10 + * + * Revision 2.0  86/09/17  15:40:14  lwall + * Baseline for netwide release. + *  + */ + +void version(); | 
