#include <stdlib.h>
#include <stdio.h>
#include <strings.h>
#include <sys/types.h>
#include <unistd.h>
#include <libintl.h>
#include <errno.h>
#include <time.h>
#include <string.h>
#include <assert.h>
#include <getopt.h>
#include <cmdparse.h>
#include <libstmf.h>
#include <signal.h>
#include <pthread.h>
#include <locale.h>
static int svcStart(int, char **, cmdOptions_t *, void *);
static int svcStop(int, char **, cmdOptions_t *, void *);
static int online();
#define VERSION_STRING_MAJOR "1"
#define VERSION_STRING_MINOR "0"
#define VERSION_STRING_MAX_LEN 10
#define TEN_MS_NANOSLEEP 10000000
optionTbl_t longOptions[] = {
{NULL, 0, 0, 0}
};
subCommandProps_t subcommands[] = {
{"start", svcStart, NULL, NULL, NULL, OPERAND_NONE, NULL},
{"stop", svcStop, NULL, NULL, NULL, OPERAND_NONE, NULL},
{NULL, NULL, NULL, NULL, NULL, 0, NULL, NULL}
};
char *cmdName;
static int
svcStop(int operandLen, char *operands[], cmdOptions_t *options,
void *args)
{
int stmfRet;
int ret = 0;
stmfState state;
boolean_t serviceOffline = B_FALSE;
struct timespec rqtp;
bzero(&rqtp, sizeof (rqtp));
rqtp.tv_nsec = TEN_MS_NANOSLEEP;
if ((stmfRet = stmfOffline()) != STMF_STATUS_SUCCESS) {
switch (stmfRet) {
case STMF_ERROR_PERM:
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("permission denied"));
break;
case STMF_ERROR_SERVICE_NOT_FOUND:
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("STMF service not found"));
break;
case STMF_ERROR_SERVICE_OFFLINE:
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("STMF service already offline"));
break;
default:
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("unable to offline service"));
break;
}
return (1);
}
while (!serviceOffline) {
stmfRet = stmfGetState(&state);
if (stmfRet != STMF_STATUS_SUCCESS) {
ret = 1;
break;
}
if (state.operationalState == STMF_SERVICE_STATE_OFFLINE) {
serviceOffline = B_TRUE;
} else {
(void) nanosleep(&rqtp, NULL);
}
}
return (ret);
}
static int
svcStart(int operandLen, char *operands[], cmdOptions_t *options,
void *args)
{
int stmfRet;
int ret = 0;
(void) stmfLoadStmfProps();
if ((stmfRet = stmfLoadConfig()) != STMF_STATUS_SUCCESS) {
switch (stmfRet) {
case STMF_ERROR_PERM:
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("permission denied"));
break;
case STMF_ERROR_SERVICE_NOT_FOUND:
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("STMF service not found"));
break;
case STMF_ERROR_SERVICE_ONLINE:
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("STMF service must be offline"));
break;
default:
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("Unable to load the configuration. "
"See /var/adm/messages for details"));
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("For information on reverting the "
"stmf:default instance to a previously "
"running configuration see the man page "
"for svccfg(8)"));
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("After reverting the instance "
"you must clear the service maintenance "
"state. See the man page for svcadm(8)"));
break;
}
return (1);
}
ret = online();
return (ret);
}
static int
online()
{
int stmfRet;
int ret = 0;
stmfState state;
boolean_t serviceOnline = B_FALSE;
struct timespec rqtp;
bzero(&rqtp, sizeof (rqtp));
rqtp.tv_nsec = TEN_MS_NANOSLEEP;
if ((stmfRet = stmfOnline()) != STMF_STATUS_SUCCESS) {
switch (stmfRet) {
case STMF_ERROR_PERM:
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("permission denied"));
break;
case STMF_ERROR_SERVICE_NOT_FOUND:
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("STMF service not found"));
break;
case STMF_ERROR_SERVICE_ONLINE:
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("STMF service already online"));
break;
default:
(void) fprintf(stderr, "%s: %s\n", cmdName,
gettext("unable to online service"));
break;
}
return (1);
}
while (!serviceOnline) {
stmfRet = stmfGetState(&state);
if (stmfRet != STMF_STATUS_SUCCESS) {
ret = 1;
break;
}
if (state.operationalState == STMF_SERVICE_STATE_ONLINE) {
serviceOnline = B_TRUE;
} else {
(void) nanosleep(&rqtp, NULL);
}
}
return (ret);
}
static char *
getExecBasename(char *execFullname)
{
char *lastSlash, *execBasename;
for (;;) {
lastSlash = strrchr(execFullname, '/');
if (lastSlash == NULL) {
execBasename = execFullname;
break;
} else {
execBasename = lastSlash + 1;
if (*execBasename == '\0') {
*lastSlash = '\0';
continue;
}
break;
}
}
return (execBasename);
}
int
main(int argc, char *argv[])
{
synTables_t synTables;
char versionString[VERSION_STRING_MAX_LEN];
int ret;
int funcRet;
void *subcommandArgs = NULL;
(void) setlocale(LC_ALL, "");
cmdName = getExecBasename(argv[0]);
(void) snprintf(versionString, VERSION_STRING_MAX_LEN, "%s.%s",
VERSION_STRING_MAJOR, VERSION_STRING_MINOR);
synTables.versionString = versionString;
synTables.longOptionTbl = &longOptions[0];
synTables.subCommandPropsTbl = &subcommands[0];
ret = cmdParse(argc, argv, synTables, subcommandArgs, &funcRet);
if (ret != 0) {
return (ret);
}
return (funcRet);
}