72d58a6f4c1551204d15661cae7980c1e75b937e
[platform/upstream/busybox.git] / util-linux / more.c
1 /*
2  * Mini more implementation for busybox
3  *
4  * Copyright (C) 1998 by Erik Andersen <andersee@debian.org>
5  *
6  * This program is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation; either version 2 of the License, or
9  * (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14  * General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19  *
20  */
21
22
23 /* Turning this off makes things a bit smaller (and less pretty) */
24 #define BB_MORE_TERM
25
26
27
28 #include "internal.h"
29 #include <stdio.h>
30 #include <fcntl.h>
31 #include <signal.h>
32
33
34 static const char more_usage[] = "[file ...]";
35
36
37 /* ED: sparc termios is broken: revert back to old termio handling. */
38 #ifdef BB_MORE_TERM
39
40
41 #if defined (__sparc__)
42 #      define USE_OLD_TERMIO
43 #      include <termio.h>
44 #      include <sys/ioctl.h>
45 #      define termios termio
46 #      define stty(fd,argp) ioctl(fd,TCSETAF,argp)
47 #else
48 #      include <termios.h>
49 #      define stty(fd,argp) tcsetattr(fd,TCSANOW,argp)
50 #endif
51
52     FILE *cin;
53     struct termios initial_settings, new_settings;
54
55     void gotsig(int sig) { 
56             stty(fileno(cin), &initial_settings);
57             exit( TRUE);
58     }
59 #endif
60
61 extern int more_main(int argc, char **argv)
62 {
63     int c, lines=0, input=0;
64     int next_page=0;
65     struct stat st;     
66     FILE *file;
67
68     if ( strcmp(*argv,"--help")==0 || strcmp(*argv,"-h")==0 ) {
69         usage (more_usage);
70     }
71     argc--;
72     argv++;
73
74     while (argc >= 0) {
75         if (argc==0) {
76             file = stdin;
77         }
78         else
79             file = fopen(*argv, "r");
80
81         if (file == NULL) {
82             perror(*argv);
83             exit(FALSE);
84         }
85         fstat(fileno(file), &st);
86
87 #ifdef BB_MORE_TERM
88         cin = fopen("/dev/tty", "r");
89         if (!cin)
90             cin = fopen("/dev/console", "r");
91 #ifdef USE_OLD_TERMIO
92         ioctl(fileno(cin),TCGETA,&initial_settings);
93 #else
94         tcgetattr(fileno(cin),&initial_settings);
95 #endif
96         new_settings = initial_settings;
97         new_settings.c_lflag &= ~ICANON;
98         new_settings.c_lflag &= ~ECHO;
99         stty(fileno(cin), &new_settings);
100         
101         (void) signal(SIGINT, gotsig);
102
103 #endif
104         while ((c = getc(file)) != EOF) {
105             if ( next_page ) {
106                 int len=0;
107                 next_page = 0;
108                 lines=0;
109                 len = fprintf(stdout, "--More-- ");
110                 if (file != stdin) {
111                     len += fprintf(stdout, "(%d%% of %ld bytes)", 
112                         (int) (100*( (double) ftell(file) / (double) st.st_size )),
113                         st.st_size);
114                 }
115                 len += fprintf(stdout, "%s",
116 #ifdef BB_MORE_TERM
117                         ""
118 #else
119                         "\n"
120 #endif
121                         );
122
123                 fflush(stdout);
124                 input = getc( cin);
125
126 #ifdef BB_MORE_TERM
127                 /* Erase the "More" message */
128                 while(len-- > 0)
129                     putc('\b', stdout);
130                 while(len++ < 79)
131                     putc(' ', stdout);
132                 while(len-- > 0)
133                     putc('\b', stdout);
134                 fflush(stdout);
135 #endif
136
137             }
138             if (input=='q')
139                 goto end;
140             if (input==' ' &&  c == '\n' )
141                 next_page = 1;
142             if ( c == '\n' && ++lines == 24 )
143                 next_page = 1;
144             putc(c, stdout);
145         }
146         fclose(file);
147         fflush(stdout);
148
149         argc--;
150         argv++;
151     }
152 end:
153 #ifdef BB_MORE_TERM
154     gotsig(0);
155 #endif  
156     exit(TRUE);
157 }
158