app_term.c
author viric@llimona
Thu, 27 Sep 2007 00:25:54 +0200
changeset 58 2cf8c513d18f
parent 53 07500c5c53cb
child 71 c209487034d7
permissions -rw-r--r--
added authors.
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
53
07500c5c53cb Adding license and web html.
viric@llimona
parents: 49
diff changeset
     1
/*
07500c5c53cb Adding license and web html.
viric@llimona
parents: 49
diff changeset
     2
    Terminal Mixer - multi-point multi-user access to terminal applications
07500c5c53cb Adding license and web html.
viric@llimona
parents: 49
diff changeset
     3
    Copyright (C) 2007  LluĂ­s Batlle i Rossell
07500c5c53cb Adding license and web html.
viric@llimona
parents: 49
diff changeset
     4
07500c5c53cb Adding license and web html.
viric@llimona
parents: 49
diff changeset
     5
    Please find the license in the provided COPYING file.
07500c5c53cb Adding license and web html.
viric@llimona
parents: 49
diff changeset
     6
*/
23
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
     7
#include <sys/types.h>
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
     8
#include <unistd.h>
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
     9
#include <errno.h>
25
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    10
#include <fcntl.h>
23
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
    11
#include <string.h>
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
    12
#include <sys/select.h>
25
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    13
#include <sys/ioctl.h>
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    14
#include <termios.h>
49
1cead94cfd99 Fixing a few bugs. The program runs well in Linux and Mac OS X.
viric@llimona
parents: 47
diff changeset
    15
#ifdef __APPLE__
1cead94cfd99 Fixing a few bugs. The program runs well in Linux and Mac OS X.
viric@llimona
parents: 47
diff changeset
    16
  #include <util.h>
1cead94cfd99 Fixing a few bugs. The program runs well in Linux and Mac OS X.
viric@llimona
parents: 47
diff changeset
    17
#else
1cead94cfd99 Fixing a few bugs. The program runs well in Linux and Mac OS X.
viric@llimona
parents: 47
diff changeset
    18
  #include <pty.h>
1cead94cfd99 Fixing a few bugs. The program runs well in Linux and Mac OS X.
viric@llimona
parents: 47
diff changeset
    19
#endif
23
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
    20
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
    21
#include "main.h"
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
    22
41
954941c6e40a Added 'nohup' option -n.
viric@llimona
parents: 29
diff changeset
    23
954941c6e40a Added 'nohup' option -n.
viric@llimona
parents: 29
diff changeset
    24
struct winsize app_winsize; /* for xterm.c */
954941c6e40a Added 'nohup' option -n.
viric@llimona
parents: 29
diff changeset
    25
25
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    26
enum
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    27
{
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    28
    STDIN,
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    29
    STDOUT,
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    30
    STDERR
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    31
};
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    32
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    33
static void set_close_on_exec(int fd)
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    34
{
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    35
    int res;
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    36
    res = fcntl(fd, F_SETFD, FD_CLOEXEC);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    37
    if (res == -1)
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    38
        error("Set close-on-exec failed");
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    39
}
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    40
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    41
static void give_terminal(int parent[], int child[])
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    42
{
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    43
    struct termios tios;
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    44
    struct winsize wsize;
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    45
    int res;
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    46
    int master, slave;
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    47
28
b73712b8370d use the termios calls to stdout instead of stdin.
viric@llimona
parents: 26
diff changeset
    48
    res = tcgetattr(1, &tios);
25
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    49
    if (res == -1)
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    50
        error("tcgetatttr");
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    51
41
954941c6e40a Added 'nohup' option -n.
viric@llimona
parents: 29
diff changeset
    52
    /* if starting using shells' &, it doesn't get ECHO set.
954941c6e40a Added 'nohup' option -n.
viric@llimona
parents: 29
diff changeset
    53
     * Let's force it. */
954941c6e40a Added 'nohup' option -n.
viric@llimona
parents: 29
diff changeset
    54
    tios.c_lflag |= ECHO | ECHOE | ECHOK | ECHONL;
954941c6e40a Added 'nohup' option -n.
viric@llimona
parents: 29
diff changeset
    55
28
b73712b8370d use the termios calls to stdout instead of stdin.
viric@llimona
parents: 26
diff changeset
    56
    res = ioctl(1, TIOCGWINSZ, &wsize);
25
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    57
    if (res == -1)
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    58
        error("ioctl TIOCGWINSZ");
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    59
41
954941c6e40a Added 'nohup' option -n.
viric@llimona
parents: 29
diff changeset
    60
    app_winsize = wsize;
954941c6e40a Added 'nohup' option -n.
viric@llimona
parents: 29
diff changeset
    61
25
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    62
    res = openpty(&master, &slave, 0, &tios, &wsize);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    63
    if (res == -1)
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    64
        error("openpty");
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    65
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    66
    parent[STDIN]  = master;
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    67
    parent[STDOUT] = master;
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    68
    parent[STDERR] = master;
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    69
    child[STDIN]  = slave;
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    70
    child[STDOUT] = slave;
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    71
    child[STDERR] = slave;
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    72
    set_close_on_exec(master);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    73
    set_close_on_exec(slave);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    74
}
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    75
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    76
static void give_pipes(int parent[], int child[])
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    77
{
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    78
    int p_stdin[2]; /* from us to mpg321 */
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    79
    int p_stdout[2]; /* from mpg321 to us */
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    80
    int p_stderr[2]; /* from mpg321 to us, its stderr */
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    81
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    82
    pipe(p_stdin);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    83
    pipe(p_stdout);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    84
    pipe(p_stderr);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    85
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    86
    parent[STDIN]  = p_stdin[1];
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    87
    parent[STDOUT] = p_stdout[0];
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    88
    parent[STDERR] = p_stderr[0];
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    89
    child[STDIN]   = p_stdin[0];
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    90
    child[STDOUT]  = p_stdout[1];
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    91
    child[STDERR]  = p_stderr[1];
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    92
    set_close_on_exec(parent[STDIN]);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    93
    set_close_on_exec(parent[STDOUT]);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    94
    set_close_on_exec(parent[STDERR]);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    95
    set_close_on_exec(child[STDIN]);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    96
    set_close_on_exec(child[STDOUT]);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    97
    set_close_on_exec(child[STDERR]);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    98
}
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
    99
