Vendor: Samba Team
Packager: Samba Team <samba@samba.org>
Name: ctdb
-Version: 1.0.108
+Version: 1.0.109
Release: 1GITHASH
Epoch: 0
License: GNU GPL version 3
%{_docdir}/ctdb/tests/bin/ctdb_transaction
%changelog
+* Thu Dec 17 2009 : Version 1.0.109
+ - Massive eventscript updates. (bz58828)
+ - Nice the daemon instead of using realtime scheduler, also use mlockall() to
+ reduce the risk of blockign due to paging.
+ - Workarounds for valgrind when forking once for each script. Valgrind consumes
+ massive cpu when terminating the scripts on virtual systems.
+ - Sync the tdb library with upstream, and use the new TDB_DISALLOW_NESTING flag.
+ - Add new command "ctdb dumpdbbackup"
+ - Start using the new tdb check framework to validate tdb files upon startup.
+ - A new framework where we can control health for individual tdb databases.
+ - Fix a crash bug in the logging code.
+ - New transaction code for persistent databases.
+ - Various other smaller fixes.
* Mon Dec 7 2009 : Version 1.0.108
- Transaction updates from Michael Adam.
- Use the new wbinfo --ping-dc instead of -p in the eventscript for samba
pid_t child;
/* Warning: this can free us! */
void (*callback)(struct ctdb_context *, int, void *);
+ int cb_status;
int fd[2];
void *private_data;
bool from_user;
return 0;
}
-/*
- Summarize status of this run of scripts.
- */
-static int script_status(struct ctdb_scripts_wire *scripts)
-{
- unsigned int i;
-
- for (i = 0; i < scripts->num_scripts; i++) {
- switch (scripts->scripts[i].status) {
- case -ENOENT:
- case -ENOEXEC:
- /* Disabled or missing; that's OK. */
- break;
- case 0:
- /* No problem. */
- break;
- default:
- return scripts->scripts[i].status;
- }
- }
-
- /* All OK! */
- return 0;
-}
-
/* called when child is finished */
static void ctdb_event_script_handler(struct event_context *ev, struct fd_event *fde,
uint16_t flags, void *p)
talloc_get_type(p, struct ctdb_event_script_state);
struct ctdb_script_wire *current = get_current_script(state);
struct ctdb_context *ctdb = state->ctdb;
- int r, status;
+ int r;
r = read(state->fd[0], ¤t->status, sizeof(current->status));
if (r < 0) {
}
current->finished = timeval_current();
+
+ /* update overall status based on this script. */
+ state->cb_status = current->status;
+
+ /* don't stop just because it vanished or was disabled. */
+ if (current->status == -ENOENT || current->status == -ENOEXEC) {
+ state->cb_status = 0;
+ }
+
/* valgrind gets overloaded if we run next script as it's still doing
* post-execution analysis, so kill finished child here. */
if (ctdb->valgrinding) {
state->child = 0;
- status = script_status(state->scripts);
-
/* Aborted or finished all scripts? We're done. */
- if (status != 0 || state->current+1 == state->scripts->num_scripts) {
+ if (state->cb_status != 0 || state->current+1 == state->scripts->num_scripts) {
DEBUG(DEBUG_INFO,(__location__ " Eventscript %s %s finished with state %d\n",
- ctdb_eventscript_call_names[state->call], state->options, status));
+ ctdb_eventscript_call_names[state->call], state->options, state->cb_status));
ctdb->event_script_timeouts = 0;
talloc_free(state);
/* Next script! */
state->current++;
- current++;
- current->status = fork_child_for_script(ctdb, state);
- if (current->status != 0) {
+ state->cb_status = fork_child_for_script(ctdb, state);
+ if (state->cb_status != 0) {
/* This calls the callback. */
talloc_free(state);
}
{
struct ctdb_event_script_state *state = talloc_get_type(p, struct ctdb_event_script_state);
struct ctdb_context *ctdb = state->ctdb;
- struct ctdb_script_wire *current = get_current_script(state);
- DEBUG(DEBUG_ERR,("Event script timed out : %s %s %s count : %u pid : %d\n",
- current->name, ctdb_eventscript_call_names[state->call], state->options, ctdb->event_script_timeouts, state->child));
+ DEBUG(DEBUG_ERR,("Event script timed out : %s %s count : %u pid : %d\n",
+ ctdb_eventscript_call_names[state->call], state->options, ctdb->event_script_timeouts, state->child));
- state->scripts->scripts[state->current].status = -ETIME;
+ state->cb_status = -ETIME;
if (kill(state->child, 0) != 0) {
DEBUG(DEBUG_ERR,("Event script child process already dead, errno %s(%d)\n", strerror(errno), errno));
state->child = 0;
}
+ state->scripts->scripts[state->current].status = state->cb_status;
+
talloc_free(state);
}
*/
static int event_script_destructor(struct ctdb_event_script_state *state)
{
- int status;
-
if (state->child) {
DEBUG(DEBUG_ERR,(__location__ " Sending SIGTERM to child pid:%d\n", state->child));
state->ctdb->current_monitor = NULL;
}
- /* Save our scripts as the last executed status, if we have them.
- * See ctdb_event_script_callback_v where we abort monitor event. */
+ /* Save our scripts as the last executed status, if we have them. */
if (state->scripts) {
talloc_free(state->ctdb->last_status[state->call]);
state->ctdb->last_status[state->call] = state->scripts;
}
}
- /* Use last status as result, or "OK" if none. */
- if (state->ctdb->last_status[state->call]) {
- status = script_status(state->ctdb->last_status[state->call]);
- } else {
- status = 0;
- }
-
/* This is allowed to free us; talloc will prevent double free anyway,
* but beware if you call this outside the destructor! */
if (state->callback) {
- state->callback(state->ctdb, status, state->private_data);
+ state->callback(state->ctdb, state->cb_status, state->private_data);
}
return 0;
const char *fmt, va_list ap)
{
struct ctdb_event_script_state *state;
+ int ret;
state = talloc(ctdb->event_script_ctx, struct ctdb_event_script_state);
CTDB_NO_MEMORY(ctdb, state);
return -1;
}
state->current = 0;
- talloc_set_destructor(state, event_script_destructor);
/* Nothing to do? */
if (state->scripts->num_scripts == 0) {
+ ctdb->event_script_timeouts = 0;
talloc_free(state);
return 0;
}
- state->scripts->scripts[0].status = fork_child_for_script(ctdb, state);
- if (state->scripts->scripts[0].status != 0) {
- /* Callback is called from destructor, with fail result. */
+ ret = fork_child_for_script(ctdb, state);
+ if (ret != 0) {
+ talloc_free(state->scripts);
talloc_free(state);
- return 0;
+ return -1;
}
+ talloc_set_destructor(state, event_script_destructor);
if (!timeval_is_zero(&state->timeout)) {
event_add_timed(ctdb->ev, state, timeval_current_ofs(state->timeout.tv_sec, state->timeout.tv_usec), ctdb_event_script_timeout, state);
} else {