/* vi: set sw=4 ts=4: */
/*
 * Copyright (C) 2010 Bernhard Reutner-Fischer <uclibc@uclibc.org>
 *
 * Licensed under LGPL v2.1 or later, see the file COPYING.LIB in this tarball.
 */

/* /etc/networks
#   network-name  number     [aliases ...]
loopback          127.0.0.0  # optional aliases

network-name: symbolic name of the netwkork
number: official number of the network in dotted quad
aliases: case sensitive optional space or tab separated list of other names
*/

#include <features.h>
#include <netdb.h>
#include <string.h>
#include <stdlib.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <errno.h>
#include <unistd.h>
#include "internal/parse_config.h"

#include <bits/uClibc_mutex.h>
__UCLIBC_MUTEX_STATIC(mylock, PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP);

#define	MAXALIASES	35
#define BUFSZ		(80) /* one line */
#define SBUFSIZE	(BUFSZ + 1 + (sizeof(char *) * MAXALIASES))

static parser_t *netp = NULL;
static struct netent nete;
static char *netbuf = NULL;
static smallint net_stayopen;

void setnetent(int stayopen)
{
	__UCLIBC_MUTEX_LOCK(mylock);
	if (netp)
		config_close(netp);
	netp = config_open(_PATH_NETWORKS);
	if (stayopen)
		net_stayopen = 1;
	__UCLIBC_MUTEX_UNLOCK(mylock);
}
libc_hidden_def(setnetent)

void endnetent(void)
{
	__UCLIBC_MUTEX_LOCK(mylock);
	if (netp) {
		config_close(netp);
		netp = NULL;
	}
	net_stayopen = 0;
	__UCLIBC_MUTEX_UNLOCK(mylock);
}
libc_hidden_def(endnetent)

int getnetent_r(struct netent *result_buf,
				char *buf, size_t buflen, struct netent **result,
				int *h_errnop
				 )
{
	char **alias, *cp = NULL;
	char **net_aliases;
	char **tok = NULL;
	const size_t aliaslen = sizeof(*net_aliases) * MAXALIASES;
	int ret = ERANGE;

	*result = NULL;
	if (buflen < aliaslen
		|| (buflen - aliaslen) < BUFSZ + 1)
		goto DONE_NOUNLOCK;

	__UCLIBC_MUTEX_LOCK(mylock);
	ret = ENOENT;
	if (netp == NULL)
		setnetent(net_stayopen);
	if (netp == NULL)
		goto DONE;
	netp->data = buf;
	netp->data_len = aliaslen;
	netp->line_len = buflen - aliaslen;
	/* <name>[[:space:]]<netnumber>[[:space:]][<aliases>] */
	if (!config_read(netp, &tok, 3, 2, "# \t/", PARSE_NORMAL)) {
		goto DONE;
	}
	result_buf->n_name = *(tok++);
	{
		struct addrinfo hints, *addri;
# define sa4_to_uint32(sa) \
	(ntohl(((struct sockaddr_in*)sa)->sin_addr.s_addr))
#ifdef __UCLIBC_HAS_IPV6__
# define sa6_to_uint8(sa) \
	(ntohl(((struct sockaddr_in6*)sa)->sin6_addr.s6_addr))
#endif
		memset(&hints, 0, sizeof(struct addrinfo));
		hints.ai_family = AF_UNSPEC;
		hints.ai_flags = AI_NUMERICHOST;
		getaddrinfo(*(tok++), NULL, &hints, &addri);
		result_buf->n_addrtype = addri->ai_family;
		result_buf->n_net =
#if 0 /*FIXME: implement me! def __UCLIBC_HAS_IPV6__ */
			addri->ai_family == AF_INET6 ? sa6_to_uint8(addri->ai_addr) :
#endif
			sa4_to_uint32(addri->ai_addr);
		freeaddrinfo(addri);
	}
	result_buf->n_aliases = alias = net_aliases = tok;
	cp = *alias;
	while (cp && *cp) {
		if (alias < &net_aliases[MAXALIASES - 1])
			*alias++ = cp;
		cp = strpbrk(cp, " \t");
		if (cp != NULL)
			*cp++ = '\0';
	}
	*alias = NULL;
	*result = result_buf;
	ret = 0;
 DONE:
	__UCLIBC_MUTEX_UNLOCK(mylock);
 DONE_NOUNLOCK:
	errno = ret;
	return errno;
}
libc_hidden_def(getnetent_r)

static void __initbuf(void)
{
	if (!netbuf) {
		netbuf = malloc(SBUFSIZE);
		if (!netbuf)
			abort();
	}
}

struct netent *getnetent(void)
{
	struct netent *result;
	int herrnop;

	__initbuf();
	getnetent_r(&nete, netbuf, SBUFSIZE, &result, &herrnop);
	return result;
}

int getnetbyname_r(const char *name,
					struct netent *result_buf, char *buf, size_t buflen,
					struct netent **result,
					int *h_errnop
					)
{
	register char **cp;
	int ret, herrnop;

	__UCLIBC_MUTEX_LOCK(mylock);
	setnetent(net_stayopen);
	while (!(ret = getnetent_r(result_buf, buf, buflen, result, &herrnop))) {
		if (strcmp(name, result_buf->n_name) == 0)
			break;
		for (cp = result_buf->n_aliases; *cp; cp++)
			if (strcmp(name, *cp) == 0)
				goto gotname;
	}
 gotname:
	if (!net_stayopen)
		endnetent();
	__UCLIBC_MUTEX_UNLOCK(mylock);
	return *result ? 0 : ret;
}
libc_hidden_def(getnetbyname_r)

struct netent *getnetbyname(const char *name)
{
	struct netent *result;
	int herrnop;

	__initbuf();
	getnetbyname_r(name, &nete, netbuf, SBUFSIZE, &result, &herrnop);
	return result;
}

int getnetbyaddr_r(uint32_t net, int type,
					struct netent *result_buf, char *buf,
					size_t buflen, struct netent **result,
					int *h_errnop)
{
	int ret, herrnop;

	__UCLIBC_MUTEX_LOCK(mylock);
	setnetent(net_stayopen);
	while (!(ret = getnetent_r(result_buf, buf, buflen, result, &herrnop))) {
		if (net == result_buf->n_net && type == result_buf->n_addrtype)
			break;
	}
	if (!net_stayopen)
		endnetent();
	__UCLIBC_MUTEX_UNLOCK(mylock);
	return *result ? 0 : ret;
}
libc_hidden_def(getnetbyaddr_r)

struct netent *getnetbyaddr(uint32_t net, int type)
{
	struct netent *result;
	int herrnop;

	__initbuf();
	getnetbyaddr_r(net, type, &nete, netbuf, SBUFSIZE, &result, &herrnop);
	return result;
}