root/usr/src/cmd/sh/bltin.c
/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License (the "License").
 * You may not use this file except in compliance with the License.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */

/*
 * Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */

/*      Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
/*        All Rights Reserved   */

/*
 *
 * UNIX shell
 *
 */


#include        "defs.h"
#include        <errno.h>
#include        "sym.h"
#include        "hash.h"
#include        <sys/types.h>
#include        <sys/times.h>

void
builtin(int type, int argc, unsigned char **argv, struct trenod *t)
{
        short index = initio(t->treio, (type != SYSEXEC));
        unsigned char *a1 = argv[1];

        switch (type)
        {

        case SYSSUSP:
                syssusp(argc,argv);
                break;

        case SYSSTOP:
                sysstop(argc,argv);
                break;

        case SYSKILL:
                syskill(argc,argv);
                break;

        case SYSFGBG:
                sysfgbg(argc,argv);
                break;

        case SYSJOBS:
                sysjobs(argc,argv);
                break;

        case SYSDOT:
                if (a1)
                {
                        int     f;

                        if ((f = pathopen(getpath(a1), a1)) < 0)
                                failed(a1, notfound);
                        else
                                execexp(0, f);
                }
                break;

        case SYSTIMES:
                {
                        struct tms tms;

                        times(&tms);
                        prt(tms.tms_cutime);
                        prc_buff(SPACE);
                        prt(tms.tms_cstime);
                        prc_buff(NL);
                }
                break;

        case SYSEXIT:
                if ( tried_to_exit++ || endjobs(JOB_STOPPED) ){
                        flags |= forcexit;      /* force exit */
                        exitsh(a1 ? stoi(a1) : retval);
                }
                break;

        case SYSNULL:
                t->treio = 0;
                break;

        case SYSCONT:
                if (loopcnt)
                {
                        execbrk = breakcnt = 1;
                        if (a1)
                                breakcnt = stoi(a1);
                        if (breakcnt > loopcnt)
                                breakcnt = loopcnt;
                        else
                                breakcnt = -breakcnt;
                }
                break;

        case SYSBREAK:
                if (loopcnt)
                {
                        execbrk = breakcnt = 1;
                        if (a1)
                                breakcnt = stoi(a1);
                        if (breakcnt > loopcnt)
                                breakcnt = loopcnt;
                }
                break;

        case SYSTRAP:
                systrap(argc,argv);
                break;

        case SYSEXEC:
                argv++;
                ioset = 0;
                if (a1 == 0) {
                        setmode(0);
                        break;
                }
                /* FALLTHROUGH */

#ifdef RES      /* Research includes login as part of the shell */

        case SYSLOGIN:
                if (!endjobs(JOB_STOPPED|JOB_RUNNING))
                        break;
                oldsigs();
                execa(argv, -1);
                done(0);
#else

        case SYSNEWGRP:
                if (flags & rshflg)
                        failed(argv[0], restricted);
                else if (!endjobs(JOB_STOPPED|JOB_RUNNING))
                        break;
                else
                {
                        flags |= forcexit; /* bad exec will terminate shell */
                        oldsigs();
                        rmtemp(0);
                        rmfunctmp();
#ifdef ACCT
                        doacct();
#endif
                        execa(argv, -1);
                        done(0);
                }

#endif

        case SYSCD:
                if (flags & rshflg)
                        failed(argv[0], restricted);
                else if ((a1 && *a1) || (a1 == 0 && (a1 = homenod.namval)))
                {
                        unsigned char *cdpath;
                        unsigned char *dir;
                        int f;

                        if ((cdpath = cdpnod.namval) == 0 ||
                             *a1 == '/' ||
                             cf(a1, ".") == 0 ||
                             cf(a1, "..") == 0 ||
                             (*a1 == '.' && (*(a1+1) == '/' || *(a1+1) == '.' && *(a1+2) == '/')))
                                cdpath = (unsigned char *)nullstr;

                        do
                        {
                                dir = cdpath;
                                cdpath = catpath(cdpath,a1);
                        }
                        while ((f = (chdir((const char *) curstak()) < 0)) &&
                            cdpath);

                        if (f) {
                                switch(errno) {
                                                case EMULTIHOP:
                                                        failed(a1, emultihop);
                                                        break;
                                                case ENOTDIR:
                                                        failed(a1, enotdir);
                                                        break;
                                                case ENOENT:
                                                        failed(a1, enoent);
                                                        break;
                                                case EACCES:
                                                        failed(a1, eacces);
                                                        break;
                                                case ENOLINK:
                                                        failed(a1, enolink);
                                                        break;
                                                default:
                                                failed(a1, baddir);
                                                break;
                                                }
                        }
                        else
                        {
                                cwd(curstak());
                                if (cf(nullstr, dir) &&
                                    *dir != ':' &&
                                        any('/', curstak()) &&
                                        flags & prompt)
                                {
                                        prs_buff(cwdget());
                                        prc_buff(NL);
                                }
                        }
                        zapcd();
                }
                else
                {
                        if (a1)
                                error(nulldir);
                        else
                                error(nohome);
                }

                break;

        case SYSSHFT:
                {
                        int places;

                        places = a1 ? stoi(a1) : 1;

                        if ((dolc -= places) < 0)
                        {
                                dolc = 0;
                                error(badshift);
                        }
                        else
                                dolv += places;
                }

                break;

        case SYSWAIT:
                syswait(argc,argv);
                break;

        case SYSREAD:
                if(argc < 2)
                        failed(argv[0],mssgargn);
                rwait = 1;
                exitval = readvar(&argv[1]);
                rwait = 0;
                break;

        case SYSSET:
                if (a1)
                {
                        int     cnt;

                        cnt = options(argc, argv);
                        if (cnt > 1)
                                setargs(argv + argc - cnt);
                }
                else if (comptr(t)->comset == 0)
                {
                        /*
                         * scan name chain and print
                         */
                        namscan(printnam);
                }
                break;

        case SYSRDONLY:
                exitval = 0;
                if (a1)
                {
                        while (*++argv)
                                attrib(lookup(*argv), N_RDONLY);
                }
                else
                        namscan(printro);

                break;

        case SYSXPORT:
                {
                        struct namnod   *n;

                        exitval = 0;
                        if (a1)
                        {
                                while (*++argv)
                                {
                                        n = lookup(*argv);
                                        if (n->namflg & N_FUNCTN)
                                                error(badexport);
                                        else
                                                attrib(n, N_EXPORT);
                                }
                        }
                        else
                                namscan(printexp);
                }
                break;

        case SYSEVAL:
                if (a1)
                        execexp(a1, &argv[2]);
                break;

#ifndef RES
        case SYSULIMIT:
                sysulimit(argc, argv);
                break;

        case SYSUMASK:
                if (a1)
                {
                        int c;
                        mode_t i;

                        i = 0;
                        while ((c = *a1++) >= '0' && c <= '7')
                                i = (i << 3) + c - '0';
                        umask(i);
                }
                else
                {
                        mode_t i;
                        int j;

                        umask(i = umask(0));
                        prc_buff('0');
                        for (j = 6; j >= 0; j -= 3)
                                prc_buff(((i >> j) & 07) +'0');
                        prc_buff(NL);
                }
                break;

#endif

        case SYSTST:
                exitval = test(argc, argv);
                break;

        case SYSECHO:
                exitval = echo(argc, argv);
                break;

        case SYSHASH:
                exitval = 0;

                if (a1)
                {
                        if (a1[0] == '-')
                        {
                                if (a1[1] == 'r')
                                        zaphash();
                                else
                                        error(badopt);
                        }
                        else
                        {
                                while (*++argv)
                                {
                                        if (hashtype(hash_cmd(*argv)) == NOTFOUND)
                                                failed(*argv, notfound);
                                }
                        }
                }
                else
                        hashpr();

                break;

        case SYSPWD:
                {
                        exitval = 0;
                        cwdprint();
                }
                break;

        case SYSRETURN:
                if (funcnt == 0)
                        error(badreturn);

                execbrk = 1;
                exitval = (a1 ? stoi(a1) : retval);
                break;

        case SYSTYPE:
                exitval = 0;
                if (a1)
                {
                        /* return success only if all names are found */
                        while (*++argv)
                                exitval |= what_is_path(*argv);
                }
                break;

        case SYSUNS:
                exitval = 0;
                if (a1)
                {
                        while (*++argv)
                                unset_name(*argv);
                }
                break;

        case SYSGETOPT: {
                int getoptval;
                struct namnod *n;
                extern unsigned char numbuf[];
                unsigned char *varnam = argv[2];
                unsigned char c[2];
                if(argc < 3) {
                        failure(argv[0],mssgargn);
                        break;
                }
                exitval = 0;
                n = lookup("OPTIND");
                optind = stoi(n->namval);
                if(argc > 3) {
                        argv[2] = dolv[0];
                        getoptval = getopt(argc-2, (char **)&argv[2], (char *)argv[1]);
                }
                else
                        getoptval = getopt(dolc+1, (char **)dolv, (char *)argv[1]);
                if(getoptval == -1) {
                        itos(optind);
                        assign(n, numbuf);
                        n = lookup(varnam);
                        assign(n, (unsigned char *)nullstr);
                        exitval = 1;
                        break;
                }
                argv[2] = varnam;
                itos(optind);
                assign(n, numbuf);
                c[0] = getoptval;
                c[1] = 0;
                n = lookup(varnam);
                assign(n, c);
                n = lookup("OPTARG");
                assign(n, (unsigned char *)optarg);
                }
                break;

        default:
                prs_buff(_gettext("unknown builtin\n"));
        }


        flushb();
        restore(index);
        chktrap();
}