]> Zhao Yanbai Git Server - minix.git/commitdiff
Importing usr.bin/paste 81/1081/1
authorThomas Cort <tcort@minix3.org>
Sun, 27 Oct 2013 23:43:02 +0000 (19:43 -0400)
committerThomas Cort <tcort@minix3.org>
Mon, 28 Oct 2013 12:31:39 +0000 (08:31 -0400)
Replaces commands/paste. No Minix-specific changes needed.

Change-Id: I92a6813502d552ad005ba6e856ce573508fc9f24

commands/Makefile
commands/paste/Makefile [deleted file]
commands/paste/paste.c [deleted file]
man/man1/Makefile
man/man1/paste.1 [deleted file]
releasetools/nbsd_ports
usr.bin/Makefile
usr.bin/paste/Makefile [new file with mode: 0644]
usr.bin/paste/paste.1 [new file with mode: 0644]
usr.bin/paste/paste.c [new file with mode: 0644]

index abd09ec4e848de88305b09435be1d847095ae6dd..b2ac29597e4de72cdc67a350b1ab7d6a38c283b9 100644 (file)
@@ -17,7 +17,7 @@ SUBDIR=       add_route arp ash at backup btrace \
        lpd lspci mail MAKEDEV \
        mined mkfifo \
        mount mt netconf \
-       nonamed od paste patch \
+       nonamed od patch \
        ping postinstall poweroff prep printroot \
        profile progressbar pr_routes ps pwdauth \
        ramdisk rarpd rawspeed rcp readclock \
