1 // SPDX-License-Identifier: GPL-2.0
2 #include <sys/select.h>
9 #include "run-command.h"
11 #include "subcmd-config.h"
14 * This is split up from the rest of git so that we can do
15 * something different on Windows.
18 static int spawned_pager
;
19 static int pager_columns
;
21 void pager_init(const char *pager_env
)
23 subcmd_config
.pager_env
= pager_env
;
26 static const char *forced_pager
;
28 void force_pager(const char *pager
)
33 static void pager_preexec(void)
36 * Work around bug in "less" by not starting it until we
45 FD_SET(0, &exception
);
46 select(1, &in
, NULL
, &exception
, NULL
);
48 setenv("LESS", "FRSX", 0);
51 static const char *pager_argv
[] = { "sh", "-c", NULL
, NULL
};
52 static struct child_process pager_process
;
54 static void wait_for_pager(void)
58 /* signal EOF to pager */
61 finish_command(&pager_process
);
64 static void wait_for_pager_signal(int signo
)
71 void setup_pager(void)
73 const char *pager
= getenv(subcmd_config
.pager_env
);
78 if (!isatty(1) && !forced_pager
)
80 if (ioctl(1, TIOCGWINSZ
, &sz
) == 0)
81 pager_columns
= sz
.ws_col
;
83 pager
= getenv("PAGER");
84 if (!(pager
|| access("/usr/bin/pager", X_OK
)))
85 pager
= "/usr/bin/pager";
86 if (!(pager
|| access("/usr/bin/less", X_OK
)))
87 pager
= "/usr/bin/less";
90 if (!*pager
|| !strcmp(pager
, "cat"))
93 spawned_pager
= 1; /* means we are emitting to terminal */
96 pager_argv
[2] = pager
;
97 pager_process
.argv
= pager_argv
;
98 pager_process
.in
= -1;
99 pager_process
.preexec_cb
= pager_preexec
;
101 if (start_command(&pager_process
))
104 /* original process continues, but writes to the pipe */
105 dup2(pager_process
.in
, 1);
107 dup2(pager_process
.in
, 2);
108 close(pager_process
.in
);
110 /* this makes sure that the parent terminates after the pager */
111 sigchain_push_common(wait_for_pager_signal
);
112 atexit(wait_for_pager
);
115 int pager_in_use(void)
117 return spawned_pager
;
120 int pager_get_columns(void)
124 s
= getenv("COLUMNS");
128 return (pager_columns
? pager_columns
: 80) - 2;