uudecode.c

来自「开放源码的编译器open watcom 1.6.0版的源代码」· C语言 代码 · 共 347 行

C
347
字号
/****************************************************************************
*
*                            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 uudecode utility
*               Decodes uuencoded files - converts from 7-bit to 8-bit chars
*
****************************************************************************/


/*
 * Copyright (c) 1983 Regents of the University of California.
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms are permitted
 * provided that the above copyright notice and this paragraph are
 * duplicated in all such forms and that any documentation,
 * advertising materials, and other materials related to such
 * distribution and use acknowledge that the software was developed
 * by the University of California, Berkeley.  The name of the
 * University may not be used to endorse or promote products derived
 * from this software without specific prior written permission.
 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
 * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
 * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
 */

/*
 * Modified 12 April 1990 by Mark Adler for use on MSDOS systems with
 * Microsoft C and Turbo C.
 *
 * Modifed 13 February 1991 by Greg Roelofs for use on VMS systems.  As
 * with the MS-DOS version, the setting of the file mode has been disabled.
 * Compile and link normally (but note that the shared-image link option
 * produces a binary only 6 blocks long, as opposed to the 137-block one
 * produced by an ordinary link).  To set up the VMS symbol to run the
 * program ("run uudecode filename" won't work), do:
 *              uudecode :== "$disk:[directory]uudecode.exe"
 * and don't forget the leading "$" or it still won't work.  The binaries
 * produced by this program are in VMS "stream-LF" format; this makes no
 * difference to VMS when running decoded executables, nor to VMS unzip,
 * but other programs such as zoo or arc may or may not require the file
 * to be "BILFed" (or "unBILFed" or whatever).  Also, unlike the other
 * flavors, VMS files don't get overwritten (a higher version is created).
 *
 * Modified 13 April 1991 by Gary Mussar to be forgiving of systems that
 * appear to be stripping trailing blanks.
 *
 * Modifed 31 August 1992 by David S. McKee for use with the WATCOM compiler.
 *
 * Modifed 7 July 1993 by David S. McKee to handle more variants of the
 *         input file.
 */

#ifdef __WATCOMC__
#define __MSDOS__
#endif
#ifdef __MSDOS__        /* For Turbo C */
#ifndef MSDOS
#define MSDOS 1
#endif
#endif

/*
 * uudecode [input]
 *
 * create the specified file, decoding as you go.
 * used with uuencode.
 */
#include <stdio.h>

#ifdef VMS
#  include <types.h>
#  include <stat.h>
#else
#  ifndef MSDOS            /* i.e., UNIX */
#    include <pwd.h>
#  endif
#  ifdef __WATCOMC__
#  include <stdlib.h>
#  include <string.h>
#  endif
#  include <sys/types.h>   /* MSDOS or UNIX */
#  include <sys/stat.h>
#endif

/* single-character decode */
#define DEC(c)  (((c) - ' ') & 077)


int main(int argc, char **argv)
{
        FILE *in, *out;
        int mode;
        int begin_flag;
        char dest[128];
        char buf[80];

#ifdef __WATCOMC__
        if( argc == 1 || !strcmp(argv[1], "?") || !strcmp(argv[1], "-?")
         || !strcmp(argv[1], "-h") ) {
            printf( "Usage: uudecode [infile]\n" );
            exit(2);
        }
#endif
        /* optional input arg */
        if (argc > 1) {
                if ((in = fopen(argv[1], "r")) == NULL) {
                        perror(argv[1]);
                        exit(1);
                }
                argv++; argc--;
        } else
                in = stdin;

        if (argc != 1) {
                printf("Usage: uudecode [infile]\n");
                exit(2);
        }

      for(;;) {
        begin_flag = 1;
        /* search for header line */
        for (;;) {
                if (fgets(buf, sizeof buf, in) == NULL) {
                    if( begin_flag == 0 ) {
                        fprintf(stderr, "No begin line\n");
                        exit(3);
                    }
                    begin_flag = 0;
                    break;
                }
                if( strnicmp( buf, "begin-", 6 ) != 0
                 && strnicmp( buf, "begin -", 7 ) != 0
                 && strnicmp( buf, "start-", 6 ) != 0
                 && strnicmp( buf, "start -", 7 ) != 0 ) {
                    if (strncmp(buf, "begin ", 6) == 0)
                        break;
                    if (strnicmp(buf, "begin ", 6) == 0) {
                        fprintf( stderr, "begin line is mixed case, assuming lower case\n" );
                        break;
                    }
                }

        }
        if( begin_flag == 0 ) break;
        (void)sscanf(buf+6, "%o %s", &mode, dest);

#if !defined(MSDOS) && !defined(VMS)    /* i.e., UNIX */
        /* handle ~user/file format */
        if (dest[0] == '~') {
                char *sl;
                struct passwd *getpwnam();
                struct passwd *user;
                char dnbuf[100], *index(), *strcat(), *strcpy();

                sl = index(dest, '/');
                if (sl == NULL) {
                        fprintf(stderr, "Illegal ~user\n");
                        exit(3);
                }
                *sl++ = 0;
                user = getpwnam(dest+1);
                if (user == NULL) {
                        fprintf(stderr, "No such user as %s\n", dest);
                        exit(4);
                }
                strcpy(dnbuf, user->pw_dir);
                strcat(dnbuf, "/");
                strcat(dnbuf, sl);
                strcpy(dest, dnbuf);
        }
#endif  /* !defined(MSDOS) && !defined(VMS) */

        /* create output file */
#ifdef MSDOS
        out = fopen(dest, "wb");        /* Binary file */
#else
        out = fopen(dest, "w");
#endif
        if (out == NULL) {
                perror(dest);
                exit(4);
        }
#if !defined(MSDOS) && !defined(VMS)    /* i.e., UNIX */
        chmod(dest, mode);
#endif

        decode(in, out);

        if (fgets(buf, sizeof buf, in) == NULL || strcmp(buf, "end\n")) {
                fprintf(stderr, "No end line\n");
                exit(5);
        }
    }
    return( 0 );
}

