xref: /onnv-gate/usr/src/cmd/lvm/rpc.metamedd/med_db.c (revision 8979:eba3282e25c3)
10Sstevel@tonic-gate /*
20Sstevel@tonic-gate  * CDDL HEADER START
30Sstevel@tonic-gate  *
40Sstevel@tonic-gate  * The contents of this file are subject to the terms of the
5*8979SRameshkumar.Ramasamy@Sun.COM  * Common Development and Distribution License (the "License").
6*8979SRameshkumar.Ramasamy@Sun.COM  * You may not use this file except in compliance with the License.
70Sstevel@tonic-gate  *
80Sstevel@tonic-gate  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
90Sstevel@tonic-gate  * or http://www.opensolaris.org/os/licensing.
100Sstevel@tonic-gate  * See the License for the specific language governing permissions
110Sstevel@tonic-gate  * and limitations under the License.
120Sstevel@tonic-gate  *
130Sstevel@tonic-gate  * When distributing Covered Code, include this CDDL HEADER in each
140Sstevel@tonic-gate  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
150Sstevel@tonic-gate  * If applicable, add the following below this CDDL HEADER, with the
160Sstevel@tonic-gate  * fields enclosed by brackets "[]" replaced with your own identifying
170Sstevel@tonic-gate  * information: Portions Copyright [yyyy] [name of copyright owner]
180Sstevel@tonic-gate  *
190Sstevel@tonic-gate  * CDDL HEADER END
200Sstevel@tonic-gate  */
210Sstevel@tonic-gate /*
22*8979SRameshkumar.Ramasamy@Sun.COM  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
230Sstevel@tonic-gate  * Use is subject to license terms.
240Sstevel@tonic-gate  */
250Sstevel@tonic-gate 
260Sstevel@tonic-gate #include "med_local.h"
270Sstevel@tonic-gate #include "med_hash.h"
280Sstevel@tonic-gate #include <sys/lvm/mdio.h>
290Sstevel@tonic-gate #include <sys/lvm/md_mdiox.h>
300Sstevel@tonic-gate #include <sys/lvm/md_crc.h>
310Sstevel@tonic-gate 
320Sstevel@tonic-gate static	int		med_db_is_inited = 0;
330Sstevel@tonic-gate static	Cache		*med_db_cache = (Cache *) NULL;
340Sstevel@tonic-gate static	med_rec_t	*med_db_medrp = NULL;
350Sstevel@tonic-gate static	int		med_db_nma = 0;
360Sstevel@tonic-gate static	int		med_db_nmu = 0;
370Sstevel@tonic-gate static	int		rec_size = roundup(sizeof (med_rec_t), DEV_BSIZE);
380Sstevel@tonic-gate static	char 		*rec_buf = NULL;
390Sstevel@tonic-gate static	int		dbfd = -1;
400Sstevel@tonic-gate 
410Sstevel@tonic-gate #define		OP_FLAGS	(O_RDWR | O_SYNC)
420Sstevel@tonic-gate #define		CR_FLAGS	(OP_FLAGS | O_CREAT)
430Sstevel@tonic-gate 
440Sstevel@tonic-gate #define	HASHSIZE	151
450Sstevel@tonic-gate #define	BSZ		4
460Sstevel@tonic-gate 
470Sstevel@tonic-gate #ifdef DEBUG
480Sstevel@tonic-gate void
med_pr(void * keyp,int keyl,void * datap,int datal)490Sstevel@tonic-gate med_pr(void *keyp, int keyl, void *datap, int datal)
500Sstevel@tonic-gate {
510Sstevel@tonic-gate 	med_med_t	*medp = (med_med_t *)keyp;
520Sstevel@tonic-gate 	int		medridx = *(int *)datap;
530Sstevel@tonic-gate 	med_rec_t	*medrp = &med_db_medrp[medridx];
540Sstevel@tonic-gate 
550Sstevel@tonic-gate 	med_eprintf(
560Sstevel@tonic-gate 	    "key (%d)[keyp=0x%08x]: setno=%ld, setname=<%s>, caller=<%s>\n",
570Sstevel@tonic-gate 	    keyl, (unsigned)keyp, medp->med_setno, medp->med_setname,
580Sstevel@tonic-gate 	    medp->med_caller);
590Sstevel@tonic-gate 	med_eprintf("data(%d)[datap=0x%x08][medrp=0x%x08]: medridx=%d\n",
600Sstevel@tonic-gate 	    datal, (unsigned)datap, (unsigned)medrp, medridx);
610Sstevel@tonic-gate }
620Sstevel@tonic-gate #endif	/* DEBUG */
630Sstevel@tonic-gate 
640Sstevel@tonic-gate static int
med_hash(void * datap,int datalen,int hsz)650Sstevel@tonic-gate med_hash(void *datap, int datalen, int hsz)
660Sstevel@tonic-gate {
670Sstevel@tonic-gate 	med_med_t	*medp = (med_med_t *)datap;
680Sstevel@tonic-gate 	int		i = datalen;
690Sstevel@tonic-gate 	char		*cp;
700Sstevel@tonic-gate 
710Sstevel@tonic-gate 	i = 0;
720Sstevel@tonic-gate 	cp = medp->med_setname;
730Sstevel@tonic-gate 	while (*cp != '\0')
740Sstevel@tonic-gate 		i += *cp++;
750Sstevel@tonic-gate 
760Sstevel@tonic-gate 	cp = medp->med_caller;
770Sstevel@tonic-gate 	while (*cp != '\0')
780Sstevel@tonic-gate 		i += *cp++;
790Sstevel@tonic-gate 
800Sstevel@tonic-gate 	i *= medp->med_setno;
810Sstevel@tonic-gate 
820Sstevel@tonic-gate 	return (i % hsz);
830Sstevel@tonic-gate }
840Sstevel@tonic-gate 
850Sstevel@tonic-gate /*ARGSUSED*/
860Sstevel@tonic-gate static int
med_comp(void * datap1,void * datap2,int datalen)870Sstevel@tonic-gate med_comp(void *datap1, void *datap2, int datalen)
880Sstevel@tonic-gate {
890Sstevel@tonic-gate 	med_med_t	*medp1 = (med_med_t *)datap1;
900Sstevel@tonic-gate 	med_med_t	*medp2 = (med_med_t *)datap2;
910Sstevel@tonic-gate 	int		ret;
920Sstevel@tonic-gate 
930Sstevel@tonic-gate 
940Sstevel@tonic-gate 	ret = medp1->med_setno - medp2->med_setno;
950Sstevel@tonic-gate 
960Sstevel@tonic-gate 	if (ret != 0)
970Sstevel@tonic-gate 		return (ret);
980Sstevel@tonic-gate 
990Sstevel@tonic-gate 	ret = strcmp(medp1->med_caller, medp2->med_caller);
1000Sstevel@tonic-gate 
1010Sstevel@tonic-gate 	if (ret != 0)
1020Sstevel@tonic-gate 		return (ret);
1030Sstevel@tonic-gate 
1040Sstevel@tonic-gate 	return (strcmp(medp1->med_setname, medp2->med_setname));
1050Sstevel@tonic-gate }
1060Sstevel@tonic-gate 
1070Sstevel@tonic-gate static void
med_kfree(void * keyp)1080Sstevel@tonic-gate med_kfree(void *keyp)
1090Sstevel@tonic-gate {
1100Sstevel@tonic-gate 	med_med_t	*medp = (med_med_t *)keyp;
1110Sstevel@tonic-gate 
1120Sstevel@tonic-gate 	(void) Free(medp->med_caller);
1130Sstevel@tonic-gate 	(void) Free(medp->med_setname);
1140Sstevel@tonic-gate 	(void) Free(keyp);
1150Sstevel@tonic-gate }
1160Sstevel@tonic-gate 
1170Sstevel@tonic-gate static int
add_key(med_med_t * medp,int medridx)1180Sstevel@tonic-gate add_key(med_med_t *medp, int medridx)
1190Sstevel@tonic-gate {
1200Sstevel@tonic-gate 	Item		*itemp;
1210Sstevel@tonic-gate 	int		len;
1220Sstevel@tonic-gate 	med_med_t	*tmedp;
1230Sstevel@tonic-gate 
1240Sstevel@tonic-gate 	if (med_db_cache == (Cache *) NULL) {
1250Sstevel@tonic-gate 		len = init_cache(&med_db_cache, HASHSIZE, BSZ, med_hash,
126*8979SRameshkumar.Ramasamy@Sun.COM 		    med_comp, med_kfree, (void (*)())NULL);
1270Sstevel@tonic-gate 		if (len == -1) {
1280Sstevel@tonic-gate 			med_eprintf("add_key(): init_cache() failed.\n");
1290Sstevel@tonic-gate 			return (-1);
1300Sstevel@tonic-gate 		}
1310Sstevel@tonic-gate 	}
1320Sstevel@tonic-gate 
1330Sstevel@tonic-gate 	len = sizeof (med_med_t);
1340Sstevel@tonic-gate 
1350Sstevel@tonic-gate 	if ((itemp = lookup_cache(med_db_cache, medp, len)) == Null_Item) {
1360Sstevel@tonic-gate 		if ((itemp = (Item *) Malloc(sizeof (*itemp))) == NULL) {
1370Sstevel@tonic-gate 			med_eprintf("add_key(): itemp = Malloc(%d)\n",
1380Sstevel@tonic-gate 			    sizeof (*itemp));
1390Sstevel@tonic-gate 			return (-1);
1400Sstevel@tonic-gate 		}
1410Sstevel@tonic-gate 		if ((tmedp = itemp->key = Malloc(len)) == NULL) {
1420Sstevel@tonic-gate 			med_eprintf("add_key(): itemp->key = Malloc(%d)\n",
1430Sstevel@tonic-gate 			    len);
1440Sstevel@tonic-gate 			return (-1);
1450Sstevel@tonic-gate 		}
1460Sstevel@tonic-gate 
1470Sstevel@tonic-gate 		*tmedp = *medp;			/* structure assignment */
1480Sstevel@tonic-gate 
1490Sstevel@tonic-gate 		tmedp->med_caller = Malloc(strlen(medp->med_caller) + 1);
1500Sstevel@tonic-gate 		if (tmedp->med_caller == NULL) {
1510Sstevel@tonic-gate 			med_eprintf(
1520Sstevel@tonic-gate 			    "add_key(): tmedp->med_caller = Malloc(%d)\n",
1530Sstevel@tonic-gate 			    strlen(medp->med_caller) + 1);
1540Sstevel@tonic-gate 			return (-1);
1550Sstevel@tonic-gate 		}
1560Sstevel@tonic-gate 		(void) strcpy(tmedp->med_caller, medp->med_caller);
1570Sstevel@tonic-gate 
1580Sstevel@tonic-gate 		tmedp->med_setname = Malloc(strlen(medp->med_setname) + 1);
1590Sstevel@tonic-gate 		if (tmedp->med_setname == NULL) {
1600Sstevel@tonic-gate 			med_eprintf(
1610Sstevel@tonic-gate 			    "add_key(): tmedp->med_setname = Malloc(%d)\n",
1620Sstevel@tonic-gate 			    strlen(medp->med_setname) + 1);
1630Sstevel@tonic-gate 			return (-1);
1640Sstevel@tonic-gate 		}
1650Sstevel@tonic-gate 		(void) strcpy(tmedp->med_setname, medp->med_setname);
1660Sstevel@tonic-gate 
1670Sstevel@tonic-gate 		itemp->keyl = len;
1680Sstevel@tonic-gate 
1690Sstevel@tonic-gate 		if ((itemp->data = Malloc(sizeof (int))) == NULL) {
1700Sstevel@tonic-gate 			med_eprintf("add_key(): itemp->data = Malloc(%d)\n",
1710Sstevel@tonic-gate 			    sizeof (med_rec_t *));
1720Sstevel@tonic-gate 			return (-1);
1730Sstevel@tonic-gate 		}
1740Sstevel@tonic-gate 
1750Sstevel@tonic-gate 		*(int *)itemp->data = medridx;
1760Sstevel@tonic-gate 
1770Sstevel@tonic-gate 		itemp->datal = sizeof (int);
1780Sstevel@tonic-gate 
1790Sstevel@tonic-gate 		if (add_cache(med_db_cache, itemp) == -1) {
1800Sstevel@tonic-gate 			med_eprintf("add_key(): add_cache() failed.\n");
1810Sstevel@tonic-gate 			return (-1);
1820Sstevel@tonic-gate 		}
1830Sstevel@tonic-gate 		return (0);
1840Sstevel@tonic-gate 	}
1850Sstevel@tonic-gate 	return (1);
1860Sstevel@tonic-gate }
1870Sstevel@tonic-gate 
1880Sstevel@tonic-gate static int
del_key(med_med_t * medp)1890Sstevel@tonic-gate del_key(med_med_t *medp)
1900Sstevel@tonic-gate {
1910Sstevel@tonic-gate 	Item		*itemp;
1920Sstevel@tonic-gate 	int		len;
1930Sstevel@tonic-gate 
1940Sstevel@tonic-gate 	if (med_db_cache == (Cache *) NULL)
1950Sstevel@tonic-gate 		return (0);
1960Sstevel@tonic-gate 
1970Sstevel@tonic-gate 	len = sizeof (med_med_t);
1980Sstevel@tonic-gate 
1990Sstevel@tonic-gate 	if ((itemp = lookup_cache(med_db_cache, medp, len)) == Null_Item)
2000Sstevel@tonic-gate 		return (0);
2010Sstevel@tonic-gate 
2020Sstevel@tonic-gate 	(void) del_cache(med_db_cache, itemp);
2030Sstevel@tonic-gate 
2040Sstevel@tonic-gate 	return (0);
2050Sstevel@tonic-gate }
2060Sstevel@tonic-gate 
2070Sstevel@tonic-gate static int
find_key(med_med_t * medp)2080Sstevel@tonic-gate find_key(med_med_t *medp)
2090Sstevel@tonic-gate {
2100Sstevel@tonic-gate 	Item		*itemp;
2110Sstevel@tonic-gate 	int		len;
2120Sstevel@tonic-gate 
2130Sstevel@tonic-gate 	if (med_db_cache == (Cache *) NULL)
2140Sstevel@tonic-gate 		return (-1);
2150Sstevel@tonic-gate 
2160Sstevel@tonic-gate 	len = sizeof (med_med_t);
2170Sstevel@tonic-gate 
2180Sstevel@tonic-gate 	if ((itemp = lookup_cache(med_db_cache, medp, len)) == Null_Item)
2190Sstevel@tonic-gate 		return (-1);
2200Sstevel@tonic-gate 
2210Sstevel@tonic-gate 	return (*(int *)itemp->data);
2220Sstevel@tonic-gate }
2230Sstevel@tonic-gate 
2240Sstevel@tonic-gate static int
add_db_keys(int medridx,med_err_t * medep)2250Sstevel@tonic-gate add_db_keys(int medridx, med_err_t *medep)
2260Sstevel@tonic-gate {
2270Sstevel@tonic-gate 	med_med_t	med;
2280Sstevel@tonic-gate 	med_rec_t	*medrp;
2290Sstevel@tonic-gate 	int		i;
2300Sstevel@tonic-gate 
231*8979SRameshkumar.Ramasamy@Sun.COM 	medrp = &med_db_medrp[medridx];
232*8979SRameshkumar.Ramasamy@Sun.COM 	med.med_setno = medrp->med_rec_sn;
233*8979SRameshkumar.Ramasamy@Sun.COM 	med.med_setname = medrp->med_rec_snm;
234*8979SRameshkumar.Ramasamy@Sun.COM 
2350Sstevel@tonic-gate 	for (i = 0; i < MD_MAXSIDES; i++) {
2360Sstevel@tonic-gate 		if (medrp->med_rec_nodes[i][0] == '\0')
2370Sstevel@tonic-gate 			continue;
2380Sstevel@tonic-gate 		med.med_caller  = medrp->med_rec_nodes[i];
2390Sstevel@tonic-gate 		if (add_key(&med, medridx) == -1)
2400Sstevel@tonic-gate 			return (med_error(medep, MDE_MED_DBKEYADDFAIL,
2410Sstevel@tonic-gate 			    medrp->med_rec_nodes[i]));
2420Sstevel@tonic-gate 	}
243*8979SRameshkumar.Ramasamy@Sun.COM 
244*8979SRameshkumar.Ramasamy@Sun.COM 	/*
245*8979SRameshkumar.Ramasamy@Sun.COM 	 * Looping through the actual list of mediator hosts
246*8979SRameshkumar.Ramasamy@Sun.COM 	 * because a mediator host may not actually be a host
247*8979SRameshkumar.Ramasamy@Sun.COM 	 * in the diskset and so access for such a host needs
248*8979SRameshkumar.Ramasamy@Sun.COM 	 * to be added.
249*8979SRameshkumar.Ramasamy@Sun.COM 	 */
250*8979SRameshkumar.Ramasamy@Sun.COM 	for (i = 0; i < MED_MAX_HOSTS; i++) {
251*8979SRameshkumar.Ramasamy@Sun.COM 		if ((medrp->med_rec_meds.n_cnt > 0) &&
252*8979SRameshkumar.Ramasamy@Sun.COM 		    (medrp->med_rec_meds.n_lst[i].a_cnt != 0)) {
253*8979SRameshkumar.Ramasamy@Sun.COM 			med.med_caller  =
254*8979SRameshkumar.Ramasamy@Sun.COM 			    medrp->med_rec_meds.n_lst[i].a_nm[0];
255*8979SRameshkumar.Ramasamy@Sun.COM 			if (add_key(&med, medridx) == -1)
256*8979SRameshkumar.Ramasamy@Sun.COM 				return (med_error(medep, MDE_MED_DBKEYADDFAIL,
257*8979SRameshkumar.Ramasamy@Sun.COM 				    medrp->med_rec_meds.n_lst[i].a_nm[0]));
258*8979SRameshkumar.Ramasamy@Sun.COM 		}
259*8979SRameshkumar.Ramasamy@Sun.COM 	}
2600Sstevel@tonic-gate 	return (0);
2610Sstevel@tonic-gate }
2620Sstevel@tonic-gate 
2630Sstevel@tonic-gate static int
del_db_keys(int medridx,med_err_t * medep)2640Sstevel@tonic-gate del_db_keys(int medridx, med_err_t *medep)
2650Sstevel@tonic-gate {
2660Sstevel@tonic-gate 	med_med_t	med;
2670Sstevel@tonic-gate 	med_rec_t	*medrp;
2680Sstevel@tonic-gate 	int		i;
2690Sstevel@tonic-gate 
270*8979SRameshkumar.Ramasamy@Sun.COM 	medrp = &med_db_medrp[medridx];
271*8979SRameshkumar.Ramasamy@Sun.COM 	med.med_setno = medrp->med_rec_sn;
272*8979SRameshkumar.Ramasamy@Sun.COM 	med.med_setname = medrp->med_rec_snm;
273*8979SRameshkumar.Ramasamy@Sun.COM 
2740Sstevel@tonic-gate 	for (i = 0; i < MD_MAXSIDES; i++) {
2750Sstevel@tonic-gate 		if (medrp->med_rec_nodes[i][0] == '\0')
2760Sstevel@tonic-gate 			continue;
2770Sstevel@tonic-gate 		med.med_caller  = medrp->med_rec_nodes[i];
2780Sstevel@tonic-gate 		if (del_key(&med) == -1)
2790Sstevel@tonic-gate 			return (med_error(medep, MDE_MED_DBKEYDELFAIL,
2800Sstevel@tonic-gate 			    medrp->med_rec_nodes[i]));
2810Sstevel@tonic-gate 	}
282*8979SRameshkumar.Ramasamy@Sun.COM 
283*8979SRameshkumar.Ramasamy@Sun.COM 	for (i = 0; i < MED_MAX_HOSTS; i++) {
284*8979SRameshkumar.Ramasamy@Sun.COM 		if ((medrp->med_rec_meds.n_cnt > 0) &&
285*8979SRameshkumar.Ramasamy@Sun.COM 		    (medrp->med_rec_meds.n_lst[i].a_cnt != 0)) {
286*8979SRameshkumar.Ramasamy@Sun.COM 			med.med_caller  =
287*8979SRameshkumar.Ramasamy@Sun.COM 			    medrp->med_rec_meds.n_lst[i].a_nm[0];
288*8979SRameshkumar.Ramasamy@Sun.COM 			if (del_key(&med) == -1)
289*8979SRameshkumar.Ramasamy@Sun.COM 				return (med_error(medep, MDE_MED_DBKEYDELFAIL,
290*8979SRameshkumar.Ramasamy@Sun.COM 				    medrp->med_rec_meds.n_lst[i].a_nm[0]));
291*8979SRameshkumar.Ramasamy@Sun.COM 		}
292*8979SRameshkumar.Ramasamy@Sun.COM 	}
2930Sstevel@tonic-gate 	return (0);
2940Sstevel@tonic-gate }
2950Sstevel@tonic-gate 
2960Sstevel@tonic-gate static int
alloc_rec_buf(med_err_t * medep)2970Sstevel@tonic-gate alloc_rec_buf(med_err_t *medep)
2980Sstevel@tonic-gate {
2990Sstevel@tonic-gate 	if (rec_buf == NULL) {
3000Sstevel@tonic-gate 		if ((rec_buf = Malloc(rec_size)) == NULL)
3010Sstevel@tonic-gate 			return (med_error(medep, errno,
3020Sstevel@tonic-gate 			    "alloc_rec_buf: Malloc()"));
3030Sstevel@tonic-gate 	}
3040Sstevel@tonic-gate 
3050Sstevel@tonic-gate 	(void) memset(rec_buf, '\0', rec_size);
3060Sstevel@tonic-gate 	return (0);
3070Sstevel@tonic-gate }
3080Sstevel@tonic-gate 
3090Sstevel@tonic-gate static void
free_rec_buf(void)3100Sstevel@tonic-gate free_rec_buf(void)
3110Sstevel@tonic-gate {
3120Sstevel@tonic-gate 	if (rec_buf == NULL)
3130Sstevel@tonic-gate 		return;
3140Sstevel@tonic-gate 
3150Sstevel@tonic-gate 	Free(rec_buf);
3160Sstevel@tonic-gate 	rec_buf = NULL;
3170Sstevel@tonic-gate }
3180Sstevel@tonic-gate 
3190Sstevel@tonic-gate static int
write_hdr(int dbfd,med_err_t * medep)3200Sstevel@tonic-gate write_hdr(
3210Sstevel@tonic-gate 	int		dbfd,
3220Sstevel@tonic-gate 	med_err_t	*medep
3230Sstevel@tonic-gate )
3240Sstevel@tonic-gate {
3250Sstevel@tonic-gate 	med_db_hdr_t	dbh;
3260Sstevel@tonic-gate 
3270Sstevel@tonic-gate 	if (alloc_rec_buf(medep))
3280Sstevel@tonic-gate 		return (-1);
3290Sstevel@tonic-gate 
3300Sstevel@tonic-gate 	(void) memset(&dbh, '\0', sizeof (med_db_hdr_t));
3310Sstevel@tonic-gate 
3320Sstevel@tonic-gate 	/* Setup the new hdr record */
3330Sstevel@tonic-gate 	dbh.med_dbh_mag = MED_DB_MAGIC;
3340Sstevel@tonic-gate 	dbh.med_dbh_rev = MED_DB_REV;
3350Sstevel@tonic-gate 	dbh.med_dbh_nm = med_db_nmu;
3360Sstevel@tonic-gate 
3370Sstevel@tonic-gate 	/* Checksum new header */
3380Sstevel@tonic-gate 	crcgen(&dbh, &dbh.med_dbh_cks, sizeof (med_db_hdr_t), NULL);
3390Sstevel@tonic-gate 
3400Sstevel@tonic-gate 	/* Position to the beginning of the file */
3410Sstevel@tonic-gate 	if (lseek(dbfd, 0, SEEK_SET) == -1)
3420Sstevel@tonic-gate 		return (med_error(medep, errno, "write_hdr: lseek()"));
3430Sstevel@tonic-gate 
3440Sstevel@tonic-gate 	/* Copy the header into the output buffer */
3450Sstevel@tonic-gate 	(void) memmove(rec_buf, &dbh, sizeof (med_db_hdr_t));
3460Sstevel@tonic-gate 
3470Sstevel@tonic-gate 	/* Write out the header */
3480Sstevel@tonic-gate 	if (write(dbfd, rec_buf, rec_size) == -1)
3490Sstevel@tonic-gate 		return (med_error(medep, errno, "write_hdr: write()"));
3500Sstevel@tonic-gate 
3510Sstevel@tonic-gate 	return (0);
3520Sstevel@tonic-gate }
3530Sstevel@tonic-gate 
3540Sstevel@tonic-gate static int
write_rec(int dbfd,med_rec_t * medrp,med_err_t * medep)3550Sstevel@tonic-gate write_rec(
3560Sstevel@tonic-gate 	int		dbfd,
3570Sstevel@tonic-gate 	med_rec_t	*medrp,
3580Sstevel@tonic-gate 	med_err_t	*medep
3590Sstevel@tonic-gate )
3600Sstevel@tonic-gate {
3610Sstevel@tonic-gate 	uint_t		save_flags = 0;
3620Sstevel@tonic-gate 	uint_t		save_cks = 0;
3630Sstevel@tonic-gate 
3640Sstevel@tonic-gate 	if (alloc_rec_buf(medep))
3650Sstevel@tonic-gate 		return (-1);
3660Sstevel@tonic-gate 
3670Sstevel@tonic-gate 	if (medrp->med_rec_data.med_dat_fl) {
3680Sstevel@tonic-gate 		save_flags = medrp->med_rec_data.med_dat_fl;
3690Sstevel@tonic-gate 		save_cks = medrp->med_rec_data.med_dat_cks;
3700Sstevel@tonic-gate 		medrp->med_rec_data.med_dat_fl = 0;
3710Sstevel@tonic-gate 		/* Checksum the new data */
3720Sstevel@tonic-gate 		crcgen(&medrp->med_rec_data, &medrp->med_rec_data.med_dat_cks,
3730Sstevel@tonic-gate 		    sizeof (med_data_t), NULL);
3740Sstevel@tonic-gate 	}
3750Sstevel@tonic-gate 
3760Sstevel@tonic-gate 	/* Checksum record */
3770Sstevel@tonic-gate 	crcgen(medrp, &medrp->med_rec_cks, sizeof (med_rec_t), NULL);
3780Sstevel@tonic-gate 
3790Sstevel@tonic-gate 	/* Load the record into the output buffer */
3800Sstevel@tonic-gate 	(void) memmove(rec_buf, medrp, sizeof (med_rec_t));
3810Sstevel@tonic-gate 
3820Sstevel@tonic-gate 	if (save_flags) {
3830Sstevel@tonic-gate 		medrp->med_rec_data.med_dat_fl = save_flags;
3840Sstevel@tonic-gate 		medrp->med_rec_data.med_dat_cks = save_cks;
3850Sstevel@tonic-gate 		/* Re-checksum the updated record */
3860Sstevel@tonic-gate 		crcgen(medrp, &medrp->med_rec_cks, sizeof (med_rec_t), NULL);
3870Sstevel@tonic-gate 	}
3880Sstevel@tonic-gate 
3890Sstevel@tonic-gate 	/* Write out the record */
3900Sstevel@tonic-gate 	if (write(dbfd, rec_buf, rec_size) == -1)
3910Sstevel@tonic-gate 		return (med_error(medep, errno, "write_rec: write()"));
3920Sstevel@tonic-gate 
3930Sstevel@tonic-gate 	return (0);
3940Sstevel@tonic-gate }
3950Sstevel@tonic-gate 
3960Sstevel@tonic-gate static int
open_dbfile(med_err_t * medep)3970Sstevel@tonic-gate open_dbfile(med_err_t *medep)
3980Sstevel@tonic-gate {
3990Sstevel@tonic-gate 	if (dbfd != -1)
4000Sstevel@tonic-gate 		return (0);
4010Sstevel@tonic-gate 
4020Sstevel@tonic-gate 	/* Open the database file */
4030Sstevel@tonic-gate 	if ((dbfd = open(MED_DB_FILE, OP_FLAGS, 0644)) == -1) {
4040Sstevel@tonic-gate 		if (errno != ENOENT)
4050Sstevel@tonic-gate 			return (med_error(medep, errno, "open_dbfile: open()"));
4060Sstevel@tonic-gate 
4070Sstevel@tonic-gate 		if ((dbfd = open(MED_DB_FILE, CR_FLAGS, 0644)) == -1)
4080Sstevel@tonic-gate 			return (med_error(medep, errno,
4090Sstevel@tonic-gate 			    "open_dbfile: open(create)"));
4100Sstevel@tonic-gate 	}
4110Sstevel@tonic-gate 
4120Sstevel@tonic-gate 	/* Try to take an advisory lock on the file */
4130Sstevel@tonic-gate 	if (lockf(dbfd, F_TLOCK, (off_t)0) == -1) {
4140Sstevel@tonic-gate 		(void) med_error(medep, errno, "open_dbfile: lockf(F_TLOCK)");
4150Sstevel@tonic-gate 		medde_perror(medep, "");
4160Sstevel@tonic-gate 		med_exit(1);
4170Sstevel@tonic-gate 	}
4180Sstevel@tonic-gate 
4190Sstevel@tonic-gate 	return (0);
4200Sstevel@tonic-gate }
4210Sstevel@tonic-gate 
4220Sstevel@tonic-gate static int
close_dbfile(med_err_t * medep)4230Sstevel@tonic-gate close_dbfile(med_err_t *medep)
4240Sstevel@tonic-gate {
4250Sstevel@tonic-gate 	if (dbfd == -1)
4260Sstevel@tonic-gate 		return (0);
4270Sstevel@tonic-gate 
4280Sstevel@tonic-gate 	/* Make sure we are at the beginning of the file */
4290Sstevel@tonic-gate 	if (lseek(dbfd, 0, SEEK_SET) == -1)
4300Sstevel@tonic-gate 		return (med_error(medep, errno, "close_dbfile: lseek()"));
4310Sstevel@tonic-gate 
4320Sstevel@tonic-gate 	/* Release the advisory lock on the file */
4330Sstevel@tonic-gate 	if (lockf(dbfd, F_ULOCK, 0LL) == -1) {
4340Sstevel@tonic-gate 		(void) med_error(medep, errno, "close_dbfile: lockf(F_ULOCK)");
4350Sstevel@tonic-gate 		medde_perror(medep, "");
4360Sstevel@tonic-gate 		med_exit(1);
4370Sstevel@tonic-gate 	}
4380Sstevel@tonic-gate 
4390Sstevel@tonic-gate 	if (close(dbfd) == -1)
4400Sstevel@tonic-gate 		return (med_error(medep, errno, "close_dbfile: close()"));
4410Sstevel@tonic-gate 
4420Sstevel@tonic-gate 	dbfd = -1;
4430Sstevel@tonic-gate 
4440Sstevel@tonic-gate 	return (0);
4450Sstevel@tonic-gate }
4460Sstevel@tonic-gate 
4470Sstevel@tonic-gate static int
med_db_del_rec(med_med_t * medp,med_err_t * medep)4480Sstevel@tonic-gate med_db_del_rec(med_med_t *medp, med_err_t *medep)
4490Sstevel@tonic-gate {
4500Sstevel@tonic-gate 	med_rec_t	*medrp = NULL;
4510Sstevel@tonic-gate 	int		i;
4520Sstevel@tonic-gate 	int		medridx = -1;
4530Sstevel@tonic-gate 
4540Sstevel@tonic-gate 
4550Sstevel@tonic-gate 	if (! med_db_is_inited)
4560Sstevel@tonic-gate 		return (med_error(medep, MDE_MED_DBNOTINIT, "med_db_del_rec"));
4570Sstevel@tonic-gate 
4580Sstevel@tonic-gate 	if ((medridx = find_key(medp)) == -1)
4590Sstevel@tonic-gate 		return (0);
4600Sstevel@tonic-gate 
4610Sstevel@tonic-gate 	/* Delete the old keys */
4620Sstevel@tonic-gate 	if (del_db_keys(medridx, medep))
4630Sstevel@tonic-gate 		return (-1);
4640Sstevel@tonic-gate 
4650Sstevel@tonic-gate 	medrp = &med_db_medrp[medridx];
4660Sstevel@tonic-gate 
4670Sstevel@tonic-gate 	/* Mark the record in core as deleted */
4680Sstevel@tonic-gate 	medrp->med_rec_fl |= MED_RFL_DEL;
4690Sstevel@tonic-gate 
4700Sstevel@tonic-gate 	/* Decrement the used slot count */
4710Sstevel@tonic-gate 	med_db_nmu--;
4720Sstevel@tonic-gate 
4730Sstevel@tonic-gate 	/* Get ready to re-write the file */
4740Sstevel@tonic-gate 	if (ftruncate(dbfd, 0) == -1)
4750Sstevel@tonic-gate 		return (med_error(medep, errno, "med_db_del_rec: ftruncate()"));
4760Sstevel@tonic-gate 
4770Sstevel@tonic-gate 	if (write_hdr(dbfd, medep))
4780Sstevel@tonic-gate 		return (-1);
4790Sstevel@tonic-gate 
4800Sstevel@tonic-gate 	for (i = 0; i < med_db_nma; i++) {
4810Sstevel@tonic-gate 		medrp = &med_db_medrp[i];
4820Sstevel@tonic-gate 
4830Sstevel@tonic-gate 		if (medrp->med_rec_fl & MED_RFL_DEL)
4840Sstevel@tonic-gate 			continue;
4850Sstevel@tonic-gate 
4860Sstevel@tonic-gate 		/* Determine our location in the file */
4870Sstevel@tonic-gate 		if ((medrp->med_rec_foff = lseek(dbfd, 0, SEEK_CUR)) == -1)
4880Sstevel@tonic-gate 			return (med_error(medep, errno,
4890Sstevel@tonic-gate 			    "med_db_del_rec: lseek()"));
4900Sstevel@tonic-gate 
4910Sstevel@tonic-gate 		if (write_rec(dbfd, medrp, medep))
4920Sstevel@tonic-gate 			return (-1);
4930Sstevel@tonic-gate 	}
4940Sstevel@tonic-gate 	return (0);
4950Sstevel@tonic-gate }
4960Sstevel@tonic-gate 
4970Sstevel@tonic-gate static int
cmp_medrec(med_rec_t * omedrp,med_rec_t * nmedrp)4980Sstevel@tonic-gate cmp_medrec(med_rec_t *omedrp, med_rec_t *nmedrp)
4990Sstevel@tonic-gate {
5000Sstevel@tonic-gate 	int	ret;
5010Sstevel@tonic-gate 	int	i;
5020Sstevel@tonic-gate 
5030Sstevel@tonic-gate 	if (omedrp->med_rec_mag != nmedrp->med_rec_mag)
5040Sstevel@tonic-gate 		return (0);
5050Sstevel@tonic-gate 
5060Sstevel@tonic-gate 	if (omedrp->med_rec_rev != nmedrp->med_rec_rev)
5070Sstevel@tonic-gate 		return (0);
5080Sstevel@tonic-gate 
5090Sstevel@tonic-gate 	/* Can't compare checksums, since the new record has no data yet */
5100Sstevel@tonic-gate 
5110Sstevel@tonic-gate 	/* Can't compare flags, since the in-core may have golden */
5120Sstevel@tonic-gate 
5130Sstevel@tonic-gate 	if (omedrp->med_rec_sn != nmedrp->med_rec_sn)
5140Sstevel@tonic-gate 		return (0);
5150Sstevel@tonic-gate 
5160Sstevel@tonic-gate 	if (strcmp(omedrp->med_rec_snm, nmedrp->med_rec_snm) != 0)
5170Sstevel@tonic-gate 		return (0);
5180Sstevel@tonic-gate 
5190Sstevel@tonic-gate 	for (i = 0; i < MD_MAXSIDES; i++) {
5200Sstevel@tonic-gate 		if (omedrp->med_rec_nodes[i][0] == '\0' &&
5210Sstevel@tonic-gate 		    nmedrp->med_rec_nodes[i][0] == '\0')
5220Sstevel@tonic-gate 			continue;
5230Sstevel@tonic-gate 
5240Sstevel@tonic-gate 		ret = strcmp(omedrp->med_rec_nodes[i],
5250Sstevel@tonic-gate 		    nmedrp->med_rec_nodes[i]);
5260Sstevel@tonic-gate 		if (ret != 0)
5270Sstevel@tonic-gate 			return (0);
5280Sstevel@tonic-gate 	}
5290Sstevel@tonic-gate 
5300Sstevel@tonic-gate 	ret = memcmp(&omedrp->med_rec_meds, &nmedrp->med_rec_meds,
5310Sstevel@tonic-gate 	    sizeof (md_h_arr_t));
5320Sstevel@tonic-gate 	if (ret != 0)
5330Sstevel@tonic-gate 		return (0);
5340Sstevel@tonic-gate 
5350Sstevel@tonic-gate 	return (1);
5360Sstevel@tonic-gate }
5370Sstevel@tonic-gate 
5380Sstevel@tonic-gate /*
5390Sstevel@tonic-gate  * Exported routines
5400Sstevel@tonic-gate  */
5410Sstevel@tonic-gate 
5420Sstevel@tonic-gate int
med_db_init(med_err_t * medep)5430Sstevel@tonic-gate med_db_init(med_err_t *medep)
5440Sstevel@tonic-gate {
5450Sstevel@tonic-gate 	int		i;
5460Sstevel@tonic-gate 	int		err = 0;
5470Sstevel@tonic-gate 	int		ret;
5480Sstevel@tonic-gate 	struct	stat	statb;
5490Sstevel@tonic-gate 	med_db_hdr_t	*dbhp;
5500Sstevel@tonic-gate 	med_rec_t	*medrp;
5510Sstevel@tonic-gate 	int		nm;
5520Sstevel@tonic-gate 	off_t		cur_off;
5530Sstevel@tonic-gate 
5540Sstevel@tonic-gate 	if (med_db_is_inited)
5550Sstevel@tonic-gate 		return (0);
5560Sstevel@tonic-gate 
5570Sstevel@tonic-gate 	if (open_dbfile(medep))
5580Sstevel@tonic-gate 		return (-1);
5590Sstevel@tonic-gate 
5600Sstevel@tonic-gate 	if (fstat(dbfd, &statb) == -1)
5610Sstevel@tonic-gate 		return (med_error(medep, errno, "med_db_init: fstat()"));
5620Sstevel@tonic-gate 
5630Sstevel@tonic-gate 	/* Empty file */
5640Sstevel@tonic-gate 	if (statb.st_size == 0)
5650Sstevel@tonic-gate 		goto out;
5660Sstevel@tonic-gate 
5670Sstevel@tonic-gate 	/* File should be a multiple of the record size */
5680Sstevel@tonic-gate 	if (((int)(statb.st_size % (off_t)rec_size)) != 0)
5690Sstevel@tonic-gate 		return (med_error(medep, MDE_MED_DBSZBAD, "med_db_init"));
5700Sstevel@tonic-gate 
5710Sstevel@tonic-gate 	if (alloc_rec_buf(medep))
5720Sstevel@tonic-gate 		return (-1);
5730Sstevel@tonic-gate 
5740Sstevel@tonic-gate 	/* Read in the file header */
5750Sstevel@tonic-gate 	if ((ret = read(dbfd, rec_buf, rec_size)) == -1)
5760Sstevel@tonic-gate 		return (med_error(medep, errno, "med_db_init: read(hdr)"));
5770Sstevel@tonic-gate 
5780Sstevel@tonic-gate 	if (ret != rec_size)
5790Sstevel@tonic-gate 		return (med_error(medep, MDE_MED_DBHDRSZBAD, "med_db_init"));
5800Sstevel@tonic-gate 
5810Sstevel@tonic-gate 	/*LINTED*/
5820Sstevel@tonic-gate 	dbhp = (med_db_hdr_t *)rec_buf;
5830Sstevel@tonic-gate 
5840Sstevel@tonic-gate 	/* Header magic is not OK */
5850Sstevel@tonic-gate 	if (dbhp->med_dbh_mag != MED_DB_MAGIC)
5860Sstevel@tonic-gate 		return (med_error(medep, MDE_MED_DBHDRMAGBAD, "med_db_init"));
5870Sstevel@tonic-gate 
5880Sstevel@tonic-gate 	/* Header revision is not OK */
5890Sstevel@tonic-gate 	if (dbhp->med_dbh_rev != MED_DB_REV)
5900Sstevel@tonic-gate 		return (med_error(medep, MDE_MED_DBHDRREVBAD, "med_db_init"));
5910Sstevel@tonic-gate 
5920Sstevel@tonic-gate 	/* Header checksum is not OK */
5930Sstevel@tonic-gate 	if (crcchk(dbhp, &dbhp->med_dbh_cks, sizeof (med_db_hdr_t), NULL))
5940Sstevel@tonic-gate 		return (med_error(medep, MDE_MED_DBHDRCKSBAD, "med_db_init"));
5950Sstevel@tonic-gate 
5960Sstevel@tonic-gate 	/* File size does not add up */
5970Sstevel@tonic-gate 	if (((off_t)((dbhp->med_dbh_nm * rec_size) + rec_size))
5980Sstevel@tonic-gate 	    != statb.st_size)
5990Sstevel@tonic-gate 		return (med_error(medep, MDE_MED_DBSZBAD, "med_db_init"));
6000Sstevel@tonic-gate 
6010Sstevel@tonic-gate 	if ((nm = dbhp->med_dbh_nm) > 0) {
6020Sstevel@tonic-gate 		/* Allocate space to hold the records to be read next */
6030Sstevel@tonic-gate 		med_db_medrp = (med_rec_t *)Calloc(nm, sizeof (med_rec_t));
6040Sstevel@tonic-gate 		if (med_db_medrp == NULL)
6050Sstevel@tonic-gate 			return (med_error(medep, errno,
6060Sstevel@tonic-gate 			    "med_db_init: Calloc(med_db_medrp)"));
6070Sstevel@tonic-gate 	}
6080Sstevel@tonic-gate 
6090Sstevel@tonic-gate 	/* Read in all the records */
6100Sstevel@tonic-gate 	for (i = 0; i < nm; i++) {
6110Sstevel@tonic-gate 		if ((cur_off = lseek(dbfd, 0, SEEK_CUR)) == -1) {
6120Sstevel@tonic-gate 			err = med_error(medep, errno,
6130Sstevel@tonic-gate 			    "med_db_init: lseek()");
6140Sstevel@tonic-gate 			goto out;
6150Sstevel@tonic-gate 		}
6160Sstevel@tonic-gate 
6170Sstevel@tonic-gate 		(void) memset(rec_buf, '\0', rec_size);
6180Sstevel@tonic-gate 
6190Sstevel@tonic-gate 		if ((ret = read(dbfd, rec_buf, rec_size)) == -1) {
6200Sstevel@tonic-gate 			err = med_error(medep, errno,
6210Sstevel@tonic-gate 			    "med_db_init: read() rec");
6220Sstevel@tonic-gate 			goto out;
6230Sstevel@tonic-gate 		}
6240Sstevel@tonic-gate 
6250Sstevel@tonic-gate 		if (ret != rec_size) {
6260Sstevel@tonic-gate 			err = med_error(medep, MDE_MED_DBRECSZBAD,
6270Sstevel@tonic-gate 			    "med_db_init");
6280Sstevel@tonic-gate 			goto out;
6290Sstevel@tonic-gate 		}
6300Sstevel@tonic-gate 
6310Sstevel@tonic-gate 		/*LINTED*/
6320Sstevel@tonic-gate 		medrp = (med_rec_t *)rec_buf;
6330Sstevel@tonic-gate 
6340Sstevel@tonic-gate 		/* Record magic is not OK */
6350Sstevel@tonic-gate 		if (medrp->med_rec_mag != MED_REC_MAGIC) {
6360Sstevel@tonic-gate 			err = med_error(medep, MDE_MED_DBRECMAGBAD,
6370Sstevel@tonic-gate 			    "med_db_init");
6380Sstevel@tonic-gate 			goto out;
6390Sstevel@tonic-gate 		}
6400Sstevel@tonic-gate 
6410Sstevel@tonic-gate 		/* Record revision is not OK */
6420Sstevel@tonic-gate 		if (medrp->med_rec_rev != MED_REC_REV) {
6430Sstevel@tonic-gate 			err = med_error(medep, MDE_MED_DBRECREVBAD,
6440Sstevel@tonic-gate 			    "med_db_init");
6450Sstevel@tonic-gate 			goto out;
6460Sstevel@tonic-gate 		}
6470Sstevel@tonic-gate 
6480Sstevel@tonic-gate 		/* Record checksum is not OK */
6490Sstevel@tonic-gate 		ret = crcchk(medrp, &medrp->med_rec_cks, sizeof (med_rec_t),
6500Sstevel@tonic-gate 		    NULL);
6510Sstevel@tonic-gate 		if (ret) {
6520Sstevel@tonic-gate 			err = med_error(medep, MDE_MED_DBRECCKSBAD,
6530Sstevel@tonic-gate 			    "med_db_init");
6540Sstevel@tonic-gate 			goto out;
6550Sstevel@tonic-gate 		}
6560Sstevel@tonic-gate 
6570Sstevel@tonic-gate 		/* Record is not where it is supposed to be */
6580Sstevel@tonic-gate 		if (medrp->med_rec_foff != cur_off) {
6590Sstevel@tonic-gate 			err = med_error(medep, MDE_MED_DBRECOFFBAD,
6600Sstevel@tonic-gate 			    "med_db_init");
6610Sstevel@tonic-gate 			goto out;
6620Sstevel@tonic-gate 		}
6630Sstevel@tonic-gate 
6640Sstevel@tonic-gate 		med_db_medrp[i] = *medrp;	/* structure assignment */
6650Sstevel@tonic-gate 	}
6660Sstevel@tonic-gate 
6670Sstevel@tonic-gate 	/* Add the keys to access this record */
6680Sstevel@tonic-gate 	for (i = 0; i < nm; i++)
6690Sstevel@tonic-gate 		if ((err = add_db_keys(i, medep)) == -1)
6700Sstevel@tonic-gate 			goto out;
6710Sstevel@tonic-gate 
6720Sstevel@tonic-gate 	med_db_nma = nm;
6730Sstevel@tonic-gate 	med_db_nmu = nm;
6740Sstevel@tonic-gate 
6750Sstevel@tonic-gate out:
6760Sstevel@tonic-gate 	if (err && med_db_medrp != NULL)
6770Sstevel@tonic-gate 		Free(med_db_medrp);
6780Sstevel@tonic-gate 
6790Sstevel@tonic-gate 	if (!err)
6800Sstevel@tonic-gate 		med_db_is_inited = 1;
6810Sstevel@tonic-gate 
6820Sstevel@tonic-gate 	return (err);
6830Sstevel@tonic-gate }
6840Sstevel@tonic-gate 
6850Sstevel@tonic-gate med_rec_t *
med_db_get_rec(med_med_t * medp,med_err_t * medep)6860Sstevel@tonic-gate med_db_get_rec(med_med_t *medp, med_err_t *medep)
6870Sstevel@tonic-gate {
6880Sstevel@tonic-gate 	int		medridx = -1;
6890Sstevel@tonic-gate 
6900Sstevel@tonic-gate 	if ((medridx = find_key(medp)) == -1) {
6910Sstevel@tonic-gate 		(void) med_error(medep, MDE_MED_DBRECNOENT, "med_db_get_rec");
6920Sstevel@tonic-gate 		return (NULL);
6930Sstevel@tonic-gate 	}
6940Sstevel@tonic-gate 
6950Sstevel@tonic-gate 	return (&med_db_medrp[medridx]);
6960Sstevel@tonic-gate }
6970Sstevel@tonic-gate 
6980Sstevel@tonic-gate med_data_t *
med_db_get_data(med_med_t * medp,med_err_t * medep)6990Sstevel@tonic-gate med_db_get_data(med_med_t *medp, med_err_t *medep)
7000Sstevel@tonic-gate {
7010Sstevel@tonic-gate 	int		medridx = -1;
7020Sstevel@tonic-gate 
7030Sstevel@tonic-gate 	if ((medridx = find_key(medp)) == -1) {
7040Sstevel@tonic-gate 		(void) med_error(medep, MDE_MED_DBRECNOENT, "med_db_get_data");
7050Sstevel@tonic-gate 		return (NULL);
7060Sstevel@tonic-gate 	}
7070Sstevel@tonic-gate 
7080Sstevel@tonic-gate 	return (&med_db_medrp[medridx].med_rec_data);
7090Sstevel@tonic-gate }
7100Sstevel@tonic-gate 
7110Sstevel@tonic-gate int
med_db_put_rec(med_med_t * medp,med_rec_t * nmedrp,med_err_t * medep)7120Sstevel@tonic-gate med_db_put_rec(med_med_t *medp, med_rec_t *nmedrp, med_err_t *medep)
7130Sstevel@tonic-gate {
7140Sstevel@tonic-gate 	med_rec_t	*medrp = NULL;
7150Sstevel@tonic-gate 	med_rec_t	*tmedrp = NULL;
7160Sstevel@tonic-gate 	int		i;
7170Sstevel@tonic-gate 	int		found = 0;
7180Sstevel@tonic-gate 	int		medridx = -1;
7190Sstevel@tonic-gate 
7200Sstevel@tonic-gate 
7210Sstevel@tonic-gate 	if (! med_db_is_inited)
7220Sstevel@tonic-gate 		return (med_error(medep, MDE_MED_DBNOTINIT, "med_db_put_rec"));
7230Sstevel@tonic-gate 
7240Sstevel@tonic-gate 	if (medp->med_setno != nmedrp->med_rec_sn)
7250Sstevel@tonic-gate 		return (med_error(medep, MDE_MED_DBARGSMISMATCH,
7260Sstevel@tonic-gate 		    "med_db_put_rec"));
7270Sstevel@tonic-gate 
7280Sstevel@tonic-gate 	/* See if we are still considered a mediator - is this a delete? */
7290Sstevel@tonic-gate 	for (i = 0; i < MED_MAX_HOSTS; i++) {
7300Sstevel@tonic-gate 		if (nmedrp->med_rec_meds.n_lst[i].a_cnt == 0)
7310Sstevel@tonic-gate 			continue;
7320Sstevel@tonic-gate 
7330Sstevel@tonic-gate 		if (strcmp(nmedrp->med_rec_meds.n_lst[i].a_nm[0],
7340Sstevel@tonic-gate 		    mynode()) == 0) {
7350Sstevel@tonic-gate 			found = 1;
7360Sstevel@tonic-gate 			break;
7370Sstevel@tonic-gate 		}
7380Sstevel@tonic-gate 	}
7390Sstevel@tonic-gate 
7400Sstevel@tonic-gate 	/* If it is a delete, make it happen */
7410Sstevel@tonic-gate 	if (! found)
7420Sstevel@tonic-gate 		return (med_db_del_rec(medp, medep));
7430Sstevel@tonic-gate 
7440Sstevel@tonic-gate 	/* See if there is an existing record */
7450Sstevel@tonic-gate 	if ((medridx = find_key(medp)) != -1) {
7460Sstevel@tonic-gate 
7470Sstevel@tonic-gate 		medrp = &med_db_medrp[medridx];
7480Sstevel@tonic-gate 
7490Sstevel@tonic-gate 		/* Delete the old keys */
7500Sstevel@tonic-gate 		if (del_db_keys(medridx, medep))
7510Sstevel@tonic-gate 			return (-1);
7520Sstevel@tonic-gate 
7530Sstevel@tonic-gate 		/* Decrement the used slot count */
7540Sstevel@tonic-gate 		med_db_nmu--;
7550Sstevel@tonic-gate 	} else {
7560Sstevel@tonic-gate 		for (i = 0; i < MED_MAX_HOSTS; i++) {
7570Sstevel@tonic-gate 			med_med_t	tmed;
7580Sstevel@tonic-gate 
7590Sstevel@tonic-gate 			if (nmedrp->med_rec_meds.n_lst[i].a_cnt == 0)
7600Sstevel@tonic-gate 				continue;
7610Sstevel@tonic-gate 
7620Sstevel@tonic-gate 			if (strcmp(nmedrp->med_rec_meds.n_lst[i].a_nm[0],
7630Sstevel@tonic-gate 			    medp->med_caller) == 0)
7640Sstevel@tonic-gate 				continue;
7650Sstevel@tonic-gate 
7660Sstevel@tonic-gate 			tmed = *medp;		/* structure assignment */
7670Sstevel@tonic-gate 
7680Sstevel@tonic-gate 			tmed.med_caller =
7690Sstevel@tonic-gate 			    Strdup(nmedrp->med_rec_meds.n_lst[i].a_nm[0]);
7700Sstevel@tonic-gate 
7710Sstevel@tonic-gate 			medridx = find_key(&tmed);
7720Sstevel@tonic-gate 
7730Sstevel@tonic-gate 			Free(tmed.med_caller);
7740Sstevel@tonic-gate 
7750Sstevel@tonic-gate 			if (medridx != -1) {
7760Sstevel@tonic-gate 				medrp = &med_db_medrp[medridx];
7770Sstevel@tonic-gate 
7780Sstevel@tonic-gate 				if (cmp_medrec(medrp, nmedrp))
7790Sstevel@tonic-gate 					return (0);
7800Sstevel@tonic-gate 			}
7810Sstevel@tonic-gate 		}
7820Sstevel@tonic-gate 	}
7830Sstevel@tonic-gate 
7840Sstevel@tonic-gate 	/* Allocate more space if needed */
7850Sstevel@tonic-gate 	if ((med_db_nmu + 1) > med_db_nma) {
7860Sstevel@tonic-gate 
7870Sstevel@tonic-gate 		/* Allocate more space to hold the new record */
7880Sstevel@tonic-gate 		tmedrp = (med_rec_t *)Calloc((med_db_nmu + 1),
7890Sstevel@tonic-gate 		    sizeof (med_rec_t));
7900Sstevel@tonic-gate 		if (tmedrp == NULL)
7910Sstevel@tonic-gate 			return (med_error(medep, errno,
7920Sstevel@tonic-gate 			    "med_db_put_rec: Re-Calloc(tmedrp)"));
7930Sstevel@tonic-gate 
7940Sstevel@tonic-gate 		/* Copy the existing information into the new area */
7950Sstevel@tonic-gate 		for (i = 0; i < med_db_nma; i++)
7960Sstevel@tonic-gate 			tmedrp[i] = med_db_medrp[i]; /* structure assignment */
7970Sstevel@tonic-gate 
7980Sstevel@tonic-gate 		med_db_nmu++;
7990Sstevel@tonic-gate 		med_db_nma = med_db_nmu;
8000Sstevel@tonic-gate 
8010Sstevel@tonic-gate 		if (med_db_medrp)
8020Sstevel@tonic-gate 			Free(med_db_medrp);
8030Sstevel@tonic-gate 
8040Sstevel@tonic-gate 		med_db_medrp = tmedrp;
8050Sstevel@tonic-gate 
8060Sstevel@tonic-gate 		medridx = med_db_nma - 1;
8070Sstevel@tonic-gate 
8080Sstevel@tonic-gate 		/* Initialize */
8090Sstevel@tonic-gate 		medrp = &med_db_medrp[medridx];
8100Sstevel@tonic-gate 		medrp->med_rec_mag = MED_REC_MAGIC;
8110Sstevel@tonic-gate 		medrp->med_rec_rev = MED_REC_REV;
8120Sstevel@tonic-gate 		medrp->med_rec_sn = nmedrp->med_rec_sn;
8130Sstevel@tonic-gate 		(void) strcpy(medrp->med_rec_snm, nmedrp->med_rec_snm);
8140Sstevel@tonic-gate 
8150Sstevel@tonic-gate 		/* Calculate the record offset */
8160Sstevel@tonic-gate 		medrp->med_rec_foff = (off_t)(((med_db_nma - 1) * rec_size) +
8170Sstevel@tonic-gate 		    rec_size);
8180Sstevel@tonic-gate 	} else {
8190Sstevel@tonic-gate 		/*
8200Sstevel@tonic-gate 		 * We did not find the record, but have space allocated.
8210Sstevel@tonic-gate 		 * Find an empty slot.
8220Sstevel@tonic-gate 		 */
8230Sstevel@tonic-gate 		if (medrp == NULL) {
8240Sstevel@tonic-gate 			for (i = 0; i < med_db_nma; i++) {
8250Sstevel@tonic-gate 				medrp = &med_db_medrp[i];
8260Sstevel@tonic-gate 
8270Sstevel@tonic-gate 				if (! (medrp->med_rec_fl & MED_RFL_DEL))
8280Sstevel@tonic-gate 					continue;
8290Sstevel@tonic-gate 
8300Sstevel@tonic-gate 				medridx = i;
8310Sstevel@tonic-gate 
8320Sstevel@tonic-gate 				/* Mark as no longer deleted */
8330Sstevel@tonic-gate 				medrp->med_rec_fl &= ~MED_RFL_DEL;
8340Sstevel@tonic-gate 
8350Sstevel@tonic-gate 				/* Initialize */
8360Sstevel@tonic-gate 				medrp->med_rec_mag = MED_REC_MAGIC;
8370Sstevel@tonic-gate 				medrp->med_rec_rev = MED_REC_REV;
8380Sstevel@tonic-gate 				medrp->med_rec_sn = nmedrp->med_rec_sn;
8390Sstevel@tonic-gate 				(void) strcpy(medrp->med_rec_snm,
8400Sstevel@tonic-gate 				    nmedrp->med_rec_snm);
8410Sstevel@tonic-gate 
8420Sstevel@tonic-gate 				/* Calculate the new offset of the record */
8430Sstevel@tonic-gate 				medrp->med_rec_foff = (off_t)
8440Sstevel@tonic-gate 				    ((med_db_nmu * rec_size) + rec_size);
8450Sstevel@tonic-gate 
8460Sstevel@tonic-gate 				/* Clear the old data */
8470Sstevel@tonic-gate 				(void) memset(&medrp->med_rec_data, '\0',
8480Sstevel@tonic-gate 				    sizeof (med_data_t));
8490Sstevel@tonic-gate 
8500Sstevel@tonic-gate 				break;
8510Sstevel@tonic-gate 			}
8520Sstevel@tonic-gate 		}
8530Sstevel@tonic-gate 		med_db_nmu++;
8540Sstevel@tonic-gate 	}
8550Sstevel@tonic-gate 
8560Sstevel@tonic-gate 	assert(medridx != -1);
8570Sstevel@tonic-gate 
8580Sstevel@tonic-gate 	/* Update the record with the new information */
8590Sstevel@tonic-gate 	medrp->med_rec_meds = nmedrp->med_rec_meds;  /* structure assignment */
8600Sstevel@tonic-gate 
8610Sstevel@tonic-gate 	for (i = 0; i < MD_MAXSIDES; i++)
8620Sstevel@tonic-gate 		(void) strcpy(medrp->med_rec_nodes[i],
8630Sstevel@tonic-gate 		    nmedrp->med_rec_nodes[i]);
8640Sstevel@tonic-gate 
8650Sstevel@tonic-gate 	if (write_hdr(dbfd, medep))
8660Sstevel@tonic-gate 		return (-1);
8670Sstevel@tonic-gate 
8680Sstevel@tonic-gate 	/* Position to record location */
8690Sstevel@tonic-gate 	if (lseek(dbfd, medrp->med_rec_foff, SEEK_SET) == -1)
8700Sstevel@tonic-gate 		return (med_error(medep, errno, "med_db_put_rec: lseek(rec)"));
8710Sstevel@tonic-gate 
8720Sstevel@tonic-gate 	if (write_rec(dbfd, medrp, medep))
8730Sstevel@tonic-gate 		return (-1);
8740Sstevel@tonic-gate 
8750Sstevel@tonic-gate 	/* Add the keys for this record */
8760Sstevel@tonic-gate 	if (add_db_keys(medridx, medep))
8770Sstevel@tonic-gate 		return (-1);
8780Sstevel@tonic-gate 
8790Sstevel@tonic-gate 	return (0);
8800Sstevel@tonic-gate }
8810Sstevel@tonic-gate 
8820Sstevel@tonic-gate int
med_db_put_data(med_med_t * medp,med_data_t * meddp,med_err_t * medep)8830Sstevel@tonic-gate med_db_put_data(med_med_t *medp, med_data_t *meddp, med_err_t *medep)
8840Sstevel@tonic-gate {
8850Sstevel@tonic-gate 	med_rec_t	*medrp = NULL;
8860Sstevel@tonic-gate 	int		medridx = -1;
8870Sstevel@tonic-gate 
8880Sstevel@tonic-gate 
8890Sstevel@tonic-gate 	if (! med_db_is_inited)
890*8979SRameshkumar.Ramasamy@Sun.COM 		return (med_error(medep, MDE_MED_DBNOTINIT, "med_db_put_data"));
8910Sstevel@tonic-gate 
8920Sstevel@tonic-gate 	if (medp->med_setno != meddp->med_dat_sn)
8930Sstevel@tonic-gate 		return (med_error(medep, MDE_MED_DBARGSMISMATCH,
8940Sstevel@tonic-gate 		    "med_db_put_data"));
8950Sstevel@tonic-gate 
8960Sstevel@tonic-gate 	if ((medridx = find_key(medp)) == -1)
8970Sstevel@tonic-gate 		return (med_error(medep, MDE_MED_DBRECNOENT,
8980Sstevel@tonic-gate 		    "med_db_put_data"));
8990Sstevel@tonic-gate 
9000Sstevel@tonic-gate 	medrp = &med_db_medrp[medridx];
9010Sstevel@tonic-gate 
9020Sstevel@tonic-gate 	medrp->med_rec_data = *meddp;		/* structure assignment */
9030Sstevel@tonic-gate 
9040Sstevel@tonic-gate 	/* Go to location of the record */
9050Sstevel@tonic-gate 	if (lseek(dbfd, medrp->med_rec_foff, SEEK_SET) == -1)
9060Sstevel@tonic-gate 		return (med_error(medep, errno, "med_db_put_data: lseek()"));
9070Sstevel@tonic-gate 
9080Sstevel@tonic-gate 	if (write_rec(dbfd, medrp, medep))
9090Sstevel@tonic-gate 		return (-1);
9100Sstevel@tonic-gate 
9110Sstevel@tonic-gate 	return (0);
9120Sstevel@tonic-gate }
9130Sstevel@tonic-gate 
9140Sstevel@tonic-gate int
med_db_finit(med_err_t * medep)9150Sstevel@tonic-gate med_db_finit(med_err_t *medep)
9160Sstevel@tonic-gate {
9170Sstevel@tonic-gate 	des_cache(&med_db_cache);
9180Sstevel@tonic-gate 	Free(med_db_medrp);
9190Sstevel@tonic-gate 	free_rec_buf();
9200Sstevel@tonic-gate 	if (close_dbfile(medep))
9210Sstevel@tonic-gate 		return (-1);
9220Sstevel@tonic-gate 	return (0);
9230Sstevel@tonic-gate }
924