diff --git a/commands/paste/Makefile b/commands/paste/Makefile
deleted file mode 100644 (file)
index c72cce9..0000000
+++ /dev/null
@@ -1,4 +0,0 @@
-PROG=  paste
-MAN=
-
-.include <bsd.prog.mk>
diff --git a/commands/paste/paste.c b/commands/paste/paste.c
deleted file mode 100644 (file)
index d0855cc..0000000
+++ /dev/null
@@ -1,456 +0,0 @@
-/* paste - laminate files              Author: David Ihnat */
-
-/* Paste - a recreation of the Unix(Tm) paste(1) command.
- *
- * syntax: paste file1 file2 ... paste -dLIST file1 file2 ... paste -s [-dLIST]
- * file1 file2 ...
- *
- * Copyright (C) 1984 by David M. Ihnat
- *
- * This program is a total rewrite of the Bell Laboratories Unix(Tm) command of
- * the same name, as of System V.  It contains no proprietary code, and
- * therefore may be used without violation of any proprietary agreements
- * whatsoever.  However, you will notice that the program is copyrighted by
- * me.  This is to assure the program does *not* fall into the public domain.
- * Thus, I may specify just what I am now: This program may be freely copied
- * and distributed, provided this notice remains; it may not be sold for
- * profit without express written consent of the author. Please note that I
- * recreated the behavior of the Unix(Tm) 'paste' command as faithfully as
- * possible, with minor exceptions (noted below); however, I haven't run a
- * full set of regression * tests.  Thus, the user of this program accepts
- * full responsibility for any effects or loss; in particular, the author is
- * not responsible for any losses, explicit or incidental, that may be
- * incurred through use of this program.
- *
- * The changes to the program, with one exception, are transparent to a user
- * familiar with the Unix command of the same name.  These changes are:
- *
- * 1) The '-s' option had a bug in the Unix version when used with multiple
- *    files.  (It would repeat each file in a list, i.e., for 
- *
- *             paste -s file1 file2 file3
- *
- * it would list
- *
- *      <file1\n><file1\n><file2\n><file1\n><file2\n><file3\n>
- *
- * I fixed this, and reported the bug to the providers of the command in Unix.
- *
- * 2) The list of valid escape sequences has been expanded to include \b,\f, 
- *    and \r.  (Just because *I* can't imagine why you'd want to use them 
- *    doesn't mean I should keep them from you.)
- *
- * 3) There is no longer any restriction on line length.
- *
- * I ask that any bugs (and, if possible, fixes) be reported to me when
- * possible.  -David Ihnat (312) 784-4544 ihuxx!ignatz
- */
-
-/* Modified to run under MINIX 1.1 by David O. Tinker  (416) 978-3636
- * (utgpu!dtinker) Sept. 19, 1987
- */
-
-/* Modified to conform to POSIX 1003.2/Draft10 standard 23rd Sept. 1990
- * Changes:
- *     - the arguments can be in any order
- *     - removed the ToUpper function
- * by Thomas Brupbacher (tobr@mw.lpc.ethz.ch)
- */
-
-#include <errno.h>
-#include <ctype.h>
-#include <stdlib.h>
-#include <string.h>
-#include <stdio.h>
-
-/* I'd love to use enums, but not everyone has them.  Portability, y'know. */
-#define NODELIM                1
-#define USAGE          2
-#define BADFILE                3
-#define TOOMANY                4
-
-#define        TAB             '\t'
-#define        NL              '\n'
-#define        BS              '\b'
-#define        FF              '\f'
-#define        CR              '\r'
-#define DEL             '\177'
-#define SPACE          ' '
-#define BACKSLASH      '\\'
-
-#define        _MAXSZ          512
-#define _MAXFILES      12
-#define        CLOSED          ((FILE *)-1)
-#define ENDLIST                ((FILE *)-2)
-
-char *cmdnam;
-
-short int sflag;
-static char default_delims[] = {TAB};  /* default delimiter string      */
-char *delims;                  /* the pointer to the delimiters        */
-int number_of_delims = 1;      /* number of delimiters to use           */
-
-int main(int argc, char **argv);
-void docol(int nfiles, char **fnamptr);
-void doserial(int nfiles, char **fnamptr);
-void delimbuild(char *strptr);
-void prerr(int etype, char *estring);
-
-int main(argc, argv)
-int argc;
-char **argv;
-{
-  char **arg_ptr;              /* used to save argv, needed for docol() etc */
-  int num_files = 0;           /* Number of filenames specified on cmd line */
-  sflag = 0;
-  delims = default_delims;     /* use default delimiters */
-
-  cmdnam = *argv;
-
-  if (argc >= 2) {
-
-       /* Skip invocation name */
-       argv++;
-       argc--;
-
-       /* Save argv */
-       arg_ptr = argv;
-       /* First, parse input options */
-
-       while (argc-- > 0) {
-               if (argv[0][0] == '-' && argv[0][1] != '\0') {
-                       switch (argv[0][1]) {
-                           case 'd':
-                               /* Delimiter character(s) */
-                               if (*(++argv) == '\0')
-                                       prerr(NODELIM, "");
-                               else
-                                       delimbuild(*(argv));
-                               argc--;
-                               break;
-
-                           case 's':   sflag++;        break;
-
-                           default:    prerr(USAGE, "");
-                       }
-                       argv++;
-               } else {
-                       num_files++;
-                       argv++;
-               }
-       }
-
-       /* If there are more than MAX_FILES files on the command
-        * line, exit with error message.                                                */
-       if (num_files > _MAXFILES) prerr(TOOMANY, "");
-
-       /* If no files specified, simply exit.  Otherwise, if not the
-        * old '-s' option, process all files. If '-s', then process
-        * files one-at-a-time. */
-
-       if (!sflag)
-               docol(num_files, arg_ptr);      /* Column paste */
-       else
-               doserial(num_files, arg_ptr);   /* Serial paste */
-
-       exit(0);
-  } else
-       prerr(USAGE, "");
-  return(0);
-}
-
-void docol(nfiles, fnamptr)
-int nfiles;
-char **fnamptr;
-{
-  char iobuff[_MAXSZ];         /* i/o buffer for the fgets */
-  short int somedone;          /* flag for blank field handling */
-
-  /* There is a strange case where all files are just ready to be
-   * closed, or will on this round.  In that case, the string of
-   * delimiters must be preserved.  delbuf[1] ->delbuf[MAXFILES+1]
-   * provides intermediate storage for closed files, if needed;
-   * delbuf[0] is the current index.
-   */
-  char delbuf[_MAXFILES + 2];
-
-  FILE *fileptr[_MAXFILES + 1];
-
-  int filecnt;                 /* Set to number of files to process */
-  register char *delimptr;     /* Cycling delimiter pointer */
-  int index;                   /* Working variable */
-  int strend;                  /* End of string in buffer */
-
-  /* Perform column paste.  First, attempt to open all files. (This
-   * could be expanded to an infinite number of files, but at the
-   * (considerable) expense of remembering the file and its current
-   * offset, then opening/reading/closing.  The commands' utility
-   * doesn't warrant the effort; at least, to me...) 
-   */
-
-  for (filecnt = 0; (nfiles > 0); fnamptr++) {
-       if ((fnamptr[0][0] == '-') && (fnamptr[0][1] != '\0')) {
-               if (fnamptr[0][1] == 'd') fnamptr++;
-       } else {
-               nfiles--;
-               if (fnamptr[0][0] == '-') {
-                       fileptr[filecnt++] = stdin;
-               } else {
-                       fileptr[filecnt] = fopen(fnamptr[0], "r");
-                       if (fileptr[filecnt++] == NULL)
-                               prerr(BADFILE, *fnamptr);
-               }
-       }
-  }
-
-  fileptr[filecnt] = ENDLIST;  /* End of list. */
-
-  /* Have all files.  Now, read a line from each file, and output to
-   * stdout.  Notice that the old 511 character limitation on the line
-   * length no longer applies, since this program doesn't do the
-   * buffering.  Do this until you go through the loop and don't
-   * successfully read from any of the files.
-   */
-  for (; filecnt;) {
-       somedone = 0;           /* Blank field handling flag */
-       delimptr = delims;      /* Start at beginning of delim list */
-       delbuf[0] = 0;          /* No squirreled delims */
-
-       for (index = 0; (fileptr[index] != ENDLIST) && filecnt; index++) {
-               /* Read a line and immediately output. If it's too
-                * big for the buffer, then dump what was read and go
-                * back for more.
-                * 
-                * Otherwise, if it is from the last file, then leave
-                * the carriage return in place; if not, replace with
-                * a delimiter (if any) 
-                */
-
-               strend = 0;     /* Set so can easily detect EOF */
-
-               if (fileptr[index] != CLOSED)
-                       while (fgets(iobuff, (_MAXSZ - 1), 
-                                       fileptr[index]) != NULL) {
-                               strend = strlen(iobuff);/* Did the buf fill? */
-
-                               if (strend == (_MAXSZ - 1)) {
-                                       /* Gosh, what a long line. */
-                                       fputs(iobuff, stdout);
-                                       strend = 0;
-                                       continue;
-                               }
-
-                               /* Ok got whole line in buffer. */
-                               break;  /* Out of loop for this file */
-                       }
-
-               /* Ended either on an EOF (well, actually NULL
-                * return-- it *could* be some sort of file error,
-                * but but if the file was opened successfully, this
-                * is unlikely. Besides, error checking on streams
-                * doesn't allow us to decide exactly what went
-                * wrong, so I'm going to be very Unix-like and
-                * ignore it!), or a closed file, or a received line.
-                * If an EOF, close the file and mark it in the list.
-                * In any case, output the delimiter of choice. 
-                */
-
-               if (!strend) {
-                       if (fileptr[index] != CLOSED) {
-                               fclose(fileptr[index]);
-                               fileptr[index] = CLOSED;
-                               filecnt--;
-                       }
-
-                       /* Is this the end of the whole thing? */
-                       if ((fileptr[index + 1] == ENDLIST) && !somedone)
-                               continue;       /* EXITS */
-
-                       /* Ok, some files not closed this line. Last file? */
-                       if (fileptr[index + 1] == ENDLIST) {
-                               if (delbuf[0]) {
-                                       fputs(&delbuf[1], stdout);
-                                       delbuf[0] = 0;
-                               }
-                               putc((int) NL, stdout);
-                               continue;       /* Next read of files */
-                       } else {
-                               /* Closed file; setup delim */
-                               if (*delimptr != DEL) {
-                                       delbuf[0]++;
-                                       delbuf[delbuf[0]] = *delimptr++;
-                                       delbuf[delbuf[0] + 1] = '\0';
-                               } else
-                                       delimptr++;
-                       }
-
-                       /* Reset end of delimiter string if necessary */
-                       if (*delimptr == '\0') delimptr = delims;
-               } else {
-                       /* Some data read. */
-                       somedone++;
-
-                       /* Any saved delims? */
-                       if (delbuf[0]) {
-                               fputs(&delbuf[1], stdout);
-                               delbuf[0] = 0;
-                       }
-
-                       /* If last file, last char will be NL. */
-                       if (fileptr[index + 1] != ENDLIST) {
-                               if (*delimptr == DEL) {
-                                       delimptr++;
-                                       iobuff[strend - 1] = '\0';/* No delim*/
-                               } else
-                                       iobuff[strend - 1] = *delimptr++;
-                       }
-                       if (*delimptr == '\0') delimptr = delims;
-
-                       /* Now dump the buffer */
-                       fputs(iobuff, stdout);
-                       fflush(stdout);
-               }
-       }
-  }
-}
-
-void doserial(nfiles, fnamptr)
-int nfiles;
-char **fnamptr;
-{
-  /* Do serial paste.  Simply scarf characters, performing
-   * one-character buffering to facilitate delim processing. 
-   */
-
-  register int charnew, charold;
-  register char *delimptr;
-
-  register FILE *fileptr;
-
-  for (; nfiles != 0; fnamptr++) {
-       if ((fnamptr[0][0] == '-') && (fnamptr[0][1] != '\0')) {
-               if (fnamptr[0][1] == 'd') fnamptr++;
-       } else {
-               if (fnamptr[0][0] == '-') {
-                       fileptr = stdin;
-               } else {
-                       fileptr = fopen(*fnamptr, "r");
-
-                       if (fileptr == NULL) prerr(BADFILE, *fnamptr);
-               }
-
-               /* The file is open; just keep taking characters,
-                * stashing them in charnew; output charold,
-                * converting to the appropriate delimiter character
-                * if needful.  After the EOF, simply output
-                * 'charold' if it's a newline; otherwise, output it
-                * and then a newline. 
-                */
-
-               delimptr = delims;      /* Set up for delimiter string */
-
-               if ((charold = getc(fileptr)) == EOF) {
-                       /* Empty file! */
-                       putc(NL, stdout);
-                       fflush(stdout);
-                       continue;       /* Go on to the next file */
-               }
-
-               /* Ok, 'charold' is set up.  Hit it! */
-
-               while ((charnew = getc(fileptr)) != EOF) {
-                       /* Ok, process the old character */
-                       if (charold == NL) {
-                               if (*delimptr != DEL)
-                                       putc((int) *delimptr++, stdout);
-
-                               /* Reset pointer at end of delimiter string */
-                               if (*delimptr == '\0') delimptr = delims;
-                       } else
-                               putc(charold, stdout);
-
-                       charold = charnew;
-               }
-
-               /* Ok, hit EOF.  Process that last character */
-
-               putc(charold, stdout);
-               if ((char) charold != NL) putc(NL, stdout);
-               fflush(stdout);
-               nfiles--;
-       }
-  }
-}
-
-void delimbuild(strptr)
-char *strptr;
-{
-  /* Process the delimiter string into something that can be used by
-   * the routines.  This involves, primarily, collapsing the backslash
-   * representations of special characters into their actual values,
-   * and terminating the string in a manner that the routines can
-   * recognize.  The set of possible backslash characters has been
-   * expanded beyond that recognized by the vanilla Unix(Tm) version. 
-   */
-
-  register char *strout;
-
-  delims = strptr;             /* delims now points to argv[...]        */
-  strout = strptr;             /* Start at the same place, anyway */
-
-  while (*strptr) {
-       if (*strptr != '\\')    /* Is it an escape character? */
-               *strout++ = *strptr++;  /* No, just transfer it */
-       else {
-               strptr++;       /* Get past escape character */
-
-               switch (*strptr) {
-                   case '0':   *strout++ = DEL;        break;
-
-                   case 't':   *strout++ = TAB;        break;
-
-                   case 'n':   *strout++ = NL; break;
-
-                   case 'b':   *strout++ = BS; break;
-
-                   case 'f':   *strout++ = FF; break;
-
-                   case 'r':   *strout++ = CR; break;
-
-                   case '\\':
-                       *strout++ = BACKSLASH;
-                       break;
-
-                   default:    *strout++ = *strptr;
-               }
-
-               strptr++;
-       }
-
-  }
-  *strout = '\0';              /* Heaven forfend that we forget this! */
-}
-
-void prerr(etype, estring)
-int etype;
-char *estring;
-{
-  switch (etype) {
-      case USAGE:
-       fprintf(stderr, "%s : Usage: %s [-s] [-d <delimiters>] file1 file2 ...\n", cmdnam, cmdnam);
-       break;
-
-      case NODELIM:
-       fprintf(stderr, "%s : no delimiters\n", cmdnam);
-       break;
-
-      case BADFILE:
-       fprintf(stderr, "%s : %s : cannot open\n", cmdnam, estring);
-       break;
-
-      case TOOMANY:
-       fprintf(stderr, "%s : too many files\n", cmdnam);
-       break;
-  }
-  exit(1);
-}
index 1f9bce7b3f54262a56965cd135eb37809af8762d..91707e2c80ebac2f4fe6e4b3250ba800f99d0299 100644 (file)
@@ -12,7 +12,7 @@ MAN=  ash.1 at.1 \
        look.1 lp.1 lspci.1 mail.1  \
        mixer.1 \
        mkproto.1 mount.1 mt.1 od.1 \
