/* * ipmi_port.c * Allocate the RMCP port (623.) with a bind so that port manager * does not try to reuse it. Only needed for Linux. * * Note that the Intel dpcproxy service also uses port 623 to listen * for incoming telnet/SOL clients, so we should not start ipmi_port * if dpcproxy is running. * * Changes: * 05/18/07 Andy Cress - created */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <fcntl.h> #include <netinet/in.h> #include <sys/types.h> #include <sys/socket.h> #include <signal.h> #ifdef TEST #include "ipmicmd.h" #endif #define RMCP_PORT 623 static char * progver = "1.4"; /* program version */ static char *progname = "ipmi_port"; /* program name */ static int sockfd = 0; static struct sockaddr_in _srcaddr; static int interval = 20; /* num sec to wait, was 60 */ static char fdebug = 0; static int mkdaemon(int fchdir, int fclose); static int mkdaemon(int fchdir, int fclose) { int fdlimit = sysconf(_SC_OPEN_MAX); /*fdlimit usu = 1024.*/ int fd = 0; int i; fdlimit = fileno(stderr); switch (fork()) { case 0: break; case -1: return -1; default: _exit(0); /* exit the original process */ } if (setsid() < 0) return -1; /* shouldn't fail */ switch (fork()) { case 0: break; case -1: return -1; default: _exit(0); } if (fchdir) i = chdir("/"); if (fclose) { /* Close stdin,stdout,stderr and replace them with /dev/null */ for (fd = 0; fd < fdlimit; fd++) close(fd); fd = open("/dev/null",O_RDWR); i = dup(0); i = dup(0); } return 0; } static int open_rmcp_port(int port); static int open_rmcp_port(int port) { int rv; /* Open a socket binding to the RMCP port */ rv = socket(AF_INET, SOCK_DGRAM, 0); if (rv < 0) return (rv); else sockfd = rv; memset(&_srcaddr, 0, sizeof(_srcaddr)); _srcaddr.sin_family = AF_INET; _srcaddr.sin_port = htons(port); _srcaddr.sin_addr.s_addr = htonl(INADDR_ANY); rv = bind(sockfd, (struct sockaddr *)&_srcaddr, sizeof(_srcaddr)); if (rv < 0) { printf("bind(%d,%d) error, rv = %d\n",port,INADDR_ANY,rv); close(sockfd); return (rv); } return(rv); } static void iport_cleanup(int sig) { if (sockfd != 0) close(sockfd); exit(EXIT_SUCCESS); } /*int ipmi_port(int argc, char **argv) */ int main(int argc, char **argv) { int rv; int c; int background = 0; struct sigaction sact; while ((c = getopt(argc, argv, "bx?")) != EOF) { switch (c) { case 'x': fdebug = 1; break; case 'b': background = 1; break; default: printf("Usage: %s [-xb] (-b means background)\n",progname); exit(1); } } if (!background) printf("%s ver %s\n", progname,progver); rv = open_rmcp_port(RMCP_PORT); if (rv != 0) { printf("open_rmcp_port(%d) failed, rv = %d\n",RMCP_PORT,rv); exit(1); } else if (!background) printf("open_rmcp_port(%d) succeeded, sleeping\n",RMCP_PORT); /* convert to a daemon if background */ if (background) { rv = mkdaemon(1,1); if (rv != 0) { printf("%s: Cannot become daemon, rv = %d\n", progname,rv); exit(1); } } /* handle signals for cleanup */ sact.sa_handler = iport_cleanup; sact.sa_flags = 0; sigemptyset(&sact.sa_mask); sigaction(SIGINT, &sact, NULL); sigaction(SIGQUIT, &sact, NULL); sigaction(SIGTERM, &sact, NULL); if (rv == 0) while(1) { sleep(interval); /*wait 60 seconds*/ } if (sockfd != 0) close(sockfd); return(rv); } /*end ipmi_port.c */