xref: /titanic_50/usr/src/lib/libshell/common/bltins/cflow.c (revision 3e14f97f673e8a630f076077de35afdd43dc1587)
1da2e3ebdSchin /***********************************************************************
2da2e3ebdSchin *                                                                      *
3da2e3ebdSchin *               This software is part of the ast package               *
4*3e14f97fSRoger A. Faulkner *          Copyright (c) 1982-2010 AT&T Intellectual Property          *
5da2e3ebdSchin *                      and is licensed under the                       *
6da2e3ebdSchin *                  Common Public License, Version 1.0                  *
77c2fbfb3SApril Chin *                    by AT&T Intellectual Property                     *
8da2e3ebdSchin *                                                                      *
9da2e3ebdSchin *                A copy of the License is available at                 *
10da2e3ebdSchin *            http://www.opensource.org/licenses/cpl1.0.txt             *
11da2e3ebdSchin *         (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9)         *
12da2e3ebdSchin *                                                                      *
13da2e3ebdSchin *              Information and Software Systems Research               *
14da2e3ebdSchin *                            AT&T Research                             *
15da2e3ebdSchin *                           Florham Park NJ                            *
16da2e3ebdSchin *                                                                      *
17da2e3ebdSchin *                  David Korn <dgk@research.att.com>                   *
18da2e3ebdSchin *                                                                      *
19da2e3ebdSchin ***********************************************************************/
20da2e3ebdSchin #pragma prototyped
21da2e3ebdSchin /*
22da2e3ebdSchin  * break [n]
23da2e3ebdSchin  * continue [n]
24da2e3ebdSchin  * return [n]
25da2e3ebdSchin  * exit [n]
26da2e3ebdSchin  *
27da2e3ebdSchin  *   David Korn
28da2e3ebdSchin  *   AT&T Labs
29da2e3ebdSchin  *   dgk@research.att.com
30da2e3ebdSchin  *
31da2e3ebdSchin  */
32da2e3ebdSchin 
33da2e3ebdSchin #include	"defs.h"
34da2e3ebdSchin #include	<ast.h>
35da2e3ebdSchin #include	<error.h>
36da2e3ebdSchin #include	"shnodes.h"
37da2e3ebdSchin #include	"builtins.h"
38da2e3ebdSchin 
39da2e3ebdSchin /*
40da2e3ebdSchin  * return and exit
41da2e3ebdSchin  */
42da2e3ebdSchin #if 0
43da2e3ebdSchin     /* for the dictionary generator */
44da2e3ebdSchin     int	b_exit(int n, register char *argv[],void *extra){}
45da2e3ebdSchin #endif
b_return(register int n,register char * argv[],void * extra)46da2e3ebdSchin int	b_return(register int n, register char *argv[],void *extra)
47da2e3ebdSchin {
48da2e3ebdSchin 	register char *arg;
497c2fbfb3SApril Chin 	register Shell_t *shp = ((Shbltin_t*)extra)->shp;
50da2e3ebdSchin 	struct checkpt *pp = (struct checkpt*)shp->jmplist;
51da2e3ebdSchin 	const char *options = (**argv=='r'?sh_optreturn:sh_optexit);
52da2e3ebdSchin 	while((n = optget(argv,options))) switch(n)
53da2e3ebdSchin 	{
54da2e3ebdSchin 	    case ':':
55da2e3ebdSchin 		if(!strmatch(argv[opt_info.index],"[+-]+([0-9])"))
56da2e3ebdSchin 			errormsg(SH_DICT,2, "%s", opt_info.arg);
57da2e3ebdSchin 		goto done;
58da2e3ebdSchin 	    case '?':
59da2e3ebdSchin 		errormsg(SH_DICT,ERROR_usage(0), "%s", opt_info.arg);
60da2e3ebdSchin 		return(2);
61da2e3ebdSchin 	}
62da2e3ebdSchin done:
63da2e3ebdSchin 	if(error_info.errors)
64da2e3ebdSchin 		errormsg(SH_DICT,ERROR_usage(2),"%s",optusage((char*)0));
65da2e3ebdSchin 	pp->mode = (**argv=='e'?SH_JMPEXIT:SH_JMPFUN);
66da2e3ebdSchin 	argv += opt_info.index;
6734f9b3eeSRoland Mainz 	n = (((arg= *argv)?(int)strtol(arg, (char**)0, 10)&SH_EXITMASK:shp->oldexit));
68da2e3ebdSchin 	/* return outside of function, dotscript and profile is exit */
69da2e3ebdSchin 	if(shp->fn_depth==0 && shp->dot_depth==0 && !sh_isstate(SH_PROFILE))
70da2e3ebdSchin 		pp->mode = SH_JMPEXIT;
71da2e3ebdSchin 	sh_exit(shp->savexit=n);
72da2e3ebdSchin 	return(1);
73da2e3ebdSchin }
74da2e3ebdSchin 
75da2e3ebdSchin 
76da2e3ebdSchin /*
77da2e3ebdSchin  * break and continue
78da2e3ebdSchin  */
79da2e3ebdSchin #if 0
80da2e3ebdSchin     /* for the dictionary generator */
81da2e3ebdSchin     int	b_continue(int n, register char *argv[],void *extra){}
82da2e3ebdSchin #endif
b_break(register int n,register char * argv[],void * extra)83da2e3ebdSchin int	b_break(register int n, register char *argv[],void *extra)
84da2e3ebdSchin {
85da2e3ebdSchin 	char *arg;
86da2e3ebdSchin 	register int cont= **argv=='c';
877c2fbfb3SApril Chin 	register Shell_t *shp = ((Shbltin_t*)extra)->shp;
88da2e3ebdSchin 	while((n = optget(argv,cont?sh_optcont:sh_optbreak))) switch(n)
89da2e3ebdSchin 	{
90da2e3ebdSchin 	    case ':':
91da2e3ebdSchin 		errormsg(SH_DICT,2, "%s", opt_info.arg);
92da2e3ebdSchin 		break;
93da2e3ebdSchin 	    case '?':
94da2e3ebdSchin 		errormsg(SH_DICT,ERROR_usage(0), "%s", opt_info.arg);
95da2e3ebdSchin 		return(2);
96da2e3ebdSchin 	}
97da2e3ebdSchin 	if(error_info.errors)
98da2e3ebdSchin 		errormsg(SH_DICT,ERROR_usage(2),"%s",optusage((char*)0));
99da2e3ebdSchin 	argv += opt_info.index;
100da2e3ebdSchin 	n=1;
101da2e3ebdSchin 	if(arg= *argv)
102da2e3ebdSchin 	{
103da2e3ebdSchin 		n = strtol(arg,&arg,10);
104da2e3ebdSchin 		if(n<=0 || *arg)
105da2e3ebdSchin 			errormsg(SH_DICT,ERROR_exit(1),e_nolabels,*argv);
106da2e3ebdSchin 	}
107da2e3ebdSchin 	if(shp->st.loopcnt)
108da2e3ebdSchin 	{
109da2e3ebdSchin 		shp->st.execbrk = shp->st.breakcnt = n;
110da2e3ebdSchin 		if(shp->st.breakcnt > shp->st.loopcnt)
111da2e3ebdSchin 			shp->st.breakcnt = shp->st.loopcnt;
112da2e3ebdSchin 		if(cont)
113da2e3ebdSchin 			shp->st.breakcnt = -shp->st.breakcnt;
114da2e3ebdSchin 	}
115da2e3ebdSchin 	return(0);
116da2e3ebdSchin }
117da2e3ebdSchin 
118