Blame modules/lookup_sss.c

Packit Service a4b2a9
/* ----------------------------------------------------------------------- *
Packit Service a4b2a9
 *
Packit Service a4b2a9
 *  lookup_sss.c - module for Linux automount to query sss service
Packit Service a4b2a9
 *
Packit Service a4b2a9
 *   Copyright 2012 Ian Kent <raven@themaw.net>
Packit Service a4b2a9
 *   Copyright 2012 Red Hat, Inc.
Packit Service a4b2a9
 *
Packit Service a4b2a9
 *   This program is free software; you can redistribute it and/or modify
Packit Service a4b2a9
 *   it under the terms of the GNU General Public License as published by
Packit Service a4b2a9
 *   the Free Software Foundation, Inc., 675 Mass Ave, Cambridge MA 02139,
Packit Service a4b2a9
 *   USA; either version 2 of the License, or (at your option) any later
Packit Service a4b2a9
 *   version; incorporated herein by reference.
Packit Service a4b2a9
 *
Packit Service a4b2a9
 * ----------------------------------------------------------------------- */
Packit Service a4b2a9
Packit Service a4b2a9
#include <stdio.h>
Packit Service a4b2a9
#include <malloc.h>
Packit Service a4b2a9
#include <stdlib.h>
Packit Service a4b2a9
#include <ctype.h>
Packit Service a4b2a9
#include <dlfcn.h>
Packit Service a4b2a9
#include <errno.h>
Packit Service a4b2a9
#include <string.h>
Packit Service a4b2a9
#include <sys/param.h>
Packit Service a4b2a9
#include <sys/types.h>
Packit Service a4b2a9
#include <sys/stat.h>
Packit Service a4b2a9
Packit Service a4b2a9
#define MODULE_LOOKUP
Packit Service a4b2a9
#include "automount.h"
Packit Service a4b2a9
#include "nsswitch.h"
Packit Service a4b2a9
Packit Service a4b2a9
#define MAPFMT_DEFAULT "sun"
Packit Service a4b2a9
Packit Service a4b2a9
/* Half a second between retries */
Packit Service a4b2a9
#define SETAUTOMOUNTENT_MASTER_INTERVAL	500000000
Packit Service a4b2a9
Packit Service a4b2a9
#define MODPREFIX "lookup(sss): "
Packit Service a4b2a9
Packit Service a4b2a9
#define SSS_SO_NAME "libsss_autofs"
Packit Service a4b2a9
Packit Service a4b2a9
int _sss_setautomntent(const char *, void **);
Packit Service a4b2a9
int _sss_getautomntent_r(char **, char **, void *);
Packit Service a4b2a9
int _sss_getautomntbyname_r(char *, char **, void *);
Packit Service a4b2a9
int _sss_endautomntent(void **);
Packit Service a4b2a9
Packit Service a4b2a9
typedef int (*setautomntent_t) (const char *, void **);
Packit Service a4b2a9
typedef int (*getautomntent_t) (char **, char **, void *);
Packit Service a4b2a9
typedef int (*getautomntbyname_t) (char *, char **, void *);
Packit Service a4b2a9
typedef int (*endautomntent_t) (void **);
Packit Service a4b2a9
Packit Service a4b2a9
struct lookup_context {
Packit Service a4b2a9
	const char *mapname;
Packit Service a4b2a9
    	void *dlhandle;
Packit Service a4b2a9
	setautomntent_t setautomntent;
Packit Service a4b2a9
	getautomntent_t getautomntent_r;
Packit Service a4b2a9
	getautomntbyname_t getautomntbyname_r;
Packit Service a4b2a9
	endautomntent_t endautomntent;
Packit Service a4b2a9
	struct parse_mod *parse;
Packit Service a4b2a9
};
Packit Service a4b2a9
Packit Service a4b2a9
int lookup_version = AUTOFS_LOOKUP_VERSION;	/* Required by protocol */
Packit Service a4b2a9
Packit Service a4b2a9
static int open_sss_lib(struct lookup_context *ctxt)
Packit Service a4b2a9
{
Packit Service a4b2a9
	char dlbuf[PATH_MAX];
Packit Service a4b2a9
	char *estr;
Packit Service a4b2a9
	void *dh;
Packit Service a4b2a9
	size_t size;
Packit Service a4b2a9
Packit Service a4b2a9
	size = snprintf(dlbuf, sizeof(dlbuf),
Packit Service a4b2a9
			"%s/%s.so", SSS_LIB_DIR, SSS_SO_NAME);
Packit Service a4b2a9
	if (size >= sizeof(dlbuf)) {
Packit Service a4b2a9
		logmsg(MODPREFIX "sss library path too long");
Packit Service a4b2a9
		return 1;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	dh = dlopen(dlbuf, RTLD_LAZY);
Packit Service a4b2a9
	if (!dh)
Packit Service a4b2a9
		return 1;
Packit Service a4b2a9
	ctxt->dlhandle = dh;
Packit Service a4b2a9
Packit Service a4b2a9
	ctxt->setautomntent = (setautomntent_t) dlsym(dh, "_sss_setautomntent");
Packit Service a4b2a9
	if (!ctxt->setautomntent)
Packit Service a4b2a9
		goto lib_names_fail;
Packit Service a4b2a9
Packit Service a4b2a9
	ctxt->getautomntent_r = (getautomntent_t) dlsym(dh, "_sss_getautomntent_r");
Packit Service a4b2a9
	if (!ctxt->getautomntent_r)
Packit Service a4b2a9
		goto lib_names_fail;
Packit Service a4b2a9
Packit Service a4b2a9
	ctxt->getautomntbyname_r = (getautomntbyname_t) dlsym(dh, "_sss_getautomntbyname_r");
Packit Service a4b2a9
	if (!ctxt->getautomntbyname_r)
Packit Service a4b2a9
		goto lib_names_fail;
Packit Service a4b2a9
Packit Service a4b2a9
	ctxt->endautomntent = (endautomntent_t) dlsym(dh, "_sss_endautomntent");
Packit Service a4b2a9
	if (!ctxt->setautomntent)
Packit Service a4b2a9
		goto lib_names_fail;
Packit Service a4b2a9
Packit Service a4b2a9
	return 0;
Packit Service a4b2a9
Packit Service a4b2a9
lib_names_fail:
Packit Service a4b2a9
	if ((estr = dlerror()) == NULL)
Packit Service a4b2a9
		logmsg(MODPREFIX "failed to locate sss library entry points");
Packit Service a4b2a9
	else
Packit Service a4b2a9
		logerr(MODPREFIX "dlsym: %s", estr);
Packit Service a4b2a9
	dlclose(dh);
Packit Service a4b2a9
Packit Service a4b2a9
	return 1;
Packit Service a4b2a9
}
Packit Service a4b2a9
Packit Service a4b2a9
static int do_init(const char *mapfmt,
Packit Service a4b2a9
		   int argc, const char *const *argv,
Packit Service a4b2a9
		   struct lookup_context *ctxt, unsigned int reinit)
Packit Service a4b2a9
{
Packit Service a4b2a9
	int ret = 0;
Packit Service a4b2a9
Packit Service a4b2a9
	if (argc < 1) {
Packit Service a4b2a9
		logerr(MODPREFIX "No map name");
Packit Service a4b2a9
		ret = 1;
Packit Service a4b2a9
		goto out;
Packit Service a4b2a9
	}
Packit Service a4b2a9
	ctxt->mapname = argv[0];
Packit Service a4b2a9
Packit Service a4b2a9
	if (!mapfmt)
Packit Service a4b2a9
		mapfmt = MAPFMT_DEFAULT;
Packit Service a4b2a9
Packit Service a4b2a9
	if (!reinit) {
Packit Service a4b2a9
		ret = open_sss_lib(ctxt);
Packit Service a4b2a9
		if (ret)
Packit Service a4b2a9
			goto out;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	if (reinit) {
Packit Service a4b2a9
		ret = reinit_parse(ctxt->parse, mapfmt, MODPREFIX, argc - 1, argv + 1);
Packit Service a4b2a9
		if (ret)
Packit Service a4b2a9
			logmsg(MODPREFIX "failed to reinit parse context");
Packit Service a4b2a9
	} else {
Packit Service a4b2a9
		ctxt->parse = open_parse(mapfmt, MODPREFIX, argc - 1, argv + 1);
Packit Service a4b2a9
		if (!ctxt->parse) {
Packit Service a4b2a9
			logmsg(MODPREFIX "failed to open parse context");
Packit Service a4b2a9
			dlclose(ctxt->dlhandle);
Packit Service a4b2a9
			ret = 1;
Packit Service a4b2a9
		}
Packit Service a4b2a9
	}
Packit Service a4b2a9
out:
Packit Service a4b2a9
	return ret;
Packit Service a4b2a9
}
Packit Service a4b2a9
Packit Service a4b2a9
int lookup_init(const char *mapfmt,
Packit Service a4b2a9
		int argc, const char *const *argv, void **context)
Packit Service a4b2a9
{
Packit Service a4b2a9
	struct lookup_context *ctxt;
Packit Service a4b2a9
	char buf[MAX_ERR_BUF];
Packit Service a4b2a9
	char *estr;
Packit Service a4b2a9
Packit Service a4b2a9
	*context = NULL;
Packit Service a4b2a9
Packit Service a4b2a9
	ctxt = malloc(sizeof(struct lookup_context));
Packit Service a4b2a9
	if (!ctxt) {
Packit Service a4b2a9
		estr = strerror_r(errno, buf, MAX_ERR_BUF);
Packit Service a4b2a9
		logerr(MODPREFIX "malloc: %s", estr);
Packit Service a4b2a9
		return 1;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	if (do_init(mapfmt, argc, argv, ctxt, 0)) {
Packit Service a4b2a9
		free(ctxt);
Packit Service a4b2a9
		return 1;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	*context = ctxt;
Packit Service a4b2a9
Packit Service a4b2a9
	return 0;
Packit Service a4b2a9
}
Packit Service a4b2a9
Packit Service a4b2a9
int lookup_reinit(const char *mapfmt,
Packit Service a4b2a9
		  int argc, const char *const *argv, void **context)
Packit Service a4b2a9
{
Packit Service a4b2a9
	struct lookup_context *ctxt = (struct lookup_context *) *context;
Packit Service a4b2a9
	struct lookup_context *new;
Packit Service a4b2a9
	char buf[MAX_ERR_BUF];
Packit Service a4b2a9
	int ret;
Packit Service a4b2a9
Packit Service a4b2a9
	new = malloc(sizeof(struct lookup_context));
Packit Service a4b2a9
	if (!new) {
Packit Service a4b2a9
		char *estr = strerror_r(errno, buf, MAX_ERR_BUF);
Packit Service a4b2a9
		logerr(MODPREFIX "malloc: %s", estr);
Packit Service a4b2a9
		return 1;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	new->parse = ctxt->parse;
Packit Service a4b2a9
	ret = do_init(mapfmt, argc, argv, new, 1);
Packit Service a4b2a9
	if (ret) {
Packit Service a4b2a9
		free(new);
Packit Service a4b2a9
		return 1;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	new->dlhandle = ctxt->dlhandle;
Packit Service a4b2a9
	new->setautomntent = ctxt->setautomntent;
Packit Service a4b2a9
	new->getautomntent_r = ctxt->getautomntent_r;
Packit Service a4b2a9
	new->getautomntbyname_r = ctxt->getautomntbyname_r;
Packit Service a4b2a9
	new->endautomntent = ctxt->endautomntent;
Packit Service a4b2a9
Packit Service a4b2a9
	*context = new;
Packit Service a4b2a9
Packit Service a4b2a9
	free(ctxt);
Packit Service a4b2a9
Packit Service a4b2a9
	return 0;
Packit Service a4b2a9
}
Packit Service a4b2a9
Packit Service a4b2a9
static int setautomntent(unsigned int logopt,
Packit Service a4b2a9
			 struct lookup_context *ctxt, const char *mapname,
Packit Service a4b2a9
			 void **sss_ctxt)
Packit Service a4b2a9
{
Packit Service a4b2a9
	int ret = ctxt->setautomntent(mapname, sss_ctxt);
Packit Service a4b2a9
	if (ret) {
Packit Service a4b2a9
		char buf[MAX_ERR_BUF];
Packit Service a4b2a9
		char *estr = strerror_r(ret, buf, MAX_ERR_BUF);
Packit Service a4b2a9
		error(logopt, MODPREFIX "setautomntent: %s", estr);
Packit Service a4b2a9
		if (*sss_ctxt)
Packit Service a4b2a9
			free(*sss_ctxt);
Packit Service a4b2a9
	}
Packit Service a4b2a9
	return ret;
Packit Service a4b2a9
}
Packit Service a4b2a9
Packit Service a4b2a9
static int setautomntent_wait(unsigned int logopt,
Packit Service a4b2a9
			      struct lookup_context *ctxt,
Packit Service a4b2a9
			      const char *mapname,
Packit Service a4b2a9
			      void **sss_ctxt, unsigned int retries)
Packit Service a4b2a9
{
Packit Service a4b2a9
	unsigned int retry = 0;
Packit Service a4b2a9
	int ret = 0;
Packit Service a4b2a9
Packit Service a4b2a9
	*sss_ctxt = NULL;
Packit Service a4b2a9
Packit Service a4b2a9
	while (++retry < retries) {
Packit Service a4b2a9
		struct timespec t = { 0, SETAUTOMOUNTENT_MASTER_INTERVAL };
Packit Service a4b2a9
		struct timespec r;
Packit Service a4b2a9
Packit Service a4b2a9
		ret = ctxt->setautomntent(mapname, sss_ctxt);
Packit Service a4b2a9
		if (ret != ENOENT)
Packit Service a4b2a9
			break;
Packit Service a4b2a9
Packit Service a4b2a9
		if (*sss_ctxt) {
Packit Service a4b2a9
			free(*sss_ctxt);
Packit Service a4b2a9
			*sss_ctxt = NULL;
Packit Service a4b2a9
		}
Packit Service a4b2a9
Packit Service a4b2a9
		while (nanosleep(&t, &r) == -1 && errno == EINTR)
Packit Service a4b2a9
			memcpy(&t, &r, sizeof(struct timespec));
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
Packit Service a4b2a9
	if (ret) {
Packit Service a4b2a9
		char buf[MAX_ERR_BUF];
Packit Service a4b2a9
		char *estr;
Packit Service a4b2a9
Packit Service a4b2a9
		if (*sss_ctxt) {
Packit Service a4b2a9
			free(*sss_ctxt);
Packit Service a4b2a9
			*sss_ctxt = NULL;
Packit Service a4b2a9
		}
Packit Service a4b2a9
Packit Service a4b2a9
		if (retry == retries)
Packit Service a4b2a9
			ret = ETIMEDOUT;
Packit Service a4b2a9
Packit Service a4b2a9
		estr = strerror_r(ret, buf, MAX_ERR_BUF);
Packit Service a4b2a9
		error(logopt, MODPREFIX "setautomntent: %s", estr);
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	return ret;
Packit Service a4b2a9
}
Packit Service a4b2a9
Packit Service a4b2a9
static int endautomntent(unsigned int logopt,
Packit Service a4b2a9
			 struct lookup_context *ctxt, void **sss_ctxt)
Packit Service a4b2a9
{
Packit Service a4b2a9
	int ret = ctxt->endautomntent(sss_ctxt);
Packit Service a4b2a9
	if (ret) {
Packit Service a4b2a9
		char buf[MAX_ERR_BUF];
Packit Service a4b2a9
		char *estr = strerror_r(ret, buf, MAX_ERR_BUF);
Packit Service a4b2a9
		error(logopt, MODPREFIX "endautomntent: %s", estr);
Packit Service a4b2a9
	}
Packit Service a4b2a9
	return ret;
Packit Service a4b2a9
}
Packit Service a4b2a9
Packit Service a4b2a9
int lookup_read_master(struct master *master, time_t age, void *context)
Packit Service a4b2a9
{
Packit Service a4b2a9
	struct lookup_context *ctxt = (struct lookup_context *) context;
Packit Service a4b2a9
	unsigned int timeout = master->default_timeout;
Packit Service a4b2a9
	unsigned int logging = master->default_logging;
Packit Service a4b2a9
	unsigned int logopt = master->logopt;
Packit Service a4b2a9
	void *sss_ctxt = NULL;
Packit Service a4b2a9
	char buf[MAX_ERR_BUF];
Packit Service a4b2a9
	char *buffer;
Packit Service a4b2a9
	size_t buffer_len;
Packit Service a4b2a9
	char *key;
Packit Service a4b2a9
	char *value = NULL;
Packit Service a4b2a9
	int count, ret;
Packit Service a4b2a9
Packit Service a4b2a9
	ret = setautomntent(logopt, ctxt, ctxt->mapname, &sss_ctxt);
Packit Service a4b2a9
	if (ret) {
Packit Service a4b2a9
		unsigned int retries;
Packit Service a4b2a9
Packit Service a4b2a9
		if (ret != ENOENT)
Packit Service a4b2a9
			return NSS_STATUS_UNAVAIL;
Packit Service a4b2a9
Packit Service a4b2a9
		retries = defaults_get_sss_master_map_wait() * 2;
Packit Service a4b2a9
		if (retries <= 0)
Packit Service a4b2a9
			return NSS_STATUS_NOTFOUND;
Packit Service a4b2a9
Packit Service a4b2a9
		ret = setautomntent_wait(logopt,
Packit Service a4b2a9
					 ctxt, ctxt->mapname, &sss_ctxt,
Packit Service a4b2a9
					 retries);
Packit Service a4b2a9
		if (ret) {
Packit Service a4b2a9
			if (ret == ENOENT)
Packit Service a4b2a9
				return NSS_STATUS_NOTFOUND;
Packit Service a4b2a9
			return NSS_STATUS_UNAVAIL;
Packit Service a4b2a9
		}
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	count = 0;
Packit Service a4b2a9
	while (1) {
Packit Service a4b2a9
	        key = NULL;
Packit Service a4b2a9
	        value = NULL;
Packit Service a4b2a9
		ret = ctxt->getautomntent_r(&key, &value, sss_ctxt);
Packit Service a4b2a9
		if (ret && ret != ENOENT) {
Packit Service a4b2a9
			char *estr = strerror_r(ret, buf, MAX_ERR_BUF);
Packit Service a4b2a9
			error(logopt, MODPREFIX "getautomntent_r: %s", estr);
Packit Service a4b2a9
			endautomntent(logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
			if (key)
Packit Service a4b2a9
				free(key);
Packit Service a4b2a9
			if (value)
Packit Service a4b2a9
				free(value);
Packit Service a4b2a9
			return NSS_STATUS_UNAVAIL;
Packit Service a4b2a9
		}
Packit Service a4b2a9
		if (ret == ENOENT) {
Packit Service a4b2a9
			if (!count) {
Packit Service a4b2a9
				char *estr = strerror_r(ret, buf, MAX_ERR_BUF);
Packit Service a4b2a9
				error(logopt, MODPREFIX "getautomntent_r: %s", estr);
Packit Service a4b2a9
				endautomntent(logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
				if (key)
Packit Service a4b2a9
					free(key);
Packit Service a4b2a9
				if (value)
Packit Service a4b2a9
					free(value);
Packit Service a4b2a9
				return NSS_STATUS_NOTFOUND;
Packit Service a4b2a9
			}
Packit Service a4b2a9
			break;
Packit Service a4b2a9
		}
Packit Service a4b2a9
		count++;
Packit Service a4b2a9
Packit Service a4b2a9
		buffer_len = strlen(key) + 1 + strlen(value) + 2;
Packit Service a4b2a9
		buffer = malloc(buffer_len);
Packit Service a4b2a9
		if (!buffer) {
Packit Service a4b2a9
			char *estr = strerror_r(errno, buf, MAX_ERR_BUF);
Packit Service a4b2a9
			error(logopt, MODPREFIX "malloc: %s", estr);
Packit Service a4b2a9
			endautomntent(logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
			free(key);
Packit Service a4b2a9
			free(value);
Packit Service a4b2a9
			return NSS_STATUS_UNAVAIL;
Packit Service a4b2a9
		}
Packit Service a4b2a9
Packit Service a4b2a9
		/*
Packit Service a4b2a9
		 * TODO: implement sun % hack for key translation for
Packit Service a4b2a9
		 * mixed case keys in schema that are single case only.
Packit Service a4b2a9
		 */
Packit Service a4b2a9
Packit Service a4b2a9
		strcpy(buffer, key);
Packit Service a4b2a9
		strcat(buffer, " ");
Packit Service a4b2a9
		strcat(buffer, value);
Packit Service a4b2a9
Packit Service a4b2a9
		/*
Packit Service a4b2a9
		 * TODO: handle cancelation. This almost certainly isn't
Packit Service a4b2a9
		 * handled properly by other lookup modules either so it
Packit Service a4b2a9
		 * should be done when cancelation is reviewed for the
Packit Service a4b2a9
		 * other modules. Ditto for the other lookup module entry
Packit Service a4b2a9
		 * points.
Packit Service a4b2a9
		 */
Packit Service a4b2a9
		master_parse_entry(buffer, timeout, logging, age);
Packit Service a4b2a9
Packit Service a4b2a9
		free(buffer);
Packit Service a4b2a9
		free(key);
Packit Service a4b2a9
		free(value);
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	endautomntent(logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
Packit Service a4b2a9
	return NSS_STATUS_SUCCESS;
Packit Service a4b2a9
}
Packit Service a4b2a9
Packit Service a4b2a9
int lookup_read_map(struct autofs_point *ap, time_t age, void *context)
Packit Service a4b2a9
{
Packit Service a4b2a9
	struct lookup_context *ctxt = (struct lookup_context *) context;
Packit Service a4b2a9
	struct map_source *source;
Packit Service a4b2a9
	struct mapent_cache *mc;
Packit Service a4b2a9
	void *sss_ctxt = NULL;
Packit Service a4b2a9
	char buf[MAX_ERR_BUF];
Packit Service a4b2a9
	char *key;
Packit Service a4b2a9
	char *value = NULL;
Packit Service a4b2a9
	char *s_key;
Packit Service a4b2a9
	int count, ret;
Packit Service a4b2a9
Packit Service a4b2a9
	source = ap->entry->current;
Packit Service a4b2a9
	ap->entry->current = NULL;
Packit Service a4b2a9
	master_source_current_signal(ap->entry);
Packit Service a4b2a9
Packit Service a4b2a9
	mc = source->mc;
Packit Service a4b2a9
Packit Service a4b2a9
	/*
Packit Service a4b2a9
	 * If we don't need to create directories (or don't need
Packit Service a4b2a9
	 * to read an amd cache:=all map) then there's no use
Packit Service a4b2a9
	 * reading the map. We always need to read the whole map
Packit Service a4b2a9
	 * for direct mounts in order to mount the triggers.
Packit Service a4b2a9
	 */
Packit Service a4b2a9
	if (ap->type != LKP_DIRECT &&
Packit Service a4b2a9
	    !(ap->flags & (MOUNT_FLAG_GHOST|MOUNT_FLAG_AMD_CACHE_ALL))) {
Packit Service a4b2a9
		debug(ap->logopt, "map read not needed, so not done");
Packit Service a4b2a9
		return NSS_STATUS_SUCCESS;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	ret = setautomntent(ap->logopt, ctxt, ctxt->mapname, &sss_ctxt);
Packit Service a4b2a9
	if (ret) {
Packit Service a4b2a9
		if (ret == ENOENT)
Packit Service a4b2a9
			return NSS_STATUS_NOTFOUND;
Packit Service a4b2a9
		return NSS_STATUS_UNAVAIL;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	count = 0;
Packit Service a4b2a9
	while (1) {
Packit Service a4b2a9
	        key = NULL;
Packit Service a4b2a9
	        value = NULL;
Packit Service a4b2a9
		ret = ctxt->getautomntent_r(&key, &value, sss_ctxt);
Packit Service a4b2a9
		if (ret && ret != ENOENT) {
Packit Service a4b2a9
			char *estr = strerror_r(ret, buf, MAX_ERR_BUF);
Packit Service a4b2a9
			error(ap->logopt,
Packit Service a4b2a9
			      MODPREFIX "getautomntent_r: %s", estr);
Packit Service a4b2a9
			endautomntent(ap->logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
			if (key)
Packit Service a4b2a9
				free(key);
Packit Service a4b2a9
			if (value)
Packit Service a4b2a9
				free(value);
Packit Service a4b2a9
			return NSS_STATUS_UNAVAIL;
Packit Service a4b2a9
		}
Packit Service a4b2a9
		if (ret == ENOENT) {
Packit Service a4b2a9
			if (!count) {
Packit Service a4b2a9
				char *estr = strerror_r(ret, buf, MAX_ERR_BUF);
Packit Service a4b2a9
				error(ap->logopt,
Packit Service a4b2a9
				      MODPREFIX "getautomntent_r: %s", estr);
Packit Service a4b2a9
				endautomntent(ap->logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
				if (key)
Packit Service a4b2a9
					free(key);
Packit Service a4b2a9
				if (value)
Packit Service a4b2a9
					free(value);
Packit Service a4b2a9
				return NSS_STATUS_NOTFOUND;
Packit Service a4b2a9
			}
Packit Service a4b2a9
			break;
Packit Service a4b2a9
		}
Packit Service a4b2a9
Packit Service a4b2a9
		/*
Packit Service a4b2a9
		 * Ignore keys beginning with '+' as plus map
Packit Service a4b2a9
		 * inclusion is only valid in file maps.
Packit Service a4b2a9
		 */
Packit Service a4b2a9
		if (*key == '+') {
Packit Service a4b2a9
			warn(ap->logopt,
Packit Service a4b2a9
			     MODPREFIX "ignoring '+' map entry - not in file map");
Packit Service a4b2a9
			free(key);
Packit Service a4b2a9
			free(value);
Packit Service a4b2a9
			continue;
Packit Service a4b2a9
		}
Packit Service a4b2a9
Packit Service a4b2a9
		if (*key == '/' && strlen(key) == 1) {
Packit Service a4b2a9
			if (ap->type == LKP_DIRECT) {
Packit Service a4b2a9
				free(key);
Packit Service a4b2a9
				free(value);
Packit Service a4b2a9
				continue;
Packit Service a4b2a9
			}
Packit Service a4b2a9
			*key = '*';
Packit Service a4b2a9
		}
Packit Service a4b2a9
Packit Service a4b2a9
		/*
Packit Service a4b2a9
		 * TODO: implement sun % hack for key translation for
Packit Service a4b2a9
		 * mixed case keys in schema that are single case only.
Packit Service a4b2a9
		 */
Packit Service a4b2a9
Packit Service a4b2a9
		s_key = sanitize_path(key, strlen(key), ap->type, ap->logopt);
Packit Service a4b2a9
		if (!s_key) {
Packit Service a4b2a9
			error(ap->logopt, MODPREFIX "invalid path %s", key);
Packit Service a4b2a9
			endautomntent(ap->logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
			free(key);
Packit Service a4b2a9
			free(value);
Packit Service a4b2a9
			return NSS_STATUS_NOTFOUND;
Packit Service a4b2a9
		}
Packit Service a4b2a9
Packit Service a4b2a9
		count++;
Packit Service a4b2a9
Packit Service a4b2a9
		cache_writelock(mc);
Packit Service a4b2a9
		cache_update(mc, source, s_key, value, age);
Packit Service a4b2a9
		cache_unlock(mc);
Packit Service a4b2a9
Packit Service a4b2a9
		free(s_key);
Packit Service a4b2a9
		free(key);
Packit Service a4b2a9
		free(value);
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	endautomntent(ap->logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
Packit Service a4b2a9
	source->age = age;
Packit Service a4b2a9
Packit Service a4b2a9
	return NSS_STATUS_SUCCESS;
Packit Service a4b2a9
}
Packit Service a4b2a9
Packit Service a4b2a9
static int lookup_one(struct autofs_point *ap,
Packit Service a4b2a9
		char *qKey, int qKey_len, struct lookup_context *ctxt)
Packit Service a4b2a9
{
Packit Service a4b2a9
	struct map_source *source;
Packit Service a4b2a9
	struct mapent_cache *mc;
Packit Service a4b2a9
	struct mapent *we;
Packit Service a4b2a9
	void *sss_ctxt = NULL;
Packit Service a4b2a9
	time_t age = monotonic_time(NULL);
Packit Service a4b2a9
	char buf[MAX_ERR_BUF];
Packit Service a4b2a9
	char *value = NULL;
Packit Service a4b2a9
	char *s_key;
Packit Service a4b2a9
	int ret;
Packit Service a4b2a9
Packit Service a4b2a9
	source = ap->entry->current;
Packit Service a4b2a9
	ap->entry->current = NULL;
Packit Service a4b2a9
	master_source_current_signal(ap->entry);
Packit Service a4b2a9
Packit Service a4b2a9
	mc = source->mc;
Packit Service a4b2a9
Packit Service a4b2a9
	ret = setautomntent(ap->logopt, ctxt, ctxt->mapname, &sss_ctxt);
Packit Service a4b2a9
	if (ret) {
Packit Service a4b2a9
		if (ret == ENOENT)
Packit Service a4b2a9
			return NSS_STATUS_NOTFOUND;
Packit Service a4b2a9
		return NSS_STATUS_UNAVAIL;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	ret = ctxt->getautomntbyname_r(qKey, &value, sss_ctxt);
Packit Service a4b2a9
	if (ret && ret != ENOENT) {
Packit Service a4b2a9
		char *estr = strerror_r(ret, buf, MAX_ERR_BUF);
Packit Service a4b2a9
		error(ap->logopt,
Packit Service a4b2a9
		      MODPREFIX "getautomntbyname_r: %s", estr);
Packit Service a4b2a9
		endautomntent(ap->logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
		if (value)
Packit Service a4b2a9
			free(value);
Packit Service a4b2a9
		return NSS_STATUS_UNAVAIL;
Packit Service a4b2a9
	}
Packit Service a4b2a9
	if (ret != ENOENT) {
Packit Service a4b2a9
		/*
Packit Service a4b2a9
		 * TODO: implement sun % hack for key translation for
Packit Service a4b2a9
		 * mixed case keys in schema that are single case only.
Packit Service a4b2a9
		 */
Packit Service a4b2a9
		s_key = sanitize_path(qKey, qKey_len, ap->type, ap->logopt);
Packit Service a4b2a9
		if (!s_key) {
Packit Service a4b2a9
			free(value);
Packit Service a4b2a9
			value = NULL;
Packit Service a4b2a9
			goto wild;
Packit Service a4b2a9
		}
Packit Service a4b2a9
		cache_writelock(mc);
Packit Service a4b2a9
		ret = cache_update(mc, source, s_key, value, age);
Packit Service a4b2a9
		cache_unlock(mc);
Packit Service a4b2a9
		endautomntent(ap->logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
		free(s_key);
Packit Service a4b2a9
		free(value);
Packit Service a4b2a9
		return NSS_STATUS_SUCCESS;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
wild:
Packit Service a4b2a9
	ret = ctxt->getautomntbyname_r("/", &value, sss_ctxt);
Packit Service a4b2a9
	if (ret && ret != ENOENT) {
Packit Service a4b2a9
		char *estr = strerror_r(ret, buf, MAX_ERR_BUF);
Packit Service a4b2a9
		error(ap->logopt,
Packit Service a4b2a9
		      MODPREFIX "getautomntbyname_r: %s", estr);
Packit Service a4b2a9
		endautomntent(ap->logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
		if (value)
Packit Service a4b2a9
			free(value);
Packit Service a4b2a9
		return NSS_STATUS_UNAVAIL;
Packit Service a4b2a9
	}
Packit Service a4b2a9
	if (ret == ENOENT) {
Packit Service a4b2a9
		ret = ctxt->getautomntbyname_r("*", &value, sss_ctxt);
Packit Service a4b2a9
		if (ret && ret != ENOENT) {
Packit Service a4b2a9
			char *estr = strerror_r(ret, buf, MAX_ERR_BUF);
Packit Service a4b2a9
			error(ap->logopt,
Packit Service a4b2a9
			      MODPREFIX "getautomntbyname_r: %s", estr);
Packit Service a4b2a9
			endautomntent(ap->logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
			if (value)
Packit Service a4b2a9
				free(value);
Packit Service a4b2a9
			return NSS_STATUS_UNAVAIL;
Packit Service a4b2a9
		}
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	if (ret == ENOENT) {
Packit Service a4b2a9
		/* Failed to find wild entry, update cache if needed */
Packit Service a4b2a9
		cache_writelock(mc);
Packit Service a4b2a9
		we = cache_lookup_distinct(mc, "*");
Packit Service a4b2a9
		if (we) {
Packit Service a4b2a9
			/* Wildcard entry existed and is now gone */
Packit Service a4b2a9
			if (we->source == source) {
Packit Service a4b2a9
				cache_delete(mc, "*");
Packit Service a4b2a9
				source->stale = 1;
Packit Service a4b2a9
			}
Packit Service a4b2a9
		}
Packit Service a4b2a9
Packit Service a4b2a9
		/* Not found in the map but found in the cache */
Packit Service a4b2a9
		struct mapent *exists = cache_lookup_distinct(mc, qKey);
Packit Service a4b2a9
		if (exists && exists->source == source) {
Packit Service a4b2a9
			if (exists->mapent) {
Packit Service a4b2a9
				free(exists->mapent);
Packit Service a4b2a9
				exists->mapent = NULL;
Packit Service a4b2a9
				source->stale = 1;
Packit Service a4b2a9
				exists->status = 0;
Packit Service a4b2a9
			}
Packit Service a4b2a9
		}
Packit Service a4b2a9
		cache_unlock(mc);
Packit Service a4b2a9
		endautomntent(ap->logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
		return NSS_STATUS_NOTFOUND;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	cache_writelock(mc);
Packit Service a4b2a9
	/* Wildcard not in map but now is */
Packit Service a4b2a9
	we = cache_lookup_distinct(mc, "*");
Packit Service a4b2a9
	if (!we)
Packit Service a4b2a9
		source->stale = 1;
Packit Service a4b2a9
	ret = cache_update(mc, source, "*", value, age);
Packit Service a4b2a9
	cache_unlock(mc);
Packit Service a4b2a9
Packit Service a4b2a9
	endautomntent(ap->logopt, ctxt, &sss_ctxt);
Packit Service a4b2a9
        free(value);
Packit Service a4b2a9
Packit Service a4b2a9
	return NSS_STATUS_SUCCESS;
Packit Service a4b2a9
}
Packit Service a4b2a9
Packit Service a4b2a9
static int check_map_indirect(struct autofs_point *ap,
Packit Service a4b2a9
			      char *key, int key_len,
Packit Service a4b2a9
			      struct lookup_context *ctxt)
Packit Service a4b2a9
{
Packit Service a4b2a9
	struct map_source *source;
Packit Service a4b2a9
	struct mapent_cache *mc;
Packit Service a4b2a9
	struct mapent *me;
Packit Service a4b2a9
	time_t now = monotonic_time(NULL);
Packit Service a4b2a9
	time_t t_last_read;
Packit Service a4b2a9
	int ret, cur_state;
Packit Service a4b2a9
Packit Service a4b2a9
	source = ap->entry->current;
Packit Service a4b2a9
	ap->entry->current = NULL;
Packit Service a4b2a9
	master_source_current_signal(ap->entry);
Packit Service a4b2a9
Packit Service a4b2a9
	mc = source->mc;
Packit Service a4b2a9
Packit Service a4b2a9
	master_source_current_wait(ap->entry);
Packit Service a4b2a9
	ap->entry->current = source;
Packit Service a4b2a9
Packit Service a4b2a9
	pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, &cur_state);
Packit Service a4b2a9
	ret = lookup_one(ap, key, key_len, ctxt);
Packit Service a4b2a9
	if (ret == NSS_STATUS_NOTFOUND) {
Packit Service a4b2a9
		pthread_setcancelstate(cur_state, NULL);
Packit Service a4b2a9
		return ret;
Packit Service a4b2a9
	} else if (ret == NSS_STATUS_UNAVAIL) {
Packit Service a4b2a9
		/*
Packit Service a4b2a9
		 * If the server is down and the entry exists in the cache
Packit Service a4b2a9
		 * and belongs to this map return success and use the entry.
Packit Service a4b2a9
		 */
Packit Service a4b2a9
		struct mapent *exists = cache_lookup(mc, key);
Packit Service a4b2a9
		if (exists && exists->source == source) {
Packit Service a4b2a9
			pthread_setcancelstate(cur_state, NULL);
Packit Service a4b2a9
			return NSS_STATUS_SUCCESS;
Packit Service a4b2a9
		}
Packit Service a4b2a9
		pthread_setcancelstate(cur_state, NULL);
Packit Service a4b2a9
Packit Service a4b2a9
		warn(ap->logopt,
Packit Service a4b2a9
		     MODPREFIX "lookup for %s failed: connection failed", key);
Packit Service a4b2a9
Packit Service a4b2a9
		return ret;
Packit Service a4b2a9
	}
Packit Service a4b2a9
	pthread_setcancelstate(cur_state, NULL);
Packit Service a4b2a9
Packit Service a4b2a9
	/*
Packit Service a4b2a9
	 * Check for map change and update as needed for
Packit Service a4b2a9
	 * following cache lookup.
Packit Service a4b2a9
	 */
Packit Service a4b2a9
	cache_readlock(mc);
Packit Service a4b2a9
	t_last_read = ap->exp_runfreq + 1;
Packit Service a4b2a9
	me = cache_lookup_first(mc);
Packit Service a4b2a9
	while (me) {
Packit Service a4b2a9
		if (me->source == source) {
Packit Service a4b2a9
			t_last_read = now - me->age;
Packit Service a4b2a9
			break;
Packit Service a4b2a9
		}
Packit Service a4b2a9
		me = cache_lookup_next(mc, me);
Packit Service a4b2a9
	}
Packit Service a4b2a9
	cache_unlock(mc);
Packit Service a4b2a9
Packit Service a4b2a9
	if (t_last_read > ap->exp_runfreq && ret & CHE_UPDATED)
Packit Service a4b2a9
		source->stale = 1;
Packit Service a4b2a9
Packit Service a4b2a9
	cache_readlock(mc);
Packit Service a4b2a9
	me = cache_lookup_distinct(mc, "*");
Packit Service a4b2a9
	if (ret == CHE_MISSING && (!me || me->source != source)) {
Packit Service a4b2a9
		cache_unlock(mc);
Packit Service a4b2a9
		return NSS_STATUS_NOTFOUND;
Packit Service a4b2a9
	}
Packit Service a4b2a9
	cache_unlock(mc);
Packit Service a4b2a9
Packit Service a4b2a9
	return NSS_STATUS_SUCCESS;
Packit Service a4b2a9
}
Packit Service a4b2a9
Packit Service a4b2a9
int lookup_mount(struct autofs_point *ap, const char *name, int name_len, void *context)
Packit Service a4b2a9
{
Packit Service a4b2a9
	struct lookup_context *ctxt = (struct lookup_context *) context;
Packit Service a4b2a9
	struct map_source *source;
Packit Service a4b2a9
	struct mapent_cache *mc;
Packit Service a4b2a9
	struct mapent *me;
Packit Service a4b2a9
	char key[KEY_MAX_LEN + 1];
Packit Service a4b2a9
	int key_len;
Packit Service a4b2a9
	char *mapent = NULL;
Packit Service a4b2a9
	char mapent_buf[MAPENT_MAX_LEN + 1];
Packit Service a4b2a9
	int ret;
Packit Service a4b2a9
Packit Service a4b2a9
	source = ap->entry->current;
Packit Service a4b2a9
	ap->entry->current = NULL;
Packit Service a4b2a9
	master_source_current_signal(ap->entry);
Packit Service a4b2a9
Packit Service a4b2a9
	mc = source->mc;
Packit Service a4b2a9
Packit Service a4b2a9
	debug(ap->logopt, MODPREFIX "looking up %s", name);
Packit Service a4b2a9
Packit Service a4b2a9
	key_len = snprintf(key, KEY_MAX_LEN + 1, "%s", name);
Packit Service a4b2a9
	if (key_len > KEY_MAX_LEN)
Packit Service a4b2a9
		return NSS_STATUS_NOTFOUND;
Packit Service a4b2a9
Packit Service a4b2a9
	/* Check if we recorded a mount fail for this key anywhere */
Packit Service a4b2a9
	me = lookup_source_mapent(ap, key, LKP_DISTINCT);
Packit Service a4b2a9
	if (me) {
Packit Service a4b2a9
		if (me->status >= monotonic_time(NULL)) {
Packit Service a4b2a9
			cache_unlock(me->mc);
Packit Service a4b2a9
			return NSS_STATUS_NOTFOUND;
Packit Service a4b2a9
		} else {
Packit Service a4b2a9
			struct mapent_cache *smc = me->mc;
Packit Service a4b2a9
			struct mapent *sme;
Packit Service a4b2a9
Packit Service a4b2a9
			if (me->mapent)
Packit Service a4b2a9
				cache_unlock(smc);
Packit Service a4b2a9
			else {
Packit Service a4b2a9
				cache_unlock(smc);
Packit Service a4b2a9
				cache_writelock(smc);
Packit Service a4b2a9
				sme = cache_lookup_distinct(smc, key);
Packit Service a4b2a9
				/* Negative timeout expired for non-existent entry. */
Packit Service a4b2a9
				if (sme && !sme->mapent) {
Packit Service a4b2a9
					if (cache_pop_mapent(sme) == CHE_FAIL)
Packit Service a4b2a9
						cache_delete(smc, key);
Packit Service a4b2a9
				}
Packit Service a4b2a9
				cache_unlock(smc);
Packit Service a4b2a9
			}
Packit Service a4b2a9
		}
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
        /*
Packit Service a4b2a9
	 * We can't check the direct mount map as if it's not in
Packit Service a4b2a9
	 * the map cache already we never get a mount lookup, so
Packit Service a4b2a9
	 * we never know about it.
Packit Service a4b2a9
	 */
Packit Service a4b2a9
	if (ap->type == LKP_INDIRECT && *key != '/') {
Packit Service a4b2a9
		int status;
Packit Service a4b2a9
		char *lkp_key;
Packit Service a4b2a9
Packit Service a4b2a9
		cache_readlock(mc);
Packit Service a4b2a9
		me = cache_lookup_distinct(mc, key);
Packit Service a4b2a9
		if (me && me->multi)
Packit Service a4b2a9
			lkp_key = strdup(me->multi->key);
Packit Service a4b2a9
		else
Packit Service a4b2a9
			lkp_key = strdup(key);
Packit Service a4b2a9
		cache_unlock(mc);
Packit Service a4b2a9
Packit Service a4b2a9
		if (!lkp_key)
Packit Service a4b2a9
			return NSS_STATUS_UNKNOWN;
Packit Service a4b2a9
Packit Service a4b2a9
		master_source_current_wait(ap->entry);
Packit Service a4b2a9
		ap->entry->current = source;
Packit Service a4b2a9
Packit Service a4b2a9
		status = check_map_indirect(ap, lkp_key, strlen(lkp_key), ctxt);
Packit Service a4b2a9
		free(lkp_key);
Packit Service a4b2a9
		if (status)
Packit Service a4b2a9
			return status;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	/*
Packit Service a4b2a9
	 * We can't take the writelock for direct mounts. If we're
Packit Service a4b2a9
	 * starting up or trying to re-connect to an existing direct
Packit Service a4b2a9
	 * mount we could be iterating through the map entries with
Packit Service a4b2a9
	 * the readlock held. But we don't need to update the cache
Packit Service a4b2a9
	 * when we're starting up so just take the readlock in that
Packit Service a4b2a9
	 */
Packit Service a4b2a9
	if (ap->flags & MOUNT_FLAG_REMOUNT)
Packit Service a4b2a9
		cache_readlock(mc);
Packit Service 979a2c
	else
Packit Service 979a2c
		cache_writelock(mc);
Packit Service a4b2a9
	me = cache_lookup(mc, key);
Packit Service a4b2a9
	/* Stale mapent => check for entry in alternate source or wildcard */
Packit Service a4b2a9
	if (me && !me->mapent) {
Packit Service a4b2a9
		while ((me = cache_lookup_key_next(me)))
Packit Service a4b2a9
			if (me->source == source)
Packit Service a4b2a9
				break;
Packit Service a4b2a9
		if (!me)
Packit Service a4b2a9
			me = cache_lookup_distinct(mc, "*");
Packit Service a4b2a9
	}
Packit Service a4b2a9
	if (me && me->mapent) {
Packit Service a4b2a9
		/*
Packit Service a4b2a9
		 * If this is a lookup add wildcard match for later validation
Packit Service a4b2a9
		 * checks and negative cache lookups.
Packit Service a4b2a9
		 */
Packit Service a4b2a9
		if (ap->type == LKP_INDIRECT && *me->key == '*' &&
Packit Service a4b2a9
		   !(ap->flags & MOUNT_FLAG_REMOUNT)) {
Packit Service a4b2a9
			ret = cache_update(mc, source, key, me->mapent, me->age);
Packit Service a4b2a9
			if (!(ret & (CHE_OK | CHE_UPDATED)))
Packit Service a4b2a9
				me = NULL;
Packit Service a4b2a9
		}
Packit Service a4b2a9
		if (me && (me->source == source || *me->key == '/')) {
Packit Service a4b2a9
			strcpy(mapent_buf, me->mapent);
Packit Service a4b2a9
			mapent = mapent_buf;
Packit Service a4b2a9
		}
Packit Service a4b2a9
	}
Packit Service a4b2a9
	cache_unlock(mc);
Packit Service a4b2a9
Packit Service a4b2a9
	if (!mapent)
Packit Service a4b2a9
		return NSS_STATUS_TRYAGAIN;
Packit Service a4b2a9
Packit Service a4b2a9
	master_source_current_wait(ap->entry);
Packit Service a4b2a9
	ap->entry->current = source;
Packit Service a4b2a9
Packit Service a4b2a9
	debug(ap->logopt, MODPREFIX "%s -> %s", key, mapent);
Packit Service a4b2a9
	ret = ctxt->parse->parse_mount(ap, key, key_len,
Packit Service a4b2a9
				       mapent, ctxt->parse->context);
Packit Service a4b2a9
	if (ret) {
Packit Service a4b2a9
		/* Don't update negative cache when re-connecting */
Packit Service a4b2a9
		if (ap->flags & MOUNT_FLAG_REMOUNT)
Packit Service a4b2a9
			return NSS_STATUS_TRYAGAIN;
Packit Service a4b2a9
		cache_writelock(mc);
Packit Service a4b2a9
		cache_update_negative(mc, source, key, ap->negative_timeout);
Packit Service a4b2a9
		cache_unlock(mc);
Packit Service a4b2a9
		return NSS_STATUS_TRYAGAIN;
Packit Service a4b2a9
	}
Packit Service a4b2a9
Packit Service a4b2a9
	return NSS_STATUS_SUCCESS;
Packit Service a4b2a9
}
Packit Service a4b2a9
Packit Service a4b2a9
int lookup_done(void *context)
Packit Service a4b2a9
{
Packit Service a4b2a9
	struct lookup_context *ctxt = (struct lookup_context *) context;
Packit Service a4b2a9
	int rv = close_parse(ctxt->parse);
Packit Service a4b2a9
	dlclose(ctxt->dlhandle);
Packit Service a4b2a9
	free(ctxt);
Packit Service a4b2a9
	return rv;
Packit Service a4b2a9
}