aboutsummaryrefslogtreecommitdiffstats
path: root/trunk/res/res_limit.c
diff options
context:
space:
mode:
Diffstat (limited to 'trunk/res/res_limit.c')
-rw-r--r--trunk/res/res_limit.c216
1 files changed, 216 insertions, 0 deletions
diff --git a/trunk/res/res_limit.c b/trunk/res/res_limit.c
new file mode 100644
index 000000000..28a5e1323
--- /dev/null
+++ b/trunk/res/res_limit.c
@@ -0,0 +1,216 @@
+/*
+ * Asterisk -- A telephony toolkit for Linux.
+ *
+ * Resource limits
+ *
+ * Copyright (c) 2006 Tilghman Lesher. All rights reserved.
+ *
+ * Tilghman Lesher <res_limit_200607@the-tilghman.com>
+ *
+ * This code is released by the author with no restrictions on usage.
+ *
+ */
+
+/*! \file
+ *
+ * \brief Resource limits
+ *
+ * \author Tilghman Lesher <res_limit_200607@the-tilghman.com>
+ */
+
+
+#include "asterisk.h"
+
+ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
+
+#define _XOPEN_SOURCE 600
+
+#include <ctype.h>
+#include <sys/time.h>
+#include <sys/resource.h>
+#include "asterisk/module.h"
+#include "asterisk/cli.h"
+
+/* Find proper rlimit for virtual memory */
+#ifdef RLIMIT_AS
+#define VMEM_DEF RLIMIT_AS
+#else
+#ifdef RLIMIT_VMEM
+#define VMEM_DEF RLIMIT_VMEM
+#endif
+#endif
+
+static struct limits {
+ int resource;
+ char limit[3];
+ char desc[40];
+} limits[] = {
+ { RLIMIT_CPU, "-t", "cpu time" },
+ { RLIMIT_FSIZE, "-f", "file size" },
+ { RLIMIT_DATA, "-d", "program data segment" },
+ { RLIMIT_STACK, "-s", "program stack size" },
+ { RLIMIT_CORE, "-c", "core file size" },
+#ifdef RLIMIT_RSS
+ { RLIMIT_RSS, "-m", "resident memory" },
+ { RLIMIT_MEMLOCK, "-l", "amount of memory locked into RAM" },
+#endif
+#ifdef RLIMIT_NPROC
+ { RLIMIT_NPROC, "-u", "number of processes" },
+#endif
+ { RLIMIT_NOFILE, "-n", "number of file descriptors" },
+#ifdef VMEM_DEF
+ { VMEM_DEF, "-v", "virtual memory" },
+#endif
+};
+
+static int str2limit(const char *string)
+{
+ size_t i;
+ for (i = 0; i < sizeof(limits) / sizeof(limits[0]); i++) {
+ if (!strcasecmp(string, limits[i].limit))
+ return limits[i].resource;
+ }
+ return -1;
+}
+
+static const char *str2desc(const char *string)
+{
+ size_t i;
+ for (i = 0; i < sizeof(limits) / sizeof(limits[0]); i++) {
+ if (!strcmp(string, limits[i].limit))
+ return limits[i].desc;
+ }
+ return "<unknown>";
+}
+
+static char *complete_ulimit(struct ast_cli_args *a)
+{
+ int which = 0, i;
+ int wordlen = strlen(a->word);
+
+ if (a->pos > 1)
+ return NULL;
+ for (i = 0; i < sizeof(limits) / sizeof(limits[0]); i++) {
+ if (!strncasecmp(limits[i].limit, a->word, wordlen)) {
+ if (++which > a->n)
+ return ast_strdup(limits[i].limit);
+ }
+ }
+ return NULL;
+}
+
+static char *handle_cli_ulimit(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
+{
+ int resource;
+ struct rlimit rlimit = { 0, 0 };
+
+ switch (cmd) {
+ case CLI_INIT:
+ e->command = "ulimit";
+ e->usage =
+ "Usage: ulimit {-d|"
+#ifdef RLIMIT_RSS
+ "-l|"
+#endif
+ "-f|"
+#ifdef RLIMIT_RSS
+ "-m|"
+#endif
+ "-s|-t|"
+#ifdef RLIMIT_NPROC
+ "-u|"
+#endif
+#ifdef VMEM_DEF
+ "-v|"
+#endif
+ "-c|-n} [<num>]\n"
+ " Shows or sets the corresponding resource limit.\n"
+ " -d Process data segment [readonly]\n"
+#ifdef RLIMIT_RSS
+ " -l Memory lock size [readonly]\n"
+#endif
+ " -f File size\n"
+#ifdef RLIMIT_RSS
+ " -m Process resident memory [readonly]\n"
+#endif
+ " -s Process stack size [readonly]\n"
+ " -t CPU usage [readonly]\n"
+#ifdef RLIMIT_NPROC
+ " -u Child processes\n"
+#endif
+#ifdef VMEM_DEF
+ " -v Process virtual memory [readonly]\n"
+#endif
+ " -c Core dump file size\n"
+ " -n Number of file descriptors\n";
+ return NULL;
+ case CLI_GENERATE:
+ return complete_ulimit(a);
+ }
+
+ if (a->argc > 3)
+ return CLI_SHOWUSAGE;
+
+ if (a->argc == 1) {
+ char arg2[3];
+ char *newargv[2] = { "ulimit", arg2 };
+ for (resource = 0; resource < sizeof(limits) / sizeof(limits[0]); resource++) {
+ struct ast_cli_args newArgs = { .argv = newargv, .argc = 2 };
+ ast_copy_string(arg2, limits[resource].limit, sizeof(arg2));
+ handle_cli_ulimit(e, CLI_HANDLER, &newArgs);
+ }
+ return CLI_SUCCESS;
+ } else {
+ resource = str2limit(a->argv[1]);
+ if (resource == -1) {
+ ast_cli(a->fd, "Unknown resource\n");
+ return CLI_FAILURE;
+ }
+
+ if (a->argc == 3) {
+ int x;
+#ifdef RLIMIT_NPROC
+ if (resource != RLIMIT_NOFILE && resource != RLIMIT_CORE && resource != RLIMIT_NPROC && resource != RLIMIT_FSIZE) {
+#else
+ if (resource != RLIMIT_NOFILE && resource != RLIMIT_CORE && resource != RLIMIT_FSIZE) {
+#endif
+ ast_cli(a->fd, "Resource not permitted to be set\n");
+ return CLI_FAILURE;
+ }
+
+ sscanf(a->argv[2], "%d", &x);
+ rlimit.rlim_max = rlimit.rlim_cur = x;
+ setrlimit(resource, &rlimit);
+ return CLI_SUCCESS;
+ } else {
+ if (!getrlimit(resource, &rlimit)) {
+ char printlimit[32];
+ const char *desc;
+ if (rlimit.rlim_max == RLIM_INFINITY)
+ ast_copy_string(printlimit, "effectively unlimited", sizeof(printlimit));
+ else
+ snprintf(printlimit, sizeof(printlimit), "limited to %d", (int) rlimit.rlim_cur);
+ desc = str2desc(a->argv[1]);
+ ast_cli(a->fd, "%c%s (%s) is %s.\n", toupper(desc[0]), desc + 1, a->argv[1], printlimit);
+ } else
+ ast_cli(a->fd, "Could not retrieve resource limits for %s: %s\n", str2desc(a->argv[1]), strerror(errno));
+ return CLI_SUCCESS;
+ }
+ }
+}
+
+static struct ast_cli_entry cli_ulimit =
+ AST_CLI_DEFINE(handle_cli_ulimit, "Set or show process resource limits");
+
+static int unload_module(void)
+{
+ return ast_cli_unregister(&cli_ulimit);
+}
+
+static int load_module(void)
+{
+ return ast_cli_register(&cli_ulimit) ? AST_MODULE_LOAD_FAILURE : AST_MODULE_LOAD_SUCCESS;
+}
+
+AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Resource limits");
+