mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-25 14:06:27 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			776 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			776 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Asterisk -- An open source telephony toolkit.
 | |
|  *
 | |
|  * Copyright (C) 1999 - 2005, Digium, Inc.
 | |
|  *
 | |
|  * Mark Spencer <markster@digium.com>
 | |
|  *
 | |
|  * Updated for Mac OSX CoreAudio
 | |
|  * by Josh Roberson <josh@asteriasgi.com>
 | |
|  *
 | |
|  * See http://www.asterisk.org for more information about
 | |
|  * the Asterisk project. Please do not directly contact
 | |
|  * any of the maintainers of this project for assistance;
 | |
|  * the project provides a web site, mailing lists and IRC
 | |
|  * channels for your use.
 | |
|  *
 | |
|  * This program is free software, distributed under the terms of
 | |
|  * the GNU General Public License Version 2. See the LICENSE file
 | |
|  * at the top of the source tree.
 | |
|  */
 | |
| 
 | |
| /*! \file
 | |
|  *
 | |
|  * \brief Mute Daemon
 | |
|  *
 | |
|  * \author Mark Spencer <markster@digium.com>
 | |
|  *
 | |
|  * Updated for Mac OSX CoreAudio
 | |
|  * \arg Josh Roberson <josh@asteriasgi.com>
 | |
|  *
 | |
|  * \note Specially written for Malcolm Davenport, but I think I'll use it too
 | |
|  * Connects to the Asterisk Manager Interface, AMI, and listens for events
 | |
|  * on certain devices. If a phone call is connected to one of the devices (phones)
 | |
|  * the local sound is muted to a lower volume during the call.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| /*! \li \ref muted.c uses the configuration file \ref muted.conf
 | |
|  * \addtogroup configuration_file Configuration Files
 | |
|  */
 | |
| 
 | |
| /*!
 | |
|  * \page muted.conf muted.conf
 | |
|  * \verbinclude muted.conf.sample
 | |
|  */
 | |
| 
 | |
| /*** MODULEINFO
 | |
| 	<support_level>deprecated</support_level>
 | |
|  ***/
 | |
| 
 | |
| #include "asterisk/autoconfig.h"
 | |
| 
 | |
| #ifdef __Darwin__
 | |
| #include <CoreAudio/AudioHardware.h>
 | |
| #include <sys/types.h>
 | |
| #include <pwd.h>
 | |
| #include <sys/stat.h>
 | |
| #elif defined(__linux__) || defined(__FreeBSD__) || defined(__DragonFly__) || defined(__GLIBC__)
 | |
| #include <sys/soundcard.h>
 | |
| #endif
 | |
| #include <stdio.h>
 | |
| #include <errno.h>
 | |
| #include <stdlib.h>
 | |
| #include <unistd.h>
 | |
| #include <fcntl.h>
 | |
| #include <string.h>
 | |
| #include <netdb.h>
 | |
| #include <sys/socket.h>
 | |
| #include <sys/ioctl.h>
 | |
| #include <netinet/in.h>
 | |
| #include <arpa/inet.h>
 | |
| 
 | |
| #define ast_strlen_zero(a)	(!(*(a)))
 | |
| 
 | |
| static char *config = "/etc/asterisk/muted.conf";
 | |
| 
 | |
| static char host[256] = "";
 | |
| static char user[256] = "";
 | |
| static char pass[256] = "";
 | |
| static int smoothfade = 0;
 | |
| static int mutelevel = 20;
 | |
| static int muted = 0;
 | |
| static int needfork = 1;
 | |
| static int debug = 0;
 | |
| static int stepsize = 3;
 | |
| #ifndef __Darwin__
 | |
| static int mixchan = SOUND_MIXER_VOLUME;
 | |
| #endif
 | |
| 
 | |
| struct subchannel {
 | |
| 	char *name;
 | |
| 	struct subchannel *next;
 | |
| };
 | |
| 
 | |
| static struct channel {
 | |
| 	char *tech;
 | |
| 	char *location;
 | |
| 	struct channel *next;
 | |
| 	struct subchannel *subs;
 | |
| } *channels;
 | |
