aboutsummaryrefslogtreecommitdiff
path: root/safetitle.c
blob: e5adac3773bbd7e1f9e12729012850055e964a21 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
/*
 *  safetitle sets the title of the terminal window to the given string,
 *  if and only if the terminal is recognized via its device attributes.
 *
 *  If multiple terminals are attached to the same GNU screen session,
 *  the title is not set.  It is theoretically possible to check every
 *  terminal too see if it is recognized and, if so, set the title --
 *  but that would make the code more complex.
 */

#include <err.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/select.h>
#include <termios.h>
#include <unistd.h>

#define WSCONS_DA (char[]){27,91,62,50,52,59,50,48,59,48,99,0}
#define SCREEN_DA (char[]){27,91,62,56,51,59,52,48,56,48,48,59,48,99,0}

int debug = 0;

int
da(int fd, char *buf, int real)
{
	char c;
	fd_set fds;
	int i, r, responses;
	struct timeval timeout;

	FD_ZERO(&fds);
	FD_SET(fd, &fds);
	timeout.tv_sec = 0;
	timeout.tv_usec = 1;
	responses = 0;
	i = 0;

	if(real) write(fd, "\033P\033[>c\033\\", 8);
	else write(fd, "\033[>c", 4);

	while(r = read(fd, &c, 1)){
loop:
		if(r==-1){
			warn("read");
			return -1;
		}
		buf[i++] = c;
		if(c==99) break;
	}
	buf[i] = '\0';
	responses++;

	/* ensure there is only one real terminal */
check:
	if(real && select(fd+1, &fds, NULL, NULL, &timeout)>0){
		if ((r = read(fd, &c, 1)) && c==0) goto check; /* extra nul */
		else goto loop;
	}
	if(responses>1){
		if(debug) fprintf(stderr, "more than one attached terminal\n");
		return -1;
	}

	return i;
}

int
main(int argc, char *argv[])
{
	char *ap, *buf;
	int r, screen, ttyfd;
	struct termios term, restore;

	r = 1;

        /* parse arguments */
        if(argc==2)
                ap = argv[1];
        else if(argc==3){
                ap = argv[2];
                if(strcmp(argv[1], "-d")==0) debug = 1;
                else goto usage;
        }
        else goto usage;

	if((ttyfd = open("/dev/tty", O_RDWR))==-1)
		err(1, "open");
	if((buf = malloc(200*sizeof(char)))==NULL)
		err(1, "malloc");

	/* enter "raw" terminal mode */
	tcgetattr(ttyfd, &restore);
	tcgetattr(ttyfd, &term);
	term.c_lflag &= ~(ICANON|ECHO);
	tcsetattr(ttyfd, TCSANOW, &term);

	/* get device attributes for real terminal */
	if(da(ttyfd, buf, 0)==-1){
		goto end;
	}
	screen = strcmp(buf, SCREEN_DA)==0;
	if(screen && da(ttyfd, buf, screen)==-1){
		goto end;
	}

	/* set title */
	if(strcmp(buf, WSCONS_DA)==0){
		if(debug) fprintf(stderr, "wrong type of terminal\n");
		goto end;
	}
	else{
		if(screen) dprintf(ttyfd, "\033P\033]2;%s\007\033\\", ap);
		else dprintf(ttyfd, "\033]2;%s\007", ap);
	}

	r = 0;
end:
	tcsetattr(ttyfd, TCSANOW, &restore);
	return r;
usage:
        fprintf(stderr, "usage: %s [-d] title\n", argv[0]);
        return 1;
}