#include <assert.h>
#include <errno.h>
#include <ctype.h>
#include <nfs/nfs.h>
#include <stdlib.h>
#include <stddef.h>
#include <string.h>
#include <strings.h>
#include <syslog.h>
#include <unistd.h>
#include <dirent.h>
#include <ndbm.h>
#include <time.h>
#include <libintl.h>
#include <sys/types.h>
#include <nfs/nfs.h>
#include <nfs/nfs_log.h>
#include "fhtab.h"
#include "nfslogd.h"
#define ROUNDUP32(val) (((val) + 3) & ~3)
#define DB_VERSION_STRING "NFSLOG_DB_VERSION"
#define DB_VERSION "1"
#define MAX_PRUNE_REC_CNT 100000
fhandle_t public_fh = { 0 };
struct db_list {
fsid_t fsid;
char *path;
DBM *db;
bool_t getall;
struct db_list *next;
};
static struct db_list *db_fs_list = NULL;
static char err_str[] = "DB I/O error has occurred";
struct link_keys {
fh_secondary_key lnkey;
int lnsize;
struct link_keys *next;
};
extern int debug;
extern time_t mapping_update_interval;
extern time_t prune_timeout;
static int fill_link_key(char *linkkey, fhandle_t *dfh, char *name);
static struct db_list *db_get_db(char *fhpath, fsid_t *fsid, int *errorp,
int create_flag);
static struct db_list *db_get_all_databases(char *fhpath, bool_t getall);
static void debug_print_fhlist(FILE *fp, fhlist_ent *fhrecp);
static void debug_print_linkinfo(FILE *fp, linkinfo_ent *fhrecp);
static void debug_print_key(FILE *fp, char *str1, char *str2, char *key,
int ksize);
static void debug_print_key_and_data(FILE *fp, char *str1, char *str2,
char *key, int ksize, char *data, int dsize);
static int store_record(struct db_list *dbp, void *keyaddr, int keysize,
void *dataaddr, int datasize, char *str);
static void *fetch_record(struct db_list *dbp, void *keyaddr, int keysize,
void *dataaddr, int *errorp, char *str);
static int delete_record(struct db_list *dbp, void *keyaddr, int keysize,
char *str);
static int db_update_fhrec(struct db_list *dbp, void *keyaddr, int keysize,
fhlist_ent *fhrecp, char *str);
static int db_update_linkinfo(struct db_list *dbp, void *keyaddr, int keysize,
linkinfo_ent *linkp, char *str);
static fhlist_ent *create_primary_struct(struct db_list *dbp, fhandle_t *dfh,
char *name, fhandle_t *fh, uint_t flags, fhlist_ent *fhrecp,
int *errorp);
static fhlist_ent *db_add_primary(struct db_list *dbp, fhandle_t *dfh,
char *name, fhandle_t *fh, uint_t flags, fhlist_ent *fhrecp,
int *errorp);
static linkinfo_ent *get_next_link(struct db_list *dbp, char *linkkey,
int *linksizep, linkinfo_ent *linkp, void **cookiep,
int *errorp, char *msg);
static void free_link_cookies(void *cookie);
static void add_mc_path(struct db_list *dbp, fhandle_t *dfh, char *name,
fhlist_ent *fhrecp, linkinfo_ent *linkp, int *errorp);
static linkinfo_ent *create_link_struct(struct db_list *dbp, fhandle_t *dfh,
char *name, fhlist_ent *fhrecp, int *errorp);
static int db_add_secondary(struct db_list *dbp, fhandle_t *dfh, char *name,
fhandle_t *fh, fhlist_ent *fhrecp);
static linkinfo_ent *update_next_link(struct db_list *dbp, char *nextkey,
int nextsize, char *prevkey, int prevsize, int *errorp);
static int update_prev_link(struct db_list *dbp, char *nextkey, int nextsize,
char *prevkey, int prevsize);
static linkinfo_ent *update_linked_list(struct db_list *dbp, char *nextkey,
int nextsize, char *prevkey, int prevsize, int *errorp);
static int db_update_primary_new_head(struct db_list *dbp,
linkinfo_ent *dellinkp, linkinfo_ent *nextlinkp, fhlist_ent *fhrecp);
static int delete_link_by_key(struct db_list *dbp, char *linkkey,
int *linksizep, int *errorp, char *errstr);
static int delete_link(struct db_list *dbp, fhandle_t *dfh, char *name,
char *nextlinkkey, int *nextlinksizep, int *errorp, char *errstr);
static int
fill_link_key(char *linkkey, fhandle_t *dfh, char *name)
{
int linksize, linksize32;
(void) memcpy(linkkey, &dfh->fh_data, dfh->fh_len);
(void) strcpy(&linkkey[dfh->fh_len], name);
linksize = dfh->fh_len + strlen(name) + 1;
linksize32 = ROUNDUP32(linksize);
if (linksize32 > linksize)
bzero(&linkkey[linksize], linksize32 - linksize);
return (linksize32);
}
static struct db_list *
db_get_db(char *fhpath, fsid_t *fsid, int *errorp, int create_flag)
{
struct db_list *p, *newp;
char fsidstr[30];
datum key, data;
*errorp = 0;
for (p = db_fs_list;
(p != NULL) && memcmp(&p->fsid, fsid, sizeof (*fsid));
p = p->next);
if (p != NULL) {
return (p);
}
if ((newp = calloc(1, sizeof (*newp))) == NULL) {
*errorp = errno;
syslog(LOG_ERR, gettext(
"db_get_db: malloc db failed: Error %s"),
strerror(*errorp));
return (NULL);
}
(void) sprintf(fsidstr, "%08x%08x", fsid->val[0], fsid->val[1]);
if ((newp->path = malloc(strlen(fhpath) + 2 + strlen(fsidstr)))
== NULL) {
*errorp = errno;
syslog(LOG_ERR, gettext(
"db_get_db: malloc dbpath failed: Error %s"),
strerror(*errorp));
goto err_exit;
}
(void) sprintf(newp->path, "%s.%s", fhpath, fsidstr);
if ((newp->db = dbm_open(newp->path, create_flag | O_RDWR, 0666))
== NULL) {
*errorp = errno;
syslog(LOG_ERR, gettext(
"db_get_db: dbm_open db '%s' failed: Error %s"),
newp->path, strerror(*errorp));
if (*errorp == 0)
*errorp = -1;
goto err_exit;
}
key.dptr = DB_VERSION_STRING;
key.dsize = strlen(DB_VERSION_STRING);
data = dbm_fetch(newp->db, key);
if (data.dptr == NULL) {
data.dptr = DB_VERSION;
data.dsize = strlen(DB_VERSION);
(void) dbm_store(newp->db, key, data, DBM_INSERT);
}
(void) memcpy(&newp->fsid, fsid, sizeof (*fsid));
newp->next = db_fs_list;
db_fs_list = newp;
if (debug > 1) {
(void) printf("db_get_db: db %s opened\n", newp->path);
}
return (newp);
err_exit:
if (newp != NULL) {
if (newp->db != NULL) {
dbm_close(newp->db);
}
if (newp->path != NULL) {
free(newp->path);
}
free(newp);
}
return (NULL);
}
static struct db_list *
db_get_all_databases(char *fhpath, bool_t getall)
{
char *dirptr, *fhdir, *fhpathname;
int len, error;
DIR *dirp;
struct dirent *dp;
fsid_t fsid;
struct db_list *dbp, *ret_dbp;
for (dbp = db_fs_list; dbp != NULL; dbp = dbp->next) {
if (strncmp(fhpath, dbp->path, strlen(fhpath)) == 0)
break;
}
if (dbp != NULL) {
if (!getall || dbp->getall)
return (dbp);
}
if ((fhdir = strdup(fhpath)) == NULL) {
syslog(LOG_ERR, gettext(
"db_get_all_databases: strdup '%s' Error '%s*'"),
fhpath, strerror(errno));
return (NULL);
}
fhpathname = NULL;
ret_dbp = NULL;
if ((dirptr = strrchr(fhdir, '/')) == NULL) {
goto exit;
}
if ((fhpathname = strdup(&dirptr[1])) == NULL) {
syslog(LOG_ERR, gettext(
"db_get_all_databases: strdup '%s' Error '%s*'"),
&dirptr[1], strerror(errno));
goto exit;
}
dirptr[1] = '\0';
if (debug > 2) {
(void) printf("db_get_all_databases: search '%s' for '%s*'\n",
fhdir, fhpathname);
}
if ((dirp = opendir(fhdir)) == NULL) {
syslog(LOG_ERR, gettext(
"db_get_all_databases: opendir '%s' Error '%s*'"),
fhdir, strerror(errno));
goto exit;
}
len = strlen(fhpathname);
while ((dp = readdir(dirp)) != NULL) {
if (strncmp(fhpathname, dp->d_name, len) == 0) {
dirptr = &dp->d_name[len + 1];
if (*(dirptr - 1) != '.') {
continue;
}
(void) sscanf(dirptr, "%08lx%08lx",
(ulong_t *)&fsid.val[0], (ulong_t *)&fsid.val[1]);
dbp = db_get_db(fhpath, &fsid, &error, 0);
if (dbp != NULL) {
ret_dbp = dbp;
if (!getall)
break;
dbp->getall = TRUE;
}
}
}
(void) closedir(dirp);
exit:
if (fhpathname != NULL)
free(fhpathname);
if (fhdir != NULL)
free(fhdir);
return (ret_dbp);
}
static void
debug_print_key(FILE *fp, char *str1, char *str2, char *key, int ksize)
{
(void) fprintf(fp, "%s: %s key (%d) ", str1, str2, ksize);
debug_opaque_print(fp, key, ksize);
if (ksize >= NFS_FHMAXDATA) {
(void) fprintf(fp, ": inode ");
debug_opaque_print(fp, &key[2], sizeof (int));
(void) fprintf(fp, ", gen ");
debug_opaque_print(fp, &key[2 + sizeof (int)], sizeof (int));
if (ksize > NFS_FHMAXDATA) {
(void) fprintf(fp, ", name '%s'", &key[NFS_FHMAXDATA]);
}
}
(void) fprintf(fp, "\n");
}
static void
debug_print_linkinfo(FILE *fp, linkinfo_ent *linkp)
{
if (linkp == NULL)
return;
(void) fprintf(fp, "linkinfo:\ndfh: ");
debug_opaque_print(fp, (void *)&linkp->dfh, sizeof (linkp->dfh));
(void) fprintf(fp, "\nname: '%s'", LN_NAME(linkp));
(void) fprintf(fp, "\nmtime 0x%x, atime 0x%x, flags 0x%x, reclen %d\n",
linkp->mtime, linkp->atime, linkp->flags, linkp->reclen);
(void) fprintf(fp, "offsets: fhkey %d, name %d, next %d, prev %d\n",
linkp->fhkey_offset, linkp->name_offset, linkp->next_offset,
linkp->prev_offset);
debug_print_key(fp, "fhkey", "", LN_FHKEY(linkp), LN_FHKEY_LEN(linkp));
debug_print_key(fp, "next", "", LN_NEXT(linkp), LN_NEXT_LEN(linkp));
debug_print_key(fp, "prev", "", LN_PREV(linkp), LN_PREV_LEN(linkp));
}
static void
debug_print_fhlist(FILE *fp, fhlist_ent *fhrecp)
{
if (fhrecp == NULL)
return;
(void) fprintf(fp, "fhrec:\nfh: ");
debug_opaque_print(fp, (void *)&fhrecp->fh, sizeof (fhrecp->fh));
(void) fprintf(fp, "name '%s', dfh: ", fhrecp->name);
debug_opaque_print(fp, (void *)&fhrecp->dfh, sizeof (fhrecp->dfh));
(void) fprintf(fp, "\nmtime 0x%x, atime 0x%x, flags 0x%x, reclen %d\n",
fhrecp->mtime, fhrecp->atime, fhrecp->flags, fhrecp->reclen);
}
static void
debug_print_key_and_data(FILE *fp, char *str1, char *str2, char *key,
int ksize, char *data, int dsize)
{
debug_print_key(fp, str1, str2, key, ksize);
(void) fprintf(fp, " ==> (%p,%d)\n", (void *)data, dsize);
if (ksize > NFS_FHMAXDATA) {
linkinfo_ent inf;
(void) memcpy(&inf, data, sizeof (linkinfo_ent));
debug_print_linkinfo(fp, &inf);
} else if (ksize == NFS_FHMAXDATA) {
fhlist_ent inf;
(void) memcpy(&inf, data, sizeof (linkinfo_ent));
debug_print_fhlist(fp, &inf);
} else {
debug_opaque_print(fp, data, dsize);
}
}
static int
store_record(struct db_list *dbp, void *keyaddr, int keysize, void *dataaddr,
int datasize, char *str)
{
datum key, data;
int error;
char *errfmt = "store_record: dbm_store failed, Error: %s\n";
char *err;
errno = 0;
key.dptr = keyaddr;
key.dsize = keysize;
data.dptr = dataaddr;
data.dsize = datasize;
if (debug > 2) {
debug_print_key_and_data(stdout, str, "dbm_store:\n ",
key.dptr, key.dsize, data.dptr, data.dsize);
}
if (dbm_store(dbp->db, key, data, DBM_REPLACE) < 0) {
error = dbm_error(dbp->db);
dbm_clearerr(dbp->db);
if (error) {
if (errno)
err = strerror(errno);
else {
err = err_str;
errno = EIO;
}
} else {
err = err_str;
errno = -1;
}
if (debug) {
debug_print_key(stderr, str, "store_record:"
"dbm_store:\n", key.dptr, key.dsize);
(void) fprintf(stderr, errfmt, err);
} else
syslog(LOG_ERR, gettext(errfmt), err);
return (errno);
}
return (0);
}
static void *
fetch_record(struct db_list *dbp, void *keyaddr, int keysize, void *dataaddr,
int *errorp, char *str)
{
datum key, data;
char *errfmt = "fetch_record: dbm_fetch failed, Error: %s\n";
char *err;
errno = 0;
*errorp = 0;
key.dptr = keyaddr;
key.dsize = keysize;
data = dbm_fetch(dbp->db, key);
if (data.dptr == NULL) {
if (dbm_error(dbp->db)) {
dbm_clearerr(dbp->db);
*errorp = EIO;
err = strerror(*errorp);
syslog(LOG_ERR, gettext(errfmt), err);
} else {
*errorp = ENOENT;
}
if (debug > 3) {
err = strerror(*errorp);
debug_print_key(stderr, str, "fetch_record:"
"dbm_fetch:\n", key.dptr, key.dsize);
(void) fprintf(stderr, errfmt, err);
}
return (NULL);
}
if ((dataaddr == NULL) &&
((dataaddr = malloc(data.dsize)) == NULL)) {
*errorp = errno;
syslog(LOG_ERR, gettext(
"%s: dbm_fetch - malloc %ld: Error %s"),
str, data.dsize, strerror(*errorp));
return (NULL);
}
(void) memcpy(dataaddr, data.dptr, data.dsize);
if (debug > 3) {
debug_print_key_and_data(stdout, str, "fetch_record:"
"dbm_fetch:\n", key.dptr, key.dsize,
dataaddr, data.dsize);
}
*errorp = 0;
return (dataaddr);
}
static int
delete_record(struct db_list *dbp, void *keyaddr, int keysize, char *str)
{
datum key;
int error = 0;
char *errfmt = "delete_record: dbm_delete failed, Error: %s\n";
char *err;
errno = 0;
key.dptr = keyaddr;
key.dsize = keysize;
if (debug > 2) {
debug_print_key(stdout, str, "delete_record:"
"dbm_delete:\n", key.dptr, key.dsize);
}
if (dbm_delete(dbp->db, key) < 0) {
error = dbm_error(dbp->db);
dbm_clearerr(dbp->db);
if (error) {
if (errno)
err = strerror(errno);
else {
err = err_str;
errno = EIO;
}
} else {
err = err_str;
errno = -1;
}
if (debug) {
debug_print_key(stderr, str, "delete_record:"
"dbm_delete:\n", key.dptr, key.dsize);
(void) fprintf(stderr, errfmt, err);
} else
syslog(LOG_ERR, gettext(errfmt), err);
}
return (errno);
}
static int
db_update_fhrec(struct db_list *dbp, void *keyaddr, int keysize,
fhlist_ent *fhrecp, char *str)
{
time_t cur_time = time(0);
if (difftime(cur_time, fhrecp->atime) >= mapping_update_interval) {
fhrecp->atime = cur_time;
return (store_record(dbp, keyaddr, keysize,
fhrecp, fhrecp->reclen, str));
}
return (0);
}
static int
db_update_linkinfo(struct db_list *dbp, void *keyaddr, int keysize,
linkinfo_ent *linkp, char *str)
{
time_t cur_time = time(0);
if (difftime(cur_time, linkp->atime) >= mapping_update_interval) {
linkp->atime = cur_time;
return (store_record(dbp, keyaddr, keysize,
linkp, linkp->reclen, str));
}
return (0);
}
static fhlist_ent *
create_primary_struct(struct db_list *dbp, fhandle_t *dfh, char *name,
fhandle_t *fh, uint_t flags, fhlist_ent *fhrecp, int *errorp)
{
int reclen, reclen1;
fhlist_ent *new_fhrecp = fhrecp;
reclen1 = offsetof(fhlist_ent, name) + strlen(name) + 1;
reclen = ROUNDUP32(reclen1);
if (fhrecp == NULL) {
if ((new_fhrecp = malloc(reclen)) == NULL) {
*errorp = errno;
syslog(LOG_ERR, gettext(
"create_primary_struct: malloc %d Error %s"),
reclen, strerror(*errorp));
return (NULL);
}
}
(void) memcpy(&new_fhrecp->fh, fh, sizeof (*fh));
(void) memcpy(&new_fhrecp->dfh, dfh, sizeof (*dfh));
new_fhrecp->flags = flags;
if (dfh == &public_fh)
new_fhrecp->flags |= PUBLIC_PATH;
else
new_fhrecp->flags &= ~PUBLIC_PATH;
new_fhrecp->mtime = time(0);
new_fhrecp->atime = new_fhrecp->mtime;
(void) strcpy(new_fhrecp->name, name);
if (reclen1 < reclen) {
bzero((char *)((uintptr_t)new_fhrecp + reclen1),
reclen - reclen1);
}
new_fhrecp->reclen = reclen;
*errorp = store_record(dbp, &fh->fh_data, fh->fh_len, new_fhrecp,
new_fhrecp->reclen, "create_primary_struct");
if (*errorp != 0) {
if (fhrecp == NULL)
free(new_fhrecp);
return (NULL);
}
return (new_fhrecp);
}
static fhlist_ent *
db_add_primary(struct db_list *dbp, fhandle_t *dfh, char *name, fhandle_t *fh,
uint_t flags, fhlist_ent *fhrecp, int *errorp)
{
fhlist_ent *new_fhrecp;
fh_primary_key fhkey;
if (debug > 2)
(void) printf("db_add_primary entered: name '%s'\n", name);
bcopy(&fh->fh_data, fhkey, fh->fh_len);
new_fhrecp = fetch_record(dbp, fhkey, fh->fh_len, (void *)fhrecp,
errorp, "db_add_primary");
if (new_fhrecp != NULL) {
*errorp = db_update_fhrec(dbp, fhkey, fh->fh_len, new_fhrecp,
"db_add_primary put fhrec");
if (debug > 2)
(void) printf("db_add_primary exits(2): name '%s'\n",
name);
return (new_fhrecp);
}
new_fhrecp = create_primary_struct(dbp, dfh, name, fh, flags,
fhrecp, errorp);
if (new_fhrecp == NULL) {
if (debug > 2)
(void) printf(
"db_add_primary exits(1): name '%s' Error %s\n",
name, ((*errorp >= 0) ? strerror(*errorp) :
"Unknown"));
return (NULL);
}
if (debug > 2)
(void) printf("db_add_primary exits(0): name '%s'\n", name);
return (new_fhrecp);
}
static linkinfo_ent *
get_next_link(struct db_list *dbp, char *linkkey, int *linksizep,
linkinfo_ent *linkp, void **cookiep, int *errorp, char *msg)
{
int linksize, nextsize;
char *nextkey;
linkinfo_ent *new_linkp = linkp;
struct link_keys *lnp;
linksize = *linksizep;
if (linksize == 0)
return (NULL);
*linksizep = 0;
new_linkp = fetch_record(dbp, linkkey, linksize, (void *)linkp,
errorp, msg);
if (new_linkp == NULL)
return (NULL);
nextsize = LN_NEXT_LEN(new_linkp);
if (nextsize == 0)
return (new_linkp);
if ((lnp = malloc(sizeof (struct link_keys))) == NULL) {
syslog(LOG_ERR, gettext("get_next_key: malloc error %s\n"),
strerror(errno));
if ((new_linkp != NULL) && (linkp == NULL))
free(new_linkp);
return (NULL);
}
(void) memcpy(lnp->lnkey, linkkey, linksize);
lnp->lnsize = linksize;
lnp->next = *(struct link_keys **)cookiep;
*cookiep = (void *)lnp;
nextkey = LN_NEXT(new_linkp);
for (; lnp != NULL; lnp = lnp->next) {
if ((nextsize == lnp->lnsize) && (memcmp(
lnp->lnkey, nextkey, nextsize) == 0)) {
if (debug) {
(void) fprintf(stderr,
"%s: get_next_link: last record invalid.\n",
msg);
debug_print_key_and_data(stderr, msg,
"invalid rec:\n ", linkkey, linksize,
(char *)new_linkp, new_linkp->reclen);
}
return (new_linkp);
}
}
(void) memcpy(linkkey, nextkey, nextsize);
*linksizep = nextsize;
return (new_linkp);
}
static void
free_link_cookies(void *cookie)
{
struct link_keys *dellnp, *lnp;
lnp = (struct link_keys *)cookie;
while (lnp != NULL) {
dellnp = lnp;
lnp = lnp->next;
free(dellnp);
}
}
static void
add_mc_path(struct db_list *dbp, fhandle_t *dfh, char *name,
fhlist_ent *fhrecp, linkinfo_ent *linkp, int *errorp)
{
fh_secondary_key linkkey;
int linksize, len;
linkinfo_ent lastlink, *lastlinkp;
void *cookie;
linksize = fill_link_key(linkkey, &fhrecp->dfh, fhrecp->name);
cookie = NULL;
do {
lastlinkp = get_next_link(dbp, linkkey, &linksize, &lastlink,
&cookie, errorp, "add_mc_path");
} while (linksize > 0);
free_link_cookies(cookie);
if (lastlinkp == NULL) {
if (debug > 1) {
(void) fprintf(stderr, "add_mc_path link is null\n");
}
return;
}
linkp->prev_offset = linkp->next_offset;
linksize = fill_link_key(LN_PREV(linkp), &lastlinkp->dfh,
LN_NAME(lastlinkp));
linkp->reclen = linkp->prev_offset + linksize;
linksize = fill_link_key(linkkey, dfh, name);
*errorp = store_record(dbp, linkkey, linksize,
linkp, linkp->reclen, "add_mc_path");
if (*errorp != 0)
return;
linksize = LN_PREV_LEN(lastlinkp);
(void) memcpy(linkkey, LN_PREV(lastlinkp), linksize);
len = fill_link_key(LN_NEXT(lastlinkp), dfh, name);
lastlinkp->prev_offset = lastlinkp->next_offset + len;
(void) memcpy(LN_PREV(lastlinkp), linkkey, linksize);
lastlinkp->reclen = lastlinkp->prev_offset + linksize;
linksize = fill_link_key(linkkey, &lastlinkp->dfh, LN_NAME(lastlinkp));
*errorp = store_record(dbp, linkkey, linksize,
lastlinkp, lastlinkp->reclen, "add_mc_path prev");
}
static linkinfo_ent *
create_link_struct(struct db_list *dbp, fhandle_t *dfh, char *name,
fhlist_ent *fhrecp, int *errorp)
{
fh_secondary_key linkkey;
int len, linksize;
linkinfo_ent *linkp;
if ((linkp = malloc(sizeof (linkinfo_ent))) == NULL) {
*errorp = errno;
syslog(LOG_ERR, gettext(
"create_link_struct: malloc failed: Error %s"),
strerror(*errorp));
return (NULL);
}
if (dfh == &public_fh)
linkp->flags |= PUBLIC_PATH;
else
linkp->flags &= ~PUBLIC_PATH;
(void) memcpy(&linkp->dfh, dfh, sizeof (*dfh));
linkp->mtime = time(0);
linkp->atime = linkp->mtime;
linkp->fhkey_offset = ROUNDUP32(offsetof(linkinfo_ent, varbuf));
len = fill_link_key(LN_FHKEY(linkp), &fhrecp->fh, name);
linkp->name_offset = linkp->fhkey_offset + fhrecp->fh.fh_len;
linkp->next_offset = linkp->fhkey_offset + len;
len = 0;
if (memcmp(&fhrecp->dfh, dfh, sizeof (*dfh)) ||
strcmp(fhrecp->name, name)) {
if (dfh == &public_fh) {
if (memcmp(&fhrecp->fh, &public_fh,
sizeof (public_fh))) {
add_mc_path(dbp, dfh, name, fhrecp, linkp,
errorp);
if (*errorp != 0) {
free(linkp);
return (NULL);
}
return (linkp);
}
} else {
len = fill_link_key(LN_NEXT(linkp), &fhrecp->dfh,
fhrecp->name);
}
}
linkp->prev_offset = linkp->next_offset + len;
linkp->reclen = linkp->prev_offset;
linksize = fill_link_key(linkkey, dfh, name);
*errorp = store_record(dbp, linkkey, linksize, linkp, linkp->reclen,
"create_link_struct");
if (*errorp != 0) {
free(linkp);
return (NULL);
}
return (linkp);
}
static int
db_add_secondary(struct db_list *dbp, fhandle_t *dfh, char *name,
fhandle_t *fh, fhlist_ent *fhrecp)
{
int nextsize, len, error;
linkinfo_ent nextlink, *newlinkp, *nextlinkp;
uint_t fhflags;
char *nextaddr;
fhlist_ent *new_fhrecp = fhrecp;
fh_primary_key fhkey;
if (debug > 2)
(void) printf("db_add_secondary entered: name '%s'\n", name);
bcopy(&fh->fh_data, fhkey, fh->fh_len);
if (fhrecp == NULL) {
new_fhrecp = fetch_record(dbp, fhkey, fh->fh_len, NULL,
&error, "db_add_secondary primary");
if (new_fhrecp == NULL) {
return (error);
}
}
error = db_update_fhrec(dbp, fhkey, fh->fh_len, new_fhrecp,
"db_add_secondary primary");
fhflags = new_fhrecp->flags;
newlinkp = create_link_struct(dbp, dfh, name, new_fhrecp, &error);
if (fhrecp == NULL) {
free(new_fhrecp);
new_fhrecp = NULL;
}
if (newlinkp == NULL) {
if (debug > 2)
(void) printf("create_link_struct '%s' Error %s\n",
name, ((error >= 0) ? strerror(error) :
"Unknown"));
return (error);
}
nextsize = LN_NEXT_LEN(newlinkp);
if (nextsize == 0) {
if (debug > 2)
(void) printf("db_add_secondary: no next link\n");
free(newlinkp);
return (0);
}
new_fhrecp = create_primary_struct(dbp, dfh, name, fh, fhflags,
new_fhrecp, &error);
if (new_fhrecp == NULL) {
if (debug > 2)
(void) printf(
"db_add_secondary: replace primary failed\n");
free(newlinkp);
return (error);
} else if (fhrecp == NULL) {
free(new_fhrecp);
}
nextaddr = LN_NEXT(newlinkp);
if (debug > 2) {
debug_print_key(stderr, "db_add_secondary", "next key\n ",
nextaddr, nextsize);
}
nextlinkp = fetch_record(dbp, nextaddr, nextsize, (void *)&nextlink,
&error, "db_add_secondary next link");
if (nextlinkp == NULL) {
if (debug > 2)
(void) printf(
"db_add_secondary: fetch next link failed\n");
free(newlinkp);
return (error);
}
len = fill_link_key(LN_PREV(nextlinkp), dfh, name);
nextlinkp->reclen = nextlinkp->prev_offset + len;
error = store_record(dbp, nextaddr, nextsize, nextlinkp,
nextlinkp->reclen, "db_add_secondary");
if (debug > 2)
(void) printf(
"db_add_secondary exits(%d): name '%s'\n", error, name);
free(newlinkp);
return (error);
}
static linkinfo_ent *
update_next_link(struct db_list *dbp, char *nextkey, int nextsize,
char *prevkey, int prevsize, int *errorp)
{
linkinfo_ent *nextlinkp, *linkp1;
if ((nextlinkp = malloc(sizeof (linkinfo_ent))) == NULL) {
*errorp = errno;
syslog(LOG_ERR, gettext(
"update_next_link: malloc next Error %s"),
strerror(*errorp));
return (NULL);
}
linkp1 = nextlinkp;
nextlinkp = fetch_record(dbp, nextkey, nextsize, nextlinkp,
errorp, "update next");
if (nextlinkp == NULL) {
*errorp = 0;
free(linkp1);
return (NULL);
}
nextlinkp->reclen = ROUNDUP32(nextlinkp->reclen -
LN_PREV_LEN(nextlinkp) + prevsize);
if (prevsize > 0) {
(void) memcpy(LN_PREV(nextlinkp), prevkey, prevsize);
}
*errorp = store_record(dbp, nextkey, nextsize, nextlinkp,
nextlinkp->reclen, "update_next");
if (*errorp != 0) {
free(nextlinkp);
nextlinkp = NULL;
}
return (nextlinkp);
}
static int
update_prev_link(struct db_list *dbp, char *nextkey, int nextsize,
char *prevkey, int prevsize)
{
linkinfo_ent prevlink, *prevlinkp;
int diff, error;
prevlinkp = fetch_record(dbp, prevkey, prevsize, &prevlink, &error,
"update prev");
if (prevlinkp == NULL) {
return (0);
}
diff = nextsize - LN_NEXT_LEN(prevlinkp);
prevlinkp->reclen = ROUNDUP32(prevlinkp->reclen + diff);
if (diff != 0) {
char *ptr = LN_PREV(prevlinkp);
prevlinkp->prev_offset += diff;
(void) memcpy(LN_PREV(prevlinkp), ptr, LN_PREV_LEN(prevlinkp));
}
if (nextsize > 0) {
(void) memcpy(LN_NEXT(prevlinkp), nextkey, nextsize);
}
error = store_record(dbp, prevkey, prevsize, prevlinkp,
prevlinkp->reclen, "update_prev");
return (error);
}
static linkinfo_ent *
update_linked_list(struct db_list *dbp, char *nextkey, int nextsize,
char *prevkey, int prevsize, int *errorp)
{
linkinfo_ent *nextlinkp = NULL;
*errorp = 0;
if (nextsize > 0) {
nextlinkp = update_next_link(dbp, nextkey, nextsize,
prevkey, prevsize, errorp);
if (nextlinkp == NULL) {
if (*errorp != 0) {
if (debug > 1) {
(void) fprintf(stderr,
"update_next_link Error %s\n",
((*errorp >= 0) ? strerror(*errorp) :
"Unknown"));
}
return (NULL);
}
}
}
if (prevsize > 0) {
*errorp = update_prev_link(dbp, nextkey, nextsize,
prevkey, prevsize);
if (*errorp != 0) {
if (debug > 1) {
(void) fprintf(stderr,
"update_prev_link Error %s\n",
((*errorp >= 0) ? strerror(*errorp) :
"Unknown"));
}
if (nextlinkp != NULL)
free(nextlinkp);
nextlinkp = NULL;
}
}
return (nextlinkp);
}
static int
db_update_primary_new_head(struct db_list *dbp, linkinfo_ent *dellinkp,
linkinfo_ent *nextlinkp, fhlist_ent *fhrecp)
{
int error;
char *name, *next_name;
fhandle_t *dfh;
fh_primary_key fhkey;
dfh = &dellinkp->dfh;
name = LN_NAME(dellinkp);
if (memcmp(&fhrecp->dfh, dfh, sizeof (*dfh)) ||
strcmp(fhrecp->name, name)) {
if (debug > 1) {
(void) fprintf(stderr,
"db_update_primary_new_head: primary "
"is for [%s,", name);
debug_opaque_print(stderr, (void *)dfh, sizeof (*dfh));
(void) fprintf(stderr, "], not [%s,", fhrecp->name);
debug_opaque_print(stderr, (void *)&fhrecp->dfh,
sizeof (fhrecp->dfh));
(void) fprintf(stderr, "]\n");
}
return (0);
}
bcopy(&fhrecp->fh.fh_data, fhkey, fhrecp->fh.fh_len);
if (nextlinkp == NULL) {
(void) delete_record(dbp,
fhkey, fhrecp->fh.fh_len,
"db_update_primary_new_head: fh delete");
return (0);
} else {
next_name = LN_NAME(nextlinkp);
fhrecp->reclen = ROUNDUP32(offsetof(fhlist_ent, name) +
strlen(next_name) + 1);
(void) memcpy(&fhrecp->dfh, &nextlinkp->dfh,
sizeof (nextlinkp->dfh));
(void) strcpy(fhrecp->name, next_name);
}
fhrecp->mtime = time(0);
fhrecp->atime = fhrecp->mtime;
error = store_record(dbp,
fhkey, fhrecp->fh.fh_len, fhrecp,
fhrecp->reclen, "db_update_primary_new_head: fh");
return (error);
}
int
db_add(char *fhpath, fhandle_t *dfh, char *name, fhandle_t *fh, uint_t flags)
{
struct db_list *dbp = NULL;
fhlist_ent fhrec, *fhrecp;
int error = 0;
if (fh == NULL) {
return (EINVAL);
}
if (fh == &public_fh) {
dbp = db_get_all_databases(fhpath, FALSE);
} else {
dbp = db_get_db(fhpath, &fh->fh_fsid, &error, O_CREAT);
}
for (; dbp != NULL; dbp = ((fh != &public_fh) ? NULL : dbp->next)) {
if (debug > 3) {
(void) printf("db_add: name '%s', db '%s'\n",
name, dbp->path);
}
fhrecp = db_add_primary(dbp, dfh, name, fh, flags,
&fhrec, &error);
if (fhrecp == NULL) {
continue;
}
if ((dfh == NULL) || (name == NULL)) {
syslog(LOG_ERR, gettext(
"db_add: dfh %p, name %p - invalid"),
(void *)dfh, (void *)name);
error = EINVAL;
continue;
}
if (fh == &public_fh) {
while ((fhrecp != NULL) && strcmp(name, fhrecp->name)) {
error = db_delete_link(fhpath, dfh,
fhrecp->name);
fhrecp = db_add_primary(dbp, dfh, name, fh,
flags, &fhrec, &error);
}
if (fhrecp == NULL) {
continue;
}
}
error = db_add_secondary(dbp, dfh, name, fh, fhrecp);
if (fhrecp != &fhrec) {
free(fhrecp);
}
}
return (error);
}
fhlist_ent *
db_lookup(char *fhpath, fhandle_t *fh, fhlist_ent *fhrecp, int *errorp)
{
struct db_list *dbp;
fh_primary_key fhkey;
if ((fhpath == NULL) || (fh == NULL) || (errorp == NULL)) {
if (errorp != NULL)
*errorp = EINVAL;
return (NULL);
}
*errorp = 0;
if (fh == &public_fh) {
dbp = db_get_all_databases(fhpath, FALSE);
} else {
dbp = db_get_db(fhpath, &fh->fh_fsid, errorp, O_CREAT);
}
if (dbp == NULL) {
return (NULL);
}
bcopy(&fh->fh_data, fhkey, fh->fh_len);
fhrecp = fetch_record(dbp, fhkey, fh->fh_len, fhrecp,
errorp, "db_lookup");
if (fhrecp != NULL) {
*errorp = db_update_fhrec(dbp, fhkey, fh->fh_len, fhrecp,
"db_lookup");
}
return (fhrecp);
}
fhlist_ent *
db_lookup_link(char *fhpath, fhandle_t *dfh, char *name, fhlist_ent *fhrecp,
int *errorp)
{
struct db_list *dbp;
fh_secondary_key linkkey;
linkinfo_ent *linkp;
int linksize, fhkeysize;
char *fhkey;
if ((fhpath == NULL) || (dfh == NULL) || (name == NULL) ||
(errorp == NULL)) {
if (errorp != NULL)
*errorp = EINVAL;
return (NULL);
}
*errorp = 0;
if (dfh == &public_fh) {
dbp = db_get_all_databases(fhpath, FALSE);
} else {
dbp = db_get_db(fhpath, &dfh->fh_fsid, errorp, O_CREAT);
}
if (dbp == NULL) {
return (NULL);
}
linksize = fill_link_key(linkkey, dfh, name);
linkp = fetch_record(dbp, linkkey, linksize, NULL, errorp,
"db_lookup_link link");
if (linkp != NULL) {
fhkeysize = LN_FHKEY_LEN(linkp);
fhkey = LN_FHKEY(linkp);
fhrecp = fetch_record(dbp, fhkey, fhkeysize,
(void *)fhrecp, errorp, "db_lookup_link fh");
if (fhrecp != NULL) {
*errorp = db_update_fhrec(dbp, fhkey, fhkeysize, fhrecp,
"db_lookup_link fhrec");
}
*errorp = db_update_linkinfo(dbp, linkkey, linksize, linkp,
"db_lookup_link link");
free(linkp);
} else {
fhrecp = NULL;
}
return (fhrecp);
}
static int
delete_link_by_key(struct db_list *dbp, char *linkkey, int *linksizep,
int *errorp, char *errstr)
{
int nextsize, prevsize, fhkeysize, linksize;
char *nextkey, *prevkey, *fhkey;
linkinfo_ent *dellinkp, *nextlinkp;
fhlist_ent *fhrecp, fhrec;
*errorp = 0;
linksize = *linksizep;
dellinkp = fetch_record(dbp, linkkey, linksize, NULL, errorp, errstr);
if (dellinkp == NULL) {
if (debug > 2) {
debug_print_key(stderr, errstr,
"link not in database\n",
linkkey, linksize);
}
*linksizep = 0;
return (ENOENT);
}
nextsize = LN_NEXT_LEN(dellinkp);
nextkey = ((nextsize > 0) ? LN_NEXT(dellinkp) : NULL);
prevsize = LN_PREV_LEN(dellinkp);
prevkey = ((prevsize > 0) ? LN_PREV(dellinkp) : NULL);
nextlinkp = update_linked_list(dbp, nextkey, nextsize,
prevkey, prevsize, errorp);
if ((nextlinkp == NULL) && (*errorp != 0)) {
free(dellinkp);
*linksizep = 0;
return (0);
}
*errorp = delete_record(dbp, linkkey, linksize, errstr);
fhkeysize = LN_FHKEY_LEN(dellinkp);
fhkey = LN_FHKEY(dellinkp);
fhrecp = fetch_record(dbp, fhkey, fhkeysize,
&fhrec, errorp, errstr);
if (fhrecp == NULL) {
if (debug > 1) {
debug_print_key(stderr, errstr,
"fetch primary for ", linkkey, linksize);
(void) fprintf(stderr, " Error %s\n",
((*errorp >= 0) ? strerror(*errorp) : "Unknown"));
}
} else if ((*errorp == 0) && (prevsize <= 0)) {
*errorp = db_update_primary_new_head(dbp, dellinkp,
nextlinkp, fhrecp);
} else {
*errorp = db_update_fhrec(dbp, fhkey, fhkeysize, fhrecp,
errstr);
}
*linksizep = nextsize;
if (nextsize > 0)
(void) memcpy(linkkey, nextkey, nextsize);
if (nextlinkp != NULL)
free(nextlinkp);
free(dellinkp);
return (0);
}
static int
delete_link(struct db_list *dbp, fhandle_t *dfh, char *name,
char *nextlinkkey, int *nextlinksizep, int *errorp, char *errstr)
{
int linkerr;
*errorp = 0;
if ((nextlinkkey != NULL) && (nextlinksizep != NULL)) {
*nextlinksizep = fill_link_key(nextlinkkey, dfh, name);
linkerr = delete_link_by_key(dbp, nextlinkkey, nextlinksizep,
errorp, errstr);
} else {
int linksize;
fh_secondary_key linkkey;
linksize = fill_link_key(linkkey, dfh, name);
linkerr = delete_link_by_key(dbp, linkkey, &linksize,
errorp, errstr);
}
return (linkerr);
}
int
db_delete_link(char *fhpath, fhandle_t *dfh, char *name)
{
struct db_list *dbp;
int error = 0;
if ((fhpath == NULL) || (dfh == NULL) || (name == NULL)) {
return (EINVAL);
}
if (dfh == &public_fh) {
dbp = db_get_all_databases(fhpath, TRUE);
} else {
dbp = db_get_db(fhpath, &dfh->fh_fsid, &error, O_CREAT);
}
for (; dbp != NULL; dbp = ((dfh == &public_fh) ? dbp->next : NULL)) {
(void) delete_link(dbp, dfh, name, NULL, NULL, &error,
"db_delete_link link");
}
return (error);
}
#ifdef DEBUG
int
db_delete(char *fhpath, fhandle_t *fh)
{
struct db_list *dbp;
int error = 0;
if ((fhpath == NULL) || (fh == NULL)) {
return (EINVAL);
}
if (fh == &public_fh) {
dbp = db_get_all_databases(fhpath, TRUE);
} else {
dbp = db_get_db(fhpath, &fh->fh_fsid, &error, O_CREAT);
}
for (; dbp != NULL; dbp = ((fh == &public_fh) ? dbp->next : NULL)) {
(void) delete_record(dbp, &fh->fh_data, fh->fh_len,
"db_delete: fh delete");
}
return (error);
}
#endif
int
db_rename_link(char *fhpath, fhandle_t *from_dfh, char *from_name,
fhandle_t *to_dfh, char *to_name)
{
int error;
struct db_list *dbp;
fhlist_ent fhrec, *fhrecp;
if ((fhpath == NULL) || (from_dfh == NULL) || (from_name == NULL) ||
(to_dfh == NULL) || (to_name == NULL)) {
return (EINVAL);
}
if (from_dfh == &public_fh) {
dbp = db_get_all_databases(fhpath, FALSE);
} else {
dbp = db_get_db(fhpath, &from_dfh->fh_fsid, &error, O_CREAT);
}
for (; dbp != NULL;
dbp = ((from_dfh != &public_fh) ? NULL : dbp->next)) {
fhrecp = db_lookup_link(fhpath, from_dfh, from_name, &fhrec,
&error);
if (fhrecp == NULL) {
continue;
}
error = db_delete_link(fhpath, from_dfh, from_name);
if (error == 0) {
error = db_add(fhpath, to_dfh, to_name, &fhrecp->fh,
fhrecp->flags);
}
}
return (error);
}
void
db_print_all_keys(char *fhpath, fsid_t *fsidp, FILE *fp)
{
struct db_list *dbp;
datum key;
int error, len;
char strkey[NFS_FHMAXDATA + MAXNAMELEN];
db_record rec;
void *ptr;
if ((fhpath == NULL) ||
((fsidp != NULL) && (fsidp == &public_fh.fh_fsid)))
return;
if (fsidp == NULL) {
(void) db_get_all_databases(fhpath, TRUE);
dbp = db_fs_list;
} else {
dbp = db_get_db(fhpath, fsidp, &error, 0);
}
if (dbp == NULL) {
return;
}
len = strlen(fhpath);
for (; dbp != NULL; dbp = ((fsidp != NULL) ? NULL : dbp->next)) {
if (strncmp(fhpath, dbp->path, len))
continue;
(void) fprintf(fp,
"\nStart print database for fsid 0x%x 0x%x\n",
dbp->fsid.val[0], dbp->fsid.val[1]);
(void) fprintf(fp, "=============================\n");
for (key = dbm_firstkey(dbp->db); key.dptr != NULL;
key = dbm_nextkey(dbp->db)) {
(void) memcpy(strkey, key.dptr, key.dsize);
debug_print_key(fp, "", "", strkey, key.dsize);
if (debug < 2)
continue;
ptr = fetch_record(dbp, key.dptr, key.dsize,
(void *)&rec, &error, "db_prt_keys");
if (ptr == NULL)
continue;
if (key.dsize == NFS_FHMAXDATA) {
debug_print_fhlist(fp, &rec.fhlist_rec);
} else if (key.dsize > NFS_FHMAXDATA) {
debug_print_linkinfo(fp, &rec.link_rec);
}
(void) fprintf(fp, "-----------------------------\n");
}
(void) fprintf(fp, "End print database for fsid 0x%x 0x%x\n",
dbp->fsid.val[0], dbp->fsid.val[1]);
}
}
void
debug_opaque_print(FILE *fp, void *buf, int size)
{
int bufoffset = 0;
char debug_str[200];
if ((buf == NULL) || (size <= 0))
return;
nfslog_opaque_print_buf(buf, size, debug_str, &bufoffset, 200);
(void) fprintf(fp, debug_str);
}
static int
links_timedout(struct db_list *pdb, fhlist_ent *pfe, time_t ts)
{
fh_secondary_key linkkey;
linkinfo_ent *linkp, link_st;
int error;
int linksize;
void *cookie;
linksize = fill_link_key(linkkey, &pfe->dfh, pfe->name);
cookie = NULL;
do {
linkp = get_next_link(pdb, linkkey, &linksize, &link_st,
&cookie, &error, "links_timedout");
if ((linkp != NULL) &&
(difftime(ts, linkp->atime) <= prune_timeout)) {
pfe = fetch_record(pdb, (void *)&pfe->fh.fh_data,
pfe->fh.fh_len, NULL, &error,
"links_timedout");
if (pfe == NULL) {
syslog(LOG_ERR, gettext(
"links_timedout: fetch fhrec error %s\n"),
strerror(error));
} else {
if (difftime(pfe->atime, linkp->atime) < 0) {
pfe->atime = linkp->atime;
(void) store_record(pdb,
(void *)&pfe->fh.fh_data,
pfe->fh.fh_len, pfe,
pfe->reclen, "links_timedout");
}
free(pfe);
}
free_link_cookies(cookie);
return (FALSE);
}
} while (linksize > 0);
free_link_cookies(cookie);
return (TRUE);
}
int
prune_dbs(char *fhpath)
{
struct db_list *pdb;
datum key;
db_record *ptr;
struct fhlist_ent *pfe;
int error, linkerr, linksize;
time_t cur_time = time(0);
fh_secondary_key linkkey;
struct thelist {
struct thelist *next;
db_record *ptr;
} thelist, *ptl;
int cnt = 0;
if (fhpath != NULL)
(void) db_get_all_databases(fhpath, TRUE);
thelist.next = NULL;
for (pdb = db_fs_list; pdb; pdb = pdb->next) {
do {
for (key = dbm_firstkey(pdb->db); key.dptr != NULL;
key = dbm_nextkey(pdb->db)) {
if (key.dsize != NFS_FHMAXDATA)
continue;
ptr = fetch_record(pdb, key.dptr, key.dsize,
NULL, &error, "dump_db");
if (ptr == NULL)
continue;
if ((ptr->fhlist_rec.flags &
(EXPORT_POINT | PUBLIC_PATH)) ||
(difftime(cur_time, ptr->fhlist_rec.atime) <=
prune_timeout)) {
free(ptr);
} else {
ptl = malloc(sizeof (struct thelist));
if (ptl == NULL) {
syslog(LOG_ERR, gettext(
"prune_dbs: malloc failed, error %s\n"),
strerror(errno));
break;
}
ptl->ptr = ptr;
ptl->next = thelist.next;
thelist.next = ptl;
cnt++;
if (cnt > MAX_PRUNE_REC_CNT) {
if (debug)
(void) fprintf(stderr,
"prune_dbs: halt search - too many records\n");
break;
}
}
}
for (ptl = thelist.next; ptl; ptl = thelist.next) {
thelist.next = ptl->next;
pfe = &(ptl->ptr->fhlist_rec);
if (links_timedout(pdb, pfe, cur_time)) {
linkerr = delete_link(pdb,
&pfe->dfh, pfe->name, linkkey,
&linksize, &error, "dump_db");
while ((linksize > 0) && !(error || linkerr)) {
linkerr = delete_link_by_key(pdb,
linkkey, &linksize,
&error, "dump_db");
if (error || linkerr) {
break;
}
}
if (linkerr) {
if (debug > 1) {
(void) fprintf(stderr,
"prune_dbs: Error primary exists ");
debug_opaque_print(stderr,
(void *)&pfe->fh,
sizeof (pfe->fh));
(void) fprintf(stderr, "\n");
}
if (debug)
syslog(LOG_ERR, gettext(
"prune_dbs: Error primary exists\n"));
(void) delete_record(pdb,
&pfe->fh.fh_data, pfe->fh.fh_len,
"prune_dbs: fh delete");
}
}
free(ptl->ptr);
free(ptl);
cnt--;
}
thelist.next = NULL;
} while (key.dptr != NULL);
}
return (0);
}