1 /* dircolors - output commands to set the LS_COLOR environment variable
2 Copyright (C) 1994, 1995, 1997, 1998, 1999 H. Peter Anvin
3 Copyright (C) 1996-1999 Free Software Foundation, Inc.
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2, or (at your option)
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software Foundation,
17 Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
23 #include <sys/types.h>
29 #include "dircolors.h"
34 /* The official name of this program (e.g., no `g' prefix). */
35 #define PROGRAM_NAME "dircolors"
37 #define AUTHORS "H. Peter Anvin"
39 #define obstack_chunk_alloc malloc
40 #define obstack_chunk_free free
51 #define APPEND_CHAR(C) obstack_1grow (&lsc_obstack, C)
52 #define APPEND_TWO_CHAR_STRING(S) \
60 /* Accumulate in this obstack the value for the LS_COLORS environment
62 static struct obstack lsc_obstack
;
64 /* Nonzero if the input file was the standard input. */
65 static int have_read_stdin
;
67 /* FIXME: associate with ls_codes? */
68 static const char *const slack_codes
[] =
70 "NORMAL", "NORM", "FILE", "DIR", "LNK", "LINK",
71 "SYMLINK", "ORPHAN", "MISSING", "FIFO", "PIPE", "SOCK", "BLK", "BLOCK",
72 "CHR", "CHAR", "DOOR", "EXEC", "LEFT", "LEFTCODE", "RIGHT", "RIGHTCODE",
73 "END", "ENDCODE", NULL
76 static const char *const ls_codes
[] =
78 "no", "no", "fi", "di", "ln", "ln", "ln", "or", "mi", "pi", "pi",
79 "so", "bd", "bd", "cd", "cd", "do", "ex", "lc", "lc", "rc", "rc", "ec", "ec"
82 static struct option
const long_options
[] =
84 {"bourne-shell", no_argument
, NULL
, 'b'},
85 {"sh", no_argument
, NULL
, 'b'},
86 {"csh", no_argument
, NULL
, 'c'},
87 {"c-shell", no_argument
, NULL
, 'c'},
88 {"print-database", no_argument
, NULL
, 'p'},
89 {GETOPT_HELP_OPTION_DECL
},
90 {GETOPT_VERSION_OPTION_DECL
},
100 fprintf (stderr
, _("Try `%s --help' for more information.\n"),
104 printf (_("Usage: %s [OPTION]... [FILE]\n"), program_name
);
106 Output commands to set the LS_COLORS environment variable.\n\
108 Determine format of output:\n\
109 -b, --sh, --bourne-shell output Bourne shell code to set LS_COLORS\n\
110 -c, --csh, --c-shell output C shell code to set LS_COLORS\n\
111 -p, --print-database output defaults\n\
112 --help display this help and exit\n\
113 --version output version information and exit\n\
115 If FILE is specified, read it to determine which colors to use for which\n\
116 file types and extensions. Otherwise, a precompiled database is used.\n\
117 For details on the format of these files, run `dircolors --print-database'.\n\
119 puts (_("\nReport bugs to <bug-fileutils@gnu.org>."));
123 exit (status
== 0 ? EXIT_SUCCESS
: EXIT_FAILURE
);
127 xstrndup (const char *s
, size_t n
)
129 char *new = strndup (s
, n
);
131 error (EXIT_FAILURE
, 0, _("virtual memory exhausted"));
135 /* If the SHELL environment variable is set to `csh' or `tcsh,'
136 assume C shell. Else Bourne shell. */
138 static enum Shell_syntax
139 guess_shell_syntax (void)
143 shell
= getenv ("SHELL");
144 if (shell
== NULL
|| *shell
== '\0')
145 return SHELL_SYNTAX_UNKNOWN
;
147 shell
= base_name (shell
);
149 if (STREQ (shell
, "csh") || STREQ (shell
, "tcsh"))
150 return SHELL_SYNTAX_C
;
152 return SHELL_SYNTAX_BOURNE
;
156 parse_line (unsigned char const *line
, char **keyword
, char **arg
)
158 unsigned char const *p
;
159 unsigned char const *keyword_start
;
160 unsigned char const *arg_start
;
165 for (p
= line
; ISSPACE (*p
); ++p
)
168 /* Ignore blank lines and shell-style comments. */
169 if (*p
== '\0' || *p
== '#')
174 while (!ISSPACE (*p
) && *p
!= '\0')
179 *keyword
= xstrndup ((const char *) keyword_start
, p
- keyword_start
);
187 while (ISSPACE (*p
));
189 if (*p
== '\0' || *p
== '#')
194 while (*p
!= '\0' && *p
!= '#')
197 for (--p
; ISSPACE (*p
); --p
)
203 *arg
= xstrndup ((const char *) arg_start
, p
- arg_start
);
206 /* FIXME: Write a string to standard out, while watching for "dangerous"
207 sequences like unescaped : and = characters. */
210 append_quoted (const char *str
)
212 int need_backslash
= 1;
220 need_backslash
= !need_backslash
;
239 /* Read the file open on FP (with name FILENAME). First, look for a
240 `TERM name' directive where name matches the current terminal type.
241 Once found, translate and accumulate the associated directives onto
242 the global obstack LSC_OBSTACK. Give a diagnostic and return nonzero
243 upon failure (unrecognized keyword is the only way to fail here).
244 Return zero otherwise. */
247 dc_parse_stream (FILE *fp
, const char *filename
)
249 size_t line_number
= 0;
251 size_t line_chars_allocated
= 0;
256 /* State for the parser. */
257 enum states
{ ST_TERMNO
, ST_TERMYES
, ST_TERMSURE
, ST_GLOBAL
};
261 /* Get terminal type */
262 term
= getenv ("TERM");
263 if (term
== NULL
|| *term
== '\0')
276 line_length
= getline (&line
, &line_chars_allocated
, fp
);
277 if (line_length
<= 0)
286 line
= (char *) (G_line
[line_number
- 1]);
287 line_length
= G_line_length
[line_number
- 1];
288 if (line_number
> G_N_LINES
)
292 parse_line ((unsigned char *) line
, &keywd
, &arg
);
299 error (0, 0, _("%s:%lu: invalid line; missing second token"),
300 filename
, (long unsigned) line_number
);
307 if (strcasecmp (keywd
, "TERM") == 0)
309 if (STREQ (arg
, term
))
311 else if (state
!= ST_TERMSURE
)
316 if (state
== ST_TERMSURE
)
317 state
= ST_TERMYES
; /* Another TERM can cancel */
319 if (state
!= ST_TERMNO
)
324 append_quoted (keywd
);
329 else if (keywd
[0] == '*')
331 append_quoted (keywd
);
336 else if (strcasecmp (keywd
, "OPTIONS") == 0
337 || strcasecmp (keywd
, "COLOR") == 0
338 || strcasecmp (keywd
, "EIGHTBIT") == 0)
346 for (i
= 0; slack_codes
[i
] != NULL
; ++i
)
347 if (strcasecmp (keywd
, slack_codes
[i
]) == 0)
350 if (slack_codes
[i
] != NULL
)
352 APPEND_TWO_CHAR_STRING (ls_codes
[i
]);
369 if (unrecognized
&& (state
== ST_TERMSURE
|| state
== ST_TERMYES
))
371 error (0, 0, _("%s:%lu: unrecognized keyword `%s'"),
372 (filename
? filename
: _("<internal>")),
373 (long unsigned) line_number
, keywd
);
386 dc_parse_file (const char *filename
)
391 if (STREQ (filename
, "-"))
398 /* OPENOPTS is a macro. It varies with the system.
399 Some systems distinguish between internal and
400 external text representations. */
402 fp
= fopen (filename
, "r");
405 error (0, errno
, "%s", filename
);
410 err
= dc_parse_stream (fp
, filename
);
412 if (fp
!= stdin
&& fclose (fp
) == EOF
)
414 error (0, errno
, "%s", filename
);
422 main (int argc
, char **argv
)
426 enum Shell_syntax syntax
= SHELL_SYNTAX_UNKNOWN
;
427 int print_database
= 0;
429 program_name
= argv
[0];
430 setlocale (LC_ALL
, "");
431 bindtextdomain (PACKAGE
, LOCALEDIR
);
432 textdomain (PACKAGE
);
434 while ((optc
= getopt_long (argc
, argv
, "bcp", long_options
, NULL
)) != -1)
437 case 'b': /* Bourne shell syntax. */
438 syntax
= SHELL_SYNTAX_BOURNE
;
441 case 'c': /* C shell syntax. */
442 syntax
= SHELL_SYNTAX_C
;
449 case_GETOPT_HELP_CHAR
;
451 case_GETOPT_VERSION_CHAR (PROGRAM_NAME
, AUTHORS
);
460 /* It doesn't make sense to use --print with either of
461 --bourne or --c-shell. */
462 if (print_database
&& syntax
!= SHELL_SYNTAX_UNKNOWN
)
465 _("the options to output dircolors' internal database and\n\
466 to select a shell syntax are mutually exclusive"));
470 if (print_database
&& argc
> 0)
473 _("no FILE arguments may be used with the option to output\n\
474 dircolors' internal database"));
478 if (!print_database
&& argc
> 1)
480 error (0, 0, _("too many arguments"));
487 for (i
= 0; i
< G_N_LINES
; i
++)
489 fwrite (G_line
[i
], 1, G_line_length
[i
], stdout
);
490 fputc ('\n', stdout
);
495 /* If shell syntax was not explicitly specified, try to guess it. */
496 if (syntax
== SHELL_SYNTAX_UNKNOWN
)
498 syntax
= guess_shell_syntax ();
499 if (syntax
== SHELL_SYNTAX_UNKNOWN
)
501 error (EXIT_FAILURE
, 0,
502 _("no SHELL environment variable, and no shell type option given"));
506 obstack_init (&lsc_obstack
);
508 err
= dc_parse_stream (NULL
, NULL
);
510 err
= dc_parse_file (argv
[0]);
514 size_t len
= obstack_object_size (&lsc_obstack
);
515 char *s
= obstack_finish (&lsc_obstack
);
519 if (syntax
== SHELL_SYNTAX_BOURNE
)
521 prefix
= "LS_COLORS='";
522 suffix
= "';\nexport LS_COLORS\n";
526 prefix
= "setenv LS_COLORS '";
529 fputs (prefix
, stdout
);
530 fwrite (s
, 1, len
, stdout
);
531 fputs (suffix
, stdout
);
537 if (have_read_stdin
&& fclose (stdin
) == EOF
)
538 error (EXIT_FAILURE
, errno
, _("standard input"));
540 exit (err
== 0 ? EXIT_SUCCESS
: EXIT_FAILURE
);