00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027 #include "asterisk.h"
00028
00029 ASTERISK_FILE_VERSION(__FILE__, "$Revision: 243485 $")
00030
00031 #include <regex.h>
00032
00033 #include "asterisk/module.h"
00034 #include "asterisk/channel.h"
00035 #include "asterisk/pbx.h"
00036 #include "asterisk/utils.h"
00037 #include "asterisk/app.h"
00038 #include "asterisk/indications.h"
00039 #include "asterisk/stringfields.h"
00040
00041
00042
00043
00044
00045
00046
00047
00048
00049
00050
00051
00052
00053
00054
00055
00056
00057
00058
00059
00060
00061
00062
00063
00064
00065
00066
00067
00068
00069
00070
00071
00072
00073
00074
00075
00076
00077
00078
00079
00080
00081
00082
00083
00084
00085
00086
00087
00088
00089
00090
00091
00092
00093
00094
00095
00096
00097
00098
00099
00100
00101
00102
00103
00104
00105
00106
00107
00108
00109
00110
00111
00112
00113
00114
00115
00116
00117
00118
00119
00120
00121
00122
00123
00124
00125
00126
00127
00128
00129
00130
00131
00132
00133
00134
00135
00136
00137
00138
00139
00140
00141
00142
00143
00144
00145
00146
00147
00148
00149
00150
00151
00152
00153
00154
00155
00156
00157
00158
00159
00160
00161
00162
00163
00164
00165
00166
00167
00168
00169
00170
00171
00172
00173
00174
00175
00176
00177
00178
00179
00180
00181
00182
00183
00184
00185
00186
00187
00188
00189
00190
00191
00192
00193
00194
00195
00196
00197
00198
00199
00200
00201
00202
00203
00204
00205
00206
00207
00208
00209
00210
00211 #define locked_copy_string(chan, dest, source, len) \
00212 do { \
00213 ast_channel_lock(chan); \
00214 ast_copy_string(dest, source, len); \
00215 ast_channel_unlock(chan); \
00216 } while (0)
00217 #define locked_string_field_set(chan, field, source) \
00218 do { \
00219 ast_channel_lock(chan); \
00220 ast_string_field_set(chan, field, source); \
00221 ast_channel_unlock(chan); \
00222 } while (0)
00223
00224 char *transfercapability_table[0x20] = {
00225 "SPEECH", "UNK", "UNK", "UNK", "UNK", "UNK", "UNK", "UNK",
00226 "DIGITAL", "RESTRICTED_DIGITAL", "UNK", "UNK", "UNK", "UNK", "UNK", "UNK",
00227 "3K1AUDIO", "DIGITAL_W_TONES", "UNK", "UNK", "UNK", "UNK", "UNK", "UNK",
00228 "VIDEO", "UNK", "UNK", "UNK", "UNK", "UNK", "UNK", "UNK", };
00229
00230 static int func_channel_read(struct ast_channel *chan, const char *function,
00231 char *data, char *buf, size_t len)
00232 {
00233 int ret = 0;
00234
00235 if (!strcasecmp(data, "audionativeformat"))
00236
00237
00238 ast_copy_string(buf, ast_getformatname(chan->nativeformats & AST_FORMAT_AUDIO_MASK), len);
00239 else if (!strcasecmp(data, "videonativeformat"))
00240
00241
00242 ast_copy_string(buf, ast_getformatname(chan->nativeformats & AST_FORMAT_VIDEO_MASK), len);
00243 else if (!strcasecmp(data, "audioreadformat"))
00244 ast_copy_string(buf, ast_getformatname(chan->readformat), len);
00245 else if (!strcasecmp(data, "audiowriteformat"))
00246 ast_copy_string(buf, ast_getformatname(chan->writeformat), len);
00247 #ifdef CHANNEL_TRACE
00248 else if (!strcasecmp(data, "trace")) {
00249 ast_channel_lock(chan);
00250 ast_copy_string(buf, ast_channel_trace_is_enabled(chan) ? "1" : "0", len);
00251 ast_channel_unlock(chan);
00252 }
00253 #endif
00254 else if (!strcasecmp(data, "tonezone") && chan->zone)
00255 locked_copy_string(chan, buf, chan->zone->country, len);
00256 else if (!strcasecmp(data, "language"))
00257 locked_copy_string(chan, buf, chan->language, len);
00258 else if (!strcasecmp(data, "musicclass"))
00259 locked_copy_string(chan, buf, chan->musicclass, len);
00260 else if (!strcasecmp(data, "parkinglot"))
00261 locked_copy_string(chan, buf, chan->parkinglot, len);
00262 else if (!strcasecmp(data, "state"))
00263 locked_copy_string(chan, buf, ast_state2str(chan->_state), len);
00264 else if (!strcasecmp(data, "channeltype"))
00265 locked_copy_string(chan, buf, chan->tech->type, len);
00266 else if (!strcasecmp(data, "transfercapability"))
00267 locked_copy_string(chan, buf, transfercapability_table[chan->transfercapability & 0x1f], len);
00268 else if (!strcasecmp(data, "callgroup")) {
00269 char groupbuf[256];
00270 locked_copy_string(chan, buf, ast_print_group(groupbuf, sizeof(groupbuf), chan->callgroup), len);
00271 } else if (!chan->tech->func_channel_read
00272 || chan->tech->func_channel_read(chan, function, data, buf, len)) {
00273 ast_log(LOG_WARNING, "Unknown or unavailable item requested: '%s'\n", data);
00274 ret = -1;
00275 }
00276
00277 return ret;
00278 }
00279
00280 static int func_channel_write(struct ast_channel *chan, const char *function,
00281 char *data, const char *value)
00282 {
00283 int ret = 0;
00284 signed char gainset;
00285
00286 if (!strcasecmp(data, "language"))
00287 locked_string_field_set(chan, language, value);
00288 else if (!strcasecmp(data, "parkinglot"))
00289 locked_string_field_set(chan, parkinglot, value);
00290 else if (!strcasecmp(data, "musicclass"))
00291 locked_string_field_set(chan, musicclass, value);
00292 #ifdef CHANNEL_TRACE
00293 else if (!strcasecmp(data, "trace")) {
00294 ast_channel_lock(chan);
00295 if (ast_true(value))
00296 ret = ast_channel_trace_enable(chan);
00297 else if (ast_false(value))
00298 ret = ast_channel_trace_disable(chan);
00299 else {
00300 ret = -1;
00301 ast_log(LOG_WARNING, "Invalid value for CHANNEL(trace).");
00302 }
00303 ast_channel_unlock(chan);
00304 }
00305 #endif
00306 else if (!strcasecmp(data, "tonezone")) {
00307 struct ast_tone_zone *new_zone;
00308 if (!(new_zone = ast_get_indication_zone(value))) {
00309 ast_log(LOG_ERROR, "Unknown country code '%s' for tonezone. Check indications.conf for available country codes.\n", value);
00310 ret = -1;
00311 } else {
00312 ast_channel_lock(chan);
00313 if (chan->zone) {
00314 chan->zone = ast_tone_zone_unref(chan->zone);
00315 }
00316 chan->zone = ast_tone_zone_ref(new_zone);
00317 ast_channel_unlock(chan);
00318 new_zone = ast_tone_zone_unref(new_zone);
00319 }
00320 } else if (!strcasecmp(data, "callgroup"))
00321 chan->callgroup = ast_get_group(value);
00322 else if (!strcasecmp(data, "txgain")) {
00323 sscanf(value, "%4hhd", &gainset);
00324 ast_channel_setoption(chan, AST_OPTION_TXGAIN, &gainset, sizeof(gainset), 0);
00325 } else if (!strcasecmp(data, "rxgain")) {
00326 sscanf(value, "%4hhd", &gainset);
00327 ast_channel_setoption(chan, AST_OPTION_RXGAIN, &gainset, sizeof(gainset), 0);
00328 } else if (!strcasecmp(data, "transfercapability")) {
00329 unsigned short i;
00330 for (i = 0; i < 0x20; i++) {
00331 if (!strcasecmp(transfercapability_table[i], value) && strcmp(value, "UNK")) {
00332 chan->transfercapability = i;
00333 break;
00334 }
00335 }
00336 } else if (!chan->tech->func_channel_write
00337 || chan->tech->func_channel_write(chan, function, data, value)) {
00338 ast_log(LOG_WARNING, "Unknown or unavailable item requested: '%s'\n",
00339 data);
00340 ret = -1;
00341 }
00342
00343 return ret;
00344 }
00345
00346 static struct ast_custom_function channel_function = {
00347 .name = "CHANNEL",
00348 .read = func_channel_read,
00349 .write = func_channel_write,
00350 };
00351
00352 static int func_channels_read(struct ast_channel *chan, const char *function, char *data, char *buf, size_t maxlen)
00353 {
00354 struct ast_channel *c = NULL;
00355 regex_t re;
00356 int res;
00357 size_t buflen = 0;
00358
00359 buf[0] = '\0';
00360
00361 if (!ast_strlen_zero(data)) {
00362 if ((res = regcomp(&re, data, REG_EXTENDED | REG_ICASE | REG_NOSUB))) {
00363 regerror(res, &re, buf, maxlen);
00364 ast_log(LOG_WARNING, "Error compiling regular expression for %s(%s): %s\n", function, data, buf);
00365 return -1;
00366 }
00367 }
00368
00369 for (c = ast_channel_walk_locked(NULL); c; ast_channel_unlock(c), c = ast_channel_walk_locked(c)) {
00370 if (ast_strlen_zero(data) || regexec(&re, c->name, 0, NULL, 0) == 0) {
00371 size_t namelen = strlen(c->name);
00372 if (buflen + namelen + (ast_strlen_zero(buf) ? 0 : 1) + 1 < maxlen) {
00373 if (!ast_strlen_zero(buf)) {
00374 strcat(buf, " ");
00375 buflen++;
00376 }
00377 strcat(buf, c->name);
00378 buflen += namelen;
00379 } else {
00380 ast_log(LOG_WARNING, "Number of channels exceeds the available buffer space. Output will be truncated!\n");
00381 }
00382 }
00383 }
00384
00385 if (!ast_strlen_zero(data)) {
00386 regfree(&re);
00387 }
00388
00389 return 0;
00390 }
00391
00392 static struct ast_custom_function channels_function = {
00393 .name = "CHANNELS",
00394 .read = func_channels_read,
00395 };
00396
00397 static int unload_module(void)
00398 {
00399 int res = 0;
00400
00401 res |= ast_custom_function_unregister(&channel_function);
00402 res |= ast_custom_function_unregister(&channels_function);
00403
00404 return res;
00405 }
00406
00407 static int load_module(void)
00408 {
00409 int res = 0;
00410
00411 res |= ast_custom_function_register(&channel_function);
00412 res |= ast_custom_function_register(&channels_function);
00413
00414 return res;
00415 }
00416
00417 AST_MODULE_INFO_STANDARD(ASTERISK_GPL_KEY, "Channel information dialplan functions");