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