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