#include <stdlib.h>
#include <string.h>
#include <strings.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <inet/ip6.h>
#include "cache.h"
static int ipaddr_compar(const void *, const void *);
static uint_t ipaddr_gethash(nss_XbyY_key_t *, int);
static void ipaddr_getlogstr(char *, char *, size_t, nss_XbyY_args_t *);
static int ipname_compar(const void *, const void *);
static uint_t ipname_gethash(nss_XbyY_key_t *, int);
static void ipname_getlogstr(char *, char *, size_t, nss_XbyY_args_t *);
#define nnam_db ctx->nsc_db[0]
#define addr_db ctx->nsc_db[1]
#define NSC_NAME_IPNODES_BYNAME "getipnodebyname"
#define NSC_NAME_IPNODES_BYADDR "getipnodebyaddr"
void
ipnode_init_ctx(nsc_ctx_t *ctx) {
ctx->dbname = NSS_DBNAM_IPNODES;
ctx->file_name = "/etc/inet/ipnodes";
ctx->db_count = 2;
nnam_db = make_cache(nsc_key_other,
NSS_DBOP_IPNODES_BYNAME,
NSC_NAME_IPNODES_BYNAME,
ipname_compar,
ipname_getlogstr,
ipname_gethash, nsc_ht_default, -1);
addr_db = make_cache(nsc_key_other,
NSS_DBOP_IPNODES_BYADDR,
NSC_NAME_IPNODES_BYADDR,
ipaddr_compar,
ipaddr_getlogstr,
ipaddr_gethash, nsc_ht_default, -1);
}
static int
ipaddr_compar(const void *n1, const void *n2) {
nsc_entry_t *e1, *e2;
int res, l1, l2;
e1 = (nsc_entry_t *)n1;
e2 = (nsc_entry_t *)n2;
if (e1->key.hostaddr.type > e2->key.hostaddr.type)
return (1);
else if (e1->key.hostaddr.type < e2->key.hostaddr.type)
return (-1);
l1 = e1->key.hostaddr.len;
l2 = e2->key.hostaddr.len;
res = memcmp(e1->key.hostaddr.addr, e2->key.hostaddr.addr,
(l2 > l1)?l1:l2);
return ((res) ? _NSC_INT_KEY_CMP(res, 0) : _NSC_INT_KEY_CMP(l1, l2));
}
static uint_t
ipaddr_gethash(nss_XbyY_key_t *key, int htsize) {
return (db_gethash(key->hostaddr.addr,
key->hostaddr.len, htsize));
}
static void
ipaddr_getlogstr(char *name, char *whoami, size_t len, nss_XbyY_args_t *argp) {
char addr[INET6_ADDRSTRLEN];
if (inet_ntop(argp->key.hostaddr.type, argp->key.hostaddr.addr, addr,
sizeof (addr)) == NULL) {
(void) snprintf(whoami, len, "%s", name);
} else {
(void) snprintf(whoami, len, "%s [key=%s addrtype=%d]",
name,
addr, argp->key.hostaddr.type);
}
}
static int
ipname_compar(const void *n1, const void *n2) {
nsc_entry_t *e1, *e2;
int res, l1, l2;
e1 = (nsc_entry_t *)n1;
e2 = (nsc_entry_t *)n2;
if (e1->key.ipnode.af_family > e2->key.ipnode.af_family)
return (1);
else if (e1->key.ipnode.af_family < e2->key.ipnode.af_family)
return (-1);
l1 = strlen(e1->key.ipnode.name);
l2 = strlen(e2->key.ipnode.name);
res = strncasecmp(e1->key.ipnode.name, e2->key.ipnode.name,
(l1 > l2)?l1:l2);
return (_NSC_INT_KEY_CMP(res, 0));
}
static uint_t
ipname_gethash(nss_XbyY_key_t *key, int htsize) {
return (cis_gethash(key->ipnode.name, htsize));
}
static void
ipname_getlogstr(char *name, char *whoami, size_t len, nss_XbyY_args_t *argp) {
(void) snprintf(whoami, len, "%s [key=%s:af=%d:flags=%d]",
name,
argp->key.ipnode.name,
argp->key.ipnode.af_family,
argp->key.ipnode.flags);
}