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