23
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   100
int fork_app(char * const command[])
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   101
{
25
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   102
    int p_parent[3];
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   103
    int p_child[3];
23
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   104
    int pid;
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   105
    int res;
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   106
29
91286c3ecebc Added getopt, and some things got based on parameters.
viric@llimona
parents: 28
diff changeset
   107
    if (command_line.s_param.run_in_subterminal)
47
8dcc45d8f3e5 xterm resizes sent on SIGWINCH. new terminal applications start a new session.
lbatlle@npdl268.bpo.hp.com
parents: 41
diff changeset
   108
    {
26
96920c3707b3 Unix sockets version works! Well, signals don't. :)
viric@llimona
parents: 25
diff changeset
   109
        give_terminal(p_parent, p_child);
47
8dcc45d8f3e5 xterm resizes sent on SIGWINCH. new terminal applications start a new session.
lbatlle@npdl268.bpo.hp.com
parents: 41
diff changeset
   110
    }
26
96920c3707b3 Unix sockets version works! Well, signals don't. :)
viric@llimona
parents: 25
diff changeset
   111
    else
96920c3707b3 Unix sockets version works! Well, signals don't. :)
viric@llimona
parents: 25
diff changeset
   112
        give_pipes(p_parent, p_child);
25
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   113
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   114
    /* globals */
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   115
    app_stdin  = p_parent[STDIN];  /* For us to write */
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   116
    app_stdout = p_parent[STDOUT]; /* For us to read */
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   117
    app_stderr = p_parent[STDERR]; /* For us to read */
23
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   118
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   119
    pid = fork();
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   120
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   121
    switch(pid)
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   122
    {
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   123
        case 0: /* child */
25
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   124
            res = dup2(p_child[STDIN], 0);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   125
            if (res == -1) error("Dup2 stdin");
23
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   126
25
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   127
            res = dup2(p_child[STDOUT], 1);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   128
            if (res == -1) error("Dup2 stdout");
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   129
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   130
            res = dup2(p_child[STDERR], 2);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   131
            if (res == -1) error("Dup2 stderr");
23
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   132
47
8dcc45d8f3e5 xterm resizes sent on SIGWINCH. new terminal applications start a new session.
lbatlle@npdl268.bpo.hp.com
parents: 41
diff changeset
   133
            if (command_line.s_param.run_in_subterminal)
8dcc45d8f3e5 xterm resizes sent on SIGWINCH. new terminal applications start a new session.
lbatlle@npdl268.bpo.hp.com
parents: 41
diff changeset
   134
            {
8dcc45d8f3e5 xterm resizes sent on SIGWINCH. new terminal applications start a new session.
lbatlle@npdl268.bpo.hp.com
parents: 41
diff changeset
   135
                int res;
8dcc45d8f3e5 xterm resizes sent on SIGWINCH. new terminal applications start a new session.
lbatlle@npdl268.bpo.hp.com
parents: 41
diff changeset
   136
                res = setsid();
8dcc45d8f3e5 xterm resizes sent on SIGWINCH. new terminal applications start a new session.
lbatlle@npdl268.bpo.hp.com
parents: 41
diff changeset
   137
                if (res < 0)
8dcc45d8f3e5 xterm resizes sent on SIGWINCH. new terminal applications start a new session.
lbatlle@npdl268.bpo.hp.com
parents: 41
diff changeset
   138
                  error("failed setsid()");
8dcc45d8f3e5 xterm resizes sent on SIGWINCH. new terminal applications start a new session.
lbatlle@npdl268.bpo.hp.com
parents: 41
diff changeset
   139
                res = ioctl(0, TIOCSCTTY, 0);
8dcc45d8f3e5 xterm resizes sent on SIGWINCH. new terminal applications start a new session.
lbatlle@npdl268.bpo.hp.com
parents: 41
diff changeset
   140
                if (res < 0)
8dcc45d8f3e5 xterm resizes sent on SIGWINCH. new terminal applications start a new session.
lbatlle@npdl268.bpo.hp.com
parents: 41
diff changeset
   141
                  error("failed ioctl(0,TIOCSCTTY,0)");
8dcc45d8f3e5 xterm resizes sent on SIGWINCH. new terminal applications start a new session.
lbatlle@npdl268.bpo.hp.com
parents: 41
diff changeset
   142
            }
23
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   143
            execvp(command[0], command);
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   144
25
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   145
            error("Cannot execvp %s", command[0]);
23
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   146
        case -1:
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   147
            error("Failed fork");
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   148
        default: /* parent */
25
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   149
            close(p_child[STDIN]);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   150
            close(p_child[STDOUT]);
5b5ca60e3646 First try on openpty.
viric@llimona
parents: 23
diff changeset
   151
            close(p_child[STDERR]);
23
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   152
    }
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   153
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   154
    return pid;
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   155
}
b3e6c6ffc69c Moving the client out.
viric@llimona
parents:
diff changeset
   156