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