mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-26 22:30:28 +00:00 
			
		
		
		
	were included almost everywhere. Remove some of the instances. git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@89424 65c4cc65-6c06-0410-ace0-fbb531ad65f3
		
			
				
	
	
		
			217 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			217 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * 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");
 | |
| 
 |