| 
 | |
| static void add_channel(char *tech, char *location)
 | |
| {
 | |
| 	struct channel *chan;
 | |
| 	chan = malloc(sizeof(struct channel));
 | |
| 	if (chan) {
 | |
| 		memset(chan, 0, sizeof(struct channel));
 | |
| 		if (!(chan->tech = strdup(tech))) {
 | |
| 			free(chan);
 | |
| 			return;
 | |
| 		}
 | |
| 		if (!(chan->location = strdup(location))) {
 | |
| 			free(chan->tech);
 | |
| 			free(chan);
 | |
| 			return;
 | |
| 		}
 | |
| 		chan->next = channels;
 | |
| 		channels = chan;
 | |
| 	}
 | |
| 
 | |
| }
 | |
| 
 | |
| static int load_config(void)
 | |
| {
 | |
| 	FILE *f;
 | |
| 	char buf[256];
 | |
| 	char *val;
 | |
| 	char *val2;
 | |
| 	int lineno=0;
 | |
| 	int x;
 | |
| 	f = fopen(config, "r");
 | |
| 	if (!f) {
 | |
| 		fprintf(stderr, "Unable to open config file '%s': %s\n", config, strerror(errno));
 | |
| 		return -1;
 | |
| 	}
 | |
| 	while(!feof(f)) {
 | |
| 		if (!fgets(buf, sizeof(buf), f)) {
 | |
| 			continue;
 | |
| 		}
 | |
| 		if (!feof(f)) {
 | |
| 			lineno++;
 | |
| 			val = strchr(buf, '#');
 | |
| 			if (val) *val = '\0';
 | |
| 			while(strlen(buf) && (buf[strlen(buf) - 1] < 33))
 | |
| 				buf[strlen(buf) - 1] = '\0';
 | |
| 			if (!strlen(buf))
 | |
| 				continue;
 | |
| 			val = buf;
 | |
| 			while(*val) {
 | |
| 				if (*val < 33)
 | |
| 					break;
 | |
| 				val++;
 | |
| 			}
 | |
| 			if (*val) {
 | |
| 				*val = '\0';
 | |
| 				val++;
 | |
| 				while(*val && (*val < 33)) val++;
 | |
| 			}
 | |
| 			if (!strcasecmp(buf, "host")) {
 | |
| 				if (val && strlen(val))
 | |
| 					strncpy(host, val, sizeof(host) - 1);
 | |
| 				else
 | |
| 					fprintf(stderr, "host needs an argument (the host) at line %d\n", lineno);
 | |
| 			} else if (!strcasecmp(buf, "user")) {
 | |
| 				if (val && strlen(val))
 | |
| 					snprintf(user, sizeof(user), "%s", val);
 | |
| 				else
 | |
| 					fprintf(stderr, "user needs an argument (the user) at line %d\n", lineno);
 | |
| 			} else if (!strcasecmp(buf, "pass")) {
 | |
| 				if (val && strlen(val))
 | |
| 					snprintf(pass, sizeof(pass), "%s", val);
 | |
| 				else
 | |
| 					fprintf(stderr, "pass needs an argument (the password) at line %d\n", lineno);
 | |
| 			} else if (!strcasecmp(buf, "smoothfade")) {
 | |
| 				smoothfade = 1;
 | |
| 			} else if (!strcasecmp(buf, "mutelevel")) {
 | |
| 				if (val && (sscanf(val, "%3d", &x) == 1) && (x > -1) && (x < 101)) {
 | |
| 					mutelevel = x;
 | |
| 				} else
 | |
| 					fprintf(stderr, "mutelevel must be a number from 0 (most muted) to 100 (no mute) at line %d\n", lineno);
 | |
| 			} else if (!strcasecmp(buf, "channel")) {
 | |
| 				if (val && strlen(val)) {
 | |
| 					val2 = strchr(val, '/');
 | |
| 					if (val2) {
 | |
| 						*val2 = '\0';
 | |
| 						val2++;
 | |
| 						add_channel(val, val2);
 | |
| 					} else
 | |
| 						fprintf(stderr, "channel needs to be of the format Tech/Location at line %d\n", lineno);
 | |
| 				} else
 | |
| 					fprintf(stderr, "channel needs an argument (the channel) at line %d\n", lineno);
 | |
| 			} else {
 | |
| 				fprintf(stderr, "ignoring unknown keyword '%s'\n", buf);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	fclose(f);
 | |
| 	if (!strlen(host))
 | |
| 		fprintf(stderr, "no 'host' specification in config file\n");
 | |
| 	else if (!strlen(user))
 | |
| 		fprintf(stderr, "no 'user' specification in config file\n");
 | |
| 	else if (!channels)
 | |
| 		fprintf(stderr, "no 'channel' specifications in config file\n");
 | |
| 	else
 | |
| 		return 0;
 | |
| 	return -1;
 | |
| }
 | |
| 
 | |
| static FILE *astf;
 | |
| #ifndef __Darwin__
 | |
| static int mixfd;
 | |
| 
 | |
| static int open_mixer(void)
 | |
| {
 | |
| 	mixfd = open("/dev/mixer", O_RDWR);
 | |
| 	if (mixfd < 0) {
 | |
| 		fprintf(stderr, "Unable to open /dev/mixer: %s\n", strerror(errno));
 | |
| 		return -1;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| #endif /* !__Darwin */
 | |
| 
 | |
| /*! Connect to the asterisk manager interface */
 | |
| static int connect_asterisk(void)
 | |
| {
 | |
| 	int sock;
 | |
| 	struct hostent *hp;
 | |
| 	char *ports;
 | |
| 	int port = 5038;
 | |
| 	struct sockaddr_in sin;
 | |
| 
 | |
| 	ports = strchr(host, ':');
 | |
| 	if (ports) {
 | |
| 		*ports = '\0';
 | |
| 		ports++;
 | |
| 		if ((sscanf(ports, "%5d", &port) != 1) || (port < 1) || (port > 65535)) {
 | |
| 			fprintf(stderr, "'%s' is not a valid port number in the hostname\n", ports);
 | |
| 			return -1;
 | |
| 		}
 | |
| 	}
 | |
| 	hp = gethostbyname(host);
 | |
| 	if (!hp) {
 | |
| 		fprintf(stderr, "Can't find host '%s'\n", host);
 | |
| 		return -1;
 | |
| 	}
 | |
| 	sock = socket(AF_INET, SOCK_STREAM, 0);
 | |
| 	if (sock < 0) {
 | |
| 		fprintf(stderr, "Failed to create socket: %s\n", strerror(errno));
 | |
| 		return -1;
 | |
| 	}
 | |
| 	sin.sin_family = AF_INET;
 | |
| 	sin.sin_port = htons(port);
 | |
| 	memcpy(&sin.sin_addr, hp->h_addr, sizeof(sin.sin_addr));
 | |
| 	if (connect(sock, (struct sockaddr *)&sin, sizeof(sin))) {
 | |
| 		fprintf(stderr, "Failed to connect to '%s' port '%d': %s\n", host, port, strerror(errno));
 | |
| 		close(sock);
 | |
| 		return -1;
 | |
| 	}
 | |
| 	astf = fdopen(sock, "r+");
 | |
| 	if (!astf) {
 | |
| 		fprintf(stderr, "fdopen failed: %s\n", strerror(errno));
 | |
| 		close(sock);
 | |
| 		return -1;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static char *get_line(void)
 | |
| {
 | |
| 	static char buf[1024];
 | |
| 	if (fgets(buf, sizeof(buf), astf)) {
 | |
| 		while(strlen(buf) && (buf[strlen(buf) - 1] < 33))
 | |
| 			buf[strlen(buf) - 1] = '\0';
 | |
| 		return buf;
 | |
| 	} else
 | |
| 		return NULL;
 | |
| }
 | |
| 
 | |
| /*! Login to the asterisk manager interface */
 | |
| static int login_asterisk(void)
 | |
| {
 | |
| 	char *welcome;
 | |
| 	char *resp;
 | |
| 	if (!(welcome = get_line())) {
 | |
| 		fprintf(stderr, "disconnected (1)\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 	fprintf(astf,
 | |
| 		"Action: Login\r\n"
 | |
| 		"Username: %s\r\n"
 | |
| 		"Secret: %s\r\n\r\n", user, pass);
 | |
| 	if (!(welcome = get_line())) {
 | |
| 		fprintf(stderr, "disconnected (2)\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 	if (strcasecmp(welcome, "Response: Success")) {
 | |
| 		fprintf(stderr, "login failed ('%s')\n", welcome);
 | |
| 		return -1;
 | |
| 	}
 | |
| 	/* Eat the rest of the event */
 | |
| 	while((resp = get_line()) && strlen(resp));
 | |
| 	if (!resp) {
 | |
| 		fprintf(stderr, "disconnected (3)\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 	fprintf(astf,
 | |
| 		"Action: Status\r\n\r\n");
 | |
| 	if (!(welcome = get_line())) {
 | |
| 		fprintf(stderr, "disconnected (4)\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 	if (strcasecmp(welcome, "Response: Success")) {
 | |
| 		fprintf(stderr, "status failed ('%s')\n", welcome);
 | |
| 		return -1;
 | |
| 	}
 | |
| 	/* Eat the rest of the event */
 | |
| 	while((resp = get_line()) && strlen(resp));
 | |
| 	if (!resp) {
 | |
| 		fprintf(stderr, "disconnected (5)\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static struct channel *find_channel(char *channel)
 | |
| {
 | |
| 	char tmp[256] = "";
 | |
| 	char *s, *t;
 | |
| 	struct channel *chan;
 | |
| 	strncpy(tmp, channel, sizeof(tmp) - 1);
 | |
| 	s = strchr(tmp, '/');
 | |
| 	if (s) {
 | |
| 		*s = '\0';
 | |
| 		s++;
 | |
| 		t = strrchr(s, '-');
 | |
| 		if (t) {
 | |
| 			*t = '\0';
 | |
| 		}
 | |
| 		if (debug)
 | |
| 			printf("Searching for '%s' tech, '%s' location\n", tmp, s);
 | |
| 		chan = channels;
 | |
| 		while(chan) {
 | |
| 			if (!strcasecmp(chan->tech, tmp) && !strcasecmp(chan->location, s)) {
 | |
| 				if (debug)
 | |
| 					printf("Found '%s'/'%s'\n", chan->tech, chan->location);
 | |
| 				break;
 | |
| 			}
 | |
| 			chan = chan->next;
 | |
| 		}
 | |
| 	} else
 | |
| 		chan = NULL;
 | |
| 	return chan;
 | |
| }
 | |
| 
 | |
| #ifndef __Darwin__
 | |
| static int getvol(void)
 | |
| {
 | |
| 	int vol;
 | |
| 
 | |
| 	if (ioctl(mixfd, MIXER_READ(mixchan), &vol)) {
 | |
| #else
 | |
| static float getvol(void)
 | |
| {
 | |
| 	float volumeL, volumeR, vol;
 | |
| 	OSStatus err;
 | |
| 	AudioDeviceID device;
 | |
| 	UInt32 size;
 | |
| 	UInt32 channels[2];
 | |
| 	AudioObjectPropertyAddress OutputAddr = { kAudioHardwarePropertyDefaultOutputDevice, kAudioObjectPropertyScopeGlobal, kAudioObjectPropertyElementMaster };
 | |
| 	AudioObjectPropertyAddress ChannelAddr = { kAudioDevicePropertyPreferredChannelsForStereo, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementWildcard };
 | |
| 	AudioObjectPropertyAddress VolumeAddr = { kAudioDevicePropertyVolumeScalar, kAudioDevicePropertyScopeOutput, };
 | |
| 
 | |
| 	size = sizeof(device);
 | |
| 	err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &OutputAddr, 0, NULL, &size, &device);
 | |
| 	size = sizeof(channels);
 | |
| 	if (!err) {
 | |
| 		err = AudioObjectGetPropertyData(device, &ChannelAddr, 0, NULL, &size, &channels);
 | |
| 	}
 | |
| 	size = sizeof(vol);
 | |
| 	if (!err) {
 | |
| 		VolumeAddr.mElement = channels[0];
 | |
| 		err = AudioObjectGetPropertyData(device, &VolumeAddr, 0, NULL, &size, &volumeL);
 | |
| 	}
 | |
| 	if (!err) {
 | |
| 		VolumeAddr.mElement = channels[1];
 | |
| 		err = AudioObjectGetPropertyData(device, &VolumeAddr, 0, NULL, &size, &volumeR);
 | |
| 	}
 | |
| 	if (!err)
 | |
| 		vol = (volumeL < volumeR) ? volumeR : volumeL;
 | |
| 	else {
 | |
| #endif
 | |
| 		fprintf(stderr, "Unable to read mixer volume: %s\n", strerror(errno));
 | |
| 		return -1;
 | |
| 	}
 | |
| 	return vol;
 | |
| }
 | |
| 
 | |
| #ifndef __Darwin__
 | |
| static int setvol(int vol)
 | |
| #else
 | |
| static int setvol(float vol)
 | |
| #endif
 | |
| {
 | |
| #ifndef __Darwin__
 | |
| 	if (ioctl(mixfd, MIXER_WRITE(mixchan), &vol)) {
 | |
| #else
 | |
| 	float volumeL = vol;
 | |
| 	float volumeR = vol;
 | |
| 	OSStatus err;
 | |
| 	AudioDeviceID device;
 | |
| 	UInt32 size;
 | |
| 	UInt32 channels[2];
 | |
| 	AudioObjectPropertyAddress OutputAddr = { kAudioHardwarePropertyDefaultOutputDevice, kAudioObjectPropertyScopeGlobal, kAudioObjectPropertyElementMaster };
 | |
| 	AudioObjectPropertyAddress ChannelAddr = { kAudioDevicePropertyPreferredChannelsForStereo, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementWildcard };
 | |
| 	AudioObjectPropertyAddress VolumeAddr = { kAudioDevicePropertyVolumeScalar, kAudioDevicePropertyScopeOutput, };
 | |
| 
 | |
| 	size = sizeof(device);
 | |
| 	err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &OutputAddr, 0, NULL, &size, &device);
 | |
| 	size = sizeof(channels);
 | |
| 	err = AudioObjectGetPropertyData(device, &ChannelAddr, 0, NULL, &size, &channels);
 | |
| 	size = sizeof(vol);
 | |
| 	if (!err) {
 | |
| 		VolumeAddr.mElement = channels[0];
 | |
| 		err = AudioObjectSetPropertyData(device, &VolumeAddr, 0, NULL, size, &volumeL);
 | |
| 	}
 | |
| 	if (!err) {
 | |
| 		VolumeAddr.mElement = channels[1];
 | |
| 		err = AudioObjectSetPropertyData(device, &VolumeAddr, 0, NULL, size, &volumeR);
 | |
| 	}
 | |
| 	if (err) {
 | |
| #endif
 | |
| 
 | |
| 		fprintf(stderr, "Unable to write mixer volume: %s\n", strerror(errno));
 | |
| 		return -1;
 | |
| 
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| #ifndef __Darwin__
 | |
| static int oldvol = 0;
 | |
| static int mutevol = 0;
 | |
| #else
 | |
| static float oldvol = 0;
 | |
| static float mutevol = 0;
 | |
| #endif
 | |
| 
 | |
| #ifndef __Darwin__
 | |
| static int mutedlevel(int orig, int level)
 | |
| {
 | |
| 	int l = orig >> 8;
 | |
| 	int r = orig & 0xff;
 | |
| 	l = (float)(level) * (float)(l) / 100.0;
 | |
| 	r = (float)(level) * (float)(r) / 100.0;
 | |
| 
 | |
| 	return (l << 8) | r;
 | |
| #else
 | |
| static float mutedlevel(float orig, float level)
 | |
| {
 | |
| 	float master = orig;
 | |
| 	master = level * master / 100.0;
 | |
| 	return master;
 | |
| #endif
 | |
| 
 | |
| }
 | |
| 
 | |
| static void mute(void)
 | |
| {
 | |
| #ifndef __Darwin__
 | |
| 	int vol;
 | |
| 	int start;
 | |
| 	int x;
 | |
| #else
 | |
| 	float vol;
 | |
| 	float start = 1.0;
 | |
| 	float x;
 | |
| #endif
 | |
| 	vol = getvol();
 | |
| 	oldvol = vol;
 | |
| 	if (smoothfade)
 | |
| #ifdef __Darwin__
 | |
| 		start = mutelevel;
 | |
| #else
 | |
| 		start = 100;
 | |
| 	else
 | |
| 		start = mutelevel;
 | |
| #endif
 | |
| 	for (x=start;x>=mutelevel;x-=stepsize) {
 | |
| 		mutevol = mutedlevel(vol, x);
 | |
| 		setvol(mutevol);
 | |
| 		/* Wait 0.01 sec */
 | |
| 		usleep(10000);
 | |
| 	}
 | |
| 	mutevol = mutedlevel(vol, mutelevel);
 | |
| 	setvol(mutevol);
 | |
| 	if (debug)
 | |
| #ifdef __Darwin__
 | |
| 		printf("Mute from '%f' to '%f'!\n", oldvol, mutevol);
 | |
| #else
 | |
| 		printf("Mute from '%04x' to '%04x'!\n", oldvol, mutevol);
 | |
| #endif
 | |
| 	muted = 1;
 | |
| }
 | |
| 
 | |
| static void unmute(void)
 | |
| {
 | |
| #ifdef __Darwin__
 | |
| 	float vol;
 | |
| 	float start;
 | |
| 	float x;
 | |
| #else
 | |
| 	int vol;
 | |
| 	int start;
 | |
| 	int x;
 | |
| #endif
 | |
| 	vol = getvol();
 | |
| 	if (debug)
 | |
| #ifdef __Darwin__
 | |
| 		printf("Unmute from '%f' (should be '%f') to '%f'!\n", vol, mutevol, oldvol);
 | |
| 	mutevol = vol;
 | |
| 	if (vol == mutevol) {
 | |
| #else
 | |
| 		printf("Unmute from '%04x' (should be '%04x') to '%04x'!\n", vol, mutevol, oldvol);
 | |
| 	if ((int)vol == mutevol) {
 | |
| #endif
 | |
| 		if (smoothfade)
 | |
| 			start = mutelevel;
 | |
| 		else
 | |
| #ifdef __Darwin__
 | |
| 			start = 1.0;
 | |
| #else
 | |
| 			start = 100;
 | |
| #endif
 | |
| 		for (x=start;x<100;x+=stepsize) {
 | |
| 			mutevol = mutedlevel(oldvol, x);
 | |
| 			setvol(mutevol);
 | |
| 			/* Wait 0.01 sec */
 | |
| 			usleep(10000);
 | |
| 		}
 | |
| 		setvol(oldvol);
 | |
| 	} else
 | |
| 		printf("Whoops, it's already been changed!\n");
 | |
| 	muted = 0;
 | |
| }
 | |
| 
 | |
| static void check_mute(void)
 | |
| {
 | |
| 	int offhook = 0;
 | |
| 	struct channel *chan;
 | |
| 	chan = channels;
 | |
| 	while(chan) {
 | |
| 		if (chan->subs) {
 | |
| 			offhook++;
 | |
| 			break;
 | |
| 		}
 | |
| 		chan = chan->next;
 | |
| 	}
 | |
| 	if (offhook && !muted)
 | |
| 		mute();
 | |
| 	else if (!offhook && muted)
 | |
| 		unmute();
 | |
| }
 | |
| 
 | |
| static void delete_sub(struct channel *chan, char *name)
 | |
| {
 | |
| 	struct subchannel *sub, *prev;
 | |
| 	prev = NULL;
 | |
| 	sub = chan->subs;
 | |
| 	while(sub) {
 | |
| 		if (!strcasecmp(sub->name, name)) {
 | |
| 			if (prev)
 | |
| 				prev->next = sub->next;
 | |
| 			else
 | |
| 				chan->subs = sub->next;
 | |
| 			free(sub->name);
 | |
| 			free(sub);
 | |
| 			return;
 | |
| 		}
 | |
| 		prev = sub;
 | |
| 		sub = sub->next;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void append_sub(struct channel *chan, char *name)
 | |
| {
 | |
| 	struct subchannel *sub;
 | |
| 	sub = chan->subs;
 | |
| 	while(sub) {
 | |
| 		if (!strcasecmp(sub->name, name))
 | |
| 			return;
 | |
| 		sub = sub->next;
 | |
| 	}
 | |
| 	sub = malloc(sizeof(struct subchannel));
 | |
| 	if (sub) {
 | |
| 		memset(sub, 0, sizeof(struct subchannel));
 | |
| 		if (!(sub->name = strdup(name))) {
 | |
| 			free(sub);
 | |
| 			return;
 | |
| 		}
 | |
| 		sub->next = chan->subs;
 | |
| 		chan->subs = sub;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void hangup_chan(char *channel)
 | |
| {
 | |
| 	struct channel *chan;
 | |
| 	if (debug)
 | |
| 		printf("Hangup '%s'\n", channel);
 | |
| 	chan = find_channel(channel);
 | |
| 	if (chan)
 | |
| 		delete_sub(chan, channel);
 | |
| 	check_mute();
 | |
| }
 | |
| 
 | |
| static void offhook_chan(char *channel)
 | |
| {
 | |
| 	struct channel *chan;
 | |
| 	if (debug)
 | |
| 		printf("Offhook '%s'\n", channel);
 | |
| 	chan = find_channel(channel);
 | |
| 	if (chan)
 | |
| 		append_sub(chan, channel);
 | |
| 	check_mute();
 | |
| }
 | |
| 
 | |
| static int wait_event(void)
 | |
| {
 | |
| 	char *resp;
 | |
| 	char event[120]="";
 | |
| 	char channel[120]="";
 | |
| 	char oldname[120]="";
 | |
| 	char newname[120]="";
 | |
| 
 | |
| 	resp = get_line();
 | |
| 	if (!resp) {
 | |
| 		fprintf(stderr, "disconnected (6)\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 	if (!strncasecmp(resp, "Event: ", strlen("Event: "))) {
 | |
| 		int event_len = -1;
 | |
| 		int channel_len = -1;
 | |
| 		int newname_len = -1;
 | |
| 		int oldname_len = -1;
 | |
| 
 | |
| 		event_len = snprintf(event, sizeof(event), "%s", resp + strlen("Event: "));
 | |
| 		/* Consume the rest of the non-event */
 | |
| 		while((resp = get_line()) && strlen(resp)) {
 | |
| 			if (!strncasecmp(resp, "Channel: ", strlen("Channel: ")))
 | |
| 				channel_len = snprintf(channel, sizeof(channel), "%s", resp + strlen("Channel: "));
 | |
| 			if (!strncasecmp(resp, "Newname: ", strlen("Newname: ")))
 | |
| 				newname_len = snprintf(newname, sizeof(newname), "%s", resp + strlen("Newname: "));
 | |
| 			if (!strncasecmp(resp, "Oldname: ", strlen("Oldname: ")))
 | |
| 				oldname_len = snprintf(oldname, sizeof(oldname), "%s", resp + strlen("Oldname: "));
 | |
| 		}
 | |
| 		if (channel_len == strlen(channel)) {
 | |
| 			if (event_len == strlen(event) && !strcasecmp(event, "Hangup"))
 | |
| 				hangup_chan(channel);
 | |
| 			else
 | |
| 				offhook_chan(channel);
 | |
| 		}
 | |
| 		if (newname_len == strlen(newname) && oldname_len == strlen(oldname)) {
 | |
| 			if (event_len == strlen(event) && !strcasecmp(event, "Rename")) {
 | |
| 				hangup_chan(oldname);
 | |
| 				offhook_chan(newname);
 | |
| 			}
 | |
| 		}
 | |
| 	} else {
 | |
| 		/* Consume the rest of the non-event */
 | |
| 		while((resp = get_line()) && strlen(resp));
 | |
| 	}
 | |
| 	if (!resp) {
 | |
| 		fprintf(stderr, "disconnected (7)\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void usage(void)
 | |
| {
 | |
| 	printf("Usage: muted [-f] [-d]\n"
 | |
| 	       "        -f : Do not fork\n"
 | |
| 	       "        -d : Debug (implies -f)\n");
 | |
| }
 | |
| 
 | |
| int main(int argc, char *argv[])
 | |
| {
 | |
| 	int x;
 | |
| 	while((x = getopt(argc, argv, "fhd")) > 0) {
 | |
| 		switch(x) {
 | |
| 		case 'd':
 | |
| 			debug = 1;
 | |
| 			needfork = 0;
 | |
| 			break;
 | |
| 		case 'f':
 | |
| 			needfork = 0;
 | |
| 			break;
 | |
| 		case 'h':
 | |
| 			/* Fall through */
 | |
| 		default:
 | |
| 			usage();
 | |
| 			exit(1);
 | |
| 		}
 | |
| 	}
 | |
| 	if (load_config())
 | |
| 		exit(1);
 | |
| #ifndef __Darwin__
 | |
| 	if (open_mixer())
 | |
| 		exit(1);
 | |
| #endif
 | |
| 	if (connect_asterisk()) {
 | |
| #ifndef __Darwin__
 | |
| 		close(mixfd);
 | |
| #endif
 | |
| 		exit(1);
 | |
| 	}
 | |
| 	if (login_asterisk()) {
 | |
| #ifndef __Darwin__
 | |
| 		close(mixfd);
 | |
| #endif
 | |
| 		fclose(astf);
 | |
| 		exit(1);
 | |
| 	}
 | |
| #ifdef HAVE_WORKING_FORK
 | |
| 	if (needfork) {
 | |
| #ifndef HAVE_SBIN_LAUNCHD
 | |
| 		if (daemon(0,0) < 0) {
 | |
| 			fprintf(stderr, "daemon() failed: %s\n", strerror(errno));
 | |
| 			exit(1);
 | |
| 		}
 | |
| #else
 | |
| 		const char *found = NULL, *paths[] = {
 | |
| 			"/Library/LaunchAgents/org.asterisk.muted.plist",
 | |
| 			"/Library/LaunchDaemons/org.asterisk.muted.plist",
 | |
| 			"contrib/init.d/org.asterisk.muted.plist",
 | |
| 			"<path-to-asterisk-source>/contrib/init.d/org.asterisk.muted.plist" };
 | |
| 		char userpath[256];
 | |
| 		struct stat unused;
 | |
| 		struct passwd *pwd = getpwuid(getuid());
 | |
| 		int i;
 | |
| 
 | |
| 		snprintf(userpath, sizeof(userpath), "%s%s", pwd->pw_dir, paths[0]);
 | |
| 		if (!stat(userpath, &unused)) {
 | |
| 			found = userpath;
 | |
| 		}
 | |
| 
 | |
| 		if (!found) {
 | |
| 			for (i = 0; i < 3; i++) {
 | |
| 				if (!stat(paths[i], &unused)) {
 | |
| 					found = paths[i];
 | |
| 					break;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		fprintf(stderr, "Mac OS X detected.  Use 'launchctl load -w %s' to launch.\n", found ? found : paths[3]);
 | |
| 		exit(1);
 | |
| #endif /* !defined(HAVE_SBIN_LAUNCHD */
 | |
| 	}
 | |
| #endif
 | |
| 	for(;;) {
 | |
| 		if (wait_event()) {
 | |
| 			fclose(astf);
 | |
| 			while(connect_asterisk()) {
 | |
| 				sleep(5);
 | |
| 			}
 | |
| 			if (login_asterisk()) {
 | |
| 				fclose(astf);
 | |
| 				exit(1);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	exit(0);
 | |
| }
 |