term.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. /*
  2. * Asterisk -- An open source telephony toolkit.
  3. *
  4. * Copyright (C) 1999 - 2005, Digium, Inc.
  5. *
  6. * Mark Spencer <markster@digium.com>
  7. *
  8. * See http://www.asterisk.org for more information about
  9. * the Asterisk project. Please do not directly contact
  10. * any of the maintainers of this project for assistance;
  11. * the project provides a web site, mailing lists and IRC
  12. * channels for your use.
  13. *
  14. * This program is free software, distributed under the terms of
  15. * the GNU General Public License Version 2. See the LICENSE file
  16. * at the top of the source tree.
  17. */
  18. /*! \file
  19. *
  20. * \brief Terminal Routines
  21. *
  22. */
  23. #include <stdio.h>
  24. #include <stdlib.h>
  25. #include <string.h>
  26. #include <sys/time.h>
  27. #include <signal.h>
  28. #include <errno.h>
  29. #include <sys/types.h>
  30. #include <sys/stat.h>
  31. #include <fcntl.h>
  32. #include <unistd.h>
  33. #include "asterisk.h"
  34. ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
  35. #include "asterisk/term.h"
  36. #include "asterisk/options.h"
  37. #include "asterisk/lock.h"
  38. #include "asterisk/utils.h"
  39. static int vt100compat = 0;
  40. static char prepdata[80] = "";
  41. static char enddata[80] = "";
  42. static char quitdata[80] = "";
  43. static const char *termpath[] = {
  44. "/usr/share/terminfo",
  45. "/usr/local/share/misc/terminfo",
  46. "/usr/lib/terminfo",
  47. NULL
  48. };
  49. /* Ripped off from Ross Ridge, but it's public domain code (libmytinfo) */
  50. static short convshort(char *s)
  51. {
  52. register int a,b;
  53. a = (int) s[0] & 0377;
  54. b = (int) s[1] & 0377;
  55. if (a == 0377 && b == 0377)
  56. return -1;
  57. if (a == 0376 && b == 0377)
  58. return -2;
  59. return a + b * 256;
  60. }
  61. int term_init(void)
  62. {
  63. char *term = getenv("TERM");
  64. char termfile[256] = "";
  65. char buffer[512] = "";
  66. int termfd = -1, parseokay = 0, i;
  67. if (!term)
  68. return 0;
  69. if (!option_console || option_nocolor || !option_nofork)
  70. return 0;
  71. for (i=0 ;; i++) {
  72. if (termpath[i] == NULL) {
  73. break;
  74. }
  75. snprintf(termfile, sizeof(termfile), "%s/%c/%s", termpath[i], *term, term);
  76. termfd = open(termfile, O_RDONLY);
  77. if (termfd > -1) {
  78. break;
  79. }
  80. }
  81. if (termfd > -1) {
  82. int actsize = read(termfd, buffer, sizeof(buffer) - 1);
  83. short sz_names = convshort(buffer + 2);
  84. short sz_bools = convshort(buffer + 4);
  85. short n_nums = convshort(buffer + 6);
  86. /* if ((sz_names + sz_bools) & 1)
  87. sz_bools++; */
  88. if (sz_names + sz_bools + n_nums < actsize) {
  89. /* Offset 13 is defined in /usr/include/term.h, though we do not
  90. * include it here, as it conflicts with include/asterisk/term.h */
  91. short max_colors = convshort(buffer + 12 + sz_names + sz_bools + 13 * 2);
  92. if (max_colors > 0) {
  93. vt100compat = 1;
  94. }
  95. parseokay = 1;
  96. }
  97. close(termfd);
  98. }
  99. if (!parseokay) {
  100. /* These comparisons should not be substrings nor case-insensitive, as
  101. * terminal types are very particular about how they treat suffixes and
  102. * capitalization. For example, terminal type 'linux-m' does NOT
  103. * support color, while 'linux' does. Not even all vt100* terminals
  104. * support color, either (e.g. 'vt100+fnkeys'). */
  105. if (!strcmp(term, "linux")) {
  106. vt100compat = 1;
  107. } else if (!strcmp(term, "xterm")) {
  108. vt100compat = 1;
  109. } else if (!strcmp(term, "xterm-color")) {
  110. vt100compat = 1;
  111. } else if (!strncmp(term, "Eterm", 5)) {
  112. /* Both entries which start with Eterm support color */
  113. vt100compat = 1;
  114. } else if (!strcmp(term, "vt100")) {
  115. vt100compat = 1;
  116. } else if (!strncmp(term, "crt", 3)) {
  117. /* Both crt terminals support color */
  118. vt100compat = 1;
  119. }
  120. }
  121. if (vt100compat) {
  122. /* Make commands show up in nice colors */
  123. snprintf(prepdata, sizeof(prepdata), "%c[%d;%d;%dm", ESC, ATTR_BRIGHT, COLOR_BROWN, COLOR_BLACK + 10);
  124. snprintf(enddata, sizeof(enddata), "%c[%d;%d;%dm", ESC, ATTR_RESET, COLOR_WHITE, COLOR_BLACK + 10);
  125. snprintf(quitdata, sizeof(quitdata), "%c[0m", ESC);
  126. }
  127. return 0;
  128. }
  129. char *term_color(char *outbuf, const char *inbuf, int fgcolor, int bgcolor, int maxout)
  130. {
  131. int attr=0;
  132. char tmp[40];
  133. if (!vt100compat) {
  134. ast_copy_string(outbuf, inbuf, maxout);
  135. return outbuf;
  136. }
  137. if (!fgcolor && !bgcolor) {
  138. ast_copy_string(outbuf, inbuf, maxout);
  139. return outbuf;
  140. }
  141. if ((fgcolor & 128) && (bgcolor & 128)) {
  142. /* Can't both be highlighted */
  143. ast_copy_string(outbuf, inbuf, maxout);
  144. return outbuf;
  145. }
  146. if (!bgcolor)
  147. bgcolor = COLOR_BLACK;
  148. if (bgcolor) {
  149. bgcolor &= ~128;
  150. bgcolor += 10;
  151. }
  152. if (fgcolor & 128) {
  153. attr = ATTR_BRIGHT;
  154. fgcolor &= ~128;
  155. }
  156. if (fgcolor && bgcolor) {
  157. snprintf(tmp, sizeof(tmp), "%d;%d", fgcolor, bgcolor);
  158. } else if (bgcolor) {
  159. snprintf(tmp, sizeof(tmp), "%d", bgcolor);
  160. } else if (fgcolor) {
  161. snprintf(tmp, sizeof(tmp), "%d", fgcolor);
  162. }
  163. if (attr) {
  164. snprintf(outbuf, maxout, "%c[%d;%sm%s%c[0;%d;%dm", ESC, attr, tmp, inbuf, ESC, COLOR_WHITE, COLOR_BLACK + 10);
  165. } else {
  166. snprintf(outbuf, maxout, "%c[%sm%s%c[0;%d;%dm", ESC, tmp, inbuf, ESC, COLOR_WHITE, COLOR_BLACK + 10);
  167. }
  168. return outbuf;
  169. }
  170. char *term_color_code(char *outbuf, int fgcolor, int bgcolor, int maxout)
  171. {
  172. int attr=0;
  173. char tmp[40];
  174. if ((!vt100compat) || (!fgcolor && !bgcolor)) {
  175. *outbuf = '\0';
  176. return outbuf;
  177. }
  178. if ((fgcolor & 128) && (bgcolor & 128)) {
  179. /* Can't both be highlighted */
  180. *outbuf = '\0';
  181. return outbuf;
  182. }
  183. if (!bgcolor)
  184. bgcolor = COLOR_BLACK;
  185. if (bgcolor) {
  186. bgcolor &= ~128;
  187. bgcolor += 10;
  188. }
  189. if (fgcolor & 128) {
  190. attr = ATTR_BRIGHT;
  191. fgcolor &= ~128;
  192. }
  193. if (fgcolor && bgcolor) {
  194. snprintf(tmp, sizeof(tmp), "%d;%d", fgcolor, bgcolor);
  195. } else if (bgcolor) {
  196. snprintf(tmp, sizeof(tmp), "%d", bgcolor);
  197. } else if (fgcolor) {
  198. snprintf(tmp, sizeof(tmp), "%d", fgcolor);
  199. }
  200. if (attr) {
  201. snprintf(outbuf, maxout, "%c[%d;%sm", ESC, attr, tmp);
  202. } else {
  203. snprintf(outbuf, maxout, "%c[%sm", ESC, tmp);
  204. }
  205. return outbuf;
  206. }
  207. char *term_strip(char *outbuf, char *inbuf, int maxout)
  208. {
  209. char *outbuf_ptr = outbuf, *inbuf_ptr = inbuf;
  210. while (outbuf_ptr < outbuf + maxout) {
  211. switch (*inbuf_ptr) {
  212. case ESC:
  213. while (*inbuf_ptr && (*inbuf_ptr != 'm'))
  214. inbuf_ptr++;
  215. break;
  216. default:
  217. *outbuf_ptr = *inbuf_ptr;
  218. outbuf_ptr++;
  219. }
  220. if (! *inbuf_ptr)
  221. break;
  222. inbuf_ptr++;
  223. }
  224. return outbuf;
  225. }
  226. char *term_prompt(char *outbuf, const char *inbuf, int maxout)
  227. {
  228. if (!vt100compat) {
  229. ast_copy_string(outbuf, inbuf, maxout);
  230. return outbuf;
  231. }
  232. snprintf(outbuf, maxout, "%c[%d;%d;%dm%c%c[%d;%d;%dm%s",
  233. ESC, ATTR_BRIGHT, COLOR_BLUE, COLOR_BLACK + 10,
  234. inbuf[0],
  235. ESC, 0, COLOR_WHITE, COLOR_BLACK + 10,
  236. inbuf + 1);
  237. return outbuf;
  238. }
  239. char *term_prep(void)
  240. {
  241. return prepdata;
  242. }
  243. char *term_end(void)
  244. {
  245. return enddata;
  246. }
  247. char *term_quit(void)
  248. {
  249. return quitdata;
  250. }