260 lines
6.9 KiB
C
260 lines
6.9 KiB
C
/* Copyright (C) 1995 Free Software Foundation, Inc.
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Library General Public License as
|
|
published by the Free Software Foundation; either version 2 of the
|
|
License, or (at your option) any later version.
|
|
|
|
The GNU C Library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Library General Public
|
|
License along with the GNU C Library; see the file COPYING.LIB. If
|
|
not, write to the Free Software Foundation, Inc., 675 Mass Ave,
|
|
Cambridge, MA 02139, USA. */
|
|
|
|
#include <getopt.h>
|
|
#include <libintl.h>
|
|
#include <locale.h>
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <sys/stat.h>
|
|
#include <errno.h>
|
|
#include "error.h"
|
|
|
|
#include "localedef.h"
|
|
|
|
/* The charmap file used. If none given DEFAULT_CHARMAP is used. */
|
|
static char *charmap_file;
|
|
|
|
/* If set output is always written, even when warning are given. */
|
|
static int force_output;
|
|
|
|
/* The input file name. */
|
|
static char *input_file;
|
|
|
|
/* Path leading to the destination directory for the produced files. */
|
|
char *output_path;
|
|
|
|
/* If this is defined be POSIX conform. */
|
|
int posix_conformance;
|
|
|
|
/* If not zero give a lot more messages. */
|
|
int verbose;
|
|
|
|
/* Long options. */
|
|
static const struct option long_options[] =
|
|
{
|
|
{ "charmap", required_argument, NULL, 'f' },
|
|
{ "debug", no_argument, NULL, 'd' },
|
|
{ "help", no_argument, NULL, 'h' },
|
|
{ "force", no_argument, NULL, 'c' },
|
|
{ "inputfile", required_argument, NULL, 'i' },
|
|
{ "posix", no_argument, &posix_conformance, 1 },
|
|
{ "verbose", no_argument, &verbose, 1},
|
|
{ "version", no_argument, NULL, 'V' },
|
|
{ NULL, 0, NULL, 0 }
|
|
};
|
|
|
|
|
|
/* Prototypes for local functions. */
|
|
static void usage (int status) __attribute__ ((noreturn));
|
|
static int construct_output_path (const char *path);
|
|
|
|
int
|
|
main(int argc, char *argv[])
|
|
{
|
|
int optchar;
|
|
int cannot_write;
|
|
int do_help = 0;
|
|
int do_version = 0;
|
|
|
|
/* Set initial values for global varaibles. */
|
|
charmap_file = NULL;
|
|
force_output = 0;
|
|
input_file = 0;
|
|
posix_conformance = getenv ("POSIXLY_CORRECT") != NULL;
|
|
verbose = 0;
|
|
|
|
/* Set locale. Do not set LC_ALL because the other categories must
|
|
not be affected (acccording to POSIX.2). */
|
|
setlocale (LC_MESSAGES, "");
|
|
setlocale (LC_CTYPE, "");
|
|
|
|
/* Initialize the message catalog. */
|
|
textdomain (PACKAGE);
|
|
|
|
while ((optchar = getopt_long (argc, argv, "cdf:hi:vV", long_options, NULL))
|
|
!= EOF)
|
|
switch (optchar)
|
|
{
|
|
case '\0':
|
|
break;
|
|
case 'c':
|
|
force_output = 1;
|
|
break;
|
|
case 'f':
|
|
if (charmap_file != NULL)
|
|
error (0, 0, gettext ("\"%s %s\" overwrites old option \"%s\""),
|
|
"-f", optarg, charmap_file);
|
|
charmap_file = optarg;
|
|
break;
|
|
case 'h':
|
|
do_help = 1;
|
|
break;
|
|
case 'i':
|
|
if (input_file != NULL)
|
|
error (0, 0, gettext ("\"%s %s\" overwrites old option \"%s\""),
|
|
"-i", optarg, input_file);
|
|
input_file = optarg;
|
|
break;
|
|
case 'v':
|
|
verbose = 1;
|
|
break;
|
|
case 'V':
|
|
do_version = 1;
|
|
break;
|
|
default:
|
|
usage (4);
|
|
break;
|
|
}
|
|
|
|
/* POSIX.2 requires to be verbose about missing characters in the
|
|
character map. */
|
|
verbose |= posix_conformance;
|
|
|
|
/* Version information is requested. */
|
|
if (do_version)
|
|
{
|
|
fprintf (stderr, "GNU %s %s\n", PACKAGE, VERSION);
|
|
exit (EXIT_SUCCESS);
|
|
}
|
|
|
|
/* Help is requested. */
|
|
if (do_help)
|
|
usage (0);
|
|
|
|
if (argc - optind != 1)
|
|
/* We need exactly one non-option parameter. */
|
|
usage (4);
|
|
|
|
/* The parameter describes the output path of the constructed files.
|
|
If the files cannot be written return a non-zero value. */
|
|
cannot_write = construct_output_path (argv[optind]);
|
|
|
|
/* Now that the parameters are processed we have to reset the local
|
|
ctype locale. (POSIX.2 4.35.5.2) */
|
|
setlocale (LC_CTYPE, "POSIX");
|
|
|
|
/* Look whether the system really allows locale definitions. */
|
|
if (sysconf (_SC_2_LOCALEDEF) < 0)
|
|
error (3, 0,
|
|
gettext ("warning: system does not define `_POSIX2_LOCALEDEF'"));
|
|
|
|
/* Process charmap file. */
|
|
charmap_read (charmap_file);
|
|
|
|
/* Now read the locale file. */
|
|
locfile_read (input_file);
|
|
|
|
/* Check all categories for consistency. */
|
|
categories_check ();
|
|
|
|
/* We are now able to write the data files. If warning were given we
|
|
do it only if it is explicitly requested (--force). */
|
|
if (error_message_count == 0 || force_output != 0)
|
|
if (cannot_write != 0)
|
|
error (0, 0, gettext ("cannot write output file `%s': %s"),
|
|
output_path, strerror (cannot_write));
|
|
else
|
|
categories_write ();
|
|
else
|
|
error (0, 0,
|
|
gettext ("no output file produced because warning were issued"));
|
|
|
|
exit (EXIT_SUCCESS);
|
|
}
|
|
|
|
|
|
/* Display usage information and exit. */
|
|
static void
|
|
usage(int status)
|
|
{
|
|
if (status != EXIT_SUCCESS)
|
|
fprintf (stderr, gettext ("Try `%s --help' for more information.\n"),
|
|
program_invocation_name);
|
|
else
|
|
printf(gettext ("\
|
|
Usage: %s [OPTION]... name\n\
|
|
Mandatory arguments to long options are mandatory for short options too.\n\
|
|
-c, --force create output even if warning messages have been issued\n\
|
|
-h, --help display this help and exit\n\
|
|
-V, --version output version information and exit\n\
|
|
\n\
|
|
-i, --inputfile=FILE source definitions are found in FILE\n\
|
|
-f, --charmap=FILE symbolic character names defined in FILE\n\
|
|
\n\
|
|
-v, --verbose print more messages\n\
|
|
--posix be strictly POSIX conform\n\
|
|
\n\
|
|
System's directory for character maps: %s\n\
|
|
locale files : %s\n\
|
|
"), program_invocation_name, CHARMAP_PATH, LOCALE_PATH);
|
|
|
|
exit (status);
|
|
}
|
|
|
|
|
|
/* The parameter to localedef describes the output path. If it does
|
|
contain a '/' character it is a relativ path. Otherwise it names the
|
|
locale this definition is for. */
|
|
static int
|
|
construct_output_path (const char *path)
|
|
{
|
|
int result = 0;
|
|
|
|
if (strchr (path, '/') == NULL)
|
|
{
|
|
/* This is a system path. */
|
|
int path_max_len = pathconf (LOCALE_PATH, _PC_PATH_MAX) + 1;
|
|
output_path = (char *) xmalloc (path_max_len);
|
|
|
|
snprintf (output_path, path_max_len, "%s/%s", LOCALE_PATH, path);
|
|
}
|
|
else
|
|
{
|
|
char *t;
|
|
/* This is a user path. */
|
|
output_path = malloc (strlen (path) + 2);
|
|
t = stpcpy (output_path, path);
|
|
*t = '\0';
|
|
}
|
|
|
|
if (euidaccess (output_path, W_OK) == -1)
|
|
/* Perhaps the directory does not exist now. Try to create it. */
|
|
if (errno == ENOENT)
|
|
{
|
|
if (mkdir (output_path, 0777) == -1)
|
|
result = errno;
|
|
}
|
|
else
|
|
result = errno;
|
|
|
|
if (result == 0)
|
|
strcat (output_path, "/");
|
|
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* Local Variables:
|
|
* mode:c
|
|
* c-basic-offset:2
|
|
* End:
|
|
*/
|