1 /****************************************************************************
2 * Copyright (c) 1998-2011,2012 Free Software Foundation, Inc. *
4 * Permission is hereby granted, free of charge, to any person obtaining a *
5 * copy of this software and associated documentation files (the *
6 * "Software"), to deal in the Software without restriction, including *
7 * without limitation the rights to use, copy, modify, merge, publish, *
8 * distribute, distribute with modifications, sublicense, and/or sell *
9 * copies of the Software, and to permit persons to whom the Software is *
10 * furnished to do so, subject to the following conditions: *
12 * The above copyright notice and this permission notice shall be included *
13 * in all copies or substantial portions of the Software. *
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS *
16 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF *
17 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. *
18 * IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, *
19 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR *
20 * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR *
21 * THE USE OR OTHER DEALINGS IN THE SOFTWARE. *
23 * Except as contained in this notice, the name(s) of the above copyright *
24 * holders shall not be used in advertising or otherwise to promote the *
25 * sale, use or other dealings in this Software without prior written *
27 ****************************************************************************/
30 * Author: Thomas E. Dickey (1998-on)
32 * $Id: ditto.c,v 1.42 2012/11/24 20:16:18 tom Exp $
34 * The program illustrates how to set up multiple screens from a single
37 * If openpty() is supported, the command line parameters are titles for
38 * the windows showing each screen's data.
40 * If openpty() is not supported, you must invoke the program by specifying
41 * another terminal on the same machine by specifying its device, e.g.,
44 #include <test.priv.h>
52 #include USE_OPENPTY_HEADER
57 #define THIS_FIFO(n) ((n) % MAX_FIFO)
58 #define NEXT_FIFO(n) THIS_FIFO((n) + 1)
61 unsigned long sequence
;
68 unsigned long sequence
;
72 * Data "owned" for a single screen. Each screen is divided into windows that
73 * show the text read from each terminal. Input from a given screen will also
74 * be read into one window per screen.
79 SCREEN
*screen
; /* this screen - curses internal data */
80 int which1
; /* this screen's index in DITTO[] array */
81 int length
; /* length of windows[] and peeks[] */
82 char **titles
; /* per-window titles */
83 WINDOW
**parents
; /* display boxes around each screen's data */
84 WINDOW
**windows
; /* display data from each screen */
85 PEEK
*peeks
; /* indices for each screen's fifo */
86 FIFO fifo
; /* fifo for this screen */
93 * Structure used to pass multiple parameters via the use_screen()
94 * single-parameter interface.
97 int source
; /* which screen did character come from */
98 int target
; /* which screen is character going to */
99 DITTO
*ditto
; /* data for all screens */
102 static void failed(const char *) GCC_NORETURN
;
103 static void usage(void) GCC_NORETURN
;
106 failed(const char *s
)
109 ExitProgram(EXIT_FAILURE
);
115 fprintf(stderr
, "usage: ditto [terminal1 ...]\n");
116 ExitProgram(EXIT_FAILURE
);
119 /* Add to the head of the fifo, checking for overflow. */
121 put_fifo(FIFO
* fifo
, int value
)
123 int next
= NEXT_FIFO(fifo
->head
);
124 if (next
== fifo
->tail
)
125 fifo
->tail
= NEXT_FIFO(fifo
->tail
);
126 fifo
->data
[next
] = value
;
131 /* Get data from the tail (oldest part) of the fifo, returning -1 if no data.
132 * Since each screen can peek into the fifo, we do not update the tail index,
133 * but modify the peek-index.
135 * FIXME - test/workaround for case where fifo gets more than a buffer
139 peek_fifo(FIFO
* fifo
, PEEK
* peek
)
142 if (peek
->sequence
< fifo
->sequence
) {
143 result
= fifo
->data
[THIS_FIFO(peek
->sequence
)];
156 char slave_name
[1024];
157 char s_option
[sizeof(slave_name
) + 80];
159 if (openpty(&amaster
, &aslave
, slave_name
, 0, 0) != 0
160 || strlen(slave_name
) > sizeof(slave_name
) - 1)
162 if (strrchr(slave_name
, '/') == 0) {
166 sprintf(s_option
, "-S%s/%d", slave_name
, aslave
);
168 execlp("xterm", "xterm", s_option
, "-title", path
, (char *) 0);
171 fp
= fdopen(amaster
, "r+");
177 if (stat(path
, &sb
) < 0)
179 if ((sb
.st_mode
& S_IFMT
) != S_IFCHR
) {
183 fp
= fopen(path
, "r+");
186 printf("opened %s\n", path
);
195 SCREEN
*sp GCC_UNUSED
,
199 DITTO
*target
= (DITTO
*) arg
;
205 scrollok(stdscr
, TRUE
);
208 target
->parents
= typeCalloc(WINDOW
*, (size_t) target
->length
);
209 target
->windows
= typeCalloc(WINDOW
*, (size_t) target
->length
);
210 target
->peeks
= typeCalloc(PEEK
, (size_t) target
->length
);
212 high
= (LINES
- 2) / target
->length
;
214 for (k
= 0; k
< target
->length
; ++k
) {
215 WINDOW
*outer
= newwin(high
, wide
, 1 + (high
* k
), 1);
216 WINDOW
*inner
= derwin(outer
, high
- 2, wide
- 2, 1, 1);
219 MvWAddStr(outer
, 0, 2, target
->titles
[k
]);
222 scrollok(inner
, TRUE
);
225 nodelay(inner
, TRUE
);
228 target
->parents
[k
] = outer
;
229 target
->windows
[k
] = inner
;
235 open_screen(DITTO
* target
, char **source
, int length
, int which1
)
239 target
->output
= open_tty(source
[which1
]);
241 target
->input
= stdin
;
242 target
->output
= stdout
;
245 target
->which1
= which1
;
246 target
->titles
= source
;
247 target
->length
= length
;
248 target
->fifo
.head
= -1;
249 target
->screen
= newterm((char *) 0, /* assume $TERM is the same */
253 if (target
->screen
== 0)
256 (void) USING_SCREEN(target
->screen
, init_screen
, target
);
262 SCREEN
*sp GCC_UNUSED
,
264 void *arg GCC_UNUSED
)
274 * Read data from the 'source' screen.
279 SCREEN
*sp GCC_UNUSED
,
283 DDATA
*data
= (DDATA
*) arg
;
284 DITTO
*ditto
= &(data
->ditto
[data
->source
]);
285 WINDOW
*win
= ditto
->windows
[data
->source
];
286 int ch
= wgetch(win
);
288 if (ch
> 0 && ch
< 256)
289 put_fifo(&(ditto
->fifo
), ch
);
297 * Write all of the data that's in fifos for the 'target' screen.
302 SCREEN
*sp GCC_UNUSED
,
304 void *arg GCC_UNUSED
)
306 DDATA
*data
= (DDATA
*) arg
;
307 DITTO
*ditto
= &(data
->ditto
[data
->target
]);
308 bool changed
= FALSE
;
311 for (which
= 0; which
< ditto
->length
; ++which
) {
312 WINDOW
*win
= ditto
->windows
[which
];
313 FIFO
*fifo
= &(data
->ditto
[which
].fifo
);
314 PEEK
*peek
= &(ditto
->peeks
[which
]);
317 while ((ch
= peek_fifo(fifo
, peek
)) > 0) {
320 waddch(win
, (chtype
) ch
);
331 show_ditto(DITTO
* data
, int count
, DDATA
* ddata
)
336 for (n
= 0; n
< count
; n
++) {
338 USING_SCREEN(data
[n
].screen
, write_screen
, (void *) ddata
);
344 handle_screen(void *arg
)
349 memset(&ddata
, 0, sizeof(ddata
));
350 ddata
.ditto
= (DITTO
*) arg
;
351 ddata
.source
= ddata
.ditto
->which1
;
352 ddata
.ditto
-= ddata
.source
; /* -> base of array */
355 ch
= read_screen(ddata
.ditto
->screen
, &ddata
);
356 if (ch
== CTRL('D')) {
357 int later
= (ddata
.source
? ddata
.source
: -1);
360 for (j
= ddata
.ditto
->length
- 1; j
> 0; --j
) {
362 pthread_cancel(ddata
.ditto
[j
].thread
);
366 pthread_cancel(ddata
.ditto
[later
].thread
);
370 show_ditto(ddata
.ditto
, ddata
.ditto
->length
, &ddata
);
377 main(int argc
, char *argv
[])
388 if ((data
= typeCalloc(DITTO
, (size_t) argc
)) == 0)
389 failed("calloc data");
393 for (j
= 0; j
< argc
; j
++) {
394 open_screen(&data
[j
], argv
, argc
, j
);
399 * For multi-threaded operation, set up a reader for each of the screens.
400 * That uses blocking I/O rather than polling for input, so no calls to
401 * napms() are needed.
403 for (j
= 0; j
< argc
; j
++) {
404 (void) pthread_create(&(data
[j
].thread
), NULL
, handle_screen
, &data
[j
]);
406 pthread_join(data
[1].thread
, NULL
);
409 * Loop, reading characters from any of the inputs and writing to all
412 for (count
= 0;; ++count
) {
415 int which
= (count
% argc
);
419 ddata
.source
= which
;
422 ch
= USING_SCREEN(data
[which
].screen
, read_screen
, &ddata
);
423 if (ch
== CTRL('D')) {
425 } else if (ch
!= ERR
) {
426 show_ditto(data
, argc
, &ddata
);
434 for (j
= argc
- 1; j
>= 0; j
--) {
435 USING_SCREEN(data
[j
].screen
, close_screen
, 0);
436 fprintf(data
[j
].output
, "**Closed\r\n");
439 * Closing before a delscreen() helps ncurses determine that there
440 * is no valid output buffer, and can remove the setbuf() data.
442 fflush(data
[j
].output
);
443 fclose(data
[j
].output
);
444 delscreen(data
[j
].screen
);
446 ExitProgram(EXIT_SUCCESS
);