/*
 * copy from in to out, decoding as you go along.
 */
void decode(in, out)
FILE *in;
FILE *out;
{
        char buf[80];
        char *bp;
        int n, i, expected;
        int found_begin;
        int after_blank;
        int end_cut;

        end_cut = 0;
        for (;;) {
                /* for each input line */
                if (fgets(buf, sizeof buf, in) == NULL) {
                        printf("Short file\n");
                        exit(10);
                }
                bp = buf;
                while( *bp == ' ' ) {
                    bp++;
                }
                if( !end_cut && (*bp == '\n'
                             || (buf[0] == '-' && buf[1] == '-')) ) {
                    after_blank = 0;
                    for(;;) {
                        if( after_blank && buf[0] == 'M' ) break;
                        if( buf[0] == ' ' || buf[0] == '\n' ) {
                            after_blank = 1;
                        } else {
                            after_blank = 0;
                        }
                        if (fgets(buf, sizeof buf, in) == NULL) {
                            printf("Short file\n");
                            exit(10);
                        }
                    }
                }

                n = DEC(buf[0]);
                if ((n <= 0) || (buf[0] == '\n'))
                        break;

                /* Calculate expected # of chars and pad if necessary */
                expected = ((n+2)/3)<<2;

                found_begin = 0;
                if( expected != strlen(buf)-1
                 && (strnicmp( buf, "end-", 4 ) == 0
                     || strnicmp( buf, "end -", 5 ) == 0
                     || strnicmp( buf, "--- end", 7 ) == 0) ) {
                    end_cut = 1;
                    for(;;) {
                        if( strnicmp( buf, "--- begin", 9 ) == 0
                         || strnicmp( buf, "begin-", 6 ) == 0
                         || strnicmp( buf, "begin -", 7 ) == 0
                         || strnicmp( buf, "start-", 6 ) == 0
                         || strnicmp( buf, "start -", 7 ) == 0 ) {
                            found_begin = 1;
                            break;
                        }
                        if (fgets(buf, sizeof buf, in) == NULL) {
                            printf("Short file\n");
                            exit(10);
                        }
                    }
                }

                if( !found_begin ) {
                    for (i = strlen(buf)-1; i <= expected; i++) buf[i] = ' ';

                    bp = &buf[1];
                    while (n > 0) {
                            outdec(bp, out, n);
                            bp += 4;
                            n -= 3;
                    }
                }
        }
}

/*
 * output a group of 3 bytes (4 input characters).
 * the input chars are pointed to by p, they are to
 * be output to file f.  n is used to tell us not to
 * output all of them at the end of the file.
 */
void outdec(p, f, n)
char *p;
FILE *f;
{
        int c1, c2, c3;

        c1 = DEC(*p) << 2 | DEC(p[1]) >> 4;
        c2 = DEC(p[1]) << 4 | DEC(p[2]) >> 2;
        c3 = DEC(p[2]) << 6 | DEC(p[3]);
        if (n >= 1)
                putc(c1, f);
        if (n >= 2)
                putc(c2, f);
        if (n >= 3)
                putc(c3, f);
}

/*
 * Return the ptr in sp at which the character c appears;
 * NULL if not found
 */

#ifndef __WATCOMC__
#define NULL    0
#endif

char *
index(sp, c)
register char *sp, c;
{
        do {
                if (*sp == c)
                        return(sp);
        } while (*sp++);
        return(NULL);
}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?