1248aee62SJacques Vidrine /* $NetBSD: nsdispatch.c,v 1.9 1999/01/25 00:16:17 lukem Exp $ */ 2248aee62SJacques Vidrine 3248aee62SJacques Vidrine /*- 4248aee62SJacques Vidrine * Copyright (c) 1997, 1998, 1999 The NetBSD Foundation, Inc. 5248aee62SJacques Vidrine * All rights reserved. 6248aee62SJacques Vidrine * 7248aee62SJacques Vidrine * This code is derived from software contributed to The NetBSD Foundation 8248aee62SJacques Vidrine * by Luke Mewburn. 9248aee62SJacques Vidrine * 10248aee62SJacques Vidrine * Redistribution and use in source and binary forms, with or without 11248aee62SJacques Vidrine * modification, are permitted provided that the following conditions 12248aee62SJacques Vidrine * are met: 13248aee62SJacques Vidrine * 1. Redistributions of source code must retain the above copyright 14248aee62SJacques Vidrine * notice, this list of conditions and the following disclaimer. 15248aee62SJacques Vidrine * 2. Redistributions in binary form must reproduce the above copyright 16248aee62SJacques Vidrine * notice, this list of conditions and the following disclaimer in the 17248aee62SJacques Vidrine * documentation and/or other materials provided with the distribution. 18248aee62SJacques Vidrine * 3. All advertising materials mentioning features or use of this software 19248aee62SJacques Vidrine * must display the following acknowledgement: 20248aee62SJacques Vidrine * This product includes software developed by the NetBSD 21248aee62SJacques Vidrine * Foundation, Inc. and its contributors. 22248aee62SJacques Vidrine * 4. Neither the name of The NetBSD Foundation nor the names of its 23248aee62SJacques Vidrine * contributors may be used to endorse or promote products derived 24248aee62SJacques Vidrine * from this software without specific prior written permission. 25248aee62SJacques Vidrine * 26248aee62SJacques Vidrine * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 27248aee62SJacques Vidrine * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 28248aee62SJacques Vidrine * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 29248aee62SJacques Vidrine * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 30248aee62SJacques Vidrine * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 31248aee62SJacques Vidrine * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 32248aee62SJacques Vidrine * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 33248aee62SJacques Vidrine * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 34248aee62SJacques Vidrine * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 35248aee62SJacques Vidrine * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 36248aee62SJacques Vidrine * POSSIBILITY OF SUCH DAMAGE. 37248aee62SJacques Vidrine */ 38248aee62SJacques Vidrine 39248aee62SJacques Vidrine #include <sys/cdefs.h> 40248aee62SJacques Vidrine #if defined(LIBC_SCCS) && !defined(lint) 41248aee62SJacques Vidrine static char *rcsid = 42248aee62SJacques Vidrine "$FreeBSD$"; 43248aee62SJacques Vidrine #endif /* LIBC_SCCS and not lint */ 44248aee62SJacques Vidrine 45248aee62SJacques Vidrine #include <sys/types.h> 46248aee62SJacques Vidrine #include <sys/param.h> 47248aee62SJacques Vidrine #include <sys/stat.h> 48248aee62SJacques Vidrine 49248aee62SJacques Vidrine #include <err.h> 50248aee62SJacques Vidrine #include <fcntl.h> 51248aee62SJacques Vidrine #define _NS_PRIVATE 52248aee62SJacques Vidrine #include <nsswitch.h> 53248aee62SJacques Vidrine #include <stdio.h> 54248aee62SJacques Vidrine #include <stdlib.h> 55248aee62SJacques Vidrine #include <string.h> 56248aee62SJacques Vidrine #include <unistd.h> 57248aee62SJacques Vidrine 58248aee62SJacques Vidrine /* 59248aee62SJacques Vidrine * default sourcelist: `files' 60248aee62SJacques Vidrine */ 61248aee62SJacques Vidrine const ns_src __nsdefaultsrc[] = { 62248aee62SJacques Vidrine { NSSRC_FILES, NS_SUCCESS }, 63248aee62SJacques Vidrine { 0 }, 64248aee62SJacques Vidrine }; 65248aee62SJacques Vidrine 66248aee62SJacques Vidrine 67248aee62SJacques Vidrine static int _nsmapsize = 0; 68248aee62SJacques Vidrine static ns_dbt *_nsmap = NULL; 69248aee62SJacques Vidrine 70248aee62SJacques Vidrine /* 71248aee62SJacques Vidrine * size of dynamic array chunk for _nsmap and _nsmap[x].srclist 72248aee62SJacques Vidrine */ 73248aee62SJacques Vidrine #define NSELEMSPERCHUNK 8 74248aee62SJacques Vidrine 75248aee62SJacques Vidrine 76248aee62SJacques Vidrine int _nscmp __P((const void *, const void *)); 77248aee62SJacques Vidrine 78248aee62SJacques Vidrine 79248aee62SJacques Vidrine int 80248aee62SJacques Vidrine _nscmp(a, b) 81248aee62SJacques Vidrine const void *a; 82248aee62SJacques Vidrine const void *b; 83248aee62SJacques Vidrine { 84248aee62SJacques Vidrine return (strcasecmp(((const ns_dbt *)a)->name, 85248aee62SJacques Vidrine ((const ns_dbt *)b)->name)); 86248aee62SJacques Vidrine } 87248aee62SJacques Vidrine 88248aee62SJacques Vidrine 89248aee62SJacques Vidrine void 90248aee62SJacques Vidrine _nsdbtaddsrc(dbt, src) 91248aee62SJacques Vidrine ns_dbt *dbt; 92248aee62SJacques Vidrine const ns_src *src; 93248aee62SJacques Vidrine { 94248aee62SJacques Vidrine if ((dbt->srclistsize % NSELEMSPERCHUNK) == 0) { 95248aee62SJacques Vidrine dbt->srclist = (ns_src *)realloc(dbt->srclist, 96248aee62SJacques Vidrine (dbt->srclistsize + NSELEMSPERCHUNK) * sizeof(ns_src)); 97248aee62SJacques Vidrine if (dbt->srclist == NULL) 98248aee62SJacques Vidrine err(1, "nsdispatch: memory allocation failure"); 99248aee62SJacques Vidrine } 100248aee62SJacques Vidrine memmove(&dbt->srclist[dbt->srclistsize++], src, sizeof(ns_src)); 101248aee62SJacques Vidrine } 102248aee62SJacques Vidrine 103248aee62SJacques Vidrine 104248aee62SJacques Vidrine void 105248aee62SJacques Vidrine _nsdbtdump(dbt) 106248aee62SJacques Vidrine const ns_dbt *dbt; 107248aee62SJacques Vidrine { 108248aee62SJacques Vidrine int i; 109248aee62SJacques Vidrine 110248aee62SJacques Vidrine printf("%s (%d source%s):", dbt->name, dbt->srclistsize, 111248aee62SJacques Vidrine dbt->srclistsize == 1 ? "" : "s"); 112248aee62SJacques Vidrine for (i = 0; i < dbt->srclistsize; i++) { 113248aee62SJacques Vidrine printf(" %s", dbt->srclist[i].name); 114248aee62SJacques Vidrine if (!(dbt->srclist[i].flags & 115248aee62SJacques Vidrine (NS_UNAVAIL|NS_NOTFOUND|NS_TRYAGAIN)) && 116248aee62SJacques Vidrine (dbt->srclist[i].flags & NS_SUCCESS)) 117248aee62SJacques Vidrine continue; 118248aee62SJacques Vidrine printf(" ["); 119248aee62SJacques Vidrine if (!(dbt->srclist[i].flags & NS_SUCCESS)) 120248aee62SJacques Vidrine printf(" SUCCESS=continue"); 121248aee62SJacques Vidrine if (dbt->srclist[i].flags & NS_UNAVAIL) 122248aee62SJacques Vidrine printf(" UNAVAIL=return"); 123248aee62SJacques Vidrine if (dbt->srclist[i].flags & NS_NOTFOUND) 124248aee62SJacques Vidrine printf(" NOTFOUND=return"); 125248aee62SJacques Vidrine if (dbt->srclist[i].flags & NS_TRYAGAIN) 126248aee62SJacques Vidrine printf(" TRYAGAIN=return"); 127248aee62SJacques Vidrine printf(" ]"); 128248aee62SJacques Vidrine } 129248aee62SJacques Vidrine printf("\n"); 130248aee62SJacques Vidrine } 131248aee62SJacques Vidrine 132248aee62SJacques Vidrine 133248aee62SJacques Vidrine const ns_dbt * 134248aee62SJacques Vidrine _nsdbtget(name) 135248aee62SJacques Vidrine const char *name; 136248aee62SJacques Vidrine { 137248aee62SJacques Vidrine static time_t confmod; 138248aee62SJacques Vidrine 139248aee62SJacques Vidrine struct stat statbuf; 140248aee62SJacques Vidrine ns_dbt dbt; 141248aee62SJacques Vidrine 142248aee62SJacques Vidrine extern FILE *_nsyyin; 143248aee62SJacques Vidrine extern int _nsyyparse __P((void)); 144248aee62SJacques Vidrine 145248aee62SJacques Vidrine dbt.name = name; 146248aee62SJacques Vidrine 147248aee62SJacques Vidrine if (confmod) { 148248aee62SJacques Vidrine if (stat(_PATH_NS_CONF, &statbuf) == -1) 149248aee62SJacques Vidrine return (NULL); 150248aee62SJacques Vidrine if (confmod < statbuf.st_mtime) { 151248aee62SJacques Vidrine int i, j; 152248aee62SJacques Vidrine 153248aee62SJacques Vidrine for (i = 0; i < _nsmapsize; i++) { 154248aee62SJacques Vidrine for (j = 0; j < _nsmap[i].srclistsize; j++) { 155248aee62SJacques Vidrine if (_nsmap[i].srclist[j].name != NULL) { 156248aee62SJacques Vidrine /*LINTED const cast*/ 157248aee62SJacques Vidrine free((void *) 158248aee62SJacques Vidrine _nsmap[i].srclist[j].name); 159248aee62SJacques Vidrine } 160248aee62SJacques Vidrine } 161248aee62SJacques Vidrine if (_nsmap[i].srclist) 162248aee62SJacques Vidrine free(_nsmap[i].srclist); 163248aee62SJacques Vidrine if (_nsmap[i].name) { 164248aee62SJacques Vidrine /*LINTED const cast*/ 165248aee62SJacques Vidrine free((void *)_nsmap[i].name); 166248aee62SJacques Vidrine } 167248aee62SJacques Vidrine } 168248aee62SJacques Vidrine if (_nsmap) 169248aee62SJacques Vidrine free(_nsmap); 170248aee62SJacques Vidrine _nsmap = NULL; 171248aee62SJacques Vidrine _nsmapsize = 0; 172248aee62SJacques Vidrine confmod = 0; 173248aee62SJacques Vidrine } 174248aee62SJacques Vidrine } 175248aee62SJacques Vidrine if (!confmod) { 176248aee62SJacques Vidrine if (stat(_PATH_NS_CONF, &statbuf) == -1) 177248aee62SJacques Vidrine return (NULL); 178248aee62SJacques Vidrine _nsyyin = fopen(_PATH_NS_CONF, "r"); 179248aee62SJacques Vidrine if (_nsyyin == NULL) 180248aee62SJacques Vidrine return (NULL); 181248aee62SJacques Vidrine _nsyyparse(); 182248aee62SJacques Vidrine (void)fclose(_nsyyin); 183248aee62SJacques Vidrine qsort(_nsmap, (size_t)_nsmapsize, sizeof(ns_dbt), _nscmp); 184248aee62SJacques Vidrine confmod = statbuf.st_mtime; 185248aee62SJacques Vidrine } 186248aee62SJacques Vidrine return (bsearch(&dbt, _nsmap, (size_t)_nsmapsize, sizeof(ns_dbt), 187248aee62SJacques Vidrine _nscmp)); 188248aee62SJacques Vidrine } 189248aee62SJacques Vidrine 190248aee62SJacques Vidrine 191248aee62SJacques Vidrine void 192248aee62SJacques Vidrine _nsdbtput(dbt) 193248aee62SJacques Vidrine const ns_dbt *dbt; 194248aee62SJacques Vidrine { 195248aee62SJacques Vidrine int i; 196248aee62SJacques Vidrine 197248aee62SJacques Vidrine for (i = 0; i < _nsmapsize; i++) { 198248aee62SJacques Vidrine if (_nscmp(dbt, &_nsmap[i]) == 0) { 199248aee62SJacques Vidrine /* overwrite existing entry */ 200248aee62SJacques Vidrine if (_nsmap[i].srclist != NULL) 201248aee62SJacques Vidrine free(_nsmap[i].srclist); 202248aee62SJacques Vidrine memmove(&_nsmap[i], dbt, sizeof(ns_dbt)); 203248aee62SJacques Vidrine return; 204248aee62SJacques Vidrine } 205248aee62SJacques Vidrine } 206248aee62SJacques Vidrine 207248aee62SJacques Vidrine if ((_nsmapsize % NSELEMSPERCHUNK) == 0) { 208248aee62SJacques Vidrine _nsmap = (ns_dbt *)realloc(_nsmap, 209248aee62SJacques Vidrine (_nsmapsize + NSELEMSPERCHUNK) * sizeof(ns_dbt)); 210248aee62SJacques Vidrine if (_nsmap == NULL) 211248aee62SJacques Vidrine err(1, "nsdispatch: memory allocation failure"); 212248aee62SJacques Vidrine } 213248aee62SJacques Vidrine memmove(&_nsmap[_nsmapsize++], dbt, sizeof(ns_dbt)); 214248aee62SJacques Vidrine } 215248aee62SJacques Vidrine 216248aee62SJacques Vidrine 217248aee62SJacques Vidrine int 218248aee62SJacques Vidrine #if __STDC__ 219248aee62SJacques Vidrine nsdispatch(void *retval, const ns_dtab disp_tab[], const char *database, 220248aee62SJacques Vidrine const char *method, const ns_src defaults[], ...) 221248aee62SJacques Vidrine #else 222248aee62SJacques Vidrine nsdispatch(retval, disp_tab, database, method, defaults, va_alist) 223248aee62SJacques Vidrine void *retval; 224248aee62SJacques Vidrine const ns_dtab disp_tab[]; 225248aee62SJacques Vidrine const char *database; 226248aee62SJacques Vidrine const char *method; 227248aee62SJacques Vidrine const ns_src defaults[]; 228248aee62SJacques Vidrine va_dcl 229248aee62SJacques Vidrine #endif 230248aee62SJacques Vidrine { 231248aee62SJacques Vidrine va_list ap; 232248aee62SJacques Vidrine int i, curdisp, result; 233248aee62SJacques Vidrine const ns_dbt *dbt; 234248aee62SJacques Vidrine const ns_src *srclist; 235248aee62SJacques Vidrine int srclistsize; 236248aee62SJacques Vidrine 237248aee62SJacques Vidrine dbt = _nsdbtget(database); 238248aee62SJacques Vidrine if (dbt != NULL) { 239248aee62SJacques Vidrine srclist = dbt->srclist; 240248aee62SJacques Vidrine srclistsize = dbt->srclistsize; 241248aee62SJacques Vidrine } else { 242248aee62SJacques Vidrine srclist = defaults; 243248aee62SJacques Vidrine srclistsize = 0; 244248aee62SJacques Vidrine while (srclist[srclistsize].name != NULL) 245248aee62SJacques Vidrine srclistsize++; 246248aee62SJacques Vidrine } 247248aee62SJacques Vidrine result = 0; 248248aee62SJacques Vidrine 249248aee62SJacques Vidrine for (i = 0; i < srclistsize; i++) { 250248aee62SJacques Vidrine for (curdisp = 0; disp_tab[curdisp].src != NULL; curdisp++) 251248aee62SJacques Vidrine if (strcasecmp(disp_tab[curdisp].src, 252248aee62SJacques Vidrine srclist[i].name) == 0) 253248aee62SJacques Vidrine break; 254248aee62SJacques Vidrine result = 0; 255248aee62SJacques Vidrine if (disp_tab[curdisp].callback) { 256248aee62SJacques Vidrine #if __STDC__ 257248aee62SJacques Vidrine va_start(ap, defaults); 258248aee62SJacques Vidrine #else 259248aee62SJacques Vidrine va_start(ap); 260248aee62SJacques Vidrine #endif 261248aee62SJacques Vidrine result = disp_tab[curdisp].callback(retval, 262248aee62SJacques Vidrine disp_tab[curdisp].cb_data, ap); 263248aee62SJacques Vidrine va_end(ap); 264248aee62SJacques Vidrine if (result & srclist[i].flags) { 265248aee62SJacques Vidrine break; 266248aee62SJacques Vidrine } 267248aee62SJacques Vidrine } 268248aee62SJacques Vidrine } 269248aee62SJacques Vidrine return (result ? result : NS_NOTFOUND); 270248aee62SJacques Vidrine } 271