14887Schin /*********************************************************************** 24887Schin * * 34887Schin * This software is part of the ast package * 4*8462SApril.Chin@Sun.COM * Copyright (c) 1982-2008 AT&T Intellectual Property * 54887Schin * and is licensed under the * 64887Schin * Common Public License, Version 1.0 * 7*8462SApril.Chin@Sun.COM * by AT&T Intellectual Property * 84887Schin * * 94887Schin * A copy of the License is available at * 104887Schin * http://www.opensource.org/licenses/cpl1.0.txt * 114887Schin * (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9) * 124887Schin * * 134887Schin * Information and Software Systems Research * 144887Schin * AT&T Research * 154887Schin * Florham Park NJ * 164887Schin * * 174887Schin * David Korn <dgk@research.att.com> * 184887Schin * * 194887Schin ***********************************************************************/ 204887Schin #pragma prototyped 214887Schin /* 224887Schin * command [-pvVx] name [arg...] 234887Schin * whence [-afvp] name... 244887Schin * 254887Schin * David Korn 264887Schin * AT&T Labs 274887Schin * 284887Schin */ 294887Schin 304887Schin #include "defs.h" 314887Schin #include <error.h> 324887Schin #include "shtable.h" 334887Schin #include "name.h" 344887Schin #include "path.h" 354887Schin #include "shlex.h" 364887Schin #include "builtins.h" 374887Schin 384887Schin #define P_FLAG 1 394887Schin #define V_FLAG 2 404887Schin #define A_FLAG 4 414887Schin #define F_FLAG 010 424887Schin #define X_FLAG 020 43*8462SApril.Chin@Sun.COM #define Q_FLAG 040 444887Schin 454887Schin static int whence(Shell_t *,char**, int); 464887Schin 474887Schin /* 484887Schin * command is called with argc==0 when checking for -V or -v option 494887Schin * In this case return 0 when -v or -V or unknown option, otherwise 504887Schin * the shift count to the command is returned 514887Schin */ 524887Schin int b_command(register int argc,char *argv[],void *extra) 534887Schin { 544887Schin register int n, flags=0; 55*8462SApril.Chin@Sun.COM register Shell_t *shp = ((Shbltin_t*)extra)->shp; 564887Schin opt_info.index = opt_info.offset = 0; 574887Schin while((n = optget(argv,sh_optcommand))) switch(n) 584887Schin { 594887Schin case 'p': 604887Schin if(sh_isoption(SH_RESTRICTED)) 614887Schin errormsg(SH_DICT,ERROR_exit(1),e_restricted,"-p"); 624887Schin sh_onstate(SH_DEFPATH); 634887Schin break; 644887Schin case 'v': 654887Schin flags |= X_FLAG; 664887Schin break; 674887Schin case 'V': 684887Schin flags |= V_FLAG; 694887Schin break; 704887Schin case 'x': 714887Schin shp->xargexit = 1; 724887Schin break; 734887Schin case ':': 744887Schin if(argc==0) 754887Schin return(0); 764887Schin errormsg(SH_DICT,2, "%s", opt_info.arg); 774887Schin break; 784887Schin case '?': 794887Schin if(argc==0) 804887Schin return(0); 814887Schin errormsg(SH_DICT,ERROR_usage(2), "%s", opt_info.arg); 824887Schin break; 834887Schin } 844887Schin if(argc==0) 854887Schin return(flags?0:opt_info.index); 864887Schin argv += opt_info.index; 874887Schin if(error_info.errors || !*argv) 884887Schin errormsg(SH_DICT,ERROR_usage(2),"%s", optusage((char*)0)); 894887Schin return(whence(shp,argv, flags)); 904887Schin } 914887Schin 924887Schin /* 934887Schin * for the whence command 944887Schin */ 954887Schin int b_whence(int argc,char *argv[],void *extra) 964887Schin { 974887Schin register int flags=0, n; 98*8462SApril.Chin@Sun.COM register Shell_t *shp = ((Shbltin_t*)extra)->shp; 994887Schin NOT_USED(argc); 1004887Schin if(*argv[0]=='t') 1014887Schin flags = V_FLAG; 1024887Schin while((n = optget(argv,sh_optwhence))) switch(n) 1034887Schin { 1044887Schin case 'a': 1054887Schin flags |= A_FLAG; 1064887Schin /* FALL THRU */ 1074887Schin case 'v': 1084887Schin flags |= V_FLAG; 1094887Schin break; 1104887Schin case 'f': 1114887Schin flags |= F_FLAG; 1124887Schin break; 1134887Schin case 'p': 1144887Schin flags |= P_FLAG; 115*8462SApril.Chin@Sun.COM flags &= ~V_FLAG; 116*8462SApril.Chin@Sun.COM break; 117*8462SApril.Chin@Sun.COM case 'q': 118*8462SApril.Chin@Sun.COM flags |= Q_FLAG; 1194887Schin break; 1204887Schin case ':': 1214887Schin errormsg(SH_DICT,2, "%s", opt_info.arg); 1224887Schin break; 1234887Schin case '?': 1244887Schin errormsg(SH_DICT,ERROR_usage(2), "%s", opt_info.arg); 1254887Schin break; 1264887Schin } 1274887Schin argv += opt_info.index; 1284887Schin if(error_info.errors || !*argv) 1294887Schin errormsg(SH_DICT,ERROR_usage(2),optusage((char*)0)); 1304887Schin return(whence(shp, argv, flags)); 1314887Schin } 1324887Schin 1334887Schin static int whence(Shell_t *shp,char **argv, register int flags) 1344887Schin { 1354887Schin register const char *name; 1364887Schin register Namval_t *np; 1374887Schin register const char *cp; 1384887Schin register int aflag,r=0; 1394887Schin register const char *msg; 1404887Schin int tofree; 1414887Schin Dt_t *root; 1424887Schin Namval_t *nq; 1434887Schin char *notused; 144*8462SApril.Chin@Sun.COM Pathcomp_t *pp=0; 1454887Schin int notrack = 1; 146*8462SApril.Chin@Sun.COM if(flags&Q_FLAG) 147*8462SApril.Chin@Sun.COM flags &= ~A_FLAG; 1484887Schin while(name= *argv++) 1494887Schin { 1504887Schin tofree=0; 1514887Schin aflag = ((flags&A_FLAG)!=0); 1524887Schin cp = 0; 1534887Schin np = 0; 1544887Schin if(flags&P_FLAG) 1554887Schin goto search; 156*8462SApril.Chin@Sun.COM if(flags&Q_FLAG) 157*8462SApril.Chin@Sun.COM goto bltins; 1584887Schin /* reserved words first */ 1594887Schin if(sh_lookup(name,shtab_reserved)) 1604887Schin { 1614887Schin sfprintf(sfstdout,"%s%s\n",name,(flags&V_FLAG)?sh_translate(is_reserved):""); 1624887Schin if(!aflag) 1634887Schin continue; 1644887Schin aflag++; 1654887Schin } 1664887Schin /* non-tracked aliases */ 1674887Schin if((np=nv_search(name,shp->alias_tree,0)) 1684887Schin && !nv_isnull(np) && !(notrack=nv_isattr(np,NV_TAGGED)) 1694887Schin && (cp=nv_getval(np))) 1704887Schin { 1714887Schin if(flags&V_FLAG) 1724887Schin { 1734887Schin if(nv_isattr(np,NV_EXPORT)) 1744887Schin msg = sh_translate(is_xalias); 1754887Schin else 1764887Schin msg = sh_translate(is_alias); 1774887Schin sfprintf(sfstdout,msg,name); 1784887Schin } 1794887Schin sfputr(sfstdout,sh_fmtq(cp),'\n'); 1804887Schin if(!aflag) 1814887Schin continue; 1824887Schin cp = 0; 1834887Schin aflag++; 1844887Schin } 1854887Schin /* built-ins and functions next */ 186*8462SApril.Chin@Sun.COM bltins: 1874887Schin root = (flags&F_FLAG)?shp->bltin_tree:shp->fun_tree; 1884887Schin if(np= nv_bfsearch(name, root, &nq, ¬used)) 1894887Schin { 1904887Schin if(is_abuiltin(np) && nv_isnull(np)) 1914887Schin goto search; 1924887Schin cp = ""; 1934887Schin if(flags&V_FLAG) 1944887Schin { 1954887Schin if(nv_isnull(np)) 1964887Schin cp = sh_translate(is_ufunction); 1974887Schin else if(is_abuiltin(np)) 198*8462SApril.Chin@Sun.COM { 199*8462SApril.Chin@Sun.COM if(nv_isattr(np,BLT_SPC)) 200*8462SApril.Chin@Sun.COM cp = sh_translate(is_spcbuiltin); 201*8462SApril.Chin@Sun.COM else 202*8462SApril.Chin@Sun.COM cp = sh_translate(is_builtin); 203*8462SApril.Chin@Sun.COM } 2044887Schin else 2054887Schin cp = sh_translate(is_function); 2064887Schin } 207*8462SApril.Chin@Sun.COM if(flags&Q_FLAG) 208*8462SApril.Chin@Sun.COM continue; 2094887Schin sfprintf(sfstdout,"%s%s\n",name,cp); 2104887Schin if(!aflag) 2114887Schin continue; 2124887Schin cp = 0; 2134887Schin aflag++; 2144887Schin } 2154887Schin search: 2164887Schin if(sh_isstate(SH_DEFPATH)) 2174887Schin { 2184887Schin cp=0; 2194887Schin notrack=1; 2204887Schin } 221*8462SApril.Chin@Sun.COM do 2224887Schin { 223*8462SApril.Chin@Sun.COM if(path_search(name,&pp,2+(aflag>1))) 224*8462SApril.Chin@Sun.COM cp = name; 225*8462SApril.Chin@Sun.COM else 2264887Schin { 227*8462SApril.Chin@Sun.COM cp = stakptr(PATH_OFFSET); 228*8462SApril.Chin@Sun.COM if(*cp==0) 229*8462SApril.Chin@Sun.COM cp = 0; 230*8462SApril.Chin@Sun.COM else if(*cp!='/') 231*8462SApril.Chin@Sun.COM { 232*8462SApril.Chin@Sun.COM cp = path_fullname(cp); 233*8462SApril.Chin@Sun.COM tofree=1; 234*8462SApril.Chin@Sun.COM } 2354887Schin } 236*8462SApril.Chin@Sun.COM if(flags&Q_FLAG) 237*8462SApril.Chin@Sun.COM r |= !cp; 238*8462SApril.Chin@Sun.COM else if(cp) 2394887Schin { 240*8462SApril.Chin@Sun.COM if(flags&V_FLAG) 2414887Schin { 242*8462SApril.Chin@Sun.COM if(*cp!= '/') 243*8462SApril.Chin@Sun.COM { 244*8462SApril.Chin@Sun.COM if(!np && (np=nv_search(name,shp->track_tree,0))) 245*8462SApril.Chin@Sun.COM sfprintf(sfstdout,"%s %s %s/%s\n",name,sh_translate(is_talias),path_pwd(0),cp); 246*8462SApril.Chin@Sun.COM else if(!np || nv_isnull(np)) 247*8462SApril.Chin@Sun.COM sfprintf(sfstdout,"%s%s\n",name,sh_translate(is_ufunction)); 248*8462SApril.Chin@Sun.COM continue; 249*8462SApril.Chin@Sun.COM } 250*8462SApril.Chin@Sun.COM sfputr(sfstdout,sh_fmtq(name),' '); 251*8462SApril.Chin@Sun.COM /* built-in version of program */ 252*8462SApril.Chin@Sun.COM if(*cp=='/' && (np=nv_search(cp,shp->bltin_tree,0))) 253*8462SApril.Chin@Sun.COM msg = sh_translate(is_builtver); 254*8462SApril.Chin@Sun.COM /* tracked aliases next */ 255*8462SApril.Chin@Sun.COM else if(aflag>1 || !notrack || strchr(name,'/')) 256*8462SApril.Chin@Sun.COM msg = sh_translate("is"); 257*8462SApril.Chin@Sun.COM else 258*8462SApril.Chin@Sun.COM msg = sh_translate(is_talias); 259*8462SApril.Chin@Sun.COM sfputr(sfstdout,msg,' '); 2604887Schin } 261*8462SApril.Chin@Sun.COM sfputr(sfstdout,sh_fmtq(cp),'\n'); 262*8462SApril.Chin@Sun.COM if(aflag) 263*8462SApril.Chin@Sun.COM { 264*8462SApril.Chin@Sun.COM if(aflag<=1) 265*8462SApril.Chin@Sun.COM aflag++; 266*8462SApril.Chin@Sun.COM if (pp) 267*8462SApril.Chin@Sun.COM pp = pp->next; 268*8462SApril.Chin@Sun.COM } 2694887Schin else 270*8462SApril.Chin@Sun.COM pp = 0; 271*8462SApril.Chin@Sun.COM if(tofree) 272*8462SApril.Chin@Sun.COM free((char*)cp); 2734887Schin } 274*8462SApril.Chin@Sun.COM else if(aflag<=1) 2754887Schin { 276*8462SApril.Chin@Sun.COM r |= 1; 277*8462SApril.Chin@Sun.COM if(flags&V_FLAG) 278*8462SApril.Chin@Sun.COM errormsg(SH_DICT,ERROR_exit(0),e_found,sh_fmtq(name)); 2794887Schin } 280*8462SApril.Chin@Sun.COM } while(pp); 2814887Schin } 2824887Schin return(r); 2834887Schin } 2844887Schin 285