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> 40333fc21eSDavid E. O'Brien __FBSDID("$FreeBSD$"); 41248aee62SJacques Vidrine 42248aee62SJacques Vidrine #include <sys/types.h> 43248aee62SJacques Vidrine #include <sys/param.h> 44248aee62SJacques Vidrine #include <sys/stat.h> 45248aee62SJacques Vidrine 462bbd7cf8SJacques Vidrine #include "namespace.h" 47248aee62SJacques Vidrine #include <err.h> 48248aee62SJacques Vidrine #include <fcntl.h> 49248aee62SJacques Vidrine #define _NS_PRIVATE 50248aee62SJacques Vidrine #include <nsswitch.h> 51248aee62SJacques Vidrine #include <stdio.h> 52248aee62SJacques Vidrine #include <stdlib.h> 53248aee62SJacques Vidrine #include <string.h> 54248aee62SJacques Vidrine #include <unistd.h> 552bbd7cf8SJacques Vidrine #include "un-namespace.h" 56248aee62SJacques Vidrine 57248aee62SJacques Vidrine /* 58248aee62SJacques Vidrine * default sourcelist: `files' 59248aee62SJacques Vidrine */ 60248aee62SJacques Vidrine const ns_src __nsdefaultsrc[] = { 61248aee62SJacques Vidrine { NSSRC_FILES, NS_SUCCESS }, 62248aee62SJacques Vidrine { 0 }, 63248aee62SJacques Vidrine }; 64248aee62SJacques Vidrine 65248aee62SJacques Vidrine 66248aee62SJacques Vidrine static int _nsmapsize = 0; 67248aee62SJacques Vidrine static ns_dbt *_nsmap = NULL; 68248aee62SJacques Vidrine 69248aee62SJacques Vidrine /* 70248aee62SJacques Vidrine * size of dynamic array chunk for _nsmap and _nsmap[x].srclist 71248aee62SJacques Vidrine */ 72248aee62SJacques Vidrine #define NSELEMSPERCHUNK 8 73248aee62SJacques Vidrine 74248aee62SJacques Vidrine 75c05ac53bSDavid E. O'Brien int _nscmp(const void *, const void *); 76248aee62SJacques Vidrine 77248aee62SJacques Vidrine 78248aee62SJacques Vidrine int 79248aee62SJacques Vidrine _nscmp(a, b) 80248aee62SJacques Vidrine const void *a; 81248aee62SJacques Vidrine const void *b; 82248aee62SJacques Vidrine { 83248aee62SJacques Vidrine return (strcasecmp(((const ns_dbt *)a)->name, 84248aee62SJacques Vidrine ((const ns_dbt *)b)->name)); 85248aee62SJacques Vidrine } 86248aee62SJacques Vidrine 87248aee62SJacques Vidrine 88248aee62SJacques Vidrine void 89248aee62SJacques Vidrine _nsdbtaddsrc(dbt, src) 90248aee62SJacques Vidrine ns_dbt *dbt; 91248aee62SJacques Vidrine const ns_src *src; 92248aee62SJacques Vidrine { 93248aee62SJacques Vidrine if ((dbt->srclistsize % NSELEMSPERCHUNK) == 0) { 94248aee62SJacques Vidrine dbt->srclist = (ns_src *)realloc(dbt->srclist, 95248aee62SJacques Vidrine (dbt->srclistsize + NSELEMSPERCHUNK) * sizeof(ns_src)); 96248aee62SJacques Vidrine if (dbt->srclist == NULL) 97ac8e56a7SBruce Evans _err(1, "nsdispatch: memory allocation failure"); 98248aee62SJacques Vidrine } 99248aee62SJacques Vidrine memmove(&dbt->srclist[dbt->srclistsize++], src, sizeof(ns_src)); 100248aee62SJacques Vidrine } 101248aee62SJacques Vidrine 102248aee62SJacques Vidrine 103248aee62SJacques Vidrine void 104248aee62SJacques Vidrine _nsdbtdump(dbt) 105248aee62SJacques Vidrine const ns_dbt *dbt; 106248aee62SJacques Vidrine { 107248aee62SJacques Vidrine int i; 108248aee62SJacques Vidrine 109248aee62SJacques Vidrine printf("%s (%d source%s):", dbt->name, dbt->srclistsize, 110248aee62SJacques Vidrine dbt->srclistsize == 1 ? "" : "s"); 111248aee62SJacques Vidrine for (i = 0; i < dbt->srclistsize; i++) { 112248aee62SJacques Vidrine printf(" %s", dbt->srclist[i].name); 113248aee62SJacques Vidrine if (!(dbt->srclist[i].flags & 114248aee62SJacques Vidrine (NS_UNAVAIL|NS_NOTFOUND|NS_TRYAGAIN)) && 115248aee62SJacques Vidrine (dbt->srclist[i].flags & NS_SUCCESS)) 116248aee62SJacques Vidrine continue; 117248aee62SJacques Vidrine printf(" ["); 118248aee62SJacques Vidrine if (!(dbt->srclist[i].flags & NS_SUCCESS)) 119248aee62SJacques Vidrine printf(" SUCCESS=continue"); 120248aee62SJacques Vidrine if (dbt->srclist[i].flags & NS_UNAVAIL) 121248aee62SJacques Vidrine printf(" UNAVAIL=return"); 122248aee62SJacques Vidrine if (dbt->srclist[i].flags & NS_NOTFOUND) 123248aee62SJacques Vidrine printf(" NOTFOUND=return"); 124248aee62SJacques Vidrine if (dbt->srclist[i].flags & NS_TRYAGAIN) 125248aee62SJacques Vidrine printf(" TRYAGAIN=return"); 126248aee62SJacques Vidrine printf(" ]"); 127248aee62SJacques Vidrine } 128248aee62SJacques Vidrine printf("\n"); 129248aee62SJacques Vidrine } 130248aee62SJacques Vidrine 131248aee62SJacques Vidrine 132248aee62SJacques Vidrine const ns_dbt * 133248aee62SJacques Vidrine _nsdbtget(name) 134248aee62SJacques Vidrine const char *name; 135248aee62SJacques Vidrine { 136248aee62SJacques Vidrine static time_t confmod; 137248aee62SJacques Vidrine 138248aee62SJacques Vidrine struct stat statbuf; 139248aee62SJacques Vidrine ns_dbt dbt; 140248aee62SJacques Vidrine 141248aee62SJacques Vidrine extern FILE *_nsyyin; 142c05ac53bSDavid E. O'Brien extern int _nsyyparse(void); 143248aee62SJacques Vidrine 144248aee62SJacques Vidrine dbt.name = name; 145248aee62SJacques Vidrine 146248aee62SJacques Vidrine if (confmod) { 147248aee62SJacques Vidrine if (stat(_PATH_NS_CONF, &statbuf) == -1) 148248aee62SJacques Vidrine return (NULL); 149248aee62SJacques Vidrine if (confmod < statbuf.st_mtime) { 150248aee62SJacques Vidrine int i, j; 151248aee62SJacques Vidrine 152248aee62SJacques Vidrine for (i = 0; i < _nsmapsize; i++) { 153248aee62SJacques Vidrine for (j = 0; j < _nsmap[i].srclistsize; j++) { 154248aee62SJacques Vidrine if (_nsmap[i].srclist[j].name != NULL) { 155248aee62SJacques Vidrine /*LINTED const cast*/ 156248aee62SJacques Vidrine free((void *) 157248aee62SJacques Vidrine _nsmap[i].srclist[j].name); 158248aee62SJacques Vidrine } 159248aee62SJacques Vidrine } 160248aee62SJacques Vidrine if (_nsmap[i].srclist) 161248aee62SJacques Vidrine free(_nsmap[i].srclist); 162248aee62SJacques Vidrine if (_nsmap[i].name) { 163248aee62SJacques Vidrine /*LINTED const cast*/ 164248aee62SJacques Vidrine free((void *)_nsmap[i].name); 165248aee62SJacques Vidrine } 166248aee62SJacques Vidrine } 167248aee62SJacques Vidrine if (_nsmap) 168248aee62SJacques Vidrine free(_nsmap); 169248aee62SJacques Vidrine _nsmap = NULL; 170248aee62SJacques Vidrine _nsmapsize = 0; 171248aee62SJacques Vidrine confmod = 0; 172248aee62SJacques Vidrine } 173248aee62SJacques Vidrine } 174248aee62SJacques Vidrine if (!confmod) { 175248aee62SJacques Vidrine if (stat(_PATH_NS_CONF, &statbuf) == -1) 176248aee62SJacques Vidrine return (NULL); 177248aee62SJacques Vidrine _nsyyin = fopen(_PATH_NS_CONF, "r"); 178248aee62SJacques Vidrine if (_nsyyin == NULL) 179248aee62SJacques Vidrine return (NULL); 180248aee62SJacques Vidrine _nsyyparse(); 181248aee62SJacques Vidrine (void)fclose(_nsyyin); 182248aee62SJacques Vidrine qsort(_nsmap, (size_t)_nsmapsize, sizeof(ns_dbt), _nscmp); 183248aee62SJacques Vidrine confmod = statbuf.st_mtime; 184248aee62SJacques Vidrine } 185248aee62SJacques Vidrine return (bsearch(&dbt, _nsmap, (size_t)_nsmapsize, sizeof(ns_dbt), 186248aee62SJacques Vidrine _nscmp)); 187248aee62SJacques Vidrine } 188248aee62SJacques Vidrine 189248aee62SJacques Vidrine 190248aee62SJacques Vidrine void 191248aee62SJacques Vidrine _nsdbtput(dbt) 192248aee62SJacques Vidrine const ns_dbt *dbt; 193248aee62SJacques Vidrine { 194248aee62SJacques Vidrine int i; 195248aee62SJacques Vidrine 196248aee62SJacques Vidrine for (i = 0; i < _nsmapsize; i++) { 197248aee62SJacques Vidrine if (_nscmp(dbt, &_nsmap[i]) == 0) { 198248aee62SJacques Vidrine /* overwrite existing entry */ 199248aee62SJacques Vidrine if (_nsmap[i].srclist != NULL) 200248aee62SJacques Vidrine free(_nsmap[i].srclist); 201248aee62SJacques Vidrine memmove(&_nsmap[i], dbt, sizeof(ns_dbt)); 202248aee62SJacques Vidrine return; 203248aee62SJacques Vidrine } 204248aee62SJacques Vidrine } 205248aee62SJacques Vidrine 206248aee62SJacques Vidrine if ((_nsmapsize % NSELEMSPERCHUNK) == 0) { 207248aee62SJacques Vidrine _nsmap = (ns_dbt *)realloc(_nsmap, 208248aee62SJacques Vidrine (_nsmapsize + NSELEMSPERCHUNK) * sizeof(ns_dbt)); 209248aee62SJacques Vidrine if (_nsmap == NULL) 210ac8e56a7SBruce Evans _err(1, "nsdispatch: memory allocation failure"); 211248aee62SJacques Vidrine } 212248aee62SJacques Vidrine memmove(&_nsmap[_nsmapsize++], dbt, sizeof(ns_dbt)); 213248aee62SJacques Vidrine } 214248aee62SJacques Vidrine 215248aee62SJacques Vidrine 216248aee62SJacques Vidrine int 217248aee62SJacques Vidrine nsdispatch(void *retval, const ns_dtab disp_tab[], const char *database, 218248aee62SJacques Vidrine const char *method, const ns_src defaults[], ...) 219248aee62SJacques Vidrine { 220248aee62SJacques Vidrine va_list ap; 221248aee62SJacques Vidrine int i, curdisp, result; 222248aee62SJacques Vidrine const ns_dbt *dbt; 223248aee62SJacques Vidrine const ns_src *srclist; 224248aee62SJacques Vidrine int srclistsize; 225248aee62SJacques Vidrine 226248aee62SJacques Vidrine dbt = _nsdbtget(database); 227248aee62SJacques Vidrine if (dbt != NULL) { 228248aee62SJacques Vidrine srclist = dbt->srclist; 229248aee62SJacques Vidrine srclistsize = dbt->srclistsize; 230248aee62SJacques Vidrine } else { 231248aee62SJacques Vidrine srclist = defaults; 232248aee62SJacques Vidrine srclistsize = 0; 233248aee62SJacques Vidrine while (srclist[srclistsize].name != NULL) 234248aee62SJacques Vidrine srclistsize++; 235248aee62SJacques Vidrine } 236248aee62SJacques Vidrine result = 0; 237248aee62SJacques Vidrine 238248aee62SJacques Vidrine for (i = 0; i < srclistsize; i++) { 239248aee62SJacques Vidrine for (curdisp = 0; disp_tab[curdisp].src != NULL; curdisp++) 240248aee62SJacques Vidrine if (strcasecmp(disp_tab[curdisp].src, 241248aee62SJacques Vidrine srclist[i].name) == 0) 242248aee62SJacques Vidrine break; 243248aee62SJacques Vidrine result = 0; 244248aee62SJacques Vidrine if (disp_tab[curdisp].callback) { 245248aee62SJacques Vidrine va_start(ap, defaults); 246248aee62SJacques Vidrine result = disp_tab[curdisp].callback(retval, 247248aee62SJacques Vidrine disp_tab[curdisp].cb_data, ap); 248248aee62SJacques Vidrine va_end(ap); 249248aee62SJacques Vidrine if (result & srclist[i].flags) { 250248aee62SJacques Vidrine break; 251248aee62SJacques Vidrine } 252248aee62SJacques Vidrine } 253248aee62SJacques Vidrine } 254248aee62SJacques Vidrine return (result ? result : NS_NOTFOUND); 255248aee62SJacques Vidrine } 256