123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330 |
- /*
- * birq
- *
- * Balance IRQ
- *
- */
- #ifdef HAVE_CONFIG_H
- #include "config.h"
- #endif /* HAVE_CONFIG_H */
- #include <stdio.h>
- #include <stdlib.h>
- #include <unistd.h>
- #include <sys/types.h>
- #include <errno.h>
- #include <assert.h>
- #include <string.h>
- #include <signal.h>
- #include <syslog.h>
- #include <fcntl.h>
- #ifdef HAVE_GETOPT_H
- #include <getopt.h>
- #endif
- #include "birq.h"
- #include "lub/log.h"
- #include "lub/list.h"
- #include "irq.h"
- #include "cpu.h"
- #include "statistics.h"
- #include "balance.h"
- #ifndef VERSION
- #define VERSION 1.0.0
- #endif
- #define QUOTE(t) #t
- #define version(v) printf("%s\n", v)
- /* Signal handlers */
- static volatile int sigterm = 0; /* Exit if 1 */
- static void sighandler(int signo);
- static void help(int status, const char *argv0);
- static struct options *opts_init(void);
- static void opts_free(struct options *opts);
- static int opts_parse(int argc, char *argv[], struct options *opts);
- /* Command line options */
- struct options {
- char *pidfile;
- int debug; /* Don't daemonize in debug mode */
- int log_facility;
- float threshold;
- };
- /*--------------------------------------------------------- */
- int main(int argc, char **argv)
- {
- int retval = -1;
- struct options *opts = NULL;
- int pidfd = -1;
- int interval = BIRQ_SHORT_INTERVAL;
- /* Signal vars */
- struct sigaction sig_act;
- sigset_t sig_set;
- /* IRQ list. It contain all found IRQs. */
- lub_list_t *irqs;
- /* IRQs need to be balanced */
- lub_list_t *balance_irqs;
- /* CPU list. It contain all found CPUs. */
- lub_list_t *cpus;
- /* Parse command line options */
- opts = opts_init();
- if (opts_parse(argc, argv, opts))
- goto err;
- /* Initialize syslog */
- openlog(argv[0], LOG_CONS, opts->log_facility);
- syslog(LOG_ERR, "Start daemon.\n");
- /* Fork the daemon */
- if (!opts->debug) {
- /* Daemonize */
- if (daemon(0, 0) < 0) {
- syslog(LOG_ERR, "Can't daemonize\n");
- goto err;
- }
- /* Write pidfile */
- if ((pidfd = open(opts->pidfile,
- O_WRONLY | O_CREAT | O_EXCL | O_TRUNC,
- S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH)) < 0) {
- syslog(LOG_WARNING, "Can't open pidfile %s: %s",
- opts->pidfile, strerror(errno));
- } else {
- char str[20];
- snprintf(str, sizeof(str), "%u\n", getpid());
- if (write(pidfd, str, strlen(str)) < 0)
- syslog(LOG_WARNING, "Can't write to %s: %s",
- opts->pidfile, strerror(errno));
- close(pidfd);
- }
- }
- /* Set signal handler */
- sigemptyset(&sig_set);
- sigaddset(&sig_set, SIGTERM);
- sigaddset(&sig_set, SIGINT);
- sigaddset(&sig_set, SIGQUIT);
- sig_act.sa_flags = 0;
- sig_act.sa_mask = sig_set;
- sig_act.sa_handler = &sighandler;
- sigaction(SIGTERM, &sig_act, NULL);
- sigaction(SIGINT, &sig_act, NULL);
- sigaction(SIGQUIT, &sig_act, NULL);
- /* Scan CPUs */
- cpus = lub_list_new(cpu_list_compare);
- scan_cpus(cpus);
- /* Prepare data structures */
- irqs = lub_list_new(irq_list_compare);
- balance_irqs = lub_list_new(irq_list_compare);
- /* Main loop */
- while (!sigterm) {
- lub_list_node_t *node;
- /* Rescan PCI devices for new IRQs. */
- scan_irqs(irqs, balance_irqs);
- /* Gather statistics on CPU load and number of interrupts. */
- gather_statistics(cpus, irqs);
- show_statistics(cpus);
- /* Choose IRQ to move to another CPU.
- Don't choose IRQ if we already have new IRQs to balance */
- if (lub_list_len(balance_irqs) == 0) {
- choose_irqs_to_move(cpus, balance_irqs,
- opts->threshold);
- }
- /* If nothing to balance */
- if (lub_list_len(balance_irqs) != 0) {
- /* Set short interval to make balancing faster. */
- interval = BIRQ_SHORT_INTERVAL;
- /* Choose new CPU for IRQs need to be balanced. */
- balance(cpus, balance_irqs, opts->threshold);
- /* Write new values to /proc/irq/<IRQ>/smp_affinity */
- apply_affinity(balance_irqs);
- /* Free list of balanced IRQs */
- while ((node = lub_list__get_tail(balance_irqs))) {
- lub_list_del(balance_irqs, node);
- lub_list_node_free(node);
- }
- } else {
- /* If nothing to balance */
- interval = BIRQ_LONG_INTERVAL;
- }
-
- /* Wait before nex iteration */
- sleep(interval);
- }
- /* Free data structures */
- irq_list_free(irqs);
- lub_list_free(balance_irqs);
- cpu_list_free(cpus);
- retval = 0;
- err:
- /* Remove pidfile */
- if (pidfd >= 0) {
- if (unlink(opts->pidfile) < 0) {
- syslog(LOG_ERR, "Can't remove pid-file %s: %s\n",
- opts->pidfile, strerror(errno));
- }
- }
- /* Free command line options */
- opts_free(opts);
- syslog(LOG_ERR, "Stop daemon.\n");
- return retval;
- }
- /*--------------------------------------------------------- */
- /*
- * Signal handler for temination signals (like SIGTERM, SIGINT, ...)
- */
- static void sighandler(int signo)
- {
- sigterm = 1;
- }
- /*--------------------------------------------------------- */
- /* Initialize option structure by defaults */
- static struct options *opts_init(void)
- {
- struct options *opts = NULL;
- opts = malloc(sizeof(*opts));
- assert(opts);
- opts->debug = 0; /* daemonize by default */
- opts->pidfile = strdup(BIRQ_PIDFILE);
- opts->log_facility = LOG_DAEMON;
- opts->threshold = BIRQ_DEFAULT_THRESHOLD;
- return opts;
- }
- /*--------------------------------------------------------- */
- /* Free option structure */
- static void opts_free(struct options *opts)
- {
- if (opts->pidfile)
- free(opts->pidfile);
- free(opts);
- }
- /*--------------------------------------------------------- */
- /* Parse command line options */
- static int opts_parse(int argc, char *argv[], struct options *opts)
- {
- static const char *shortopts = "hvp:dO:t:";
- #ifdef HAVE_GETOPT_H
- static const struct option longopts[] = {
- {"help", 0, NULL, 'h'},
- {"version", 0, NULL, 'v'},
- {"pid", 1, NULL, 'p'},
- {"debug", 0, NULL, 'd'},
- {"facility", 1, NULL, 'O'},
- {"threshold", 1, NULL, 't'},
- {NULL, 0, NULL, 0}
- };
- #endif
- optind = 1;
- while(1) {
- int opt;
- #ifdef HAVE_GETOPT_H
- opt = getopt_long(argc, argv, shortopts, longopts, NULL);
- #else
- opt = getopt(argc, argv, shortopts);
- #endif
- if (-1 == opt)
- break;
- switch (opt) {
- case 'p':
- if (opts->pidfile)
- free(opts->pidfile);
- opts->pidfile = strdup(optarg);
- break;
- case 'd':
- opts->debug = 1;
- break;
- case 'O':
- if (lub_log_facility(optarg, &(opts->log_facility))) {
- fprintf(stderr, "Error: Illegal syslog facility %s.\n", optarg);
- help(-1, argv[0]);
- exit(-1);
- }
- break;
- case 't':
- {
- char *endptr;
- float thresh;
- thresh = strtof(optarg, &endptr);
- if (endptr == optarg)
- thresh = opts->threshold;
- opts->threshold = thresh;
- if (thresh > 100.00) {
- fprintf(stderr, "Error: Illegal threshold value %s.\n", optarg);
- help(-1, argv[0]);
- exit(-1);
- }
- }
- break;
- case 'h':
- help(0, argv[0]);
- exit(0);
- break;
- case 'v':
- version(VERSION);
- exit(0);
- break;
- default:
- help(-1, argv[0]);
- exit(-1);
- break;
- }
- }
- return 0;
- }
- /*--------------------------------------------------------- */
- /* Print help message */
- static void help(int status, const char *argv0)
- {
- const char *name = NULL;
- if (!argv0)
- return;
- /* Find the basename */
- name = strrchr(argv0, '/');
- if (name)
- name++;
- else
- name = argv0;
- if (status != 0) {
- fprintf(stderr, "Try `%s -h' for more information.\n",
- name);
- } else {
- printf("Usage: %s [options]\n", name);
- printf("Daemon to store user configuration (i.e. commands). "
- "The part of the klish project.\n");
- printf("Options:\n");
- printf("\t-v, --version\tPrint version.\n");
- printf("\t-h, --help\tPrint this help.\n");
- printf("\t-d, --debug\tDebug mode. Don't daemonize.\n");
- printf("\t-p <path>, --pid=<path>\tFile to save daemon's PID to.\n");
- printf("\t-O, --facility\tSyslog facility. Default is DAEMON.\n");
- printf("\t-t <float>, --threshold=<float>\tThreshold to consider CPU is overloaded, in percents.\n");
- }
- }
|