/* * Copyright (C) 2002 Jeff Dike (jdike@karaya.com) * Licensed under the GPL */ #include <stdio.h> #include <unistd.h> #include <errno.h> #include "user_util.h" #include "user.h" #include "helper.h" #include "mconsole.h" #include "os.h" #include "choose-mode.h" #include "mode.h" struct dog_data { int stdin; int stdout; int close_me[2]; }; static void pre_exec(void *d) { struct dog_data *data = d; dup2(data->stdin, 0); dup2(data->stdout, 1); dup2(data->stdout, 2); os_close_file(data->stdin); os_close_file(data->stdout); os_close_file(data->close_me[0]); os_close_file(data->close_me[1]); } int start_watchdog(int *in_fd_ret, int *out_fd_ret, char *sock) { struct dog_data data; int in_fds[2], out_fds[2], pid, n, err; char pid_buf[sizeof("nnnnn\0")], c; char *pid_args[] = { "/usr/bin/uml_watchdog", "-pid", pid_buf, NULL }; char *mconsole_args[] = { "/usr/bin/uml_watchdog", "-mconsole", NULL, NULL }; char **args = NULL; err = os_pipe(in_fds, 1, 0); if(err < 0){ printk("harddog_open - os_pipe failed, err = %d\n", -err); goto out; } err = os_pipe(out_fds, 1, 0); if(err < 0){ printk("harddog_open - os_pipe failed, err = %d\n", -err); goto out_close_in; } data.stdin = out_fds[0]; data.stdout = in_fds[1]; data.close_me[0] = out_fds[1]; data.close_me[1] = in_fds[0]; if(sock != NULL){ mconsole_args[2] = sock; args = mconsole_args; } else { /* XXX The os_getpid() is not SMP correct */ sprintf(pid_buf, "%d", CHOOSE_MODE(tracing_pid, os_getpid())); args = pid_args; } pid = run_helper(pre_exec, &data, args, NULL); os_close_file(out_fds[0]); os_close_file(in_fds[1]); if(pid < 0){ err = -pid; printk("harddog_open - run_helper failed, errno = %d\n", -err); goto out_close_out; } n = os_read_file(in_fds[0], &c, sizeof(c)); if(n == 0){ printk("harddog_open - EOF on watchdog pipe\n"); helper_wait(pid); err = -EIO; goto out_close_out; } else if(n < 0){ printk("harddog_open - read of watchdog pipe failed, " "err = %d\n", -n); helper_wait(pid); err = n; goto out_close_out; } *in_fd_ret = in_fds[0]; *out_fd_ret = out_fds[1]; return(0); out_close_in: os_close_file(in_fds[0]); os_close_file(in_fds[1]); out_close_out: os_close_file(out_fds[0]); os_close_file(out_fds[1]); out: return(err); } void stop_watchdog(int in_fd, int out_fd) { os_close_file(in_fd); os_close_file(out_fd); } int ping_watchdog(int fd) { int n; char c = '\n'; n = os_write_file(fd, &c, sizeof(c)); if(n != sizeof(c)){ printk("ping_watchdog - write failed, err = %d\n", -n); if(n < 0) return(n); return(-EIO); } return 1; } /* * Overrides for Emacs so that we follow Linus's tabbing style. * Emacs will notice this stuff at the end of the file and automatically * adjust the settings for this buffer only. This must remain at the end * of the file. * --------------------------------------------------------------------------- * Local variables: * c-file-style: "linux" * End: */