-       paste.1 ping.1 playwave.1 prep.1 \
+       ping.1 playwave.1 prep.1 \
        profile.1 ps.1 rcp.1 recwave.1 \
        remsync.1 rget.1 rlogin.1 rsh.1 rz.1 \
        shar.1 sleep.1 spell.1 \
diff --git a/man/man1/paste.1 b/man/man1/paste.1
deleted file mode 100644 (file)
index cdbd3b1..0000000
+++ /dev/null
@@ -1,50 +0,0 @@
-.TH PASTE 1
-.SH NAME
-paste \- paste multiple files together
-.SH SYNOPSIS
-\fBpaste\fR [\fB\-s\fR]\fR [\fB\-d\fI list\fR] \fIfile...\fR
-.br
-.de FL
-.TP
-\\fB\\$1\\fR
-\\$2
-..
-.de EX
-.TP 20
-\\fB\\$1\\fR
-# \\$2
-..
-.SH OPTIONS
-.TP 5
-.B \-d
-# Set delimiter used to separate columns to \fIlist\fR.
-.TP 5
-.B \-s
-# Print files sequentially, file \fIk\fR on line \fIk\fR.
-.SH EXAMPLES
-.TP 20
-.B paste file1 file2
-# Print \fIfile1\fR in col 1, \fIfile2\fR in col 2
-.TP 20
-.B paste \-s f1 f2
-# Print \fIf1\fR on line 1 and \fIf2\fR on line 2
-.TP 20
-.B paste -d : file1 file2
-# Print the lines separated by a colon
-.SH DESCRIPTION
-.PP
-\fIPaste\fR concatenates corresponding lines of the given input files
-and writes them to standard output. The lines of the different files
-are separated by the delimiters given with the option \-s\fR. If
-no list is given, a tab is substituted for every linefeed, except the last one.
-If end-of-file is hit on an input file, subsequent lines are empty.
-Suppose a set of \fIk\fR files each has one word per line.  
-Then the \fIpaste\fR output will have \fIk\fR columns, 
-with the contents of file \fIj\fR in column \fIj\fR.  
-If the \fB\-s\fR flag is given, then the first
-file is on line 1, the second file on line 2, etc.  
-In effect, \fB\-s\fR turns the output sideways.
-.PP
-If a list of delimiters is given, they are used in turn.  The C escape
-sequences \\n, \\t, \\\\, and \\0 are used for linefeed, tab, backslash, and
-the null string, respectively.
index b99bcc9788bf6bb81b3c9fdc3c76e6fa55c6b386..87b77b9519396e6c40c1fcbc05c5ebb390b3cf96 100644 (file)
 2012/10/17 12:00:00,usr.bin/nvi
 2010/05/14 17:28:23,usr.bin/newgrp
 2012/10/17 12:00:00,usr.bin/passwd
