2007-10-17 23:34:54 +00:00
|
|
|
// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
|
|
|
|
// vim: ts=8 sw=2 smarttab
|
|
|
|
/*
|
|
|
|
* Ceph - scalable distributed file system
|
|
|
|
*
|
|
|
|
* Copyright (C) 2004-2006 Sage Weil <sage@newdream.net>
|
|
|
|
*
|
|
|
|
* This is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License version 2.1, as published by the Free Software
|
|
|
|
* Foundation. See file COPYING.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <iostream>
|
|
|
|
#include <string>
|
|
|
|
using namespace std;
|
|
|
|
|
2012-01-23 14:18:53 +00:00
|
|
|
#include "include/ceph_features.h"
|
|
|
|
|
2011-02-22 16:38:45 +00:00
|
|
|
#include "common/config.h"
|
2011-06-02 22:50:22 +00:00
|
|
|
#include "common/strtol.h"
|
2007-10-17 23:34:54 +00:00
|
|
|
|
|
|
|
#include "mon/MonMap.h"
|
|
|
|
#include "mds/MDS.h"
|
2010-11-04 18:30:59 +00:00
|
|
|
#include "mds/Dumper.h"
|
2011-01-29 00:25:31 +00:00
|
|
|
#include "mds/Resetter.h"
|
2007-10-17 23:34:54 +00:00
|
|
|
|
|
|
|
#include "msg/SimpleMessenger.h"
|
|
|
|
|
|
|
|
#include "common/Timer.h"
|
2011-02-20 17:18:03 +00:00
|
|
|
#include "common/ceph_argparse.h"
|
2011-11-21 21:32:45 +00:00
|
|
|
#include "common/pick_address.h"
|
2007-10-17 23:34:54 +00:00
|
|
|
|
2012-02-12 00:39:27 +00:00
|
|
|
#include "global/global_init.h"
|
|
|
|
#include "global/signal_handler.h"
|
|
|
|
#include "global/pidfile.h"
|
|
|
|
|
2008-03-10 05:32:16 +00:00
|
|
|
#include "mon/MonClient.h"
|
2007-10-17 23:34:54 +00:00
|
|
|
|
2010-02-04 21:48:23 +00:00
|
|
|
#include "auth/KeyRing.h"
|
|
|
|
|
2009-03-10 21:57:24 +00:00
|
|
|
void usage()
|
|
|
|
{
|
2011-09-21 23:28:43 +00:00
|
|
|
derr << "usage: ceph-mds -i name [flags] [[--journal_check rank]|[--hot-standby][rank]]\n"
|
2010-12-17 00:25:29 +00:00
|
|
|
<< " -m monitorip:port\n"
|
|
|
|
<< " connect to monitor at given address\n"
|
|
|
|
<< " --debug_mds n\n"
|
2011-01-25 22:07:48 +00:00
|
|
|
<< " debug MDS level (e.g. 10)\n"
|
2011-01-28 17:45:33 +00:00
|
|
|
<< " --dump-journal rank filename\n"
|
2011-12-01 16:19:47 +00:00
|
|
|
<< " dump the MDS journal for rank.\n"
|
2011-01-28 17:45:33 +00:00
|
|
|
<< " --journal-check rank\n"
|
|
|
|
<< " replay the journal for rank, then exit\n"
|
|
|
|
<< " --hot-standby rank\n"
|
|
|
|
<< " stat up as a hot standby for rank\n"
|
2011-01-29 00:25:31 +00:00
|
|
|
<< " --reset-journal rank\n"
|
|
|
|
<< " discard the MDS journal for rank, and replace it with a single\n"
|
|
|
|
<< " event that updates/resets inotable and sessionmap on replay.\n"
|
2011-01-25 22:07:48 +00:00
|
|
|
<< dendl;
|
2009-03-10 22:22:52 +00:00
|
|
|
generic_server_usage();
|
2009-03-10 21:57:24 +00:00
|
|
|
}
|
|
|
|
|
2011-06-02 22:50:22 +00:00
|
|
|
static int do_cmds_special_action(const std::string &action,
|
|
|
|
const std::string &dump_file, int rank)
|
|
|
|
{
|
2011-06-21 19:28:16 +00:00
|
|
|
SimpleMessenger *messenger = new SimpleMessenger(g_ceph_context);
|
2012-01-16 20:00:55 +00:00
|
|
|
int r = messenger->bind(g_conf->public_addr, getpid());
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
2011-06-21 19:28:16 +00:00
|
|
|
MonClient mc(g_ceph_context);
|
2011-06-02 22:50:22 +00:00
|
|
|
if (mc.build_initial_monmap() < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (action == "dump-journal") {
|
2011-09-28 04:42:35 +00:00
|
|
|
dout(0) << "dumping journal for mds." << rank << " to " << dump_file << dendl;
|
2011-06-02 22:50:22 +00:00
|
|
|
Dumper *journal_dumper = new Dumper(messenger, &mc);
|
|
|
|
journal_dumper->init(rank);
|
|
|
|
journal_dumper->dump(dump_file.c_str());
|
|
|
|
mc.shutdown();
|
|
|
|
}
|
|
|
|
else if (action == "undump-journal") {
|
2011-09-28 04:42:35 +00:00
|
|
|
dout(0) << "undumping journal for mds." << rank << " from " << dump_file << dendl;
|
2011-06-02 22:50:22 +00:00
|
|
|
Dumper *journal_dumper = new Dumper(messenger, &mc);
|
|
|
|
journal_dumper->init(rank);
|
|
|
|
journal_dumper->undump(dump_file.c_str());
|
|
|
|
mc.shutdown();
|
|
|
|
}
|
|
|
|
else if (action == "reset-journal") {
|
|
|
|
dout(0) << "resetting journal" << dendl;
|
|
|
|
Resetter *jr = new Resetter(messenger, &mc);
|
|
|
|
jr->init(rank);
|
|
|
|
jr->reset();
|
|
|
|
mc.shutdown();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
assert(0);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void set_special_action(std::string &dest, const std::string &act)
|
|
|
|
{
|
|
|
|
if (!dest.empty()) {
|
|
|
|
derr << "Parse error! Can't specify more than one action. You "
|
|
|
|
<< "specified both " << act << " and " << dest << "\n" << dendl;
|
|
|
|
usage();
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
dest = act;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int parse_rank(const char *opt_name, const std::string &val)
|
|
|
|
{
|
|
|
|
std::string err;
|
|
|
|
int ret = strict_strtol(val.c_str(), 10, &err);
|
|
|
|
if (!err.empty()) {
|
|
|
|
derr << "error parsing " << opt_name << ": failed to parse rank. "
|
|
|
|
<< "It must be an int." << "\n" << dendl;
|
|
|
|
usage();
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-02-12 00:33:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
MDS *mds = NULL;
|
|
|
|
|
|
|
|
|
|
|
|
static void handle_mds_signal(int signum)
|
|
|
|
{
|
|
|
|
if (mds)
|
|
|
|
mds->handle_signal(signum);
|
|
|
|
}
|
|
|
|
|
2008-01-01 22:04:31 +00:00
|
|
|
int main(int argc, const char **argv)
|
2007-10-17 23:34:54 +00:00
|
|
|
{
|
2008-01-01 22:04:31 +00:00
|
|
|
vector<const char*> args;
|
2007-10-17 23:34:54 +00:00
|
|
|
argv_to_vec(argc, argv, args);
|
2008-11-05 22:54:13 +00:00
|
|
|
env_to_vec(args);
|
2010-07-01 15:55:52 +00:00
|
|
|
|
2011-06-17 03:15:24 +00:00
|
|
|
global_init(args, CEPH_ENTITY_TYPE_MDS, CODE_ENVIRONMENT_DAEMON, 0);
|
2007-10-17 23:34:54 +00:00
|
|
|
|
|
|
|
// mds specific args
|
2011-01-28 17:45:33 +00:00
|
|
|
int shadow = 0;
|
2011-06-02 22:50:22 +00:00
|
|
|
int rank = -1;
|
|
|
|
std::string dump_file;
|
|
|
|
|
|
|
|
std::string val, action;
|
|
|
|
for (std::vector<const char*>::iterator i = args.begin(); i != args.end(); ) {
|
2011-08-24 01:12:06 +00:00
|
|
|
if (ceph_argparse_double_dash(args, i)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else if (ceph_argparse_witharg(args, i, &val, "--dump-journal", (char*)NULL)) {
|
2011-06-02 22:50:22 +00:00
|
|
|
set_special_action(action, "dump-journal");
|
|
|
|
rank = parse_rank("dump-journal", val);
|
|
|
|
if (i == args.end()) {
|
|
|
|
derr << "error parsing --dump-journal: you must give a second "
|
|
|
|
<< "dump-journal argument: the filename to dump the journal to. "
|
|
|
|
<< "\n" << dendl;
|
|
|
|
usage();
|
|
|
|
}
|
|
|
|
dump_file = *i++;
|
|
|
|
}
|
|
|
|
else if (ceph_argparse_witharg(args, i, &val, "--undump-journal", (char*)NULL)) {
|
|
|
|
set_special_action(action, "undump-journal");
|
|
|
|
rank = parse_rank("undump-journal", val);
|
|
|
|
if (i == args.end()) {
|
|
|
|
derr << "error parsing --undump-journal: you must give a second "
|
|
|
|
<< "undump-journal argument: the filename to undump the journal from. "
|
|
|
|
<< "\n" << dendl;
|
|
|
|
usage();
|
|
|
|
}
|
|
|
|
dump_file = *i++;
|
|
|
|
}
|
|
|
|
else if (ceph_argparse_witharg(args, i, &val, "--reset-journal", (char*)NULL)) {
|
|
|
|
set_special_action(action, "reset-journal");
|
|
|
|
rank = parse_rank("reset-journal", val);
|
|
|
|
}
|
|
|
|
else if (ceph_argparse_witharg(args, i, &val, "--journal-check", (char*)NULL)) {
|
|
|
|
int r = parse_rank("journal-check", val);
|
2010-12-03 00:36:22 +00:00
|
|
|
if (shadow) {
|
|
|
|
dout(0) << "Error: can only select one standby state" << dendl;
|
|
|
|
return -1;
|
|
|
|
}
|
2011-09-28 04:42:35 +00:00
|
|
|
dout(0) << "requesting oneshot_replay for mds." << r << dendl;
|
2010-12-03 00:36:22 +00:00
|
|
|
shadow = MDSMap::STATE_ONESHOT_REPLAY;
|
2011-09-14 21:54:35 +00:00
|
|
|
char rb[32];
|
|
|
|
snprintf(rb, sizeof(rb), "%d", r);
|
|
|
|
g_conf->set_val("mds_standby_for_rank", rb);
|
|
|
|
g_conf->apply_changes(NULL);
|
2011-06-02 22:50:22 +00:00
|
|
|
}
|
|
|
|
else if (ceph_argparse_witharg(args, i, &val, "--hot-standby", (char*)NULL)) {
|
|
|
|
int r = parse_rank("hot-standby", val);
|
2010-12-03 00:36:22 +00:00
|
|
|
if (shadow) {
|
|
|
|
dout(0) << "Error: can only select one standby state" << dendl;
|
|
|
|
return -1;
|
|
|
|
}
|
2011-09-28 04:42:35 +00:00
|
|
|
dout(0) << "requesting standby_replay for mds." << r << dendl;
|
2010-12-03 00:36:22 +00:00
|
|
|
shadow = MDSMap::STATE_STANDBY_REPLAY;
|
2011-09-14 21:54:35 +00:00
|
|
|
char rb[32];
|
|
|
|
snprintf(rb, sizeof(rb), "%d", r);
|
|
|
|
g_conf->set_val("mds_standby_for_rank", rb);
|
|
|
|
g_conf->apply_changes(NULL);
|
2011-06-02 22:50:22 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
derr << "Error: can't understand argument: " << *i << "\n" << dendl;
|
2011-03-28 23:10:33 +00:00
|
|
|
usage();
|
2010-11-04 18:30:59 +00:00
|
|
|
}
|
2007-10-17 23:34:54 +00:00
|
|
|
}
|
2011-06-02 22:50:22 +00:00
|
|
|
|
2011-11-21 21:32:45 +00:00
|
|
|
pick_addresses(g_ceph_context);
|
|
|
|
|
2011-06-02 22:50:22 +00:00
|
|
|
// Check for special actions
|
|
|
|
if (!action.empty()) {
|
|
|
|
return do_cmds_special_action(action, dump_file, rank);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Normal startup
|
2011-06-07 17:38:54 +00:00
|
|
|
if (g_conf->name.has_default_id()) {
|
2011-09-21 23:28:43 +00:00
|
|
|
derr << "must specify '-i name' with the ceph-mds instance name" << dendl;
|
2009-03-11 20:04:03 +00:00
|
|
|
usage();
|
2009-03-11 21:58:58 +00:00
|
|
|
}
|
2007-10-17 23:34:54 +00:00
|
|
|
|
2011-12-01 17:17:00 +00:00
|
|
|
global_print_banner();
|
|
|
|
|
2011-06-21 19:28:16 +00:00
|
|
|
SimpleMessenger *messenger = new SimpleMessenger(g_ceph_context);
|
2012-01-04 21:54:11 +00:00
|
|
|
messenger->set_cluster_protocol(CEPH_MDS_PROTOCOL);
|
|
|
|
|
2012-01-16 20:00:55 +00:00
|
|
|
int r = messenger->bind(g_conf->public_addr, getpid());
|
|
|
|
if (r < 0)
|
|
|
|
exit(1);
|
2012-01-17 22:23:58 +00:00
|
|
|
|
2011-06-07 17:38:54 +00:00
|
|
|
cout << "starting " << g_conf->name << " at " << messenger->get_ms_addr()
|
2011-06-02 22:50:22 +00:00
|
|
|
<< std::endl;
|
|
|
|
messenger->register_entity(entity_name_t::MDS(-1));
|
|
|
|
uint64_t supported =
|
|
|
|
CEPH_FEATURE_UID |
|
|
|
|
CEPH_FEATURE_NOSRCADDR |
|
2011-08-27 16:56:45 +00:00
|
|
|
CEPH_FEATURE_DIRLAYOUTHASH |
|
|
|
|
CEPH_FEATURE_PGID64;
|
2012-01-13 23:08:17 +00:00
|
|
|
uint64_t required =
|
|
|
|
CEPH_FEATURE_OSDREPLYMUX;
|
|
|
|
messenger->set_default_policy(SimpleMessenger::Policy::client(supported, required));
|
2011-06-02 22:50:22 +00:00
|
|
|
messenger->set_policy(entity_name_t::TYPE_MON,
|
|
|
|
SimpleMessenger::Policy::client(supported,
|
2011-08-27 16:56:45 +00:00
|
|
|
CEPH_FEATURE_UID |
|
|
|
|
CEPH_FEATURE_PGID64));
|
2011-06-02 22:50:22 +00:00
|
|
|
messenger->set_policy(entity_name_t::TYPE_MDS,
|
|
|
|
SimpleMessenger::Policy::lossless_peer(supported,
|
|
|
|
CEPH_FEATURE_UID));
|
|
|
|
messenger->set_policy(entity_name_t::TYPE_CLIENT,
|
|
|
|
SimpleMessenger::Policy::stateful_server(supported, 0));
|
|
|
|
|
|
|
|
if (shadow != MDSMap::STATE_ONESHOT_REPLAY)
|
2011-06-21 19:28:16 +00:00
|
|
|
global_init_daemonize(g_ceph_context, 0);
|
|
|
|
common_init_finish(g_ceph_context);
|
2011-06-02 22:50:22 +00:00
|
|
|
|
2008-03-10 05:32:16 +00:00
|
|
|
// get monmap
|
2011-06-21 19:28:16 +00:00
|
|
|
MonClient mc(g_ceph_context);
|
2009-06-26 20:49:12 +00:00
|
|
|
if (mc.build_initial_monmap() < 0)
|
2008-01-31 22:43:18 +00:00
|
|
|
return -1;
|
2011-06-21 19:28:16 +00:00
|
|
|
global_init_chdir(g_ceph_context);
|
2007-10-17 23:34:54 +00:00
|
|
|
|
2011-06-02 22:50:22 +00:00
|
|
|
messenger->start();
|
2010-11-04 18:30:59 +00:00
|
|
|
|
2012-02-12 00:33:51 +00:00
|
|
|
// set up signal handlers, now that we've daemonized/forked.
|
|
|
|
init_async_signal_handler();
|
|
|
|
register_async_signal_handler(SIGHUP, sighup_handler);
|
|
|
|
register_async_signal_handler_oneshot(SIGINT, handle_mds_signal);
|
|
|
|
register_async_signal_handler_oneshot(SIGTERM, handle_mds_signal);
|
|
|
|
|
2011-06-02 22:50:22 +00:00
|
|
|
// start mds
|
2012-02-12 00:33:51 +00:00
|
|
|
mds = new MDS(g_conf->name.get_id().c_str(), messenger, &mc);
|
2011-06-02 22:50:22 +00:00
|
|
|
|
|
|
|
// in case we have to respawn...
|
|
|
|
mds->orig_argc = argc;
|
|
|
|
mds->orig_argv = argv;
|
|
|
|
|
|
|
|
if (shadow)
|
|
|
|
mds->init(shadow);
|
|
|
|
else
|
|
|
|
mds->init();
|
|
|
|
|
|
|
|
messenger->wait();
|
|
|
|
|
2012-02-12 00:33:51 +00:00
|
|
|
unregister_async_signal_handler(SIGINT, handle_mds_signal);
|
|
|
|
unregister_async_signal_handler(SIGTERM, handle_mds_signal);
|
|
|
|
|
2011-06-02 22:50:22 +00:00
|
|
|
// yuck: grab the mds lock, so we can be sure that whoever in *mds
|
|
|
|
// called shutdown finishes what they were doing.
|
|
|
|
mds->mds_lock.Lock();
|
|
|
|
mds->mds_lock.Unlock();
|
|
|
|
|
2012-02-12 00:39:27 +00:00
|
|
|
pidfile_remove();
|
|
|
|
|
2011-06-02 22:50:22 +00:00
|
|
|
// only delete if it was a clean shutdown (to aid memory leak
|
|
|
|
// detection, etc.). don't bother if it was a suicide.
|
|
|
|
if (mds->is_stopped())
|
|
|
|
delete mds;
|
|
|
|
|
|
|
|
// cd on exit, so that gmon.out (if any) goes into a separate directory for each node.
|
|
|
|
char s[20];
|
|
|
|
snprintf(s, sizeof(s), "gmon/%d", getpid());
|
|
|
|
if ((mkdir(s, 0755) == 0) && (chdir(s) == 0)) {
|
2011-09-21 23:28:43 +00:00
|
|
|
dout(0) << "ceph-mds: gmon.out should be in " << s << dendl;
|
2010-11-04 18:30:59 +00:00
|
|
|
}
|
2011-06-02 22:50:22 +00:00
|
|
|
|
|
|
|
generic_dout(0) << "stopped." << dendl;
|
2007-10-17 23:34:54 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|