1*4887Schin /***********************************************************************
2*4887Schin *                                                                      *
3*4887Schin *               This software is part of the ast package               *
4*4887Schin *           Copyright (c) 1982-2007 AT&T Knowledge Ventures            *
5*4887Schin *                      and is licensed under the                       *
6*4887Schin *                  Common Public License, Version 1.0                  *
7*4887Schin *                      by AT&T Knowledge Ventures                      *
8*4887Schin *                                                                      *
9*4887Schin *                A copy of the License is available at                 *
10*4887Schin *            http://www.opensource.org/licenses/cpl1.0.txt             *
11*4887Schin *         (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9)         *
12*4887Schin *                                                                      *
13*4887Schin *              Information and Software Systems Research               *
14*4887Schin *                            AT&T Research                             *
15*4887Schin *                           Florham Park NJ                            *
16*4887Schin *                                                                      *
17*4887Schin *                  David Korn <dgk@research.att.com>                   *
18*4887Schin *                                                                      *
19*4887Schin ***********************************************************************/
20*4887Schin #pragma prototyped
21*4887Schin /*
22*4887Schin  * ulimit [-HSacdfmnstuv] [limit]
23*4887Schin  *
24*4887Schin  *   David Korn
25*4887Schin  *   AT&T Labs
26*4887Schin  *
27*4887Schin  */
28*4887Schin 
29*4887Schin #include	<ast.h>
30*4887Schin #include	<sfio.h>
31*4887Schin #include	<error.h>
32*4887Schin #include	<shell.h>
33*4887Schin #include	"builtins.h"
34*4887Schin #include	"name.h"
35*4887Schin #include	"ulimit.h"
36*4887Schin #ifndef SH_DICT
37*4887Schin #   define SH_DICT	"libshell"
38*4887Schin #endif
39*4887Schin 
40*4887Schin #ifdef _no_ulimit
41*4887Schin 	int	b_ulimit(int argc,char *argv[],void *extra)
42*4887Schin 	{
43*4887Schin 		NOT_USED(argc);
44*4887Schin 		NOT_USED(argv);
45*4887Schin 		NOT_USED(extra);
46*4887Schin 		errormsg(SH_DICT,ERROR_exit(2),e_nosupport);
47*4887Schin 		return(0);
48*4887Schin 	}
49*4887Schin #else
50*4887Schin 
51*4887Schin static int infof(Opt_t* op, Sfio_t* sp, const char* s, Optdisc_t* dp)
52*4887Schin {
53*4887Schin 	register const Limit_t*	tp;
54*4887Schin 
55*4887Schin 	for (tp = shtab_limits; tp->option; tp++)
56*4887Schin 	{
57*4887Schin 		sfprintf(sp, "[%c=%d:%s?The %s", tp->option, tp - shtab_limits + 1, tp->name, tp->description);
58*4887Schin 		if(tp->type != LIM_COUNT)
59*4887Schin 			sfprintf(sp, " in %ss", e_units[tp->type]);
60*4887Schin 		sfprintf(sp, ".]");
61*4887Schin 	}
62*4887Schin         return(1);
63*4887Schin }
64*4887Schin 
65*4887Schin #define HARD	2
66*4887Schin #define SOFT	4
67*4887Schin 
68*4887Schin int	b_ulimit(int argc,char *argv[],void *extra)
69*4887Schin {
70*4887Schin 	register char *limit;
71*4887Schin 	register int mode=0, n;
72*4887Schin 	register unsigned long hit = 0;
73*4887Schin 	Shell_t *shp = (Shell_t*)extra;
74*4887Schin #ifdef _lib_getrlimit
75*4887Schin 	struct rlimit rlp;
76*4887Schin #endif /* _lib_getrlimit */
77*4887Schin 	const Limit_t* tp;
78*4887Schin 	char* conf;
79*4887Schin 	int label, unit, noargs, nosupport;
80*4887Schin 	rlim_t i;
81*4887Schin 	char tmp[32];
82*4887Schin         Optdisc_t disc;
83*4887Schin         memset(&disc, 0, sizeof(disc));
84*4887Schin         disc.version = OPT_VERSION;
85*4887Schin         disc.infof = infof;
86*4887Schin 	opt_info.disc = &disc;
87*4887Schin 	while((n = optget(argv,sh_optulimit))) switch(n)
88*4887Schin 	{
89*4887Schin 		case 'H':
90*4887Schin 			mode |= HARD;
91*4887Schin 			continue;
92*4887Schin 		case 'S':
93*4887Schin 			mode |= SOFT;
94*4887Schin 			continue;
95*4887Schin 		case 'a':
96*4887Schin 			hit = ~0;
97*4887Schin 			break;
98*4887Schin 		default:
99*4887Schin 			if(n < 0)
100*4887Schin 				hit |= (1L<<(-(n+1)));
101*4887Schin 			else
102*4887Schin 				errormsg(SH_DICT,2, e_notimp, opt_info.name);
103*4887Schin 			break;
104*4887Schin 		case ':':
105*4887Schin 			errormsg(SH_DICT,2, "%s", opt_info.arg);
106*4887Schin 			break;
107*4887Schin 		case '?':
108*4887Schin 			errormsg(SH_DICT,ERROR_usage(2), "%s", opt_info.arg);
109*4887Schin 			break;
110*4887Schin 	}
111*4887Schin 	opt_info.disc = 0;
112*4887Schin 	limit = argv[opt_info.index];
113*4887Schin 	/* default to -f */
114*4887Schin 	if(noargs=(hit==0))
115*4887Schin 		for(n=0; shtab_limits[n].option; n++)
116*4887Schin 			if(shtab_limits[n].index == RLIMIT_FSIZE)
117*4887Schin 			{
118*4887Schin 				hit |= (1L<<n);
119*4887Schin 				break;
120*4887Schin 			}
121*4887Schin 	/* only one option at a time for setting */
122*4887Schin 	label = (hit&(hit-1));
123*4887Schin 	if(error_info.errors || (limit && label) || argc>opt_info.index+1)
124*4887Schin 		errormsg(SH_DICT,ERROR_usage(2),optusage((char*)0));
125*4887Schin 	if(mode==0)
126*4887Schin 		mode = (HARD|SOFT);
127*4887Schin 	for(tp = shtab_limits; tp->option && hit; tp++,hit>>=1)
128*4887Schin 	{
129*4887Schin 		if(!(hit&1))
130*4887Schin 			continue;
131*4887Schin 		nosupport = (n = tp->index) == RLIMIT_UNKNOWN;
132*4887Schin 		unit = shtab_units[tp->type];
133*4887Schin 		if(limit)
134*4887Schin 		{
135*4887Schin 			if(shp->subshell)
136*4887Schin 				sh_subfork();
137*4887Schin 			if(strcmp(limit,e_unlimited)==0)
138*4887Schin 				i = INFINITY;
139*4887Schin 			else
140*4887Schin 			{
141*4887Schin 				char *last;
142*4887Schin 				if((i=sh_strnum(limit,&last,2))==INFINITY || *last)
143*4887Schin 					errormsg(SH_DICT,ERROR_system(1),e_number,limit);
144*4887Schin 				i *= unit;
145*4887Schin 			}
146*4887Schin 			if(nosupport)
147*4887Schin 				errormsg(SH_DICT,ERROR_system(1),e_readonly,tp->name);
148*4887Schin 			else
149*4887Schin 			{
150*4887Schin #ifdef _lib_getrlimit
151*4887Schin 				if(getrlimit(n,&rlp) <0)
152*4887Schin 					errormsg(SH_DICT,ERROR_system(1),e_number,limit);
153*4887Schin 				if(mode&HARD)
154*4887Schin 					rlp.rlim_max = i;
155*4887Schin 				if(mode&SOFT)
156*4887Schin 					rlp.rlim_cur = i;
157*4887Schin 				if(setrlimit(n,&rlp) <0)
158*4887Schin 					errormsg(SH_DICT,ERROR_system(1),e_overlimit,limit);
159*4887Schin #else
160*4887Schin 				if((i=vlimit(n,i)) < 0)
161*4887Schin 					errormsg(SH_DICT,ERROR_system(1),e_number,limit);
162*4887Schin #endif /* _lib_getrlimit */
163*4887Schin 			}
164*4887Schin 		}
165*4887Schin 		else
166*4887Schin 		{
167*4887Schin 			if(!nosupport)
168*4887Schin 			{
169*4887Schin #ifdef  _lib_getrlimit
170*4887Schin 				if(getrlimit(n,&rlp) <0)
171*4887Schin 					errormsg(SH_DICT,ERROR_system(1),e_number,limit);
172*4887Schin 				if(mode&HARD)
173*4887Schin 					i = rlp.rlim_max;
174*4887Schin 				if(mode&SOFT)
175*4887Schin 					i = rlp.rlim_cur;
176*4887Schin #else
177*4887Schin #   ifdef _lib_ulimit
178*4887Schin 				n--;
179*4887Schin #   endif /* _lib_ulimit */
180*4887Schin 				i = -1;
181*4887Schin 				if((i=vlimit(n,i)) < 0)
182*4887Schin 					errormsg(SH_DICT,ERROR_system(1),e_number,limit);
183*4887Schin #endif /* _lib_getrlimit */
184*4887Schin 			}
185*4887Schin 			if(label)
186*4887Schin 			{
187*4887Schin 				if(tp->type != LIM_COUNT)
188*4887Schin 					sfsprintf(tmp,sizeof(tmp),"%s (%ss)", tp->description, e_units[tp->type]);
189*4887Schin 				else
190*4887Schin 					sfsprintf(tmp,sizeof(tmp),"%s", tp->name);
191*4887Schin 				sfprintf(sfstdout,"%-30s (-%c)  ",tmp,tp->option);
192*4887Schin 			}
193*4887Schin 			if(nosupport)
194*4887Schin 			{
195*4887Schin 				if(!tp->conf || !*(conf = astconf(tp->conf, NiL, NiL)))
196*4887Schin 					conf = (char*)e_nosupport;
197*4887Schin 				sfputr(sfstdout,conf,'\n');
198*4887Schin 			}
199*4887Schin 			else if(i!=INFINITY || noargs)
200*4887Schin 			{
201*4887Schin 				if(!noargs)
202*4887Schin 					i += (unit-1);
203*4887Schin 				sfprintf(sfstdout,"%I*d\n",sizeof(i),i/unit);
204*4887Schin 			}
205*4887Schin 			else
206*4887Schin 				sfputr(sfstdout,e_unlimited,'\n');
207*4887Schin 		}
208*4887Schin 	}
209*4887Schin 	return(0);
210*4887Schin }
211*4887Schin #endif /* _no_ulimit */
212