blob: 6a4e348b1ca8f240177ee41dd8bc90ae3308b9c1 [file] [log] [blame]
/*
* Copyright (c) 2000-2001 Silicon Graphics, Inc.
* All Rights Reserved.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it would be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <unistd.h>
#include <sys/stat.h>
#include <time.h>
#include <errno.h>
#include <assert.h>
#include <string.h>
#include <uuid/uuid.h>
#include <xfs/xfs.h>
#include <xfs/jdm.h> /* only for util.h include */
#include "config.h"
#include "types.h"
#include "util.h" /* only for strncpyterm */
#include "mlog.h"
#include "dlog.h"
#include "global.h"
#include "getopt.h"
#include "swap.h"
/* declarations of externally defined global symbols *************************/
extern void usage(void);
extern bool_t pipeline;
/* forward declarations of locally defined static functions ******************/
#ifdef DUMP
static char * prompt_label(char *bufp, size_t bufsz);
#endif /* DUMP */
/* definition of locally defined global variables ****************************/
/* definition of locally defined static variables *****************************/
/* definition of locally defined global functions ****************************/
global_hdr_t *
global_hdr_alloc(int argc, char *argv[])
{
global_hdr_t *ghdrp;
int c;
char *dumplabel;
#ifdef DUMP
char labelbuf[GLOBAL_HDR_STRING_SZ];
struct stat64 statb;
#endif /* DUMP */
int rval;
/* sanity checks
*/
assert(sizeof(time32_t) == GLOBAL_HDR_TIME_SZ);
assert(sizeof(uuid_t) == GLOBAL_HDR_UUID_SZ);
/* allocate a global hdr
*/
ghdrp = (global_hdr_t *)calloc(1, sizeof(global_hdr_t));
assert(ghdrp);
/* fill in the magic number
*/
strncpy(ghdrp->gh_magic, GLOBAL_HDR_MAGIC, GLOBAL_HDR_MAGIC_SZ);
/* fill in the hdr version
*/
ghdrp->gh_version = GLOBAL_HDR_VERSION;
/* fill in the timestamp: all changes made at or after this moment
* will be included in increments on this base. This may be
* overridden with the GETOPT_DUMPTIME option.
*/
ghdrp->gh_timestamp = (time32_t) time(0);
/* fill in the host id: typecast to fit into a 64 bit field
*/
ghdrp->gh_ipaddr = (uint64_t)(unsigned long)gethostid();
#ifdef DUMP
uuid_generate(ghdrp->gh_dumpid);
#endif /* DUMP */
#ifdef RESTORE
uuid_clear(ghdrp->gh_dumpid);
#endif /* RESTORE */
/* fill in the hostname
*/
rval = gethostname(ghdrp->gh_hostname, GLOBAL_HDR_STRING_SZ);
if (rval) {
mlog(MLOG_NORMAL | MLOG_ERROR,
_("unable to determine hostname: %s\n"),
strerror(errno));
return 0;
}
if (!strlen(ghdrp->gh_hostname)) {
mlog(MLOG_NORMAL | MLOG_ERROR,
_("hostname length is zero\n"));
return 0;
}
/* scan the command line for the dump session label
*/
dumplabel = 0;
optind = 1;
opterr = 0;
while ((c = getopt(argc, argv, GETOPT_CMDSTRING)) != EOF) {
switch (c) {
case GETOPT_DUMPLABEL:
if (dumplabel) {
mlog(MLOG_NORMAL,
_("too many -%c arguments: "
"\"-%c %s\" already given\n"),
c,
c,
dumplabel);
usage();
return 0;
}
if (! optarg || optarg[0] == '-') {
mlog(MLOG_NORMAL | MLOG_ERROR,
_("-%c argument missing\n"),
c);
usage();
return 0;
}
dumplabel = optarg;
break;
#ifdef RESTORE
case GETOPT_SESSIONID:
if (! uuid_is_null(ghdrp->gh_dumpid)) {
mlog(MLOG_NORMAL | MLOG_ERROR,
_("too many -%c arguments\n"),
c);
usage();
return 0;
}
if (! optarg || optarg[0] == '-') {
mlog(MLOG_NORMAL | MLOG_ERROR,
_("-%c argument missing\n"),
c);
usage();
return 0;
}
if (!uuid_parse(optarg, ghdrp->gh_dumpid)) {
mlog(MLOG_NORMAL | MLOG_ERROR,
_("-%c argument not a valid uuid\n"),
c);
usage();
return 0;
}
break;
#endif /* RESTORE */
#ifdef DUMP
case GETOPT_DUMPTIME:
/* Use the timestamp of the specified file for the
* dump time, rather than using the current time.
*/
if (! optarg || optarg[0] == '-') {
mlog(MLOG_NORMAL | MLOG_ERROR,
_("-%c argument missing\n"),
c);
usage();
return 0;
}
rval = stat64(optarg, &statb);
if (rval) {
mlog(MLOG_NORMAL | MLOG_ERROR,
_("unable to stat %s: %s\n"),
optarg,
strerror(errno));
usage();
return 0;
}
ghdrp->gh_timestamp = statb.st_mtime;
break;
case GETOPT_FMT2COMPAT:
ghdrp->gh_version = GLOBAL_HDR_VERSION_2;
break;
#endif /* DUMP */
}
}
#ifdef DUMP
/* if no dump label specified, no pipes in use, and dialogs
* are allowed, prompt for one
*/
if (!dumplabel && dlog_allowed()) {
dumplabel = prompt_label(labelbuf, sizeof(labelbuf));
}
#endif /* DUMP */
if (!dumplabel || !strlen(dumplabel)) {
#ifdef DUMP
if (!pipeline) {
mlog(MLOG_VERBOSE | MLOG_WARNING,
_("no session label specified\n"));
}
#endif /* DUMP */
dumplabel = "";
}
strncpyterm(ghdrp->gh_dumplabel,
dumplabel,
sizeof(ghdrp->gh_dumplabel));
return ghdrp;
}
void
global_hdr_free(global_hdr_t *ghdrp)
{
free((void *)ghdrp);
}
/* global_hdr_checksum_set - fill in the global media file header checksum.
* utility function for use by drive-specific strategies.
*/
void
global_hdr_checksum_set(global_hdr_t *hdrp)
{
uint32_t *beginp = (uint32_t *)&hdrp[0];
uint32_t *endp = (uint32_t *)&hdrp[1];
uint32_t *p;
uint32_t accum;
hdrp->gh_checksum = 0;
accum = 0;
for (p = beginp; p < endp; p++) {
accum += INT_GET(*p, ARCH_CONVERT);
}
INT_SET(hdrp->gh_checksum, ARCH_CONVERT, (int32_t)(~accum + 1));
}
/* global_hdr_checksum_check - check the global media file header checksum.
* utility function for use by drive-specific strategies.
* returns BOOL_TRUE if ok, BOOL_FALSE if bad
*/
bool_t
global_hdr_checksum_check(global_hdr_t *hdrp)
{
uint32_t *beginp = (uint32_t *)&hdrp[0];
uint32_t *endp = (uint32_t *)&hdrp[1];
uint32_t *p;
uint32_t accum;
accum = 0;
for (p = beginp; p < endp; p++) {
accum += INT_GET(*p, ARCH_CONVERT);
}
return accum == 0 ? BOOL_TRUE : BOOL_FALSE;
}
/* global_version_check - if we know this version number, return BOOL_TRUE
* else return BOOL_FALSE
*/
bool_t
global_version_check(uint32_t version)
{
switch (version) {
case GLOBAL_HDR_VERSION_0:
case GLOBAL_HDR_VERSION_1:
case GLOBAL_HDR_VERSION_2:
case GLOBAL_HDR_VERSION_3:
return BOOL_TRUE;
default:
return BOOL_FALSE;
}
}
/* definition of locally defined static functions ****************************/
#ifdef DUMP
#define PREAMBLEMAX 3
#define QUERYMAX 1
#define CHOICEMAX 1
#define ACKMAX 3
#define POSTAMBLEMAX 3
#define DLOG_TIMEOUT 300
/* ARGSUSED */
static void
prompt_label_cb(void *uctxp, dlog_pcbp_t pcb, void *pctxp)
{
/* query: ask for a dump label
*/
(*pcb)(pctxp,
_("please enter label for this dump session"));
}
static char *
prompt_label(char *bufp, size_t bufsz)
{
fold_t fold;
char *preamblestr[PREAMBLEMAX];
size_t preamblecnt;
char *ackstr[ACKMAX];
size_t ackcnt;
char *postamblestr[POSTAMBLEMAX];
size_t postamblecnt;
const ix_t abortix = 1;
const ix_t okix = 2;
ix_t responseix;
preamblecnt = 0;
fold_init(fold, _("dump label dialog"), '=');
preamblestr[preamblecnt++ ] = "\n";
preamblestr[preamblecnt++] = fold;
preamblestr[preamblecnt++ ] = "\n\n";
assert(preamblecnt <= PREAMBLEMAX);
dlog_begin(preamblestr, preamblecnt);
responseix = dlog_string_query(prompt_label_cb,
0,
bufp,
bufsz,
DLOG_TIMEOUT,
abortix,/* timeout ix */
IXMAX, /* sigint ix */
IXMAX, /* sighup ix */
IXMAX, /* sigquit ix */
okix); /* ok ix */
ackcnt = 0;
if (responseix == okix) {
ackstr[ackcnt++ ] = _("session label entered: \"");
ackstr[ackcnt++] = bufp;
ackstr[ackcnt++ ] = "\"\n";
} else {
ackstr[ackcnt++ ] = _("session label left blank\n");
}
assert(ackcnt <= ACKMAX);
dlog_string_ack(ackstr,
ackcnt);
postamblecnt = 0;
fold_init(fold, _("end dialog"), '-');
postamblestr[postamblecnt++ ] = "\n";
postamblestr[postamblecnt++] = fold;
postamblestr[postamblecnt++ ] = "\n\n";
assert(postamblecnt <= POSTAMBLEMAX);
dlog_end(postamblestr,
postamblecnt);
if (responseix == okix) {
return bufp;
} else {
return 0;
}
}
#endif /* DUMP */