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