chiark / gitweb /
*** empty log message ***
[sympathy.git] / src / terminal.c
1 /*
2  * terminal.c:
3  *
4  * Copyright (c) 2008 James McKenzie <james@fishsoup.dhs.org>,
5  * All rights reserved.
6  *
7  */
8
9 static char rcsid[] = "$Id$";
10
11 /*
12  * $Log$
13  * Revision 1.3  2008/02/13 09:12:21  james
14  * *** empty log message ***
15  *
16  * Revision 1.2  2008/02/13 01:08:18  james
17  * *** empty log message ***
18  *
19  * Revision 1.1  2008/02/12 22:36:46  james
20  * *** empty log message ***
21  *
22  * Revision 1.1  2008/02/09 15:47:28  james
23  * *** empty log message ***
24  *
25  * Revision 1.2  2008/02/07 11:11:14  staffcvs
26  * *** empty log message ***
27  *
28  * Revision 1.1  2008/02/07 01:02:52  james
29  * *** empty log message ***
30  *
31  * Revision 1.3  2008/02/06 17:53:28  james
32  * *** empty log message ***
33  *
34  * Revision 1.2  2008/02/04 02:05:06  james
35  * *** empty log message ***
36  *
37  * Revision 1.1  2008/02/04 01:32:39  james
38  * *** empty log message ***
39  *
40  */
41
42 #include "project.h"
43
44
45 typedef struct TERMINAL_struct
46 {
47   TTY_SIGNATURE;
48   struct termios orig_termios;
49   struct TERMINAL_struct *next;
50 } TERMINAL;
51
52
53 static TERMINAL *terminal_list = NULL;
54 int terminal_winches;
55
56
57
58 static void
59 terminal_close (TTY * _t)
60 {
61   TERMINAL *t = (TERMINAL *) _t;
62   TERMINAL **ptr = &terminal_list;
63
64   if (!t)
65     return;
66
67   /* Take out of cleanup list */
68   while (*ptr && (*ptr != t))
69     ptr = &((*ptr)->next);
70
71   if (*ptr)
72     *ptr = t->next;
73
74   tcsetattr (t->wfd, TCSANOW, &t->orig_termios);
75
76   set_blocking (t->rfd);
77   set_blocking (t->wfd);
78
79   free (t);
80 }
81
82
83 void
84 terminal_atexit (void)
85 {
86   while (terminal_list)
87     terminal_close ((TTY *) terminal_list);
88 }
89
90 static void
91 sigint (int dummy)
92 {
93   terminal_atexit ();
94   exit (-1);
95 }
96
97 static void
98 sigwinch (int not)
99 {
100   terminal_winches++;
101 }
102
103
104 void
105 terminal_getsize (TTY * _t)
106 {
107   TERMINAL *t = (TERMINAL *) _t;
108   struct winsize sz = { 0 };
109
110   if (!t)
111     return;
112
113   if (ioctl (t->wfd, TIOCGWINSZ, &sz))
114     {
115       t->size.x = CRT_COLS;
116       t->size.y = CRT_ROWS;
117     }
118   else
119     {
120       t->size.x = sz.ws_col;
121       t->size.y = sz.ws_row;
122     }
123 }
124
125
126 void
127 terminal_dispatch (void)
128 {
129   TERMINAL *t;
130
131
132   if (!terminal_winches)
133     return;
134
135   terminal_winches = 0;
136
137   for (t = terminal_list; t; t = t->next)
138     terminal_getsize (t);
139
140 }
141
142
143 static int
144 terminal_read (TTY * _t, void *buf, int len)
145 {
146   TERMINAL *t = (TERMINAL *) _t;
147   int red, done = 0;
148
149   terminal_dispatch ();
150
151   do
152     {
153
154       red = wrap_read (t->rfd, buf, len);
155       if (red < 0)
156         return -1;
157       if (!red)
158         return done;
159
160       buf += red;
161       len -= red;
162       done += red;
163     }
164   while (len);
165
166
167   return done;
168 }
169
170
171 static int
172 terminal_write (TTY * _t, void *buf, int len)
173 {
174   int writ, done = 0;
175   TERMINAL *t = (TERMINAL *) _t;
176
177   terminal_dispatch ();
178
179   do
180     {
181
182       writ = wrap_write (t->wfd, buf, len);
183       if (writ < 0)
184         return -1;
185       if (!writ)
186         sleep (1);
187
188       buf += writ;
189       len -= writ;
190       done += writ;
191     }
192   while (len);
193
194
195   return done;
196 }
197
198
199 void
200 terminal_register_handlers (void)
201 {
202   struct sigaction sa = { 0 };
203
204   sa.sa_handler = sigwinch;
205   sa.sa_flags = SA_RESTART;
206   sigaction (SIGWINCH, &sa, NULL);
207
208   sa.sa_handler = sigint;
209   sa.sa_flags = SA_RESTART;
210   sigaction (SIGINT, &sa, NULL);
211 }
212
213 TTY *
214 terminal_open (int rfd, int wfd)
215 {
216   TERMINAL *t;
217   pid_t child;
218   char name[1024];
219   struct termios termios;
220
221   t = (TERMINAL *) malloc (sizeof (TERMINAL));
222
223   t->rfd = rfd;
224   t->wfd = wfd;
225
226   tcgetattr (wfd, &t->orig_termios);
227
228   t->next = terminal_list;
229   terminal_list = t;
230
231   tcgetattr (wfd, &termios);
232
233   set_nonblocking (rfd);
234   set_nonblocking (wfd);
235
236
237   raw_termios (&termios);
238
239   tcsetattr (wfd, TCSANOW, &termios);
240
241   t->recv = terminal_read;
242   t->xmit = terminal_write;
243   t->close = terminal_close;
244
245
246   terminal_getsize ((TTY *) t);
247
248   return (TTY *) t;
249 }