#include <stdlib.h>
#include <libscf.h>
#include <string.h>
#include "nscd_switch.h"
#include "nscd_log.h"
#include "nscd_door.h"
extern int _whoami;
nscd_smf_state_t *nscd_smf_service_state;
static rwlock_t nscd_smf_service_state_lock = DEFAULTRWLOCK;
nscd_rc_t
_nscd_alloc_service_state_table()
{
int i;
nscd_smf_service_state = calloc(NSCD_NUM_SMF_FMRI,
sizeof (nscd_smf_state_t));
if (nscd_smf_service_state == NULL)
return (NSCD_NO_MEMORY);
for (i = 1; i < NSCD_NUM_SMF_FMRI; i++)
NSCD_SMF_SVC_STATE(i) = NSCD_SVC_STATE_UNINITED;
return (NSCD_SUCCESS);
}
static int
query_smf_state(int srci)
{
int ret = NSCD_SVC_STATE_UNINITED;
char *state = NULL;
char *me = "query_smf_state";
state = smf_get_state(NSCD_SMF_SVC_FMRI(srci));
if (state == NULL)
return (ret);
_NSCD_LOG(NSCD_LOG_SMF_MONITOR, NSCD_LOG_LEVEL_DEBUG)
(me, "%s -- %s\n", state, NSCD_SMF_SVC_FMRI(srci));
(void) rw_wrlock(&nscd_smf_service_state_lock);
if (nscd_smf_service_state[srci].src_name == NULL)
nscd_smf_service_state[srci].src_name =
NSCD_NSW_SRC_NAME(srci);
if (strcmp(state, SCF_STATE_STRING_UNINIT) == 0)
NSCD_SMF_SVC_STATE(srci) = SCF_STATE_UNINIT;
else if (strcmp(state, SCF_STATE_STRING_MAINT) == 0)
NSCD_SMF_SVC_STATE(srci) = SCF_STATE_MAINT;
else if (strcmp(state, SCF_STATE_STRING_OFFLINE) == 0)
NSCD_SMF_SVC_STATE(srci) = SCF_STATE_OFFLINE;
else if (strcmp(state, SCF_STATE_STRING_DISABLED) == 0)
NSCD_SMF_SVC_STATE(srci) = SCF_STATE_DISABLED;
else if (strcmp(state, SCF_STATE_STRING_ONLINE) == 0)
NSCD_SMF_SVC_STATE(srci) = SCF_STATE_ONLINE;
else if (strcmp(state, SCF_STATE_STRING_DEGRADED) == 0)
NSCD_SMF_SVC_STATE(srci) = SCF_STATE_DEGRADED;
ret = NSCD_SMF_SVC_STATE(srci);
(void) rw_unlock(&nscd_smf_service_state_lock);
free(state);
return (ret);
}
static void *
set_smf_state(void *arg)
{
int i;
int st;
(void) thr_setname(thr_self(), "set_smf_state");
if (_whoami == NSCD_FORKER)
thr_exit(0);
while (1) {
for (i = 1; i < NSCD_NUM_SMF_FMRI; i++) {
st = query_smf_state(i);
if (st == NSCD_SVC_STATE_UNINITED)
break;
}
(void) sleep(NSCD_SW_CFG_G.check_smf_state_interval_g);
}
}
nscd_rc_t
_nscd_init_smf_monitor(void)
{
int errnum;
char *me = "_nscd_init_smf_monitor";
_NSCD_LOG(NSCD_LOG_SMF_MONITOR, NSCD_LOG_LEVEL_DEBUG)
(me, "initializing the smf monitor\n");
if (thr_create(NULL, 0, set_smf_state,
NULL, THR_DETACHED, NULL) != 0) {
errnum = errno;
_NSCD_LOG(NSCD_LOG_SMF_MONITOR, NSCD_LOG_LEVEL_ERROR)
(me, "thr_create: %s\n", strerror(errnum));
return (NSCD_THREAD_CREATE_ERROR);
}
return (NSCD_SUCCESS);
}
int
_nscd_get_smf_state(int srci, int dbi, int recheck)
{
int s;
char *n;
n = NSCD_NSW_SRC_NAME(srci);
if ((*n == 'f' || *n == 'c' || *n == 'd' || *n == 'a') &&
(strcmp(NSCD_NSW_SRC_NAME(srci), "files") == 0 ||
strcmp(NSCD_NSW_SRC_NAME(srci), "compat") == 0 ||
strcmp(NSCD_NSW_SRC_NAME(srci), "ad") == 0 ||
strcmp(NSCD_NSW_SRC_NAME(srci), "dns") == 0)) {
return (SCF_STATE_ONLINE);
}
if (*n == 'u' && strcmp(NSCD_NSW_SRC_NAME(srci), "user") == 0) {
if (strcmp(NSCD_NSW_DB_NAME(dbi), NSS_DBNAM_PRINTERS) == 0)
return (NSCD_SVC_STATE_UNSUPPORTED_SRC);
else
return (SCF_STATE_ONLINE);
}
if (srci >= _nscd_cfg_num_nsw_src)
return (NSCD_SVC_STATE_FOREIGN_SRC);
if (recheck == 1)
return (query_smf_state(srci));
(void) rw_rdlock(&nscd_smf_service_state_lock);
s = NSCD_SMF_SVC_STATE(srci);
(void) rw_unlock(&nscd_smf_service_state_lock);
if (s != NSCD_SVC_STATE_UNINITED && s < SCF_STATE_ONLINE)
s = query_smf_state(srci);
return (s);
}