+2012/10/17 12:00:00,usr.bin/paste
 2013/10/24 12:00:00,usr.bin/pr
 2013/10/17 12:00:00,usr.bin/printenv
 2012/10/17 12:00:00,usr.bin/printf
index d58ff85160c208f75a391041a3ef2be5455836b7..dca7e585c98c98d698b5de07c8ae3ba5e2f35dd9 100644 (file)
@@ -19,7 +19,7 @@ SUBDIR= asa \
        mkdep mktemp \
        \
        nbperf newgrp nice nl nohup nvi \
-       passwd pr \
+       passwd paste pr \
        printenv printf pwhash \
        renice rev \
        \
diff --git a/usr.bin/paste/Makefile b/usr.bin/paste/Makefile
new file mode 100644 (file)
index 0000000..3a9f487
--- /dev/null
@@ -0,0 +1,6 @@
+#      $NetBSD: Makefile,v 1.5 2009/04/14 22:15:24 lukem Exp $
+#      from: @(#)Makefile      8.1 (Berkeley) 6/6/93
+
+PROG=  paste
+
+.include <bsd.prog.mk>
diff --git a/usr.bin/paste/paste.1 b/usr.bin/paste/paste.1
new file mode 100644 (file)
index 0000000..2cfd0b2
--- /dev/null
@@ -0,0 +1,118 @@
+.\"    $NetBSD: paste.1,v 1.7 2003/08/07 11:15:28 agc Exp $
+.\"
+.\" Copyright (c) 1989, 1990, 1993
+.\"    The Regents of the University of California.  All rights reserved.
+.\"
+.\" This code is derived from software contributed to Berkeley by
+.\" Adam S. Moskowitz and the Institute of Electrical and Electronics
+.\" Engineers, Inc.
+.\"
+.\" Redistribution and use in source and binary forms, with or without
+.\" modification, are permitted provided that the following conditions
+.\" are met:
+.\" 1. Redistributions of source code must retain the above copyright
+.\"    notice, this list of conditions and the following disclaimer.
+.\" 2. Redistributions in binary form must reproduce the above copyright
+.\"    notice, this list of conditions and the following disclaimer in the
+.\"    documentation and/or other materials provided with the distribution.
+.\" 3. Neither the name of the University nor the names of its contributors
+.\"    may be used to endorse or promote products derived from this software
+.\"    without specific prior written permission.
+.\"
+.\" THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+.\" ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+.\" IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+.\" ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+.\" FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+.\" DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+.\" OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+.\" HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+.\" LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+.\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+.\" SUCH DAMAGE.
+.\"
+.\"    from: @(#)paste.1       8.1 (Berkeley) 6/6/93
+.\"    $NetBSD: paste.1,v 1.7 2003/08/07 11:15:28 agc Exp $
+.\"
+.Dd June 6, 1993
+.Dt PASTE 1
+.Os
+.Sh NAME
+.Nm paste
+.Nd merge corresponding or subsequent lines of files
+.Sh SYNOPSIS
+.Nm
+.Op Fl s
+.Op Fl d Ar list
+.Ar file ...
+.Sh DESCRIPTION
+The
+.Nm
+utility concatenates the corresponding lines of the given input files,
+replacing all but the last file's newline characters with a single tab
+character, and writes the resulting lines to standard output.
+If end-of-file is reached on an input file while other input files
+still contain data, the file is treated as if it were an endless source
+of empty lines.
+.Pp
+The options are as follows:
+.Bl -tag -width Fl
+.It Fl d  Ar list
+Use one or more of the provided characters to replace the newline
+characters instead of the default tab.
+The characters in
+.Ar list
+are used circularly, i.e., when
+.Ar list
+is exhausted the first character from
+.Ar list
+is reused.
+This continues until a line from the last input file (in default operation)
+or the last line in each file (using the -s option) is displayed, at which
+time
+.Nm
+begins selecting characters from the beginning of
+.Ar list
+again.
+.Pp
+The following special characters can also be used in list:
+.Pp
+.Bl -tag -width flag -compact
+.It Li \en
+newline character
+.It Li \et
+tab character
+.It Li \e\e
+backslash character
+.It Li \e0
+Empty string (not a null character).
+.El
+.Pp
+Any other character preceded by a backslash is equivalent to the
+character itself.
+.It Fl s
+Concatenate all of the lines of each separate input file in command line
+order.
+The newline character of every line except the last line in each input
+file is replaced with the tab character, unless otherwise specified by
+the -d option.
+.El
+.Pp
+If
+.Ql Fl
+is specified for one or more of the input files, the standard
+input is used; standard input is read one line at a time, circularly,
+for each instance of
+.Ql Fl .
+.Pp
+The
+.Nm
+utility exits 0 on success, and \*[Gt]0 if an error occurs.
+.Sh SEE ALSO
+.Xr cut 1
+.Sh STANDARDS
+The
+.Nm
+utility is expected to be
+.St -p1003.2
+compatible.
diff --git a/usr.bin/paste/paste.c b/usr.bin/paste/paste.c
new file mode 100644 (file)
index 0000000..5698131
--- /dev/null
@@ -0,0 +1,233 @@
+/*     $NetBSD: paste.c,v 1.16 2011/09/06 18:24:43 joerg Exp $ */
+
+/*
+ * Copyright (c) 1989, 1993
+ *     The Regents of the University of California.  All rights reserved.
+ *
+ * This code is derived from software contributed to Berkeley by
+ * Adam S. Moskowitz of Menlo Consulting.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of the University nor the names of its contributors
+ *    may be used to endorse or promote products derived from this software
+ *    without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+#include <sys/cdefs.h>
+#ifndef lint
+__COPYRIGHT("@(#) Copyright (c) 1989, 1993\
+ The Regents of the University of California.  All rights reserved.");
+#endif /* not lint */
+
+#ifndef lint
+/*static char sccsid[] = "from: @(#)paste.c    8.1 (Berkeley) 6/6/93";*/
+__RCSID("$NetBSD: paste.c,v 1.16 2011/09/06 18:24:43 joerg Exp $");
+#endif /* not lint */
+
+#include <sys/types.h>
+#include <err.h>
+#include <errno.h>
+#include <limits.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <unistd.h>
+
+static void    parallel(int, char **);
+static void    sequential(char **);
+static int     tr(char *);
+__dead static void     usage(void);
+
+static char dflt_delim[] = "\t";
+static char *delim = dflt_delim;
+static int delimcnt = 1;
+
+int
+main(int argc, char **argv)
+{
+       int ch, seq;
+
+       seq = 0;
+       while ((ch = getopt(argc, argv, "d:s")) != -1) {
+               switch (ch) {
+               case 'd':
+                       delim = strdup(optarg);
+                       delimcnt = tr(delim);
+                       break;
+               case 's':
+                       seq = 1;
+                       break;
+               case '?':
+               default:
+                       usage();
+               }
+       }
+       argc -= optind;
+       argv += optind;
+
+       if (seq)
+               sequential(argv);
+       else
+               parallel(argc, argv);
+       exit(0);
+}
+
+static void
+parallel(int argc, char **argv)
+{
+       char ch, *dp, *line;
+       FILE **fpp, *fp;
+       size_t line_len;
+       int cnt, output;
+
+       fpp = calloc(argc, sizeof *fpp);
+       if (fpp == NULL)
+               err(1, "calloc");
+
+       for (cnt = 0; cnt < argc; cnt++) {
+               if (strcmp(argv[cnt], "-") == 0)
+                       fpp[cnt] = stdin;
+               else if (!(fpp[cnt] = fopen(argv[cnt], "r")))
+                       err(1, "%s", argv[cnt]);
+       }
+
+       for (;;) {
+               /* Start with the NUL at the end of 'delim' ... */
+               dp = delim + delimcnt;
+               output = 0;
+               for (cnt = 0; cnt < argc; cnt++) {
+                       fp = fpp[cnt];
+                       if (fp == NULL)
+                               continue;
+                       line = fgetln(fp, &line_len);
+                       if (line == NULL) {
+                               /* Assume EOF */
+                               if (fp != stdin)
+                                       fclose(fp);
+                               fpp[cnt] = NULL;
+                               continue;
+                       }
+                       /* Output enough separators to catch up */
+                       do {
+                               ch = *dp++;
+                               if (ch)
+                                       putchar(ch);
+                               if (dp >= delim + delimcnt)
+                                       dp = delim;
+                       } while (++output <= cnt);
+                       /* Remove any trailing newline - check for last line */
+                       if (line[line_len - 1] == '\n')
+                               line_len--;
+                       printf("%.*s", (int)line_len, line);
+               }
+
+               if (!output)
+                       break;
+
+               /* Add separators to end of line */
+               while (++output <= cnt) {
+                       ch = *dp++;
+                       if (ch)
+                               putchar(ch);
+                       if (dp >= delim + delimcnt)
+                               dp = delim;
+               }
+               putchar('\n');
+       }
+
+       free(fpp);
+}
+
+static void
+sequential(char **argv)
+{
+       FILE *fp;
+       int cnt;
+       char ch, *p, *dp;
+       char buf[_POSIX2_LINE_MAX + 1];
+
+       for (; (p = *argv) != NULL; ++argv) {
+               if (p[0] == '-' && !p[1])
+                       fp = stdin;
+               else if (!(fp = fopen(p, "r"))) {
+                       warn("%s", p);
+                       continue;
+               }
+               if (fgets(buf, sizeof(buf), fp)) {
+                       for (cnt = 0, dp = delim;;) {
+                               if (!(p = strchr(buf, '\n')))
+                                       err(1, "%s: input line too long.",
+                                           *argv);
+                               *p = '\0';
+                               (void)printf("%s", buf);
+                               if (!fgets(buf, sizeof(buf), fp))
+                                       break;
+                               if ((ch = *dp++) != 0)
+                                       putchar(ch);
+                               if (++cnt == delimcnt) {
+                                       dp = delim;
+                                       cnt = 0;
+                               }
+                       }
+                       putchar('\n');
+               }
+               if (fp != stdin)
+                       (void)fclose(fp);
+       }
+}
+
+static int
+tr(char *arg)
+{
+       int cnt;
+       char ch, *p;
+
+       for (p = arg, cnt = 0; (ch = *p++); ++arg, ++cnt)
+               if (ch == '\\')
+                       switch(ch = *p++) {
+                       case 'n':
+                               *arg = '\n';
+                               break;
+                       case 't':
+                               *arg = '\t';
+                               break;
+                       case '0':
+                               *arg = '\0';
+                               break;
+                       default:
+                               *arg = ch;
+                               break;
+               } else
+                       *arg = ch;
+
+       if (!cnt)
+               errx(1, "no delimiters specified.");
+       *arg = '\0';
+       return(cnt);
+}
+
+static void
+usage(void)
+{
+       (void)fprintf(stderr, "paste: [-s] [-d delimiters] file ...\n");
+       exit(1);
+}