1 #include <sys/select.h>
8 #include "run-command.h"
10 #include "subcmd-config.h"
13 * This is split up from the rest of git so that we can do
14 * something different on Windows.
17 static int spawned_pager
;
18 static int pager_columns
;
20 void pager_init(const char *pager_env
)
22 subcmd_config
.pager_env
= pager_env
;
25 static void pager_preexec(void)
28 * Work around bug in "less" by not starting it until we
35 select(1, &in
, NULL
, &in
, NULL
);
37 setenv("LESS", "FRSX", 0);
40 static const char *pager_argv
[] = { "sh", "-c", NULL
, NULL
};
41 static struct child_process pager_process
;
43 static void wait_for_pager(void)
47 /* signal EOF to pager */
50 finish_command(&pager_process
);
53 static void wait_for_pager_signal(int signo
)
60 void setup_pager(void)
62 const char *pager
= getenv(subcmd_config
.pager_env
);
67 if (ioctl(1, TIOCGWINSZ
, &sz
) == 0)
68 pager_columns
= sz
.ws_col
;
70 pager
= getenv("PAGER");
71 if (!(pager
|| access("/usr/bin/pager", X_OK
)))
72 pager
= "/usr/bin/pager";
73 if (!(pager
|| access("/usr/bin/less", X_OK
)))
74 pager
= "/usr/bin/less";
77 if (!*pager
|| !strcmp(pager
, "cat"))
80 spawned_pager
= 1; /* means we are emitting to terminal */
83 pager_argv
[2] = pager
;
84 pager_process
.argv
= pager_argv
;
85 pager_process
.in
= -1;
86 pager_process
.preexec_cb
= pager_preexec
;
88 if (start_command(&pager_process
))
91 /* original process continues, but writes to the pipe */
92 dup2(pager_process
.in
, 1);
94 dup2(pager_process
.in
, 2);
95 close(pager_process
.in
);
97 /* this makes sure that the parent terminates after the pager */
98 sigchain_push_common(wait_for_pager_signal
);
99 atexit(wait_for_pager
);
102 int pager_in_use(void)
104 return spawned_pager
;
107 int pager_get_columns(void)
111 s
= getenv("COLUMNS");
115 return (pager_columns
? pager_columns
: 80) - 2;