Enhancements for zaptel+bsd (bug #1781)
[asterisk/asterisk.git] / apps / app_flash.c
1 /*
2  * Asterisk -- A telephony toolkit for Linux.
3  *
4  * App to flash a zap trunk
5  * 
6  * Copyright (C) 1999, Mark Spencer
7  *
8  * Mark Spencer <markster@linux-support.net>
9  *
10  * This program is free software, distributed under the terms of
11  * the GNU General Public License
12  */
13  
14 #include <asterisk/lock.h>
15 #include <asterisk/file.h>
16 #include <asterisk/logger.h>
17 #include <asterisk/channel.h>
18 #include <asterisk/pbx.h>
19 #include <asterisk/module.h>
20 #include <asterisk/translate.h>
21 #include <asterisk/image.h>
22 #include <asterisk/options.h>
23 #include <sys/ioctl.h>
24 #ifdef __linux__
25 #include <linux/zaptel.h>
26 #else
27 #include <zaptel.h>
28 #endif /* __linux__ */
29 #include <string.h>
30 #include <errno.h>
31 #include <stdlib.h>
32 #include <pthread.h>
33
34 static char *tdesc = "Flash zap trunk application";
35
36 static char *app = "Flash";
37
38 static char *synopsis = "Flashes a Zap Trunk";
39
40 static char *descrip = 
41 "  Flash(): Sends a flash on a zap trunk.  This is only a hack for\n"
42 "people who want to perform transfers and such via AGI and is generally\n"
43 "quite useless otherwise.  Returns 0 on success or -1 if this is not\n"
44 "a zap trunk\n";
45
46 STANDARD_LOCAL_USER;
47
48 LOCAL_USER_DECL;
49
50 static inline int zt_wait_event(int fd)
51 {
52         /* Avoid the silly zt_waitevent which ignores a bunch of events */
53         int i,j=0;
54         i = ZT_IOMUX_SIGEVENT;
55         if (ioctl(fd, ZT_IOMUX, &i) == -1) return -1;
56         if (ioctl(fd, ZT_GETEVENT, &j) == -1) return -1;
57         return j;
58 }
59
60 static int flash_exec(struct ast_channel *chan, void *data)
61 {
62         int res = -1;
63         int x;
64         struct localuser *u;
65         struct zt_params ztp;
66         LOCAL_USER_ADD(u);
67         if (!strcasecmp(chan->type, "Zap")) {
68                 memset(&ztp, 0, sizeof(ztp));
69                 res = ioctl(chan->fds[0], ZT_GET_PARAMS, &ztp);
70                 if (!res) {
71                         if (ztp.sigtype & __ZT_SIG_FXS) {
72                                 x = ZT_FLASH;
73                                 res = ioctl(chan->fds[0], ZT_HOOK, &x);
74                                 if (!res || (errno == EINPROGRESS)) {
75                                         if (res) {
76                                                 /* Wait for the event to finish */
77                                                 zt_wait_event(chan->fds[0]);
78                                         }
79                                         res = ast_safe_sleep(chan, 1000);
80                                         if (option_verbose > 2)
81                                                 ast_verbose(VERBOSE_PREFIX_3 "Flashed channel %s\n", chan->name);
82                                 } else
83                                         ast_log(LOG_WARNING, "Unable to flash channel %s: %s\n", chan->name, strerror(errno));
84                         } else
85                                 ast_log(LOG_WARNING, "%s is not an FXO Channel\n", chan->name);
86                 } else
87                         ast_log(LOG_WARNING, "Unable to get parameters of %s: %s\n", chan->name, strerror(errno));
88         } else
89                 ast_log(LOG_WARNING, "%s is not a Zap channel\n", chan->name);
90         LOCAL_USER_REMOVE(u);
91         return res;
92 }
93
94 int unload_module(void)
95 {
96         STANDARD_HANGUP_LOCALUSERS;
97         return ast_unregister_application(app);
98 }
99
100 int load_module(void)
101 {
102         return ast_register_application(app, flash_exec, synopsis, descrip);
103 }
104
105 char *description(void)
106 {
107         return tdesc;
108 }
109
110 int usecount(void)
111 {
112         int res;
113         STANDARD_USECOUNT(res);
114         return res;
115 }
116
117 char *key()
118 {
119         return ASTERISK_GPL_KEY;
120 }