Merge branch 'jk/reflog-date' into next
[git] / pager.c
1 #include "cache.h"
2 #include "run-command.h"
3 #include "sigchain.h"
4
5 /*
6  * This is split up from the rest of git so that we can do
7  * something different on Windows.
8  */
9
10 static int spawned_pager;
11
12 #ifndef WIN32
13 static void pager_preexec(void)
14 {
15         /*
16          * Work around bug in "less" by not starting it until we
17          * have real input
18          */
19         fd_set in;
20
21         FD_ZERO(&in);
22         FD_SET(0, &in);
23         select(1, &in, NULL, &in, NULL);
24 }
25 #endif
26
27 static const char *pager_argv[] = { "sh", "-c", NULL, NULL };
28 static struct child_process pager_process;
29
30 static void wait_for_pager(void)
31 {
32         fflush(stdout);
33         fflush(stderr);
34         /* signal EOF to pager */
35         close(1);
36         close(2);
37         finish_command(&pager_process);
38 }
39
40 static void wait_for_pager_signal(int signo)
41 {
42         wait_for_pager();
43         sigchain_pop(signo);
44         raise(signo);
45 }
46
47 void setup_pager(void)
48 {
49         const char *pager = getenv("GIT_PAGER");
50
51         if (!isatty(1))
52                 return;
53         if (!pager) {
54                 if (!pager_program)
55                         git_config(git_default_config, NULL);
56                 pager = pager_program;
57         }
58         if (!pager)
59                 pager = getenv("PAGER");
60         if (!pager)
61                 pager = "less";
62         else if (!*pager || !strcmp(pager, "cat"))
63                 return;
64
65         spawned_pager = 1; /* means we are emitting to terminal */
66
67         /* spawn the pager */
68         pager_argv[2] = pager;
69         pager_process.argv = pager_argv;
70         pager_process.in = -1;
71         if (!getenv("LESS")) {
72                 static const char *env[] = { "LESS=FRSX", NULL };
73                 pager_process.env = env;
74         }
75 #ifndef WIN32
76         pager_process.preexec_cb = pager_preexec;
77 #endif
78         if (start_command(&pager_process))
79                 return;
80
81         /* original process continues, but writes to the pipe */
82         dup2(pager_process.in, 1);
83         if (isatty(2))
84                 dup2(pager_process.in, 2);
85         close(pager_process.in);
86
87         /* this makes sure that the parent terminates after the pager */
88         sigchain_push_common(wait_for_pager_signal);
89         atexit(wait_for_pager);
90 }
91
92 int pager_in_use(void)
93 {
94         const char *env;
95
96         if (spawned_pager)
97                 return 1;
98
99         env = getenv("GIT_PAGER_IN_USE");
100         return env ? git_config_bool("GIT_PAGER_IN_USE", env) : 0;
101 }