chiark / gitweb /
*** empty log message ***
[sympathy.git] / src / ptty.c
1 /*
2  * ptty.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.11  2008/02/26 23:23:17  james
14  * *** empty log message ***
15  *
16  * Revision 1.10  2008/02/24 00:42:53  james
17  * *** empty log message ***
18  *
19  * Revision 1.9  2008/02/23 13:05:58  staffcvs
20  * *** empty log message ***
21  *
22  * Revision 1.8  2008/02/23 11:48:37  james
23  * *** empty log message ***
24  *
25  * Revision 1.7  2008/02/22 17:07:00  james
26  * *** empty log message ***
27  *
28  * Revision 1.6  2008/02/22 14:51:54  james
29  * *** empty log message ***
30  *
31  * Revision 1.5  2008/02/15 23:52:12  james
32  * *** empty log message ***
33  *
34  * Revision 1.4  2008/02/14 10:39:14  james
35  * *** empty log message ***
36  *
37  * Revision 1.3  2008/02/13 09:12:21  james
38  * *** empty log message ***
39  *
40  * Revision 1.2  2008/02/12 22:36:46  james
41  * *** empty log message ***
42  *
43  * Revision 1.1  2008/02/09 15:47:28  james
44  * *** empty log message ***
45  *
46  * Revision 1.2  2008/02/07 11:11:14  staffcvs
47  * *** empty log message ***
48  *
49  * Revision 1.1  2008/02/07 01:02:52  james
50  * *** empty log message ***
51  *
52  * Revision 1.3  2008/02/06 17:53:28  james
53  * *** empty log message ***
54  *
55  * Revision 1.2  2008/02/04 02:05:06  james
56  * *** empty log message ***
57  *
58  * Revision 1.1  2008/02/04 01:32:39  james
59  * *** empty log message ***
60  *
61  */
62
63 #include "project.h"
64
65
66 typedef struct
67 {
68   TTY_SIGNATURE;
69   int fd;
70   pid_t child;
71 } PTTY;
72
73
74 static void
75 ptty_close (TTY * _t)
76 {
77   PTTY *t = (PTTY *) _t;
78
79   if (!t)
80     return;
81
82   close (t->fd);
83   free (t);
84 }
85
86
87
88 static int
89 ptty_read (TTY * _t, void *buf, int len)
90 {
91   PTTY *t = (PTTY *) _t;
92   int red, done = 0;
93
94   do
95     {
96
97       red = wrap_read (t->fd, buf, len);
98       if (red < 0)
99         return -1;
100       if (!red)
101         return done;
102
103       buf += red;
104       len -= red;
105       done += red;
106     }
107   while (len);
108
109
110   return done;
111 }
112
113
114 static int
115 ptty_write (TTY * _t, void *buf, int len)
116 {
117   int writ, done = 0;
118   PTTY *t = (PTTY *) _t;
119
120   do
121     {
122
123       writ = wrap_write (t->fd, buf, len);
124       if (writ < 0)
125         return -1;
126       if (!writ)
127         sleep (1);
128
129       buf += writ;
130       len -= writ;
131       done += writ;
132     }
133   while (len);
134
135
136   return done;
137 }
138
139 TTY *
140 ptty_open (char *path, char *argv[])
141 {
142   PTTY *t;
143   pid_t child;
144   char name[1024];
145   struct winsize winsize = { 0 };
146   struct termios ctermios = { 0 };
147   int fd;
148   char *default_argv[] = { "-", (char *) 0 };
149
150
151   client_termios (&ctermios);
152   winsize.ws_row = VT102_ROWS;
153   winsize.ws_col = VT102_COLS_80;
154
155   child = forkpty (&fd, name, &ctermios, &winsize);
156
157   switch (child)
158     {
159     case -1:                   /*boo hiss */
160       return NULL;
161     case 0:                    /*waaah */
162       setenv ("TERM", "xterm", 1);
163       if (!path)
164         path = "/bin/sh";
165
166       if (!argv)
167         argv = default_argv;
168
169       execv (path, argv);
170       _exit (-1);
171     }
172
173   set_nonblocking (fd);
174
175 #if 0
176   {
177     struct termios termios = { 0 };
178
179     tcgetattr (fd, &termios);
180     default_termios (&termios);
181     tcsetattr (fd, TCSANOW, &termios);
182   }
183 #endif
184
185   t = (PTTY *) malloc (sizeof (PTTY));
186
187   strncpy (t->name, name, sizeof (t->name));
188   t->name[sizeof (t->name) - 1] = 0;
189
190   t->recv = ptty_read;
191   t->xmit = ptty_write;
192   t->close = ptty_close;
193   t->fd = fd;
194   t->child = child;
195   t->rfd = t->fd;
196   t->wfd = t->fd;
197   t->size.x = winsize.ws_row;
198   t->size.y = winsize.ws_col;
199   t->blocked = 0;
200   t->hanging_up = 0;
201
202   return (TTY *) t;
203 }