xref: /onnv-gate/usr/src/cmd/svr4pkg/pkgchk/checkmap.c (revision 9869:9bff8d14ecc3)
19781SMoriah.Waterland@Sun.COM /*
29781SMoriah.Waterland@Sun.COM  * CDDL HEADER START
39781SMoriah.Waterland@Sun.COM  *
49781SMoriah.Waterland@Sun.COM  * The contents of this file are subject to the terms of the
59781SMoriah.Waterland@Sun.COM  * Common Development and Distribution License (the "License").
69781SMoriah.Waterland@Sun.COM  * You may not use this file except in compliance with the License.
79781SMoriah.Waterland@Sun.COM  *
89781SMoriah.Waterland@Sun.COM  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
99781SMoriah.Waterland@Sun.COM  * or http://www.opensolaris.org/os/licensing.
109781SMoriah.Waterland@Sun.COM  * See the License for the specific language governing permissions
119781SMoriah.Waterland@Sun.COM  * and limitations under the License.
129781SMoriah.Waterland@Sun.COM  *
139781SMoriah.Waterland@Sun.COM  * When distributing Covered Code, include this CDDL HEADER in each
149781SMoriah.Waterland@Sun.COM  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
159781SMoriah.Waterland@Sun.COM  * If applicable, add the following below this CDDL HEADER, with the
169781SMoriah.Waterland@Sun.COM  * fields enclosed by brackets "[]" replaced with your own identifying
179781SMoriah.Waterland@Sun.COM  * information: Portions Copyright [yyyy] [name of copyright owner]
189781SMoriah.Waterland@Sun.COM  *
199781SMoriah.Waterland@Sun.COM  * CDDL HEADER END
209781SMoriah.Waterland@Sun.COM  */
219781SMoriah.Waterland@Sun.COM 
229781SMoriah.Waterland@Sun.COM /*
23*9869SCasper.Dik@Sun.COM  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
249781SMoriah.Waterland@Sun.COM  * Use is subject to license terms.
259781SMoriah.Waterland@Sun.COM  */
269781SMoriah.Waterland@Sun.COM 
279781SMoriah.Waterland@Sun.COM /* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
289781SMoriah.Waterland@Sun.COM /* All Rights Reserved */
299781SMoriah.Waterland@Sun.COM 
309781SMoriah.Waterland@Sun.COM 
319781SMoriah.Waterland@Sun.COM #include <stdio.h>
329781SMoriah.Waterland@Sun.COM #include <string.h>
339781SMoriah.Waterland@Sun.COM #include <limits.h>
349781SMoriah.Waterland@Sun.COM #include <errno.h>
359781SMoriah.Waterland@Sun.COM #include <stdlib.h>
369781SMoriah.Waterland@Sun.COM #include <unistd.h>
379781SMoriah.Waterland@Sun.COM #include <dirent.h>
389781SMoriah.Waterland@Sun.COM #include <sys/types.h>
399781SMoriah.Waterland@Sun.COM #include <sys/stat.h>
409781SMoriah.Waterland@Sun.COM #include <sys/param.h>
419781SMoriah.Waterland@Sun.COM #include <sys/mman.h>
429781SMoriah.Waterland@Sun.COM #include <pkgstrct.h>
439781SMoriah.Waterland@Sun.COM #include <pkglocs.h>
449781SMoriah.Waterland@Sun.COM #include <locale.h>
459781SMoriah.Waterland@Sun.COM #include <libintl.h>
469781SMoriah.Waterland@Sun.COM #include <pkglib.h>
479781SMoriah.Waterland@Sun.COM #include "libadm.h"
489781SMoriah.Waterland@Sun.COM #include "libinst.h"
499781SMoriah.Waterland@Sun.COM 
509781SMoriah.Waterland@Sun.COM extern int	qflag, lflag, Lflag, pkgcnt;
519781SMoriah.Waterland@Sun.COM extern short	npaths;
529781SMoriah.Waterland@Sun.COM 
539781SMoriah.Waterland@Sun.COM extern char	*basedir, *pathlist[], *ppathlist[], **pkg, **environ;
549781SMoriah.Waterland@Sun.COM 
559781SMoriah.Waterland@Sun.COM extern short	used[];
569781SMoriah.Waterland@Sun.COM extern struct cfent **eptlist;
579781SMoriah.Waterland@Sun.COM 
58*9869SCasper.Dik@Sun.COM /* ckentry.c */
59*9869SCasper.Dik@Sun.COM extern int	ckentry(int, int, struct cfent *, VFP_T *, PKGserver);
609781SMoriah.Waterland@Sun.COM 
61*9869SCasper.Dik@Sun.COM #define	NXTENTRY(P, VFP, SRV) \
62*9869SCasper.Dik@Sun.COM 		(maptyp ? srchcfile((P), "*", (SRV)) : \
639781SMoriah.Waterland@Sun.COM 		gpkgmapvfp((P), (VFP)))
649781SMoriah.Waterland@Sun.COM 
659781SMoriah.Waterland@Sun.COM #define	MSG_ARCHIVE	"NOTE: some pathnames are in private formats " \
669781SMoriah.Waterland@Sun.COM 			    "and cannot be verified"
679781SMoriah.Waterland@Sun.COM #define	WRN_NOPKG	"WARNING: no pathnames were associated with <%s>"
689781SMoriah.Waterland@Sun.COM #define	WRN_NOPATH	"WARNING: no information associated with pathname <%s>"
699781SMoriah.Waterland@Sun.COM #define	EMPTY_PKG "WARNING: Package <%s> is installed but empty"
709781SMoriah.Waterland@Sun.COM #define	ERR_NOMEM	"unable to allocate dynamic memory, errno=%d"
719781SMoriah.Waterland@Sun.COM #define	ERR_PKGMAP	"unable to open pkgmap file <%s>"
729781SMoriah.Waterland@Sun.COM #define	ERR_ENVFILE	"unable to open environment file <%s>"
739781SMoriah.Waterland@Sun.COM 
749781SMoriah.Waterland@Sun.COM static struct cfent entry;
759781SMoriah.Waterland@Sun.COM 
769781SMoriah.Waterland@Sun.COM static int	shellmatch(char *, char *);
779781SMoriah.Waterland@Sun.COM static int is_partial_path_in_DB(char *, char *);
789781SMoriah.Waterland@Sun.COM 
799781SMoriah.Waterland@Sun.COM int	selpath(char *, int);
809781SMoriah.Waterland@Sun.COM int	selpkg(char *);
819781SMoriah.Waterland@Sun.COM 
829781SMoriah.Waterland@Sun.COM /*
839781SMoriah.Waterland@Sun.COM  * This routine checks all files which are referenced in the pkgmap which is
849781SMoriah.Waterland@Sun.COM  * identified by the mapfile arg. When the package is installed, the mapfile
859781SMoriah.Waterland@Sun.COM  * may be the contents file or a separate pkgmap (maptyp tells the function
869781SMoriah.Waterland@Sun.COM  * which it is). The variable uninst tells the function whether the package
879781SMoriah.Waterland@Sun.COM  * is in the installed state or not. The envfile entry is usually a pkginfo
889781SMoriah.Waterland@Sun.COM  * file, but it could be any environment parameter list.
899781SMoriah.Waterland@Sun.COM  */
909781SMoriah.Waterland@Sun.COM 
919781SMoriah.Waterland@Sun.COM int
checkmap(int maptyp,int uninst,char * mapfile,char * envfile,char * pkginst,char * path,int pathtype)929781SMoriah.Waterland@Sun.COM checkmap(int maptyp, int uninst, char *mapfile, char *envfile,
939781SMoriah.Waterland@Sun.COM 		char *pkginst, char *path, int pathtype)
949781SMoriah.Waterland@Sun.COM {
959781SMoriah.Waterland@Sun.COM 	FILE		*fp;
969781SMoriah.Waterland@Sun.COM 	char		*cl = NULL;
979781SMoriah.Waterland@Sun.COM 	char		*value;
989781SMoriah.Waterland@Sun.COM 	char		param[MAX_PKG_PARAM_LENGTH];
999781SMoriah.Waterland@Sun.COM 	int		count;
1009781SMoriah.Waterland@Sun.COM 	int		errflg;
1019781SMoriah.Waterland@Sun.COM 	int		n;
1029781SMoriah.Waterland@Sun.COM 	int		selected;
1039781SMoriah.Waterland@Sun.COM 	struct pinfo	*pinfo;
1049781SMoriah.Waterland@Sun.COM 	VFP_T		*vfp = (VFP_T *)NULL;
105*9869SCasper.Dik@Sun.COM 	PKGserver	server;
1069781SMoriah.Waterland@Sun.COM 
1079781SMoriah.Waterland@Sun.COM 	if (envfile != NULL) {
1089781SMoriah.Waterland@Sun.COM 		if ((fp = fopen(envfile, "r")) == NULL) {
1099781SMoriah.Waterland@Sun.COM 			progerr(gettext(ERR_ENVFILE), envfile);
1109781SMoriah.Waterland@Sun.COM 			return (-1);
1119781SMoriah.Waterland@Sun.COM 		}
1129781SMoriah.Waterland@Sun.COM 		param[0] = '\0';
1139781SMoriah.Waterland@Sun.COM 		while (value = fpkgparam(fp, param)) {
1149781SMoriah.Waterland@Sun.COM 			if (strcmp("PATH", param) != 0) {
1159781SMoriah.Waterland@Sun.COM 				/*
1169781SMoriah.Waterland@Sun.COM 				 * If checking an uninstalled package, we
1179781SMoriah.Waterland@Sun.COM 				 * only want two parameters. If we took all
1189781SMoriah.Waterland@Sun.COM 				 * of them, including path definitions, we
1199781SMoriah.Waterland@Sun.COM 				 * wouldn't be looking in the right places in
1209781SMoriah.Waterland@Sun.COM 				 * the reloc and root directories.
1219781SMoriah.Waterland@Sun.COM 				 */
1229781SMoriah.Waterland@Sun.COM 				if (uninst) {
1239781SMoriah.Waterland@Sun.COM 					if ((strncmp("PKG_SRC_NOVERIFY", param,
1249781SMoriah.Waterland@Sun.COM 					    16) == 0) && value) {
1259781SMoriah.Waterland@Sun.COM 						logerr(gettext(MSG_ARCHIVE));
1269781SMoriah.Waterland@Sun.COM 						putparam(param, value);
1279781SMoriah.Waterland@Sun.COM 					}
1289781SMoriah.Waterland@Sun.COM 					if ((strncmp("CLASSES", param,
1299781SMoriah.Waterland@Sun.COM 					    7) == 0) && value)
1309781SMoriah.Waterland@Sun.COM 						putparam(param, value);
1319781SMoriah.Waterland@Sun.COM 				} else
1329781SMoriah.Waterland@Sun.COM 					putparam(param, value);
1339781SMoriah.Waterland@Sun.COM 			}
1349781SMoriah.Waterland@Sun.COM 
1359781SMoriah.Waterland@Sun.COM 			free(value);
1369781SMoriah.Waterland@Sun.COM 
1379781SMoriah.Waterland@Sun.COM 			param[0] = '\0';
1389781SMoriah.Waterland@Sun.COM 		}
1399781SMoriah.Waterland@Sun.COM 		(void) fclose(fp);
1409781SMoriah.Waterland@Sun.COM 		basedir = getenv("BASEDIR");
1419781SMoriah.Waterland@Sun.COM 	}
1429781SMoriah.Waterland@Sun.COM 
1439781SMoriah.Waterland@Sun.COM 	/*
1449781SMoriah.Waterland@Sun.COM 	 * If we are using a contents file for the map, this locks the
1459781SMoriah.Waterland@Sun.COM 	 * contents file in order to freeze the database and assure it
1469781SMoriah.Waterland@Sun.COM 	 * remains synchronized with the file system against which it is
1479781SMoriah.Waterland@Sun.COM 	 * being compared. There is no practical way to lock another pkgmap
1489781SMoriah.Waterland@Sun.COM 	 * on some unknown medium so we don't bother.
1499781SMoriah.Waterland@Sun.COM 	 */
1509781SMoriah.Waterland@Sun.COM 	if (maptyp) {	/* If this is the contents file */
151*9869SCasper.Dik@Sun.COM 		if (!socfile(&server, B_FALSE) ||
152*9869SCasper.Dik@Sun.COM 		    pkgopenfilter(server, pkgcnt == 1 ? pkginst : NULL) != 0) {
1539781SMoriah.Waterland@Sun.COM 			progerr(gettext(ERR_PKGMAP), "contents");
1549781SMoriah.Waterland@Sun.COM 			return (-1);
1559781SMoriah.Waterland@Sun.COM 		}
1569781SMoriah.Waterland@Sun.COM 	} else {
1579781SMoriah.Waterland@Sun.COM 		if (vfpOpen(&vfp, mapfile, "r", VFP_NONE) != 0) {
1589781SMoriah.Waterland@Sun.COM 			progerr(gettext(ERR_PKGMAP), mapfile);
1599781SMoriah.Waterland@Sun.COM 			return (-1);
1609781SMoriah.Waterland@Sun.COM 		}
1619781SMoriah.Waterland@Sun.COM 	}
1629781SMoriah.Waterland@Sun.COM 
1639781SMoriah.Waterland@Sun.COM 	if ((cl = getenv("CLASSES")) != NULL)
1649781SMoriah.Waterland@Sun.COM 		cl_sets(qstrdup(cl));
1659781SMoriah.Waterland@Sun.COM 
1669781SMoriah.Waterland@Sun.COM 	errflg = count = 0;
1679781SMoriah.Waterland@Sun.COM 
1689781SMoriah.Waterland@Sun.COM 	do {
169*9869SCasper.Dik@Sun.COM 		if ((n = NXTENTRY(&entry, vfp, server)) == 0) {
1709781SMoriah.Waterland@Sun.COM 			break;
1719781SMoriah.Waterland@Sun.COM 		}
1729781SMoriah.Waterland@Sun.COM 		/*
1739781SMoriah.Waterland@Sun.COM 		 * Search for partial paths in the ext DB.
1749781SMoriah.Waterland@Sun.COM 		 */
1759781SMoriah.Waterland@Sun.COM 		if (pathtype) {
1769781SMoriah.Waterland@Sun.COM 			/* LINTED warning: statement has no consequent: if */
1779781SMoriah.Waterland@Sun.COM 			if (is_partial_path_in_DB(entry.path, path)) {
1789781SMoriah.Waterland@Sun.COM 				/* Check this entry */
1799781SMoriah.Waterland@Sun.COM 				;
180*9869SCasper.Dik@Sun.COM 			} else if (entry.ftype == 's' || entry.ftype == 'l') {
1819781SMoriah.Waterland@Sun.COM 				if (is_partial_path_in_DB(
1829781SMoriah.Waterland@Sun.COM 				/* LINTED warning: statement has no consequen */
1839781SMoriah.Waterland@Sun.COM 					entry.ainfo.local, path)) {
1849781SMoriah.Waterland@Sun.COM 					/* Check this entry */
1859781SMoriah.Waterland@Sun.COM 					;
1869781SMoriah.Waterland@Sun.COM 				} else {
1879781SMoriah.Waterland@Sun.COM 					continue;
1889781SMoriah.Waterland@Sun.COM 				}
1899781SMoriah.Waterland@Sun.COM 			} else {
1909781SMoriah.Waterland@Sun.COM 				/* Skip to next DB entry */
1919781SMoriah.Waterland@Sun.COM 				continue;
1929781SMoriah.Waterland@Sun.COM 			}
1939781SMoriah.Waterland@Sun.COM 		}
1949781SMoriah.Waterland@Sun.COM 
1959781SMoriah.Waterland@Sun.COM 		if (n < 0) {
1969781SMoriah.Waterland@Sun.COM 			char	*errstr = getErrstr();
1979781SMoriah.Waterland@Sun.COM 			logerr(gettext("ERROR: garbled entry"));
1989781SMoriah.Waterland@Sun.COM 			logerr(gettext("pathname: %s"),
1999781SMoriah.Waterland@Sun.COM 			    (entry.path && *entry.path) ? entry.path :
2009781SMoriah.Waterland@Sun.COM 			    "Unknown");
2019781SMoriah.Waterland@Sun.COM 			logerr(gettext("problem: %s"),
2029781SMoriah.Waterland@Sun.COM 			    (errstr && *errstr) ? errstr : "Unknown");
2039781SMoriah.Waterland@Sun.COM 			exit(99);
2049781SMoriah.Waterland@Sun.COM 		}
2059781SMoriah.Waterland@Sun.COM 		if (n == 0)
2069781SMoriah.Waterland@Sun.COM 			break; /* done with file */
2079781SMoriah.Waterland@Sun.COM 
2089781SMoriah.Waterland@Sun.COM 		/*
2099781SMoriah.Waterland@Sun.COM 		 * The class list may not be complete for good reason, so
2109781SMoriah.Waterland@Sun.COM 		 * there's no complaining if this returns an index of -1.
2119781SMoriah.Waterland@Sun.COM 		 */
2129781SMoriah.Waterland@Sun.COM 		if (cl != NULL)
2139781SMoriah.Waterland@Sun.COM 			entry.pkg_class_idx = cl_idx(entry.pkg_class);
2149781SMoriah.Waterland@Sun.COM 
2159781SMoriah.Waterland@Sun.COM 		if (maptyp && pkginst != NULL) {
2169781SMoriah.Waterland@Sun.COM 			/*
2179781SMoriah.Waterland@Sun.COM 			 * check to see if the entry we just read
2189781SMoriah.Waterland@Sun.COM 			 * is associated with one of the packages
2199781SMoriah.Waterland@Sun.COM 			 * we have listed on the command line
2209781SMoriah.Waterland@Sun.COM 			 */
2219781SMoriah.Waterland@Sun.COM 			selected = 0;
2229781SMoriah.Waterland@Sun.COM 			pinfo = entry.pinfo;
2239781SMoriah.Waterland@Sun.COM 			while (pinfo) {
2249781SMoriah.Waterland@Sun.COM 				if (selpkg(pinfo->pkg)) {
2259781SMoriah.Waterland@Sun.COM 					selected++;
2269781SMoriah.Waterland@Sun.COM 					break;
2279781SMoriah.Waterland@Sun.COM 				}
2289781SMoriah.Waterland@Sun.COM 				pinfo = pinfo->next;
2299781SMoriah.Waterland@Sun.COM 			}
2309781SMoriah.Waterland@Sun.COM 			if (!selected)
2319781SMoriah.Waterland@Sun.COM 				continue; /* not selected */
2329781SMoriah.Waterland@Sun.COM 		}
2339781SMoriah.Waterland@Sun.COM 
2349781SMoriah.Waterland@Sun.COM 		/*
2359781SMoriah.Waterland@Sun.COM 		 * Check to see if the pathname associated with the entry
2369781SMoriah.Waterland@Sun.COM 		 * we just read is associated with the list of paths we
2379781SMoriah.Waterland@Sun.COM 		 * supplied on the command line
2389781SMoriah.Waterland@Sun.COM 		 */
2399781SMoriah.Waterland@Sun.COM 		if (!selpath(entry.path, pathtype))
2409781SMoriah.Waterland@Sun.COM 			continue; /* not selected */
2419781SMoriah.Waterland@Sun.COM 
2429781SMoriah.Waterland@Sun.COM 		/*
2439781SMoriah.Waterland@Sun.COM 		 * Determine if this is a package object wanting
2449781SMoriah.Waterland@Sun.COM 		 * verification. Metafiles are always checked, otherwise, we
2459781SMoriah.Waterland@Sun.COM 		 * rely on the class to discriminate.
2469781SMoriah.Waterland@Sun.COM 		 */
2479781SMoriah.Waterland@Sun.COM 		if (entry.ftype != 'i')
2489781SMoriah.Waterland@Sun.COM 			/* If there's no class list... */
2499781SMoriah.Waterland@Sun.COM 			if (cl != NULL)
2509781SMoriah.Waterland@Sun.COM 				/*
2519781SMoriah.Waterland@Sun.COM 				 * ... or this entry isn't in that class list
2529781SMoriah.Waterland@Sun.COM 				 * or it's in a private format, then don't
2539781SMoriah.Waterland@Sun.COM 				 * check it.
2549781SMoriah.Waterland@Sun.COM 				 */
2559781SMoriah.Waterland@Sun.COM 				if (entry.pkg_class_idx == -1 ||
2569781SMoriah.Waterland@Sun.COM 				    cl_svfy(entry.pkg_class_idx) == NOVERIFY)
2579781SMoriah.Waterland@Sun.COM 					continue;
2589781SMoriah.Waterland@Sun.COM 
2599781SMoriah.Waterland@Sun.COM 		count++;
260*9869SCasper.Dik@Sun.COM 		if (ckentry((envfile ? 1 : 0), maptyp, &entry, vfp, server))
2619781SMoriah.Waterland@Sun.COM 			errflg++;
2629781SMoriah.Waterland@Sun.COM 	} while (n != 0);
2639781SMoriah.Waterland@Sun.COM 
2649781SMoriah.Waterland@Sun.COM 	if (maptyp)
2659781SMoriah.Waterland@Sun.COM 		relslock();
266*9869SCasper.Dik@Sun.COM 	else
267*9869SCasper.Dik@Sun.COM 		(void) vfpClose(&vfp);
2689781SMoriah.Waterland@Sun.COM 
2699781SMoriah.Waterland@Sun.COM 	if (environ) {
2709781SMoriah.Waterland@Sun.COM 		/* free up environment resources */
2719781SMoriah.Waterland@Sun.COM 		for (n = 0; environ[n]; n++)
2729781SMoriah.Waterland@Sun.COM 			free(environ[n]);
2739781SMoriah.Waterland@Sun.COM 		free(environ);
2749781SMoriah.Waterland@Sun.COM 		environ = NULL;
2759781SMoriah.Waterland@Sun.COM 	}
2769781SMoriah.Waterland@Sun.COM 
2779781SMoriah.Waterland@Sun.COM 	if (maptyp) {
2789781SMoriah.Waterland@Sun.COM 		/*
2799781SMoriah.Waterland@Sun.COM 		 * make sure each listed package was associated with
2809781SMoriah.Waterland@Sun.COM 		 * an entry from the prototype or pkgmap
2819781SMoriah.Waterland@Sun.COM 		 */
2829781SMoriah.Waterland@Sun.COM 		(void) selpkg(NULL);
2839781SMoriah.Waterland@Sun.COM 	}
2849781SMoriah.Waterland@Sun.COM 	if (!qflag && !lflag && !Lflag) {
2859781SMoriah.Waterland@Sun.COM 		/*
2869781SMoriah.Waterland@Sun.COM 		 * make sure each listed pathname was associated with an entry
2879781SMoriah.Waterland@Sun.COM 		 * from the prototype or pkgmap
2889781SMoriah.Waterland@Sun.COM 		 */
2899781SMoriah.Waterland@Sun.COM 		(void) selpath(NULL, pathtype);
2909781SMoriah.Waterland@Sun.COM 	}
2919781SMoriah.Waterland@Sun.COM 	return (errflg);
2929781SMoriah.Waterland@Sun.COM }
2939781SMoriah.Waterland@Sun.COM 
2949781SMoriah.Waterland@Sun.COM int
selpkg(char * p)2959781SMoriah.Waterland@Sun.COM selpkg(char *p)
2969781SMoriah.Waterland@Sun.COM {
2979781SMoriah.Waterland@Sun.COM 	static char *selected;
2989781SMoriah.Waterland@Sun.COM 	char buf[80];
2999781SMoriah.Waterland@Sun.COM 	char *root;
3009781SMoriah.Waterland@Sun.COM 	register int i;
3019781SMoriah.Waterland@Sun.COM 
3029781SMoriah.Waterland@Sun.COM 	if (p == NULL) {
3039781SMoriah.Waterland@Sun.COM 		if (selected == NULL) {
3049781SMoriah.Waterland@Sun.COM 			if (pkgcnt) {
3059781SMoriah.Waterland@Sun.COM 				for (i = 0; i < pkgcnt; ++i) {
3069781SMoriah.Waterland@Sun.COM 					/* bugid 1227628 */
3079781SMoriah.Waterland@Sun.COM 					root = get_inst_root();
3089781SMoriah.Waterland@Sun.COM 					if (root)
3099781SMoriah.Waterland@Sun.COM 						(void) snprintf(buf,
3109781SMoriah.Waterland@Sun.COM 						sizeof (buf),
3119781SMoriah.Waterland@Sun.COM 						"%s/var/sadm/pkg/%s/pkginfo",
3129781SMoriah.Waterland@Sun.COM 						root, pkg[i]);
3139781SMoriah.Waterland@Sun.COM 					else
3149781SMoriah.Waterland@Sun.COM 						(void) snprintf(buf,
3159781SMoriah.Waterland@Sun.COM 						sizeof (buf),
3169781SMoriah.Waterland@Sun.COM 						"/var/sadm/pkg/%s/pkginfo",
3179781SMoriah.Waterland@Sun.COM 						pkg[i]);
3189781SMoriah.Waterland@Sun.COM 
3199781SMoriah.Waterland@Sun.COM 					if (access(buf, F_OK))
3209781SMoriah.Waterland@Sun.COM 						logerr(gettext(WRN_NOPKG),
3219781SMoriah.Waterland@Sun.COM 							pkg[i]);
3229781SMoriah.Waterland@Sun.COM 					else
3239781SMoriah.Waterland@Sun.COM 						logerr(gettext(EMPTY_PKG),
3249781SMoriah.Waterland@Sun.COM 							pkg[i]);
3259781SMoriah.Waterland@Sun.COM 				}
3269781SMoriah.Waterland@Sun.COM 			}
3279781SMoriah.Waterland@Sun.COM 		} else {
3289781SMoriah.Waterland@Sun.COM 			for (i = 0; i < pkgcnt; ++i) {
3299781SMoriah.Waterland@Sun.COM 				if (selected[i] == NULL) {
3309781SMoriah.Waterland@Sun.COM 					root = get_inst_root();
3319781SMoriah.Waterland@Sun.COM 					if (root)
3329781SMoriah.Waterland@Sun.COM 						(void) snprintf(buf,
3339781SMoriah.Waterland@Sun.COM 						sizeof (buf),
3349781SMoriah.Waterland@Sun.COM 						"%s/var/sadm/pkg/%s/pkginfo",
3359781SMoriah.Waterland@Sun.COM 						root, pkg[i]);
3369781SMoriah.Waterland@Sun.COM 					else
3379781SMoriah.Waterland@Sun.COM 						(void) snprintf(buf,
3389781SMoriah.Waterland@Sun.COM 						sizeof (buf),
3399781SMoriah.Waterland@Sun.COM 						"/var/sadm/pkg/%s/pkginfo",
3409781SMoriah.Waterland@Sun.COM 						pkg[i]);
3419781SMoriah.Waterland@Sun.COM 
3429781SMoriah.Waterland@Sun.COM 					if (access(buf, F_OK))
3439781SMoriah.Waterland@Sun.COM 						logerr(gettext(WRN_NOPKG),
3449781SMoriah.Waterland@Sun.COM 							pkg[i]);
3459781SMoriah.Waterland@Sun.COM 					else
3469781SMoriah.Waterland@Sun.COM 						logerr(gettext(EMPTY_PKG),
3479781SMoriah.Waterland@Sun.COM 							pkg[i]);
3489781SMoriah.Waterland@Sun.COM 				}
3499781SMoriah.Waterland@Sun.COM 			}
3509781SMoriah.Waterland@Sun.COM 		}
3519781SMoriah.Waterland@Sun.COM 		return (0); /* return value not important */
3529781SMoriah.Waterland@Sun.COM 	} else if (pkgcnt == 0)
3539781SMoriah.Waterland@Sun.COM 		return (1);
3549781SMoriah.Waterland@Sun.COM 	else if (selected == NULL) {
3559781SMoriah.Waterland@Sun.COM 		selected =
3569781SMoriah.Waterland@Sun.COM 		    (char *)calloc((unsigned)(pkgcnt+1), sizeof (char));
3579781SMoriah.Waterland@Sun.COM 		if (selected == NULL) {
3589781SMoriah.Waterland@Sun.COM 			progerr(gettext(ERR_NOMEM), errno);
3599781SMoriah.Waterland@Sun.COM 			exit(99);
3609781SMoriah.Waterland@Sun.COM 			/*NOTREACHED*/
3619781SMoriah.Waterland@Sun.COM 		}
3629781SMoriah.Waterland@Sun.COM 	}
3639781SMoriah.Waterland@Sun.COM 
3649781SMoriah.Waterland@Sun.COM 	for (i = 0; i < pkgcnt; ++i) {
3659781SMoriah.Waterland@Sun.COM 		if (pkgnmchk(p, pkg[i], 0) == 0) {
3669781SMoriah.Waterland@Sun.COM 			if (selected != NULL)
3679781SMoriah.Waterland@Sun.COM 				selected[i] = 'b';
3689781SMoriah.Waterland@Sun.COM 			return (1);
3699781SMoriah.Waterland@Sun.COM 		}
3709781SMoriah.Waterland@Sun.COM 	}
3719781SMoriah.Waterland@Sun.COM 	return (0);
3729781SMoriah.Waterland@Sun.COM }
3739781SMoriah.Waterland@Sun.COM 
3749781SMoriah.Waterland@Sun.COM int
selpath(char * path,int partial_path)3759781SMoriah.Waterland@Sun.COM selpath(char *path, int partial_path)
3769781SMoriah.Waterland@Sun.COM {
3779781SMoriah.Waterland@Sun.COM 	int n;
3789781SMoriah.Waterland@Sun.COM 
3799781SMoriah.Waterland@Sun.COM 	if (!npaths)
3809781SMoriah.Waterland@Sun.COM 		return (1); /* everything is selectable */
3819781SMoriah.Waterland@Sun.COM 
3829781SMoriah.Waterland@Sun.COM 	for (n = 0; n < npaths; n++) {
3839781SMoriah.Waterland@Sun.COM 		if (path == NULL) {
3849781SMoriah.Waterland@Sun.COM 			if (!used[n])
3859781SMoriah.Waterland@Sun.COM 				logerr(gettext(WRN_NOPATH),
3869781SMoriah.Waterland@Sun.COM 					partial_path ? ppathlist[n] :
3879781SMoriah.Waterland@Sun.COM 					pathlist[n]);
3889781SMoriah.Waterland@Sun.COM 		} else if (partial_path) {
3899781SMoriah.Waterland@Sun.COM 			used[n] = 1;
3909781SMoriah.Waterland@Sun.COM 			return (1);
3919781SMoriah.Waterland@Sun.COM 		} else if (!shellmatch(pathlist[n], path)) {
3929781SMoriah.Waterland@Sun.COM 			used[n] = 1;
3939781SMoriah.Waterland@Sun.COM 			return (1);
3949781SMoriah.Waterland@Sun.COM 		}
3959781SMoriah.Waterland@Sun.COM 	}
3969781SMoriah.Waterland@Sun.COM 	return (0); /* not selected */
3979781SMoriah.Waterland@Sun.COM }
3989781SMoriah.Waterland@Sun.COM 
3999781SMoriah.Waterland@Sun.COM static int
shellmatch(char * spec,char * path)4009781SMoriah.Waterland@Sun.COM shellmatch(char *spec, char *path)
4019781SMoriah.Waterland@Sun.COM {
4029781SMoriah.Waterland@Sun.COM 	/* Check if the value is NULL */
4039781SMoriah.Waterland@Sun.COM 	if (spec == NULL || path == NULL)
4049781SMoriah.Waterland@Sun.COM 		return (1);
4059781SMoriah.Waterland@Sun.COM 
4069781SMoriah.Waterland@Sun.COM 	while (*spec && (*spec == *path)) {
4079781SMoriah.Waterland@Sun.COM 		spec++, path++;
4089781SMoriah.Waterland@Sun.COM 	}
4099781SMoriah.Waterland@Sun.COM 	if ((*spec == *path) || (*spec == '*'))
4109781SMoriah.Waterland@Sun.COM 		return (0);
4119781SMoriah.Waterland@Sun.COM 	return (1);
4129781SMoriah.Waterland@Sun.COM }
4139781SMoriah.Waterland@Sun.COM 
4149781SMoriah.Waterland@Sun.COM static int
is_partial_path_in_DB(char * srcpath,char * trgtpath)4159781SMoriah.Waterland@Sun.COM is_partial_path_in_DB(char *srcpath, char *trgtpath)
4169781SMoriah.Waterland@Sun.COM {
4179781SMoriah.Waterland@Sun.COM 	if (strstr(srcpath, trgtpath) == NULL) {
4189781SMoriah.Waterland@Sun.COM 		return (0);
4199781SMoriah.Waterland@Sun.COM 	} else {
4209781SMoriah.Waterland@Sun.COM 		return (1);
4219781SMoriah.Waterland@Sun.COM 	}
4229781SMoriah.Waterland@Sun.COM }
423