tee.c
来自「开放源码的编译器open watcom 1.6.0版的源代码」· C语言 代码 · 共 129 行
C
129 行
/****************************************************************************
*
* Open Watcom Project
*
* Portions Copyright (c) 1983-2002 Sybase, Inc. All Rights Reserved.
*
* ========================================================================
*
* This file contains Original Code and/or Modifications of Original
* Code as defined in and that are subject to the Sybase Open Watcom
* Public License version 1.0 (the 'License'). You may not use this file
* except in compliance with the License. BY USING THIS FILE YOU AGREE TO
* ALL TERMS AND CONDITIONS OF THE LICENSE. A copy of the License is
* provided with the Original Code and Modifications, and is also
* available at www.sybase.com/developer/opensource.
*
* The Original Code and all software distributed under the License are
* distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
* EXPRESS OR IMPLIED, AND SYBASE AND ALL CONTRIBUTORS HEREBY DISCLAIM
* ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR
* NON-INFRINGEMENT. Please see the License for the specific language
* governing rights and limitations under the License.
*
* ========================================================================
*
* Description: POSIX tee utility
* Sends input to two outputs
*
****************************************************************************/
#include <errno.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <malloc.h>
#include "getopt.h"
#include "misc.h"
char *OptEnvVar = "tee";
static const char *usageTxt[] = {
"Usage: tee [-?a] [files]",
"\tfiles : files to tee",
"\t\t Input is copied to each output `file' as well as to",
"\t\t the standard output.",
"\tOptions: -? : print this list",
"\t\t -a : append, rather than overwrite, outputs",
NULL
};
#define BUFFER_SIZE 16384
char buffer[ BUFFER_SIZE ];
void main( int argc, char *argv[] ) {
int ch;
int *out_fh;
int i;
int bytes_read;
int append_flag;
append_flag = 0;
for(;;) {
ch = GetOpt( &argc, argv, "a", usageTxt );
if( ch == -1 ) break;
if( ch == 'a' ) {
append_flag = 1;
}
}
if( argc < 2 ) {
Quit( usageTxt, "must specify at least 1 output file\n" );
}
--argc; /* ignore argv[0] */
++argv;
setmode( STDIN_FILENO, O_BINARY );
setmode( STDOUT_FILENO, O_BINARY );
out_fh = alloca( sizeof( int ) * argc );
if( out_fh == NULL ) {
Die( "not enough memory for file handles\n" );
}
/* open output files */
for( i = 0; i < argc; ++i ) {
out_fh[i] = open( argv[i], append_flag ?
( O_WRONLY | O_APPEND | O_CREAT | O_BINARY ) :
( O_WRONLY | O_TRUNC | O_CREAT | O_BINARY ),
S_IREAD | S_IWRITE );
if( out_fh[i] == -1 ) {
Die( "unable to open %s for writing: %s\n", argv[i], strerror( errno ) );
}
}
/* do the tee-ing */
for(;;) {
bytes_read = read( STDIN_FILENO, buffer, BUFFER_SIZE );
if( bytes_read < 0 ) {
#ifdef __NT__
{
extern int __stdcall GetLastError( void );
if( GetLastError() == 109 ) { // BROKEN_PIPE
break;
}
}
#endif
Die( "error reading (stdin): %s\n", strerror( errno ) );
}
if( bytes_read == 0 ) break;
if( write( STDOUT_FILENO, buffer, bytes_read ) != bytes_read ) {
Die( "error writing (stdout): %s\n", strerror( errno ) );
}
for( i = 0; i < argc; ++i ) {
if( write( out_fh[i], buffer, bytes_read ) != bytes_read ) {
Die( "error writing %s: %s\n", strerror( errno ) );
}
}
}
for( i = 0; i < argc; ++i ) {
close( out_fh[ i ] );
}
exit( 0 );
}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?